ExpatParser.hxx 4 KB
Newer Older
1
/*
2
 * Copyright 2003-2018 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
 * 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

#include "check.h"
24
#include "util/Compiler.h"
25 26 27

#include <expat.h>

28 29
#include <stdexcept>

30
class InputStream;
31

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

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

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

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

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

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

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

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

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

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

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

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

81
	void Parse(InputStream &is);
82

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

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

92 93 94 95 96 97 98 99 100 101 102 103 104
/**
 * 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);
	}

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

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

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

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

126 127
	gcc_pure
	static const char *GetAttributeCase(const XML_Char **atts,
128
					    const char *name) noexcept {
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 156
		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);
	}
};

157
#endif