You can not select more than 25 topics Topics must start with a chinese character,a letter or number, can include dashes ('-') and can be up to 35 characters long.

substitute.h 34 kB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664
  1. //
  2. // Copyright 2017 The Abseil Authors.
  3. //
  4. // Licensed under the Apache License, Version 2.0 (the "License");
  5. // you may not use this file except in compliance with the License.
  6. // You may obtain a copy of the License at
  7. //
  8. // https://www.apache.org/licenses/LICENSE-2.0
  9. //
  10. // Unless required by applicable law or agreed to in writing, software
  11. // distributed under the License is distributed on an "AS IS" BASIS,
  12. // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. // See the License for the specific language governing permissions and
  14. // limitations under the License.
  15. //
  16. // -----------------------------------------------------------------------------
  17. // File: substitute.h
  18. // -----------------------------------------------------------------------------
  19. //
  20. // This package contains functions for efficiently performing string
  21. // substitutions using a format string with positional notation:
  22. // `Substitute()` and `SubstituteAndAppend()`.
  23. //
  24. // Unlike printf-style format specifiers, `Substitute()` functions do not need
  25. // to specify the type of the substitution arguments. Supported arguments
  26. // following the format string, such as strings, string_views, ints,
  27. // floats, and bools, are automatically converted to strings during the
  28. // substitution process. (See below for a full list of supported types.)
  29. //
  30. // `Substitute()` does not allow you to specify *how* to format a value, beyond
  31. // the default conversion to string. For example, you cannot format an integer
  32. // in hex.
  33. //
  34. // The format string uses positional identifiers indicated by a dollar sign ($)
  35. // and single digit positional ids to indicate which substitution arguments to
  36. // use at that location within the format string.
  37. //
  38. // A '$$' sequence in the format string causes a literal '$' character to be
  39. // output.
  40. //
  41. // Example 1:
  42. // std::string s = Substitute("$1 purchased $0 $2 for $$10. Thanks $1!",
  43. // 5, "Bob", "Apples");
  44. // EXPECT_EQ("Bob purchased 5 Apples for $10. Thanks Bob!", s);
  45. //
  46. // Example 2:
  47. // std::string s = "Hi. ";
  48. // SubstituteAndAppend(&s, "My name is $0 and I am $1 years old.", "Bob", 5);
  49. // EXPECT_EQ("Hi. My name is Bob and I am 5 years old.", s);
  50. //
  51. // Supported types:
  52. // * absl::string_view, std::string, const char* (null is equivalent to "")
  53. // * int32_t, int64_t, uint32_t, uint64_t
  54. // * float, double
  55. // * bool (Printed as "true" or "false")
  56. // * pointer types other than char* (Printed as "0x<lower case hex string>",
  57. // except that null is printed as "NULL")
  58. //
  59. // If an invalid format string is provided, Substitute returns an empty string
  60. // and SubstituteAndAppend does not change the provided output string.
  61. // A format string is invalid if it:
  62. // * ends in an unescaped $ character,
  63. // e.g. "Hello $", or
  64. // * calls for a position argument which is not provided,
  65. // e.g. Substitute("Hello $2", "world"), or
  66. // * specifies a non-digit, non-$ character after an unescaped $ character,
  67. // e.g. "Hello $f".
  68. // In debug mode, i.e. #ifndef NDEBUG, such errors terminate the program.
  69. #ifndef ABSL_STRINGS_SUBSTITUTE_H_
  70. #define ABSL_STRINGS_SUBSTITUTE_H_
  71. #include <cstring>
  72. #include <string>
  73. #include <type_traits>
  74. #include <vector>
  75. #include "absl/base/macros.h"
  76. #include "absl/base/port.h"
  77. #include "absl/strings/ascii.h"
  78. #include "absl/strings/escaping.h"
  79. #include "absl/strings/numbers.h"
  80. #include "absl/strings/str_cat.h"
  81. #include "absl/strings/str_split.h"
  82. #include "absl/strings/string_view.h"
  83. #include "absl/strings/strip.h"
  84. namespace absl
  85. {
  86. ABSL_NAMESPACE_BEGIN
  87. namespace substitute_internal
  88. {
  89. // Arg
  90. //
  91. // This class provides an argument type for `absl::Substitute()` and
  92. // `absl::SubstituteAndAppend()`. `Arg` handles implicit conversion of various
  93. // types to a string. (`Arg` is very similar to the `AlphaNum` class in
  94. // `StrCat()`.)
  95. //
  96. // This class has implicit constructors.
  97. class Arg
  98. {
  99. public:
  100. // Overloads for string-y things
  101. //
  102. // Explicitly overload `const char*` so the compiler doesn't cast to `bool`.
  103. Arg(const char* value) // NOLINT(runtime/explicit)
  104. :
  105. piece_(absl::NullSafeStringView(value))
  106. {
  107. }
  108. template<typename Allocator>
  109. Arg( // NOLINT
  110. const std::basic_string<char, std::char_traits<char>, Allocator>&
  111. value
  112. ) noexcept
  113. :
  114. piece_(value)
  115. {
  116. }
  117. Arg(absl::string_view value) // NOLINT(runtime/explicit)
  118. :
  119. piece_(value)
  120. {
  121. }
  122. // Overloads for primitives
  123. //
  124. // No overloads are available for signed and unsigned char because if people
  125. // are explicitly declaring their chars as signed or unsigned then they are
  126. // probably using them as 8-bit integers and would probably prefer an integer
  127. // representation. However, we can't really know, so we make the caller decide
  128. // what to do.
  129. Arg(char value) // NOLINT(runtime/explicit)
  130. :
  131. piece_(scratch_, 1)
  132. {
  133. scratch_[0] = value;
  134. }
  135. Arg(short value) // NOLINT(*)
  136. :
  137. piece_(scratch_, numbers_internal::FastIntToBuffer(value, scratch_) - scratch_)
  138. {
  139. }
  140. Arg(unsigned short value) // NOLINT(*)
  141. :
  142. piece_(scratch_, numbers_internal::FastIntToBuffer(value, scratch_) - scratch_)
  143. {
  144. }
  145. Arg(int value) // NOLINT(runtime/explicit)
  146. :
  147. piece_(scratch_, numbers_internal::FastIntToBuffer(value, scratch_) - scratch_)
  148. {
  149. }
  150. Arg(unsigned int value) // NOLINT(runtime/explicit)
  151. :
  152. piece_(scratch_, numbers_internal::FastIntToBuffer(value, scratch_) - scratch_)
  153. {
  154. }
  155. Arg(long value) // NOLINT(*)
  156. :
  157. piece_(scratch_, numbers_internal::FastIntToBuffer(value, scratch_) - scratch_)
  158. {
  159. }
  160. Arg(unsigned long value) // NOLINT(*)
  161. :
  162. piece_(scratch_, numbers_internal::FastIntToBuffer(value, scratch_) - scratch_)
  163. {
  164. }
  165. Arg(long long value) // NOLINT(*)
  166. :
  167. piece_(scratch_, numbers_internal::FastIntToBuffer(value, scratch_) - scratch_)
  168. {
  169. }
  170. Arg(unsigned long long value) // NOLINT(*)
  171. :
  172. piece_(scratch_, numbers_internal::FastIntToBuffer(value, scratch_) - scratch_)
  173. {
  174. }
  175. Arg(float value) // NOLINT(runtime/explicit)
  176. :
  177. piece_(scratch_, numbers_internal::SixDigitsToBuffer(value, scratch_))
  178. {
  179. }
  180. Arg(double value) // NOLINT(runtime/explicit)
  181. :
  182. piece_(scratch_, numbers_internal::SixDigitsToBuffer(value, scratch_))
  183. {
  184. }
  185. Arg(bool value) // NOLINT(runtime/explicit)
  186. :
  187. piece_(value ? "true" : "false")
  188. {
  189. }
  190. Arg(Hex hex); // NOLINT(runtime/explicit)
  191. Arg(Dec dec); // NOLINT(runtime/explicit)
  192. // vector<bool>::reference and const_reference require special help to convert
  193. // to `Arg` because it requires two user defined conversions.
  194. template<typename T, absl::enable_if_t<std::is_class<T>::value && (std::is_same<T, std::vector<bool>::reference>::value || std::is_same<T, std::vector<bool>::const_reference>::value)>* = nullptr>
  195. Arg(T value) // NOLINT(google-explicit-constructor)
  196. :
  197. Arg(static_cast<bool>(value))
  198. {
  199. }
  200. // `void*` values, with the exception of `char*`, are printed as
  201. // "0x<hex value>". However, in the case of `nullptr`, "NULL" is printed.
  202. Arg(const void* value); // NOLINT(runtime/explicit)
  203. // Normal enums are already handled by the integer formatters.
  204. // This overload matches only scoped enums.
  205. template<typename T, typename = typename std::enable_if<std::is_enum<T>{} && !std::is_convertible<T, int>{}>::type>
  206. Arg(T value) // NOLINT(google-explicit-constructor)
  207. :
  208. Arg(static_cast<typename std::underlying_type<T>::type>(value))
  209. {
  210. }
  211. Arg(const Arg&) = delete;
  212. Arg& operator=(const Arg&) = delete;
  213. absl::string_view piece() const
  214. {
  215. return piece_;
  216. }
  217. private:
  218. absl::string_view piece_;
  219. char scratch_[numbers_internal::kFastToBufferSize];
  220. };
  221. // Internal helper function. Don't call this from outside this implementation.
  222. // This interface may change without notice.
  223. void SubstituteAndAppendArray(std::string* output, absl::string_view format, const absl::string_view* args_array, size_t num_args);
  224. #if defined(ABSL_BAD_CALL_IF)
  225. constexpr int CalculateOneBit(const char* format)
  226. {
  227. // Returns:
  228. // * 2^N for '$N' when N is in [0-9]
  229. // * 0 for correct '$' escaping: '$$'.
  230. // * -1 otherwise.
  231. return (*format < '0' || *format > '9') ? (*format == '$' ? 0 : -1) : (1 << (*format - '0'));
  232. }
  233. constexpr const char* SkipNumber(const char* format)
  234. {
  235. return !*format ? format : (format + 1);
  236. }
  237. constexpr int PlaceholderBitmask(const char* format)
  238. {
  239. return !*format ? 0 : *format != '$' ? PlaceholderBitmask(format + 1) :
  240. (CalculateOneBit(format + 1) | PlaceholderBitmask(SkipNumber(format + 1)));
  241. }
  242. #endif // ABSL_BAD_CALL_IF
  243. } // namespace substitute_internal
  244. //
  245. // PUBLIC API
  246. //
  247. // SubstituteAndAppend()
  248. //
  249. // Substitutes variables into a given format string and appends to a given
  250. // output string. See file comments above for usage.
  251. //
  252. // The declarations of `SubstituteAndAppend()` below consist of overloads
  253. // for passing 0 to 10 arguments, respectively.
  254. //
  255. // NOTE: A zero-argument `SubstituteAndAppend()` may be used within variadic
  256. // templates to allow a variable number of arguments.
  257. //
  258. // Example:
  259. // template <typename... Args>
  260. // void VarMsg(std::string* boilerplate, absl::string_view format,
  261. // const Args&... args) {
  262. // absl::SubstituteAndAppend(boilerplate, format, args...);
  263. // }
  264. //
  265. inline void SubstituteAndAppend(std::string* output, absl::string_view format)
  266. {
  267. substitute_internal::SubstituteAndAppendArray(output, format, nullptr, 0);
  268. }
  269. inline void SubstituteAndAppend(std::string* output, absl::string_view format, const substitute_internal::Arg& a0)
  270. {
  271. const absl::string_view args[] = {a0.piece()};
  272. substitute_internal::SubstituteAndAppendArray(output, format, args, ABSL_ARRAYSIZE(args));
  273. }
  274. inline void SubstituteAndAppend(std::string* output, absl::string_view format, const substitute_internal::Arg& a0, const substitute_internal::Arg& a1)
  275. {
  276. const absl::string_view args[] = {a0.piece(), a1.piece()};
  277. substitute_internal::SubstituteAndAppendArray(output, format, args, ABSL_ARRAYSIZE(args));
  278. }
  279. inline void SubstituteAndAppend(std::string* output, absl::string_view format, const substitute_internal::Arg& a0, const substitute_internal::Arg& a1, const substitute_internal::Arg& a2)
  280. {
  281. const absl::string_view args[] = {a0.piece(), a1.piece(), a2.piece()};
  282. substitute_internal::SubstituteAndAppendArray(output, format, args, ABSL_ARRAYSIZE(args));
  283. }
  284. inline void SubstituteAndAppend(std::string* output, absl::string_view format, const substitute_internal::Arg& a0, const substitute_internal::Arg& a1, const substitute_internal::Arg& a2, const substitute_internal::Arg& a3)
  285. {
  286. const absl::string_view args[] = {a0.piece(), a1.piece(), a2.piece(), a3.piece()};
  287. substitute_internal::SubstituteAndAppendArray(output, format, args, ABSL_ARRAYSIZE(args));
  288. }
  289. inline void SubstituteAndAppend(std::string* output, absl::string_view format, const substitute_internal::Arg& a0, const substitute_internal::Arg& a1, const substitute_internal::Arg& a2, const substitute_internal::Arg& a3, const substitute_internal::Arg& a4)
  290. {
  291. const absl::string_view args[] = {a0.piece(), a1.piece(), a2.piece(), a3.piece(), a4.piece()};
  292. substitute_internal::SubstituteAndAppendArray(output, format, args, ABSL_ARRAYSIZE(args));
  293. }
  294. inline void SubstituteAndAppend(std::string* output, absl::string_view format, const substitute_internal::Arg& a0, const substitute_internal::Arg& a1, const substitute_internal::Arg& a2, const substitute_internal::Arg& a3, const substitute_internal::Arg& a4, const substitute_internal::Arg& a5)
  295. {
  296. const absl::string_view args[] = {a0.piece(), a1.piece(), a2.piece(), a3.piece(), a4.piece(), a5.piece()};
  297. substitute_internal::SubstituteAndAppendArray(output, format, args, ABSL_ARRAYSIZE(args));
  298. }
  299. inline void SubstituteAndAppend(std::string* output, absl::string_view format, const substitute_internal::Arg& a0, const substitute_internal::Arg& a1, const substitute_internal::Arg& a2, const substitute_internal::Arg& a3, const substitute_internal::Arg& a4, const substitute_internal::Arg& a5, const substitute_internal::Arg& a6)
  300. {
  301. const absl::string_view args[] = {a0.piece(), a1.piece(), a2.piece(), a3.piece(), a4.piece(), a5.piece(), a6.piece()};
  302. substitute_internal::SubstituteAndAppendArray(output, format, args, ABSL_ARRAYSIZE(args));
  303. }
  304. inline void SubstituteAndAppend(
  305. std::string* output, absl::string_view format, const substitute_internal::Arg& a0, const substitute_internal::Arg& a1, const substitute_internal::Arg& a2, const substitute_internal::Arg& a3, const substitute_internal::Arg& a4, const substitute_internal::Arg& a5, const substitute_internal::Arg& a6, const substitute_internal::Arg& a7
  306. )
  307. {
  308. const absl::string_view args[] = {a0.piece(), a1.piece(), a2.piece(), a3.piece(), a4.piece(), a5.piece(), a6.piece(), a7.piece()};
  309. substitute_internal::SubstituteAndAppendArray(output, format, args, ABSL_ARRAYSIZE(args));
  310. }
  311. inline void SubstituteAndAppend(
  312. std::string* output, absl::string_view format, const substitute_internal::Arg& a0, const substitute_internal::Arg& a1, const substitute_internal::Arg& a2, const substitute_internal::Arg& a3, const substitute_internal::Arg& a4, const substitute_internal::Arg& a5, const substitute_internal::Arg& a6, const substitute_internal::Arg& a7, const substitute_internal::Arg& a8
  313. )
  314. {
  315. const absl::string_view args[] = {a0.piece(), a1.piece(), a2.piece(), a3.piece(), a4.piece(), a5.piece(), a6.piece(), a7.piece(), a8.piece()};
  316. substitute_internal::SubstituteAndAppendArray(output, format, args, ABSL_ARRAYSIZE(args));
  317. }
  318. inline void SubstituteAndAppend(
  319. std::string* output, absl::string_view format, const substitute_internal::Arg& a0, const substitute_internal::Arg& a1, const substitute_internal::Arg& a2, const substitute_internal::Arg& a3, const substitute_internal::Arg& a4, const substitute_internal::Arg& a5, const substitute_internal::Arg& a6, const substitute_internal::Arg& a7, const substitute_internal::Arg& a8, const substitute_internal::Arg& a9
  320. )
  321. {
  322. const absl::string_view args[] = {
  323. a0.piece(), a1.piece(), a2.piece(), a3.piece(), a4.piece(), a5.piece(), a6.piece(), a7.piece(), a8.piece(), a9.piece()};
  324. substitute_internal::SubstituteAndAppendArray(output, format, args, ABSL_ARRAYSIZE(args));
  325. }
  326. #if defined(ABSL_BAD_CALL_IF)
  327. // This body of functions catches cases where the number of placeholders
  328. // doesn't match the number of data arguments.
  329. void SubstituteAndAppend(std::string* output, const char* format)
  330. ABSL_BAD_CALL_IF(
  331. substitute_internal::PlaceholderBitmask(format) != 0,
  332. "There were no substitution arguments "
  333. "but this format string either has a $[0-9] in it or contains "
  334. "an unescaped $ character (use $$ instead)"
  335. );
  336. void SubstituteAndAppend(std::string* output, const char* format, const substitute_internal::Arg& a0)
  337. ABSL_BAD_CALL_IF(substitute_internal::PlaceholderBitmask(format) != 1, "There was 1 substitution argument given, but "
  338. "this format string is missing its $0, contains "
  339. "one of $1-$9, or contains an unescaped $ character (use "
  340. "$$ instead)");
  341. void SubstituteAndAppend(std::string* output, const char* format, const substitute_internal::Arg& a0, const substitute_internal::Arg& a1)
  342. ABSL_BAD_CALL_IF(
  343. substitute_internal::PlaceholderBitmask(format) != 3,
  344. "There were 2 substitution arguments given, but this format string is "
  345. "missing its $0/$1, contains one of $2-$9, or contains an "
  346. "unescaped $ character (use $$ instead)"
  347. );
  348. void SubstituteAndAppend(std::string* output, const char* format, const substitute_internal::Arg& a0, const substitute_internal::Arg& a1, const substitute_internal::Arg& a2)
  349. ABSL_BAD_CALL_IF(
  350. substitute_internal::PlaceholderBitmask(format) != 7,
  351. "There were 3 substitution arguments given, but "
  352. "this format string is missing its $0/$1/$2, contains one of "
  353. "$3-$9, or contains an unescaped $ character (use $$ instead)"
  354. );
  355. void SubstituteAndAppend(std::string* output, const char* format, const substitute_internal::Arg& a0, const substitute_internal::Arg& a1, const substitute_internal::Arg& a2, const substitute_internal::Arg& a3)
  356. ABSL_BAD_CALL_IF(
  357. substitute_internal::PlaceholderBitmask(format) != 15,
  358. "There were 4 substitution arguments given, but "
  359. "this format string is missing its $0-$3, contains one of "
  360. "$4-$9, or contains an unescaped $ character (use $$ instead)"
  361. );
  362. void SubstituteAndAppend(std::string* output, const char* format, const substitute_internal::Arg& a0, const substitute_internal::Arg& a1, const substitute_internal::Arg& a2, const substitute_internal::Arg& a3, const substitute_internal::Arg& a4)
  363. ABSL_BAD_CALL_IF(
  364. substitute_internal::PlaceholderBitmask(format) != 31,
  365. "There were 5 substitution arguments given, but "
  366. "this format string is missing its $0-$4, contains one of "
  367. "$5-$9, or contains an unescaped $ character (use $$ instead)"
  368. );
  369. void SubstituteAndAppend(std::string* output, const char* format, const substitute_internal::Arg& a0, const substitute_internal::Arg& a1, const substitute_internal::Arg& a2, const substitute_internal::Arg& a3, const substitute_internal::Arg& a4, const substitute_internal::Arg& a5)
  370. ABSL_BAD_CALL_IF(
  371. substitute_internal::PlaceholderBitmask(format) != 63,
  372. "There were 6 substitution arguments given, but "
  373. "this format string is missing its $0-$5, contains one of "
  374. "$6-$9, or contains an unescaped $ character (use $$ instead)"
  375. );
  376. void SubstituteAndAppend(
  377. std::string* output, const char* format, const substitute_internal::Arg& a0, const substitute_internal::Arg& a1, const substitute_internal::Arg& a2, const substitute_internal::Arg& a3, const substitute_internal::Arg& a4, const substitute_internal::Arg& a5, const substitute_internal::Arg& a6
  378. )
  379. ABSL_BAD_CALL_IF(
  380. substitute_internal::PlaceholderBitmask(format) != 127,
  381. "There were 7 substitution arguments given, but "
  382. "this format string is missing its $0-$6, contains one of "
  383. "$7-$9, or contains an unescaped $ character (use $$ instead)"
  384. );
  385. void SubstituteAndAppend(
  386. std::string* output, const char* format, const substitute_internal::Arg& a0, const substitute_internal::Arg& a1, const substitute_internal::Arg& a2, const substitute_internal::Arg& a3, const substitute_internal::Arg& a4, const substitute_internal::Arg& a5, const substitute_internal::Arg& a6, const substitute_internal::Arg& a7
  387. )
  388. ABSL_BAD_CALL_IF(
  389. substitute_internal::PlaceholderBitmask(format) != 255,
  390. "There were 8 substitution arguments given, but "
  391. "this format string is missing its $0-$7, contains one of "
  392. "$8-$9, or contains an unescaped $ character (use $$ instead)"
  393. );
  394. void SubstituteAndAppend(
  395. std::string* output, const char* format, const substitute_internal::Arg& a0, const substitute_internal::Arg& a1, const substitute_internal::Arg& a2, const substitute_internal::Arg& a3, const substitute_internal::Arg& a4, const substitute_internal::Arg& a5, const substitute_internal::Arg& a6, const substitute_internal::Arg& a7, const substitute_internal::Arg& a8
  396. )
  397. ABSL_BAD_CALL_IF(
  398. substitute_internal::PlaceholderBitmask(format) != 511,
  399. "There were 9 substitution arguments given, but "
  400. "this format string is missing its $0-$8, contains a $9, or "
  401. "contains an unescaped $ character (use $$ instead)"
  402. );
  403. void SubstituteAndAppend(
  404. std::string* output, const char* format, const substitute_internal::Arg& a0, const substitute_internal::Arg& a1, const substitute_internal::Arg& a2, const substitute_internal::Arg& a3, const substitute_internal::Arg& a4, const substitute_internal::Arg& a5, const substitute_internal::Arg& a6, const substitute_internal::Arg& a7, const substitute_internal::Arg& a8, const substitute_internal::Arg& a9
  405. )
  406. ABSL_BAD_CALL_IF(
  407. substitute_internal::PlaceholderBitmask(format) != 1023,
  408. "There were 10 substitution arguments given, but this "
  409. "format string either doesn't contain all of $0 through $9 or "
  410. "contains an unescaped $ character (use $$ instead)"
  411. );
  412. #endif // ABSL_BAD_CALL_IF
  413. // Substitute()
  414. //
  415. // Substitutes variables into a given format string. See file comments above
  416. // for usage.
  417. //
  418. // The declarations of `Substitute()` below consist of overloads for passing 0
  419. // to 10 arguments, respectively.
  420. //
  421. // NOTE: A zero-argument `Substitute()` may be used within variadic templates to
  422. // allow a variable number of arguments.
  423. //
  424. // Example:
  425. // template <typename... Args>
  426. // void VarMsg(absl::string_view format, const Args&... args) {
  427. // std::string s = absl::Substitute(format, args...);
  428. ABSL_MUST_USE_RESULT inline std::string Substitute(absl::string_view format)
  429. {
  430. std::string result;
  431. SubstituteAndAppend(&result, format);
  432. return result;
  433. }
  434. ABSL_MUST_USE_RESULT inline std::string Substitute(
  435. absl::string_view format, const substitute_internal::Arg& a0
  436. )
  437. {
  438. std::string result;
  439. SubstituteAndAppend(&result, format, a0);
  440. return result;
  441. }
  442. ABSL_MUST_USE_RESULT inline std::string Substitute(
  443. absl::string_view format, const substitute_internal::Arg& a0, const substitute_internal::Arg& a1
  444. )
  445. {
  446. std::string result;
  447. SubstituteAndAppend(&result, format, a0, a1);
  448. return result;
  449. }
  450. ABSL_MUST_USE_RESULT inline std::string Substitute(
  451. absl::string_view format, const substitute_internal::Arg& a0, const substitute_internal::Arg& a1, const substitute_internal::Arg& a2
  452. )
  453. {
  454. std::string result;
  455. SubstituteAndAppend(&result, format, a0, a1, a2);
  456. return result;
  457. }
  458. ABSL_MUST_USE_RESULT inline std::string Substitute(
  459. absl::string_view format, const substitute_internal::Arg& a0, const substitute_internal::Arg& a1, const substitute_internal::Arg& a2, const substitute_internal::Arg& a3
  460. )
  461. {
  462. std::string result;
  463. SubstituteAndAppend(&result, format, a0, a1, a2, a3);
  464. return result;
  465. }
  466. ABSL_MUST_USE_RESULT inline std::string Substitute(
  467. absl::string_view format, const substitute_internal::Arg& a0, const substitute_internal::Arg& a1, const substitute_internal::Arg& a2, const substitute_internal::Arg& a3, const substitute_internal::Arg& a4
  468. )
  469. {
  470. std::string result;
  471. SubstituteAndAppend(&result, format, a0, a1, a2, a3, a4);
  472. return result;
  473. }
  474. ABSL_MUST_USE_RESULT inline std::string Substitute(
  475. absl::string_view format, const substitute_internal::Arg& a0, const substitute_internal::Arg& a1, const substitute_internal::Arg& a2, const substitute_internal::Arg& a3, const substitute_internal::Arg& a4, const substitute_internal::Arg& a5
  476. )
  477. {
  478. std::string result;
  479. SubstituteAndAppend(&result, format, a0, a1, a2, a3, a4, a5);
  480. return result;
  481. }
  482. ABSL_MUST_USE_RESULT inline std::string Substitute(
  483. absl::string_view format, const substitute_internal::Arg& a0, const substitute_internal::Arg& a1, const substitute_internal::Arg& a2, const substitute_internal::Arg& a3, const substitute_internal::Arg& a4, const substitute_internal::Arg& a5, const substitute_internal::Arg& a6
  484. )
  485. {
  486. std::string result;
  487. SubstituteAndAppend(&result, format, a0, a1, a2, a3, a4, a5, a6);
  488. return result;
  489. }
  490. ABSL_MUST_USE_RESULT inline std::string Substitute(
  491. absl::string_view format, const substitute_internal::Arg& a0, const substitute_internal::Arg& a1, const substitute_internal::Arg& a2, const substitute_internal::Arg& a3, const substitute_internal::Arg& a4, const substitute_internal::Arg& a5, const substitute_internal::Arg& a6, const substitute_internal::Arg& a7
  492. )
  493. {
  494. std::string result;
  495. SubstituteAndAppend(&result, format, a0, a1, a2, a3, a4, a5, a6, a7);
  496. return result;
  497. }
  498. ABSL_MUST_USE_RESULT inline std::string Substitute(
  499. absl::string_view format, const substitute_internal::Arg& a0, const substitute_internal::Arg& a1, const substitute_internal::Arg& a2, const substitute_internal::Arg& a3, const substitute_internal::Arg& a4, const substitute_internal::Arg& a5, const substitute_internal::Arg& a6, const substitute_internal::Arg& a7, const substitute_internal::Arg& a8
  500. )
  501. {
  502. std::string result;
  503. SubstituteAndAppend(&result, format, a0, a1, a2, a3, a4, a5, a6, a7, a8);
  504. return result;
  505. }
  506. ABSL_MUST_USE_RESULT inline std::string Substitute(
  507. absl::string_view format, const substitute_internal::Arg& a0, const substitute_internal::Arg& a1, const substitute_internal::Arg& a2, const substitute_internal::Arg& a3, const substitute_internal::Arg& a4, const substitute_internal::Arg& a5, const substitute_internal::Arg& a6, const substitute_internal::Arg& a7, const substitute_internal::Arg& a8, const substitute_internal::Arg& a9
  508. )
  509. {
  510. std::string result;
  511. SubstituteAndAppend(&result, format, a0, a1, a2, a3, a4, a5, a6, a7, a8, a9);
  512. return result;
  513. }
  514. #if defined(ABSL_BAD_CALL_IF)
  515. // This body of functions catches cases where the number of placeholders
  516. // doesn't match the number of data arguments.
  517. std::string Substitute(const char* format)
  518. ABSL_BAD_CALL_IF(substitute_internal::PlaceholderBitmask(format) != 0, "There were no substitution arguments "
  519. "but this format string either has a $[0-9] in it or "
  520. "contains an unescaped $ character (use $$ instead)");
  521. std::string Substitute(const char* format, const substitute_internal::Arg& a0)
  522. ABSL_BAD_CALL_IF(
  523. substitute_internal::PlaceholderBitmask(format) != 1,
  524. "There was 1 substitution argument given, but "
  525. "this format string is missing its $0, contains one of $1-$9, "
  526. "or contains an unescaped $ character (use $$ instead)"
  527. );
  528. std::string Substitute(const char* format, const substitute_internal::Arg& a0, const substitute_internal::Arg& a1)
  529. ABSL_BAD_CALL_IF(
  530. substitute_internal::PlaceholderBitmask(format) != 3,
  531. "There were 2 substitution arguments given, but "
  532. "this format string is missing its $0/$1, contains one of "
  533. "$2-$9, or contains an unescaped $ character (use $$ instead)"
  534. );
  535. std::string Substitute(const char* format, const substitute_internal::Arg& a0, const substitute_internal::Arg& a1, const substitute_internal::Arg& a2)
  536. ABSL_BAD_CALL_IF(
  537. substitute_internal::PlaceholderBitmask(format) != 7,
  538. "There were 3 substitution arguments given, but "
  539. "this format string is missing its $0/$1/$2, contains one of "
  540. "$3-$9, or contains an unescaped $ character (use $$ instead)"
  541. );
  542. std::string Substitute(const char* format, const substitute_internal::Arg& a0, const substitute_internal::Arg& a1, const substitute_internal::Arg& a2, const substitute_internal::Arg& a3)
  543. ABSL_BAD_CALL_IF(
  544. substitute_internal::PlaceholderBitmask(format) != 15,
  545. "There were 4 substitution arguments given, but "
  546. "this format string is missing its $0-$3, contains one of "
  547. "$4-$9, or contains an unescaped $ character (use $$ instead)"
  548. );
  549. std::string Substitute(const char* format, const substitute_internal::Arg& a0, const substitute_internal::Arg& a1, const substitute_internal::Arg& a2, const substitute_internal::Arg& a3, const substitute_internal::Arg& a4)
  550. ABSL_BAD_CALL_IF(
  551. substitute_internal::PlaceholderBitmask(format) != 31,
  552. "There were 5 substitution arguments given, but "
  553. "this format string is missing its $0-$4, contains one of "
  554. "$5-$9, or contains an unescaped $ character (use $$ instead)"
  555. );
  556. std::string Substitute(const char* format, const substitute_internal::Arg& a0, const substitute_internal::Arg& a1, const substitute_internal::Arg& a2, const substitute_internal::Arg& a3, const substitute_internal::Arg& a4, const substitute_internal::Arg& a5)
  557. ABSL_BAD_CALL_IF(
  558. substitute_internal::PlaceholderBitmask(format) != 63,
  559. "There were 6 substitution arguments given, but "
  560. "this format string is missing its $0-$5, contains one of "
  561. "$6-$9, or contains an unescaped $ character (use $$ instead)"
  562. );
  563. std::string Substitute(const char* format, const substitute_internal::Arg& a0, const substitute_internal::Arg& a1, const substitute_internal::Arg& a2, const substitute_internal::Arg& a3, const substitute_internal::Arg& a4, const substitute_internal::Arg& a5, const substitute_internal::Arg& a6)
  564. ABSL_BAD_CALL_IF(
  565. substitute_internal::PlaceholderBitmask(format) != 127,
  566. "There were 7 substitution arguments given, but "
  567. "this format string is missing its $0-$6, contains one of "
  568. "$7-$9, or contains an unescaped $ character (use $$ instead)"
  569. );
  570. std::string Substitute(const char* format, const substitute_internal::Arg& a0, const substitute_internal::Arg& a1, const substitute_internal::Arg& a2, const substitute_internal::Arg& a3, const substitute_internal::Arg& a4, const substitute_internal::Arg& a5, const substitute_internal::Arg& a6, const substitute_internal::Arg& a7)
  571. ABSL_BAD_CALL_IF(
  572. substitute_internal::PlaceholderBitmask(format) != 255,
  573. "There were 8 substitution arguments given, but "
  574. "this format string is missing its $0-$7, contains one of "
  575. "$8-$9, or contains an unescaped $ character (use $$ instead)"
  576. );
  577. std::string Substitute(
  578. const char* format, const substitute_internal::Arg& a0, const substitute_internal::Arg& a1, const substitute_internal::Arg& a2, const substitute_internal::Arg& a3, const substitute_internal::Arg& a4, const substitute_internal::Arg& a5, const substitute_internal::Arg& a6, const substitute_internal::Arg& a7, const substitute_internal::Arg& a8
  579. )
  580. ABSL_BAD_CALL_IF(
  581. substitute_internal::PlaceholderBitmask(format) != 511,
  582. "There were 9 substitution arguments given, but "
  583. "this format string is missing its $0-$8, contains a $9, or "
  584. "contains an unescaped $ character (use $$ instead)"
  585. );
  586. std::string Substitute(
  587. const char* format, const substitute_internal::Arg& a0, const substitute_internal::Arg& a1, const substitute_internal::Arg& a2, const substitute_internal::Arg& a3, const substitute_internal::Arg& a4, const substitute_internal::Arg& a5, const substitute_internal::Arg& a6, const substitute_internal::Arg& a7, const substitute_internal::Arg& a8, const substitute_internal::Arg& a9
  588. )
  589. ABSL_BAD_CALL_IF(
  590. substitute_internal::PlaceholderBitmask(format) != 1023,
  591. "There were 10 substitution arguments given, but this "
  592. "format string either doesn't contain all of $0 through $9 or "
  593. "contains an unescaped $ character (use $$ instead)"
  594. );
  595. #endif // ABSL_BAD_CALL_IF
  596. ABSL_NAMESPACE_END
  597. } // namespace absl
  598. #endif // ABSL_STRINGS_SUBSTITUTE_H_