Switch from cstring_view to string_view

This commit is contained in:
Victor Zverovich
2017-07-18 19:40:48 -07:00
parent a8d6f309c8
commit 2f4f49fd60
11 changed files with 344 additions and 245 deletions

View File

@ -213,7 +213,7 @@ void report_error(FormatFunc func, int error_code,
} // namespace } // namespace
FMT_FUNC void system_error::init( FMT_FUNC void system_error::init(
int err_code, cstring_view format_str, args args) { int err_code, string_view format_str, args args) {
error_code_ = err_code; error_code_ = err_code;
memory_buffer buffer; memory_buffer buffer;
format_system_error(buffer, err_code, vformat(format_str, args)); format_system_error(buffer, err_code, vformat(format_str, args));
@ -337,7 +337,7 @@ FMT_FUNC int internal::utf16_to_utf8::convert(wstring_view s) {
} }
FMT_FUNC void windows_error::init( FMT_FUNC void windows_error::init(
int err_code, cstring_view format_str, args args) { int err_code, string_view format_str, args args) {
error_code_ = err_code; error_code_ = err_code;
memory_buffer buffer; memory_buffer buffer;
internal::format_windows_error(buffer, err_code, vformat(format_str, args)); internal::format_windows_error(buffer, err_code, vformat(format_str, args));
@ -419,17 +419,17 @@ FMT_FUNC void report_windows_error(
} }
#endif #endif
FMT_FUNC void vprint(std::FILE *f, cstring_view format_str, args args) { FMT_FUNC void vprint(std::FILE *f, string_view format_str, args args) {
memory_buffer buffer; memory_buffer buffer;
vformat_to(buffer, format_str, args); vformat_to(buffer, format_str, args);
std::fwrite(buffer.data(), 1, buffer.size(), f); std::fwrite(buffer.data(), 1, buffer.size(), f);
} }
FMT_FUNC void vprint(cstring_view format_str, args args) { FMT_FUNC void vprint(string_view format_str, args args) {
vprint(stdout, format_str, args); vprint(stdout, format_str, args);
} }
FMT_FUNC void vprint_colored(Color c, cstring_view format, args args) { FMT_FUNC void vprint_colored(Color c, string_view format, args args) {
char escape[] = "\x1b[30m"; char escape[] = "\x1b[30m";
escape[3] = static_cast<char>('0' + c); escape[3] = static_cast<char>('0' + c);
std::fputs(escape, stdout); std::fputs(escape, stdout);

View File

@ -474,59 +474,15 @@ class basic_string_view {
typedef basic_string_view<char> string_view; typedef basic_string_view<char> string_view;
typedef basic_string_view<wchar_t> wstring_view; typedef basic_string_view<wchar_t> wstring_view;
/**
\rst
A reference to a null-terminated string. It can be constructed from a C
string or ``std::string``.
You can use one of the following typedefs for common character types:
+---------------+-----------------------------+
| Type | Definition |
+===============+=============================+
| cstring_view | basic_cstring_view<char> |
+---------------+-----------------------------+
| wcstring_view | basic_cstring_view<wchar_t> |
+---------------+-----------------------------+
This class is most useful as a parameter type to allow passing
different types of strings to a function, for example::
template <typename... Args>
std::string format(cstring_view format_str, const Args & ... args);
format("{}", 42);
format(std::string("{}"), 42);
\endrst
*/
template <typename Char>
class basic_cstring_view {
private:
const Char *data_;
public:
/** Constructs a string reference object from a C string. */
basic_cstring_view(const Char *s) : data_(s) {}
/**
\rst
Constructs a string reference from an ``std::string`` object.
\endrst
*/
basic_cstring_view(const std::basic_string<Char> &s) : data_(s.c_str()) {}
/** Returns the pointer to a C string. */
const Char *c_str() const { return data_; }
};
typedef basic_cstring_view<char> cstring_view;
typedef basic_cstring_view<wchar_t> wcstring_view;
/** A formatting error such as invalid format string. */ /** A formatting error such as invalid format string. */
class format_error : public std::runtime_error { class format_error : public std::runtime_error {
public: public:
explicit format_error(cstring_view message) explicit format_error(const char *message)
: std::runtime_error(message.c_str()) {} : std::runtime_error(message) {}
explicit format_error(const std::string &message)
: std::runtime_error(message) {}
~format_error() throw(); ~format_error() throw();
}; };
@ -1197,7 +1153,6 @@ template <> constexpr Type gettype<unsigned char *>() { return CSTRING; }
template <> constexpr Type gettype<const unsigned char *>() { return CSTRING; } template <> constexpr Type gettype<const unsigned char *>() { return CSTRING; }
template <> constexpr Type gettype<std::string>() { return STRING; } template <> constexpr Type gettype<std::string>() { return STRING; }
template <> constexpr Type gettype<string_view>() { return STRING; } template <> constexpr Type gettype<string_view>() { return STRING; }
template <> constexpr Type gettype<cstring_view>() { return CSTRING; }
template <> constexpr Type gettype<wchar_t *>() { return TSTRING; } template <> constexpr Type gettype<wchar_t *>() { return TSTRING; }
template <> constexpr Type gettype<const wchar_t *>() { return TSTRING; } template <> constexpr Type gettype<const wchar_t *>() { return TSTRING; }
template <> constexpr Type gettype<std::wstring>() { return TSTRING; } template <> constexpr Type gettype<std::wstring>() { return TSTRING; }
@ -1335,7 +1290,6 @@ class value {
FMT_MAKE_VALUE(const unsigned char *, ustring.value, CSTRING) FMT_MAKE_VALUE(const unsigned char *, ustring.value, CSTRING)
FMT_MAKE_STR_VALUE(const std::string &, STRING) FMT_MAKE_STR_VALUE(const std::string &, STRING)
FMT_MAKE_STR_VALUE(string_view, STRING) FMT_MAKE_STR_VALUE(string_view, STRING)
FMT_MAKE_VALUE_(cstring_view, string.value, CSTRING, value.c_str())
#define FMT_MAKE_WSTR_VALUE(Type, TYPE) \ #define FMT_MAKE_WSTR_VALUE(Type, TYPE) \
value(typename wchar_helper<Type, Char>::supported value) { \ value(typename wchar_helper<Type, Char>::supported value) { \
@ -1978,18 +1932,112 @@ class arg_formatter_base {
} }
}; };
template <typename Char, typename Context> template <typename Char>
class context_base { class null_terminating_iterator;
template <typename Char>
const Char *pointer_from(null_terminating_iterator<Char> it);
// An iterator that produces a null terminator on *end.
template <typename Char>
class null_terminating_iterator {
public:
typedef std::ptrdiff_t difference_type;
null_terminating_iterator() : ptr_(0), end_(0) {}
null_terminating_iterator(const Char *ptr, const Char *end)
: ptr_(ptr), end_(end) {}
Char operator*() const {
return ptr_ != end_ ? *ptr_ : 0;
}
null_terminating_iterator operator++() {
++ptr_;
return *this;
}
null_terminating_iterator operator++(int) {
null_terminating_iterator result(*this);
++ptr_;
return result;
}
null_terminating_iterator operator--() {
--ptr_;
return *this;
}
null_terminating_iterator operator+(difference_type n) {
return null_terminating_iterator(ptr_ + n, end_);
}
null_terminating_iterator operator+=(difference_type n) {
ptr_ += n;
return *this;
}
difference_type operator-(null_terminating_iterator other) const {
return ptr_ - other.ptr_;
}
bool operator!=(null_terminating_iterator other) const {
return ptr_ != other.ptr_;
}
bool operator>=(null_terminating_iterator other) const {
return ptr_ >= other.ptr_;
}
friend const Char *pointer_from<Char>(null_terminating_iterator it);
private: private:
const Char *ptr_; const Char *ptr_;
const Char *end_;
};
template <
typename T,
typename Char,
typename std::enable_if<
std::is_same<T, null_terminating_iterator<Char>>::value, int>::type = 0>
null_terminating_iterator<Char> to_iterator(basic_string_view<Char> v) {
const Char *s = v.data();
return null_terminating_iterator<Char>(s, s + v.size());
}
template <
typename T,
typename Char,
typename std::enable_if<std::is_same<T, const Char*>::value, int>::type = 0>
const Char *to_iterator(const basic_string_view<Char> v) {
return v.data();
}
template <typename T>
const T *pointer_from(const T *p) { return p; }
template <typename Char>
const Char *pointer_from(null_terminating_iterator<Char> it) {
return it.ptr_;
}
template <typename Char, typename Context>
class context_base {
public:
typedef null_terminating_iterator<Char> iterator;
private:
iterator pos_;
basic_args<Context> args_; basic_args<Context> args_;
int next_arg_index_; int next_arg_index_;
protected: protected:
typedef basic_arg<Context> format_arg; typedef basic_arg<Context> format_arg;
context_base(const Char *format_str, basic_args<Context> args) context_base(basic_string_view<Char> format_str, basic_args<Context> args)
: ptr_(format_str), args_(args), next_arg_index_(0) {} : pos_(to_iterator<iterator>(format_str)), args_(args), next_arg_index_(0) {}
~context_base() {} ~context_base() {}
basic_args<Context> args() const { return args_; } basic_args<Context> args() const { return args_; }
@ -2027,8 +2075,8 @@ class context_base {
} }
public: public:
// Returns a pointer to the current position in the format string. // Returns an iterator to the current position in the format string.
const Char *&ptr() { return ptr_; } iterator &pos() { return pos_; }
}; };
} // namespace internal } // namespace internal
@ -2091,13 +2139,14 @@ class basic_context :
stored in the object so make sure they have appropriate lifetimes. stored in the object so make sure they have appropriate lifetimes.
\endrst \endrst
*/ */
basic_context(const Char *format_str, basic_args<basic_context> args) basic_context(
basic_string_view<Char> format_str, basic_args<basic_context> args)
: Base(format_str, args) {} : Base(format_str, args) {}
// Parses argument id and returns corresponding argument. // Parses argument id and returns corresponding argument.
format_arg parse_arg_id(); format_arg parse_arg_id();
using Base::ptr; using Base::pos;
}; };
/** /**
@ -2106,7 +2155,7 @@ class basic_context :
*/ */
class system_error : public std::runtime_error { class system_error : public std::runtime_error {
private: private:
void init(int err_code, cstring_view format_str, args args); void init(int err_code, string_view format_str, args args);
protected: protected:
int error_code_; int error_code_;
@ -2133,7 +2182,7 @@ class system_error : public std::runtime_error {
\endrst \endrst
*/ */
template <typename... Args> template <typename... Args>
system_error(int error_code, cstring_view message, const Args & ... args) system_error(int error_code, string_view message, const Args & ... args)
: std::runtime_error("") { : std::runtime_error("") {
init(error_code, message, make_args(args...)); init(error_code, message, make_args(args...));
} }
@ -2392,7 +2441,7 @@ class basic_writer {
Writes *value* to the buffer. Writes *value* to the buffer.
\endrst \endrst
*/ */
void write(fmt::basic_string_view<Char> value) { void write(basic_string_view<Char> value) {
const Char *str = value.data(); const Char *str = value.data();
buffer_.append(str, str + value.size()); buffer_.append(str, str + value.size());
} }
@ -2794,7 +2843,7 @@ FMT_API void report_system_error(int error_code,
/** A Windows error. */ /** A Windows error. */
class windows_error : public system_error { class windows_error : public system_error {
private: private:
FMT_API void init(int error_code, cstring_view format_str, args args); FMT_API void init(int error_code, string_view format_str, args args);
public: public:
/** /**
@ -2826,7 +2875,7 @@ class windows_error : public system_error {
\endrst \endrst
*/ */
template <typename... Args> template <typename... Args>
windows_error(int error_code, cstring_view message, const Args & ... args) { windows_error(int error_code, string_view message, const Args & ... args) {
init(error_code, message, make_args(args...)); init(error_code, message, make_args(args...));
} }
}; };
@ -2840,7 +2889,7 @@ FMT_API void report_windows_error(int error_code,
enum Color { BLACK, RED, GREEN, YELLOW, BLUE, MAGENTA, CYAN, WHITE }; enum Color { BLACK, RED, GREEN, YELLOW, BLUE, MAGENTA, CYAN, WHITE };
FMT_API void vprint_colored(Color c, cstring_view format, args args); FMT_API void vprint_colored(Color c, string_view format, args args);
/** /**
Formats a string and prints it to stdout using ANSI escape sequences Formats a string and prints it to stdout using ANSI escape sequences
@ -2849,36 +2898,36 @@ FMT_API void vprint_colored(Color c, cstring_view format, args args);
print_colored(fmt::RED, "Elapsed time: {0:.2f} seconds", 1.23); print_colored(fmt::RED, "Elapsed time: {0:.2f} seconds", 1.23);
*/ */
template <typename... Args> template <typename... Args>
inline void print_colored(Color c, cstring_view format_str, inline void print_colored(Color c, string_view format_str,
const Args & ... args) { const Args & ... args) {
vprint_colored(c, format_str, make_args(args...)); vprint_colored(c, format_str, make_args(args...));
} }
template <typename ArgFormatter, typename Char, typename Context> template <typename ArgFormatter, typename Char, typename Context>
void vformat_to(basic_buffer<Char> &buffer, basic_cstring_view<Char> format_str, void vformat_to(basic_buffer<Char> &buffer, basic_string_view<Char> format_str,
basic_args<Context> args); basic_args<Context> args);
inline void vformat_to(buffer &buf, cstring_view format_str, args args) { inline void vformat_to(buffer &buf, string_view format_str, args args) {
vformat_to<arg_formatter<char>>(buf, format_str, args); vformat_to<arg_formatter<char>>(buf, format_str, args);
} }
inline void vformat_to(wbuffer &buf, wcstring_view format_str, wargs args) { inline void vformat_to(wbuffer &buf, wstring_view format_str, wargs args) {
vformat_to<arg_formatter<wchar_t>>(buf, format_str, args); vformat_to<arg_formatter<wchar_t>>(buf, format_str, args);
} }
template <typename... Args> template <typename... Args>
inline void format_to(buffer &buf, cstring_view format_str, inline void format_to(buffer &buf, string_view format_str,
const Args & ... args) { const Args & ... args) {
vformat_to(buf, format_str, make_args(args...)); vformat_to(buf, format_str, make_args(args...));
} }
template <typename... Args> template <typename... Args>
inline void format_to(wbuffer &buf, wcstring_view format_str, inline void format_to(wbuffer &buf, wstring_view format_str,
const Args & ... args) { const Args & ... args) {
vformat_to(buf, format_str, make_args<wcontext>(args...)); vformat_to(buf, format_str, make_args<wcontext>(args...));
} }
inline std::string vformat(cstring_view format_str, args args) { inline std::string vformat(string_view format_str, args args) {
memory_buffer buffer; memory_buffer buffer;
vformat_to(buffer, format_str, args); vformat_to(buffer, format_str, args);
return to_string(buffer); return to_string(buffer);
@ -2894,22 +2943,22 @@ inline std::string vformat(cstring_view format_str, args args) {
\endrst \endrst
*/ */
template <typename... Args> template <typename... Args>
inline std::string format(cstring_view format_str, const Args & ... args) { inline std::string format(string_view format_str, const Args & ... args) {
return vformat(format_str, make_args(args...)); return vformat(format_str, make_args(args...));
} }
inline std::wstring vformat(wcstring_view format_str, wargs args) { inline std::wstring vformat(wstring_view format_str, wargs args) {
wmemory_buffer buffer; wmemory_buffer buffer;
vformat_to(buffer, format_str, args); vformat_to(buffer, format_str, args);
return to_string(buffer); return to_string(buffer);
} }
template <typename... Args> template <typename... Args>
inline std::wstring format(wcstring_view format_str, const Args & ... args) { inline std::wstring format(wstring_view format_str, const Args & ... args) {
return vformat(format_str, make_args<wcontext>(args...)); return vformat(format_str, make_args<wcontext>(args...));
} }
FMT_API void vprint(std::FILE *f, cstring_view format_str, args args); FMT_API void vprint(std::FILE *f, string_view format_str, args args);
/** /**
\rst \rst
@ -2921,12 +2970,12 @@ FMT_API void vprint(std::FILE *f, cstring_view format_str, args args);
\endrst \endrst
*/ */
template <typename... Args> template <typename... Args>
inline void print(std::FILE *f, cstring_view format_str, inline void print(std::FILE *f, string_view format_str,
const Args & ... args) { const Args & ... args) {
vprint(f, format_str, make_args(args...)); vprint(f, format_str, make_args(args...));
} }
FMT_API void vprint(cstring_view format_str, args args); FMT_API void vprint(string_view format_str, args args);
/** /**
\rst \rst
@ -2938,7 +2987,7 @@ FMT_API void vprint(cstring_view format_str, args args);
\endrst \endrst
*/ */
template <typename... Args> template <typename... Args>
inline void print(cstring_view format_str, const Args & ... args) { inline void print(string_view format_str, const Args & ... args) {
vprint(format_str, make_args(args...)); vprint(format_str, make_args(args...));
} }
@ -3092,21 +3141,22 @@ inline bool is_name_start(Char c) {
return ('a' <= c && c <= 'z') || ('A' <= c && c <= 'Z') || '_' == c; return ('a' <= c && c <= 'z') || ('A' <= c && c <= 'Z') || '_' == c;
} }
// Parses an unsigned integer advancing s to the end of the parsed input. // Parses an unsigned integer advancing it to the end of the parsed input.
// This function assumes that the first character of s is a digit. // This function assumes that the first character of it is a digit and a
template <typename Char> // presence of a non-digit character at the end.
unsigned parse_nonnegative_int(const Char *&s) { template <typename Iterator>
assert('0' <= *s && *s <= '9'); unsigned parse_nonnegative_int(Iterator &it) {
assert('0' <= *it && *it <= '9');
unsigned value = 0; unsigned value = 0;
do { do {
unsigned new_value = value * 10 + (*s++ - '0'); unsigned new_value = value * 10 + (*it++ - '0');
// Check if value wrapped around. // Check if value wrapped around.
if (new_value < value) { if (new_value < value) {
value = (std::numeric_limits<unsigned>::max)(); value = (std::numeric_limits<unsigned>::max)();
break; break;
} }
value = new_value; value = new_value;
} while ('0' <= *s && *s <= '9'); } while ('0' <= *it && *it <= '9');
// Convert to unsigned to prevent a warning. // Convert to unsigned to prevent a warning.
unsigned max_int = (std::numeric_limits<int>::max)(); unsigned max_int = (std::numeric_limits<int>::max)();
if (value > max_int) if (value > max_int)
@ -3138,15 +3188,15 @@ struct is_unsigned {
} }
}; };
template <typename Char, typename Context> template <typename Iterator, typename Context>
void check_sign(const Char *&s, const basic_arg<Context> &arg) { void check_sign(Iterator &it, const basic_arg<Context> &arg) {
char sign = static_cast<char>(*s); char sign = static_cast<char>(*it);
require_numeric_argument(arg, sign); require_numeric_argument(arg, sign);
if (visit(is_unsigned(), arg)) { if (visit(is_unsigned(), arg)) {
FMT_THROW(format_error(fmt::format( FMT_THROW(format_error(fmt::format(
"format specifier '{}' requires signed argument", sign))); "format specifier '{}' requires signed argument", sign)));
} }
++s; ++it;
} }
template <typename Char, typename Context> template <typename Char, typename Context>
@ -3228,25 +3278,26 @@ template <typename Char>
inline typename basic_context<Char>::format_arg inline typename basic_context<Char>::format_arg
basic_context<Char>::parse_arg_id() { basic_context<Char>::parse_arg_id() {
format_arg arg; format_arg arg;
const Char *&s = this->ptr(); auto &it = this->pos();
if (!internal::is_name_start(*s)) { if (!internal::is_name_start(*it)) {
const char *error = 0; const char *error = 0;
arg = *s < '0' || *s > '9' ? arg = *it < '0' || *it > '9' ?
this->next_arg(error) : this->next_arg(error) :
get_arg(internal::parse_nonnegative_int(s), error); get_arg(internal::parse_nonnegative_int(it), error);
if (error) { if (error) {
FMT_THROW(format_error( FMT_THROW(format_error(
*s != '}' && *s != ':' ? "invalid format string" : error)); *it != '}' && *it != ':' ? "invalid format string" : error));
} }
return arg; return arg;
} }
const Char *start = s; auto start = it;
Char c; Char c;
do { do {
c = *++s; c = *++it;
} while (internal::is_name_start(c) || ('0' <= c && c <= '9')); } while (internal::is_name_start(c) || ('0' <= c && c <= '9'));
const char *error = 0; const char *error = 0;
arg = get_arg(basic_string_view<Char>(start, s - start), error); arg = get_arg(basic_string_view<Char>(
internal::pointer_from(start), it - start), error);
if (error) if (error)
FMT_THROW(format_error(error)); FMT_THROW(format_error(error));
return arg; return arg;
@ -3256,15 +3307,15 @@ inline typename basic_context<Char>::format_arg
template <typename ArgFormatter, typename Char, typename Context> template <typename ArgFormatter, typename Char, typename Context>
void do_format_arg(basic_buffer<Char> &buffer, const basic_arg<Context>& arg, void do_format_arg(basic_buffer<Char> &buffer, const basic_arg<Context>& arg,
Context &ctx) { Context &ctx) {
const Char *&s = ctx.ptr(); auto &it = ctx.pos();
basic_format_specs<Char> spec; basic_format_specs<Char> spec;
if (*s == ':') { if (*it == ':') {
if (visit(internal::custom_formatter<Char, Context>(buffer, ctx), arg)) if (visit(internal::custom_formatter<Char, Context>(buffer, ctx), arg))
return; return;
++s; ++it;
// Parse fill and alignment. // Parse fill and alignment.
if (Char c = *s) { if (Char c = *it) {
const Char *p = s + 1; auto p = it + 1;
spec.align_ = ALIGN_DEFAULT; spec.align_ = ALIGN_DEFAULT;
do { do {
switch (*p) { switch (*p) {
@ -3282,57 +3333,57 @@ void do_format_arg(basic_buffer<Char> &buffer, const basic_arg<Context>& arg,
break; break;
} }
if (spec.align_ != ALIGN_DEFAULT) { if (spec.align_ != ALIGN_DEFAULT) {
if (p != s) { if (p != it) {
if (c == '}') break; if (c == '}') break;
if (c == '{') if (c == '{')
FMT_THROW(format_error("invalid fill character '{'")); FMT_THROW(format_error("invalid fill character '{'"));
s += 2; it += 2;
spec.fill_ = c; spec.fill_ = c;
} else ++s; } else ++it;
if (spec.align_ == ALIGN_NUMERIC) if (spec.align_ == ALIGN_NUMERIC)
internal::require_numeric_argument(arg, '='); internal::require_numeric_argument(arg, '=');
break; break;
} }
} while (--p >= s); } while (--p >= it);
} }
// Parse sign. // Parse sign.
switch (*s) { switch (*it) {
case '+': case '+':
internal::check_sign(s, arg); internal::check_sign(it, arg);
spec.flags_ |= SIGN_FLAG | PLUS_FLAG; spec.flags_ |= SIGN_FLAG | PLUS_FLAG;
break; break;
case '-': case '-':
internal::check_sign(s, arg); internal::check_sign(it, arg);
spec.flags_ |= MINUS_FLAG; spec.flags_ |= MINUS_FLAG;
break; break;
case ' ': case ' ':
internal::check_sign(s, arg); internal::check_sign(it, arg);
spec.flags_ |= SIGN_FLAG; spec.flags_ |= SIGN_FLAG;
break; break;
} }
if (*s == '#') { if (*it == '#') {
internal::require_numeric_argument(arg, '#'); internal::require_numeric_argument(arg, '#');
spec.flags_ |= HASH_FLAG; spec.flags_ |= HASH_FLAG;
++s; ++it;
} }
// Parse zero flag. // Parse zero flag.
if (*s == '0') { if (*it == '0') {
internal::require_numeric_argument(arg, '0'); internal::require_numeric_argument(arg, '0');
spec.align_ = ALIGN_NUMERIC; spec.align_ = ALIGN_NUMERIC;
spec.fill_ = '0'; spec.fill_ = '0';
++s; ++it;
} }
// Parse width. // Parse width.
if ('0' <= *s && *s <= '9') { if ('0' <= *it && *it <= '9') {
spec.width_ = internal::parse_nonnegative_int(s); spec.width_ = internal::parse_nonnegative_int(it);
} else if (*s == '{') { } else if (*it == '{') {
++s; ++it;
auto width_arg = ctx.parse_arg_id(); auto width_arg = ctx.parse_arg_id();
if (*s++ != '}') if (*it++ != '}')
FMT_THROW(format_error("invalid format string")); FMT_THROW(format_error("invalid format string"));
ulong_long width = visit(internal::width_handler(), width_arg); ulong_long width = visit(internal::width_handler(), width_arg);
if (width > (std::numeric_limits<int>::max)()) if (width > (std::numeric_limits<int>::max)())
@ -3341,15 +3392,15 @@ void do_format_arg(basic_buffer<Char> &buffer, const basic_arg<Context>& arg,
} }
// Parse precision. // Parse precision.
if (*s == '.') { if (*it == '.') {
++s; ++it;
spec.precision_ = 0; spec.precision_ = 0;
if ('0' <= *s && *s <= '9') { if ('0' <= *it && *it <= '9') {
spec.precision_ = internal::parse_nonnegative_int(s); spec.precision_ = internal::parse_nonnegative_int(it);
} else if (*s == '{') { } else if (*it == '{') {
++s; ++it;
auto precision_arg = ctx.parse_arg_id(); auto precision_arg = ctx.parse_arg_id();
if (*s++ != '}') if (*it++ != '}')
FMT_THROW(format_error("invalid format string")); FMT_THROW(format_error("invalid format string"));
ulong_long precision = ulong_long precision =
visit(internal::precision_handler(), precision_arg); visit(internal::precision_handler(), precision_arg);
@ -3367,11 +3418,11 @@ void do_format_arg(basic_buffer<Char> &buffer, const basic_arg<Context>& arg,
} }
// Parse type. // Parse type.
if (*s != '}' && *s) if (*it != '}' && *it)
spec.type_ = static_cast<char>(*s++); spec.type_ = static_cast<char>(*it++);
} }
if (*s != '}') if (*it != '}')
FMT_THROW(format_error("missing '}' in format string")); FMT_THROW(format_error("missing '}' in format string"));
// Format argument. // Format argument.
@ -3380,28 +3431,29 @@ void do_format_arg(basic_buffer<Char> &buffer, const basic_arg<Context>& arg,
/** Formats arguments and writes the output to the buffer. */ /** Formats arguments and writes the output to the buffer. */
template <typename ArgFormatter, typename Char, typename Context> template <typename ArgFormatter, typename Char, typename Context>
void vformat_to(basic_buffer<Char> &buffer, basic_cstring_view<Char> format_str, void vformat_to(basic_buffer<Char> &buffer, basic_string_view<Char> format_str,
basic_args<Context> args) { basic_args<Context> args) {
basic_context<Char> ctx(format_str.c_str(), args); basic_context<Char> ctx(format_str, args);
const Char *&s = ctx.ptr(); auto &it = ctx.pos();
const Char *start = s; auto start = it;
while (*s) { using internal::pointer_from;
Char c = *s++; while (*it) {
Char c = *it++;
if (c != '{' && c != '}') continue; if (c != '{' && c != '}') continue;
if (*s == c) { if (*it == c) {
buffer.append(start, s); buffer.append(pointer_from(start), pointer_from(it));
start = ++s; start = ++it;
continue; continue;
} }
if (c == '}') if (c == '}')
FMT_THROW(format_error("unmatched '}' in format string")); FMT_THROW(format_error("unmatched '}' in format string"));
buffer.append(start, s - 1); buffer.append(pointer_from(start), pointer_from(it) - 1);
do_format_arg<ArgFormatter>(buffer, ctx.parse_arg_id(), ctx); do_format_arg<ArgFormatter>(buffer, ctx.parse_arg_id(), ctx);
if (*s != '}') if (*it != '}')
FMT_THROW(format_error(fmt::format("unknown format specifier"))); FMT_THROW(format_error(fmt::format("unknown format specifier")));
start = ++s; start = ++it;
} }
buffer.append(start, s); buffer.append(pointer_from(start), pointer_from(it));
} }
} // namespace fmt } // namespace fmt

View File

@ -27,7 +27,7 @@ FMT_FUNC void write(std::ostream &os, buffer &buf) {
} }
} }
FMT_FUNC void vprint(std::ostream &os, cstring_view format_str, args args) { FMT_FUNC void vprint(std::ostream &os, string_view format_str, args args) {
memory_buffer buffer; memory_buffer buffer;
vformat_to(buffer, format_str, args); vformat_to(buffer, format_str, args);
internal::write(os, buffer); internal::write(os, buffer);

View File

@ -90,7 +90,7 @@ void format_value(basic_buffer<Char> &buf, const T &value,
buf, internal::make_arg< basic_context<Char> >(str), ctx); buf, internal::make_arg< basic_context<Char> >(str), ctx);
} }
FMT_API void vprint(std::ostream &os, cstring_view format_str, args args); FMT_API void vprint(std::ostream &os, string_view format_str, args args);
/** /**
\rst \rst
@ -102,7 +102,7 @@ FMT_API void vprint(std::ostream &os, cstring_view format_str, args args);
\endrst \endrst
*/ */
template <typename... Args> template <typename... Args>
inline void print(std::ostream &os, cstring_view format_str, inline void print(std::ostream &os, string_view format_str,
const Args & ... args) { const Args & ... args) {
vprint(os, format_str, make_args(args...)); vprint(os, format_str, make_args(args...));
} }

View File

@ -72,7 +72,7 @@ fmt::BufferedFile::BufferedFile(
fmt::cstring_view filename, fmt::cstring_view mode) { fmt::cstring_view filename, fmt::cstring_view mode) {
FMT_RETRY_VAL(file_, FMT_SYSTEM(fopen(filename.c_str(), mode.c_str())), 0); FMT_RETRY_VAL(file_, FMT_SYSTEM(fopen(filename.c_str(), mode.c_str())), 0);
if (!file_) if (!file_)
throw system_error(errno, "cannot open file {}", filename); throw system_error(errno, "cannot open file {}", filename.c_str());
} }
void fmt::BufferedFile::close() { void fmt::BufferedFile::close() {
@ -103,7 +103,7 @@ fmt::File::File(fmt::cstring_view path, int oflag) {
FMT_RETRY(fd_, FMT_POSIX_CALL(open(path.c_str(), oflag, mode))); FMT_RETRY(fd_, FMT_POSIX_CALL(open(path.c_str(), oflag, mode)));
#endif #endif
if (fd_ == -1) if (fd_ == -1)
throw system_error(errno, "cannot open file {}", path); throw system_error(errno, "cannot open file {}", path.c_str());
} }
fmt::File::~File() FMT_NOEXCEPT { fmt::File::~File() FMT_NOEXCEPT {

View File

@ -66,6 +66,54 @@
namespace fmt { namespace fmt {
/**
\rst
A reference to a null-terminated string. It can be constructed from a C
string or ``std::string``.
You can use one of the following typedefs for common character types:
+---------------+-----------------------------+
| Type | Definition |
+===============+=============================+
| cstring_view | basic_cstring_view<char> |
+---------------+-----------------------------+
| wcstring_view | basic_cstring_view<wchar_t> |
+---------------+-----------------------------+
This class is most useful as a parameter type to allow passing
different types of strings to a function, for example::
template <typename... Args>
std::string format(cstring_view format_str, const Args & ... args);
format("{}", 42);
format(std::string("{}"), 42);
\endrst
*/
template <typename Char>
class basic_cstring_view {
private:
const Char *data_;
public:
/** Constructs a string reference object from a C string. */
basic_cstring_view(const Char *s) : data_(s) {}
/**
\rst
Constructs a string reference from an ``std::string`` object.
\endrst
*/
basic_cstring_view(const std::basic_string<Char> &s) : data_(s.c_str()) {}
/** Returns the pointer to a C string. */
const Char *c_str() const { return data_; }
};
typedef basic_cstring_view<char> cstring_view;
typedef basic_cstring_view<wchar_t> wcstring_view;
// An error code. // An error code.
class ErrorCode { class ErrorCode {
private: private:
@ -166,12 +214,12 @@ public:
// of MinGW that define fileno as a macro. // of MinGW that define fileno as a macro.
int (fileno)() const; int (fileno)() const;
void vprint(cstring_view format_str, const args &args) { void vprint(string_view format_str, const args &args) {
fmt::vprint(file_, format_str, args); fmt::vprint(file_, format_str, args);
} }
template <typename... Args> template <typename... Args>
inline void print(cstring_view format_str, const Args & ... args) { inline void print(string_view format_str, const Args & ... args) {
vprint(format_str, make_args(args...)); vprint(format_str, make_args(args...));
} }
}; };

View File

@ -3,9 +3,9 @@
namespace fmt { namespace fmt {
template <typename Char> template <typename Char>
void printf(basic_writer<Char> &w, basic_cstring_view<Char> format, args args); void printf(basic_writer<Char> &w, basic_string_view<Char> format, args args);
FMT_FUNC int vfprintf(std::FILE *f, cstring_view format, printf_args args) { FMT_FUNC int vfprintf(std::FILE *f, string_view format, printf_args args) {
memory_buffer buffer; memory_buffer buffer;
printf(buffer, format, args); printf(buffer, format, args);
std::size_t size = buffer.size(); std::size_t size = buffer.size();

View File

@ -307,17 +307,18 @@ class printf_context :
typedef internal::context_base<Char, printf_context> Base; typedef internal::context_base<Char, printf_context> Base;
typedef typename Base::format_arg format_arg; typedef typename Base::format_arg format_arg;
typedef basic_format_specs<Char> format_specs; typedef basic_format_specs<Char> format_specs;
typedef typename Base::iterator iterator;
void parse_flags(format_specs &spec, const Char *&s); void parse_flags(format_specs &spec, iterator &it);
// Returns the argument with specified index or, if arg_index is equal // Returns the argument with specified index or, if arg_index is equal
// to the maximum unsigned value, the next argument. // to the maximum unsigned value, the next argument.
format_arg get_arg( format_arg get_arg(
const Char *s, iterator it,
unsigned arg_index = (std::numeric_limits<unsigned>::max)()); unsigned arg_index = (std::numeric_limits<unsigned>::max)());
// Parses argument index, flags and width and returns the argument index. // Parses argument index, flags and width and returns the argument index.
unsigned parse_header(const Char *&s, format_specs &spec); unsigned parse_header(iterator &it, format_specs &spec);
public: public:
/** /**
@ -327,18 +328,18 @@ class printf_context :
appropriate lifetimes. appropriate lifetimes.
\endrst \endrst
*/ */
explicit printf_context(basic_cstring_view<Char> format_str, explicit printf_context(basic_string_view<Char> format_str,
basic_args<printf_context> args) basic_args<printf_context> args)
: Base(format_str.c_str(), args) {} : Base(format_str, args) {}
/** Formats stored arguments and writes the output to the buffer. */ /** Formats stored arguments and writes the output to the buffer. */
FMT_API void format(basic_buffer<Char> &buffer); FMT_API void format(basic_buffer<Char> &buffer);
}; };
template <typename Char, typename AF> template <typename Char, typename AF>
void printf_context<Char, AF>::parse_flags(format_specs &spec, const Char *&s) { void printf_context<Char, AF>::parse_flags(format_specs &spec, iterator &it) {
for (;;) { for (;;) {
switch (*s++) { switch (*it++) {
case '-': case '-':
spec.align_ = ALIGN_LEFT; spec.align_ = ALIGN_LEFT;
break; break;
@ -355,7 +356,7 @@ void printf_context<Char, AF>::parse_flags(format_specs &spec, const Char *&s) {
spec.flags_ |= HASH_FLAG; spec.flags_ |= HASH_FLAG;
break; break;
default: default:
--s; --it;
return; return;
} }
} }
@ -363,27 +364,27 @@ void printf_context<Char, AF>::parse_flags(format_specs &spec, const Char *&s) {
template <typename Char, typename AF> template <typename Char, typename AF>
typename printf_context<Char, AF>::format_arg printf_context<Char, AF>::get_arg( typename printf_context<Char, AF>::format_arg printf_context<Char, AF>::get_arg(
const Char *s, unsigned arg_index) { iterator it, unsigned arg_index) {
(void)s; (void)it;
const char *error = 0; const char *error = 0;
format_arg arg = arg_index == std::numeric_limits<unsigned>::max() ? format_arg arg = arg_index == std::numeric_limits<unsigned>::max() ?
this->next_arg(error) : Base::get_arg(arg_index - 1, error); this->next_arg(error) : Base::get_arg(arg_index - 1, error);
if (error) if (error)
FMT_THROW(format_error(!*s ? "invalid format string" : error)); FMT_THROW(format_error(!*it ? "invalid format string" : error));
return arg; return arg;
} }
template <typename Char, typename AF> template <typename Char, typename AF>
unsigned printf_context<Char, AF>::parse_header( unsigned printf_context<Char, AF>::parse_header(
const Char *&s, format_specs &spec) { iterator &it, format_specs &spec) {
unsigned arg_index = std::numeric_limits<unsigned>::max(); unsigned arg_index = std::numeric_limits<unsigned>::max();
Char c = *s; Char c = *it;
if (c >= '0' && c <= '9') { if (c >= '0' && c <= '9') {
// Parse an argument index (if followed by '$') or a width possibly // Parse an argument index (if followed by '$') or a width possibly
// preceded with '0' flag(s). // preceded with '0' flag(s).
unsigned value = internal::parse_nonnegative_int(s); unsigned value = internal::parse_nonnegative_int(it);
if (*s == '$') { // value is an argument index if (*it == '$') { // value is an argument index
++s; ++it;
arg_index = value; arg_index = value;
} else { } else {
if (c == '0') if (c == '0')
@ -396,49 +397,51 @@ unsigned printf_context<Char, AF>::parse_header(
} }
} }
} }
parse_flags(spec, s); parse_flags(spec, it);
// Parse width. // Parse width.
if (*s >= '0' && *s <= '9') { if (*it >= '0' && *it <= '9') {
spec.width_ = internal::parse_nonnegative_int(s); spec.width_ = internal::parse_nonnegative_int(it);
} else if (*s == '*') { } else if (*it == '*') {
++s; ++it;
spec.width_ = visit(internal::PrintfWidthHandler<Char>(spec), get_arg(s)); spec.width_ = visit(internal::PrintfWidthHandler<Char>(spec), get_arg(it));
} }
return arg_index; return arg_index;
} }
template <typename Char, typename AF> template <typename Char, typename AF>
void printf_context<Char, AF>::format(basic_buffer<Char> &buffer) { void printf_context<Char, AF>::format(basic_buffer<Char> &buffer) {
const Char *start = this->ptr(); auto start = this->pos();
const Char *s = start; auto it = start;
while (*s) { using internal::pointer_from;
Char c = *s++; while (*it) {
Char c = *it++;
if (c != '%') continue; if (c != '%') continue;
if (*s == c) { if (*it == c) {
buffer.append(start, s); buffer.append(pointer_from(start), pointer_from(it));
start = ++s; start = ++it;
continue; continue;
} }
buffer.append(start, s - 1); buffer.append(pointer_from(start), pointer_from(it) - 1);
format_specs spec; format_specs spec;
spec.align_ = ALIGN_RIGHT; spec.align_ = ALIGN_RIGHT;
// Parse argument index, flags and width. // Parse argument index, flags and width.
unsigned arg_index = parse_header(s, spec); unsigned arg_index = parse_header(it, spec);
// Parse precision. // Parse precision.
if (*s == '.') { if (*it == '.') {
++s; ++it;
if ('0' <= *s && *s <= '9') { if ('0' <= *it && *it <= '9') {
spec.precision_ = static_cast<int>(internal::parse_nonnegative_int(s)); spec.precision_ = static_cast<int>(internal::parse_nonnegative_int(it));
} else if (*s == '*') { } else if (*it == '*') {
++s; ++it;
spec.precision_ = visit(internal::PrintfPrecisionHandler(), get_arg(s)); spec.precision_ =
visit(internal::PrintfPrecisionHandler(), get_arg(it));
} }
} }
format_arg arg = get_arg(s, arg_index); format_arg arg = get_arg(it, arg_index);
if (spec.flag(HASH_FLAG) && visit(internal::IsZeroInt(), arg)) if (spec.flag(HASH_FLAG) && visit(internal::IsZeroInt(), arg))
spec.flags_ &= ~internal::to_unsigned<int>(HASH_FLAG); spec.flags_ &= ~internal::to_unsigned<int>(HASH_FLAG);
if (spec.fill_ == '0') { if (spec.fill_ == '0') {
@ -450,41 +453,41 @@ void printf_context<Char, AF>::format(basic_buffer<Char> &buffer) {
// Parse length and convert the argument to the required type. // Parse length and convert the argument to the required type.
using internal::convert_arg; using internal::convert_arg;
switch (*s++) { switch (*it++) {
case 'h': case 'h':
if (*s == 'h') if (*it == 'h')
convert_arg<signed char>(arg, *++s); convert_arg<signed char>(arg, *++it);
else else
convert_arg<short>(arg, *s); convert_arg<short>(arg, *it);
break; break;
case 'l': case 'l':
if (*s == 'l') if (*it == 'l')
convert_arg<fmt::long_long>(arg, *++s); convert_arg<fmt::long_long>(arg, *++it);
else else
convert_arg<long>(arg, *s); convert_arg<long>(arg, *it);
break; break;
case 'j': case 'j':
convert_arg<intmax_t>(arg, *s); convert_arg<intmax_t>(arg, *it);
break; break;
case 'z': case 'z':
convert_arg<std::size_t>(arg, *s); convert_arg<std::size_t>(arg, *it);
break; break;
case 't': case 't':
convert_arg<std::ptrdiff_t>(arg, *s); convert_arg<std::ptrdiff_t>(arg, *it);
break; break;
case 'L': case 'L':
// printf produces garbage when 'L' is omitted for long double, no // printf produces garbage when 'L' is omitted for long double, no
// need to do the same. // need to do the same.
break; break;
default: default:
--s; --it;
convert_arg<void>(arg, *s); convert_arg<void>(arg, *it);
} }
// Parse type. // Parse type.
if (!*s) if (!*it)
FMT_THROW(format_error("invalid format string")); FMT_THROW(format_error("invalid format string"));
spec.type_ = static_cast<char>(*s++); spec.type_ = static_cast<char>(*it++);
if (arg.is_integral()) { if (arg.is_integral()) {
// Normalize type. // Normalize type.
switch (spec.type_) { switch (spec.type_) {
@ -498,12 +501,12 @@ void printf_context<Char, AF>::format(basic_buffer<Char> &buffer) {
} }
} }
start = s; start = it;
// Format argument. // Format argument.
visit(AF(buffer, spec), arg); visit(AF(buffer, spec), arg);
} }
buffer.append(start, s); buffer.append(pointer_from(start), pointer_from(it));
} }
// Formats a value. // Formats a value.
@ -514,14 +517,14 @@ void format_value(basic_buffer<Char> &buf, const T &value,
} }
template <typename Char> template <typename Char>
void printf(basic_buffer<Char> &buf, basic_cstring_view<Char> format, void printf(basic_buffer<Char> &buf, basic_string_view<Char> format,
basic_args<printf_context<Char>> args) { basic_args<printf_context<Char>> args) {
printf_context<Char>(format, args).format(buf); printf_context<Char>(format, args).format(buf);
} }
typedef basic_args<printf_context<char>> printf_args; typedef basic_args<printf_context<char>> printf_args;
inline std::string vsprintf(cstring_view format, printf_args args) { inline std::string vsprintf(string_view format, printf_args args) {
memory_buffer buffer; memory_buffer buffer;
printf(buffer, format, args); printf(buffer, format, args);
return to_string(buffer); return to_string(buffer);
@ -537,24 +540,24 @@ inline std::string vsprintf(cstring_view format, printf_args args) {
\endrst \endrst
*/ */
template <typename... Args> template <typename... Args>
inline std::string sprintf(cstring_view format_str, const Args & ... args) { inline std::string sprintf(string_view format_str, const Args & ... args) {
return vsprintf(format_str, make_args<printf_context<char>>(args...)); return vsprintf(format_str, make_args<printf_context<char>>(args...));
} }
inline std::wstring vsprintf( inline std::wstring vsprintf(
wcstring_view format, basic_args<printf_context<wchar_t>> args) { wstring_view format, basic_args<printf_context<wchar_t>> args) {
wmemory_buffer buffer; wmemory_buffer buffer;
printf(buffer, format, args); printf(buffer, format, args);
return to_string(buffer); return to_string(buffer);
} }
template <typename... Args> template <typename... Args>
inline std::wstring sprintf(wcstring_view format_str, const Args & ... args) { inline std::wstring sprintf(wstring_view format_str, const Args & ... args) {
auto vargs = make_args<printf_context<wchar_t>>(args...); auto vargs = make_args<printf_context<wchar_t>>(args...);
return vsprintf(format_str, vargs); return vsprintf(format_str, vargs);
} }
FMT_API int vfprintf(std::FILE *f, cstring_view format, printf_args args); FMT_API int vfprintf(std::FILE *f, string_view format, printf_args args);
/** /**
\rst \rst
@ -566,12 +569,12 @@ FMT_API int vfprintf(std::FILE *f, cstring_view format, printf_args args);
\endrst \endrst
*/ */
template <typename... Args> template <typename... Args>
inline int fprintf(std::FILE *f, cstring_view format_str, const Args & ... args) { inline int fprintf(std::FILE *f, string_view format_str, const Args & ... args) {
auto vargs = make_args<printf_context<char>>(args...); auto vargs = make_args<printf_context<char>>(args...);
return vfprintf(f, format_str, vargs); return vfprintf(f, format_str, vargs);
} }
inline int vprintf(cstring_view format, printf_args args) { inline int vprintf(string_view format, printf_args args) {
return vfprintf(stdout, format, args); return vfprintf(stdout, format, args);
} }
@ -585,11 +588,11 @@ inline int vprintf(cstring_view format, printf_args args) {
\endrst \endrst
*/ */
template <typename... Args> template <typename... Args>
inline int printf(cstring_view format_str, const Args & ... args) { inline int printf(string_view format_str, const Args & ... args) {
return vprintf(format_str, make_args<printf_context<char>>(args...)); return vprintf(format_str, make_args<printf_context<char>>(args...));
} }
inline int vfprintf(std::ostream &os, cstring_view format_str, printf_args args) { inline int vfprintf(std::ostream &os, string_view format_str, printf_args args) {
memory_buffer buffer; memory_buffer buffer;
printf(buffer, format_str, args); printf(buffer, format_str, args);
internal::write(os, buffer); internal::write(os, buffer);
@ -606,7 +609,7 @@ inline int vfprintf(std::ostream &os, cstring_view format_str, printf_args args)
\endrst \endrst
*/ */
template <typename... Args> template <typename... Args>
inline int fprintf(std::ostream &os, cstring_view format_str, inline int fprintf(std::ostream &os, string_view format_str,
const Args & ... args) { const Args & ... args) {
auto vargs = make_args<printf_context<char>>(args...); auto vargs = make_args<printf_context<char>>(args...);
return vfprintf(os, format_str, vargs); return vfprintf(os, format_str, vargs);

View File

@ -16,17 +16,19 @@
namespace fmt { namespace fmt {
void format_value(buffer &buf, const std::tm &tm, context &ctx) { void format_value(buffer &buf, const std::tm &tm, context &ctx) {
const char *&s = ctx.ptr(); auto &it = ctx.pos();
if (*s == ':') if (*it == ':')
++s; ++it;
const char *end = s; auto end = it;
while (*end && *end != '}') while (*end && *end != '}')
++end; ++end;
if (*end != '}') if (*end != '}')
FMT_THROW(format_error("missing '}' in format string")); FMT_THROW(format_error("missing '}' in format string"));
memory_buffer format; memory_buffer format;
format.append(s, end + 1); format.reserve(end - it + 1);
format[format.size() - 1] = '\0'; using internal::pointer_from;
format.append(pointer_from(it), pointer_from(end));
format.push_back('\0');
std::size_t start = buf.size(); std::size_t start = buf.size();
for (;;) { for (;;) {
std::size_t size = buf.capacity() - start; std::size_t size = buf.capacity() - start;
@ -45,7 +47,7 @@ void format_value(buffer &buf, const std::tm &tm, context &ctx) {
const std::size_t MIN_GROWTH = 10; const std::size_t MIN_GROWTH = 10;
buf.reserve(buf.capacity() + (size > MIN_GROWTH ? size : MIN_GROWTH)); buf.reserve(buf.capacity() + (size > MIN_GROWTH ? size : MIN_GROWTH));
} }
s = end; it = end;
} }
} }

View File

@ -45,7 +45,7 @@ class CustomPrintfArgFormatter : public printf_arg_formatter<char> {
} }
}; };
std::string custom_vformat(fmt::cstring_view format_str, fmt::args args) { std::string custom_vformat(fmt::string_view format_str, fmt::args args) {
fmt::memory_buffer buffer; fmt::memory_buffer buffer;
// Pass custom argument formatter as a template arg to vwrite. // Pass custom argument formatter as a template arg to vwrite.
fmt::vformat_to<CustomArgFormatter>(buffer, format_str, args); fmt::vformat_to<CustomArgFormatter>(buffer, format_str, args);

View File

@ -57,7 +57,6 @@ using fmt::basic_writer;
using fmt::format; using fmt::format;
using fmt::format_error; using fmt::format_error;
using fmt::string_view; using fmt::string_view;
using fmt::cstring_view;
using fmt::memory_buffer; using fmt::memory_buffer;
using fmt::wmemory_buffer; using fmt::wmemory_buffer;
using fmt::fill; using fmt::fill;
@ -145,11 +144,6 @@ TEST(StringViewTest, ConvertToString) {
EXPECT_EQ("abc", s); EXPECT_EQ("abc", s);
} }
TEST(CStringViewTest, Ctor) {
EXPECT_STREQ("abc", cstring_view("abc").c_str());
EXPECT_STREQ("defg", cstring_view(std::string("defg")).c_str());
}
#if FMT_USE_TYPE_TRAITS #if FMT_USE_TYPE_TRAITS
TEST(WriterTest, NotCopyConstructible) { TEST(WriterTest, NotCopyConstructible) {
EXPECT_FALSE(std::is_copy_constructible<basic_writer<char> >::value); EXPECT_FALSE(std::is_copy_constructible<basic_writer<char> >::value);
@ -464,7 +458,7 @@ TEST(FormatterTest, ArgErrors) {
template <int N> template <int N>
struct TestFormat { struct TestFormat {
template <typename... Args> template <typename... Args>
static std::string format(fmt::cstring_view format_str, const Args & ... args) { static std::string format(fmt::string_view format_str, const Args & ... args) {
return TestFormat<N - 1>::format(format_str, N - 1, args...); return TestFormat<N - 1>::format(format_str, N - 1, args...);
} }
}; };
@ -472,7 +466,7 @@ struct TestFormat {
template <> template <>
struct TestFormat<0> { struct TestFormat<0> {
template <typename... Args> template <typename... Args>
static std::string format(fmt::cstring_view format_str, const Args & ... args) { static std::string format(fmt::string_view format_str, const Args & ... args) {
return fmt::format(format_str, args...); return fmt::format(format_str, args...);
} }
}; };
@ -1233,10 +1227,6 @@ TEST(FormatterTest, FormatStringView) {
EXPECT_EQ("test", format("{0}", string_view("test"))); EXPECT_EQ("test", format("{0}", string_view("test")));
} }
TEST(FormatterTest, FormatCStringView) {
EXPECT_EQ("test", format("{0}", cstring_view("test")));
}
void format_value(fmt::buffer &buf, const Date &d, fmt::context &) { void format_value(fmt::buffer &buf, const Date &d, fmt::context &) {
fmt::format_to(buf, "{}-{}-{}", d.year(), d.month(), d.day()); fmt::format_to(buf, "{}-{}-{}", d.year(), d.month(), d.day());
} }
@ -1512,7 +1502,7 @@ class MockArgFormatter : public fmt::internal::arg_formatter_base<char> {
void operator()(fmt::internal::custom_value<char>) {} void operator()(fmt::internal::custom_value<char>) {}
}; };
void custom_vformat(fmt::cstring_view format_str, fmt::args args) { void custom_vformat(fmt::string_view format_str, fmt::args args) {
fmt::memory_buffer buffer; fmt::memory_buffer buffer;
fmt::vformat_to<MockArgFormatter>(buffer, format_str, args); fmt::vformat_to<MockArgFormatter>(buffer, format_str, args);
} }
@ -1526,3 +1516,7 @@ void custom_format(const char *format_str, const Args & ... args) {
TEST(FormatTest, CustomArgFormatter) { TEST(FormatTest, CustomArgFormatter) {
custom_format("{}", 42); custom_format("{}", 42);
} }
TEST(FormatTest, NonNullTerminatedFormatString) {
EXPECT_EQ("42", format(string_view("{}foo", 2), 42));
}