Response.hxx 3.58 KB
Newer Older
1
/*
Max Kellermann's avatar
Max Kellermann committed
2
 * Copyright 2003-2021 The Music Player Daemon Project
3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24
 * http://www.musicpd.org
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License along
 * with this program; if not, write to the Free Software Foundation, Inc.,
 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
 */

#ifndef MPD_RESPONSE_HXX
#define MPD_RESPONSE_HXX

#include "protocol/Ack.hxx"

25
#include <fmt/core.h>
26
#if FMT_VERSION < 70000 || FMT_VERSION >= 80000
27 28 29
#include <fmt/format.h>
#endif

30
#include <cstddef>
31

32
template<typename T> struct ConstBuffer;
33
class Client;
34
class TagMask;
35 36 37 38

class Response {
	Client &client;

39 40 41 42 43 44
	/**
	 * This command's index in the command list.  Used to generate
	 * error messages.
	 */
	const unsigned list_index;

45 46 47
	/**
	 * This command's name.  Used to generate error messages.
	 */
48
	const char *command = "";
49

50
public:
51
	Response(Client &_client, unsigned _list_index) noexcept
52
		:client(_client), list_index(_list_index) {}
53 54 55 56

	Response(const Response &) = delete;
	Response &operator=(const Response &) = delete;

57 58 59 60 61 62
	/**
	 * Returns a const reference to the associated #Client object.
	 * This should only be used to access a client's settings, to
	 * determine how to format the response.  For this reason, the
	 * returned reference is "const".
	 */
63
	const Client &GetClient() const noexcept {
64 65 66
		return client;
	}

67 68 69 70
	/**
	 * Accessor for Client::tag_mask.  Can be used if caller wants
	 * to avoid including Client.hxx.
	 */
71
	[[gnu::pure]]
72
	TagMask GetTagMask() const noexcept;
73

74
	void SetCommand(const char *_command) noexcept {
75 76 77
		command = _command;
	}

78 79
	bool Write(const void *data, size_t length) noexcept;
	bool Write(const char *data) noexcept;
80

81 82 83 84
	bool VFmt(fmt::string_view format_str, fmt::format_args args) noexcept;

	template<typename S, typename... Args>
	bool Fmt(const S &format_str, Args&&... args) noexcept {
Max Kellermann's avatar
Max Kellermann committed
85 86 87 88
#if FMT_VERSION >= 90000
		return VFmt(format_str,
			    fmt::make_format_args(args...));
#elif FMT_VERSION >= 70000
89 90 91 92 93 94 95 96 97 98
		return VFmt(fmt::to_string_view(format_str),
			    fmt::make_args_checked<Args...>(format_str,
							    args...));
#else
		/* expensive fallback for older libfmt versions */
		const auto result = fmt::format(format_str, args...);
		return Write(result.data(), result.size());
#endif
	}

99 100 101 102 103 104 105 106
	/**
	 * Write a binary chunk; this writes the "binary" line, the
	 * given chunk and the trailing newline.
	 *
	 * @return true on success
	 */
	bool WriteBinary(ConstBuffer<void> payload) noexcept;

107
	void Error(enum ack code, const char *msg) noexcept;
108 109 110 111 112 113 114

	void VFmtError(enum ack code,
		       fmt::string_view format_str, fmt::format_args args) noexcept;

	template<typename S, typename... Args>
	void FmtError(enum ack code,
		      const S &format_str, Args&&... args) noexcept {
Max Kellermann's avatar
Max Kellermann committed
115 116 117 118
#if FMT_VERSION >= 90000
		return VFmtError(code, format_str,
				 fmt::make_format_args(args...));
#elif FMT_VERSION >= 70000
119 120 121 122 123 124 125 126 127
		return VFmtError(code, fmt::to_string_view(format_str),
				 fmt::make_args_checked<Args...>(format_str,
								 args...));
#else
		/* expensive fallback for older libfmt versions */
		const auto result = fmt::format(format_str, args...);
		return Error(code, result.c_str());
#endif
	}
128 129 130
};

#endif