ExpatParser.hxx 3.98 KB
Newer Older
1
/*
Max Kellermann's avatar
Max Kellermann committed
2
 * Copyright 2003-2019 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
 * 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_EXPAT_HXX
#define MPD_EXPAT_HXX

23
#include "util/Compiler.h"
24 25 26

#include <expat.h>

27 28
#include <stdexcept>

29
class InputStream;
30

31 32
class ExpatError final : public std::runtime_error {
public:
33
	explicit ExpatError(XML_Error code)
34 35
		:std::runtime_error(XML_ErrorString(code)) {}

36
	explicit ExpatError(XML_Parser parser)
37 38 39
		:ExpatError(XML_GetErrorCode(parser)) {}
};

40 41 42 43
struct ExpatNamespaceSeparator {
	char separator;
};

44 45 46 47
class ExpatParser final {
	const XML_Parser parser;

public:
48
	explicit ExpatParser(void *userData)
49 50 51 52
		:parser(XML_ParserCreate(nullptr)) {
		XML_SetUserData(parser, userData);
	}

53 54 55 56 57
	ExpatParser(ExpatNamespaceSeparator ns, void *userData)
		:parser(XML_ParserCreateNS(nullptr, ns.separator)) {
		XML_SetUserData(parser, userData);
	}

58 59 60 61
	~ExpatParser() {
		XML_ParserFree(parser);
	}

62 63 64
	ExpatParser(const ExpatParser &) = delete;
	ExpatParser &operator=(const ExpatParser &) = delete;

65
	void SetElementHandler(XML_StartElementHandler start,
66
			       XML_EndElementHandler end) noexcept {
67 68 69
		XML_SetElementHandler(parser, start, end);
	}

70
	void SetCharacterDataHandler(XML_CharacterDataHandler charhndl) noexcept {
71 72 73
		XML_SetCharacterDataHandler(parser, charhndl);
	}

74
	void Parse(const char *data, size_t length, bool is_final=false);
75

76 77 78 79
	void CompleteParse() {
		Parse("", 0, true);
	}

80
	void Parse(InputStream &is);
81

82 83
	gcc_pure
	static const char *GetAttribute(const XML_Char **atts,
84
					const char *name) noexcept;
85

86 87
	gcc_pure
	static const char *GetAttributeCase(const XML_Char **atts,
88
					    const char *name) noexcept;
89 90
};

91 92 93 94 95 96 97 98 99 100 101 102 103
/**
 * A specialization of #ExpatParser that provides the most common
 * callbacks as virtual methods.
 */
class CommonExpatParser {
	ExpatParser parser;

public:
	CommonExpatParser():parser(this) {
		parser.SetElementHandler(StartElement, EndElement);
		parser.SetCharacterDataHandler(CharacterData);
	}

104 105 106 107 108 109
	explicit CommonExpatParser(ExpatNamespaceSeparator ns)
		:parser(ns, this) {
		parser.SetElementHandler(StartElement, EndElement);
		parser.SetCharacterDataHandler(CharacterData);
	}

110 111 112
	template<typename... Args>
	void Parse(Args&&... args) {
		parser.Parse(std::forward<Args>(args)...);
113 114
	}

115 116 117 118
	void CompleteParse() {
		parser.CompleteParse();
	}

119 120
	gcc_pure
	static const char *GetAttribute(const XML_Char **atts,
121
					const char *name) noexcept {
122 123 124
		return ExpatParser::GetAttribute(atts, name);
	}

125 126
	gcc_pure
	static const char *GetAttributeCase(const XML_Char **atts,
127
					    const char *name) noexcept {
128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155
		return ExpatParser::GetAttributeCase(atts, name);
	}

protected:
	virtual void StartElement(const XML_Char *name,
				  const XML_Char **atts) = 0;
	virtual void EndElement(const XML_Char *name) = 0;
	virtual void CharacterData(const XML_Char *s, int len) = 0;

private:
	static void XMLCALL StartElement(void *user_data, const XML_Char *name,
					 const XML_Char **atts) {
		CommonExpatParser &p = *(CommonExpatParser *)user_data;
		p.StartElement(name, atts);
	}

	static void XMLCALL EndElement(void *user_data, const XML_Char *name) {
		CommonExpatParser &p = *(CommonExpatParser *)user_data;
		p.EndElement(name);
	}

	static void XMLCALL CharacterData(void *user_data,
					  const XML_Char *s, int len) {
		CommonExpatParser &p = *(CommonExpatParser *)user_data;
		p.CharacterData(s, len);
	}
};

156
#endif