Directory.cxx 5.88 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21
/*
 * Copyright (C) 2003-2014 The Music Player Daemon Project
 * 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.
 */

#include "config.h"
#include "Directory.hxx"
22
#include "lib/upnp/Util.hxx"
23
#include "lib/expat/ExpatParser.hxx"
24 25 26
#include "Tags.hxx"
#include "tag/TagBuilder.hxx"
#include "tag/TagTable.hxx"
27
#include "util/NumberParser.hxx"
28

29
#include <algorithm>
30 31 32 33
#include <string>

#include <string.h>

34 35 36 37 38
UPnPDirContent::~UPnPDirContent()
{
	/* this destructor exists here just so it won't get inlined */
}

39 40 41 42 43 44 45 46
gcc_pure gcc_nonnull_all
static bool
CompareStringLiteral(const char *literal, const char *value, size_t length)
{
	return length == strlen(literal) &&
		memcmp(literal, value, length) == 0;
}

47 48
gcc_pure
static UPnPDirObject::ItemClass
49
ParseItemClass(const char *name, size_t length)
50
{
51 52
	if (CompareStringLiteral("object.item.audioItem.musicTrack",
				 name, length))
53
		return UPnPDirObject::ItemClass::MUSIC;
54 55
	else if (CompareStringLiteral("object.item.playlistItem",
				      name, length))
56 57 58 59 60
		return UPnPDirObject::ItemClass::PLAYLIST;
	else
		return UPnPDirObject::ItemClass::UNKNOWN;
}

61 62
gcc_pure
static int
63
ParseDuration(const char *duration)
64
{
65 66 67 68 69 70 71 72 73 74
	char *endptr;

	unsigned result = ParseUnsigned(duration, &endptr);
	if (endptr == duration || *endptr != ':')
		return 0;

	result *= 60;
	duration = endptr + 1;
	result += ParseUnsigned(duration, &endptr);
	if (endptr == duration || *endptr != ':')
75
		return 0;
76 77 78 79 80 81 82 83

	result *= 60;
	duration = endptr + 1;
	result += ParseUnsigned(duration, &endptr);
	if (endptr == duration || *endptr != 0)
		return 0;

	return result;
84 85
}

86 87 88 89 90 91 92
/**
 * Transform titles to turn '/' into '_' to make them acceptable path
 * elements. There is a very slight risk of collision in doing
 * this. Twonky returns directory names (titles) like 'Artist/Album'.
 */
gcc_pure
static std::string
93
titleToPathElt(std::string &&s)
94 95 96 97 98
{
	std::replace(s.begin(), s.end(), '/', '_');
	return s;
}

99 100 101 102
/**
 * An XML parser which builds directory contents from DIDL lite input.
 */
class UPnPDirParser final : public CommonExpatParser {
103 104
	UPnPDirContent &m_dir;

105 106 107 108 109
	enum {
		NONE,
		RES,
		CLASS,
	} state;
110 111 112 113 114 115 116 117 118 119 120 121 122

	/**
	 * If not equal to #TAG_NUM_OF_ITEM_TYPES, then we're
	 * currently reading an element containing a tag value.  The
	 * value is being constructed in #value.
	 */
	TagType tag_type;

	/**
	 * The text inside the current element.
	 */
	std::string value;

123
	UPnPDirObject m_tobj;
124
	TagBuilder tag;
125 126 127

public:
	UPnPDirParser(UPnPDirContent& dir)
128
		:m_dir(dir),
129
		 state(NONE),
130
		 tag_type(TAG_NUM_OF_ITEM_TYPES)
131 132 133 134 135 136
	{
	}

protected:
	virtual void StartElement(const XML_Char *name, const XML_Char **attrs)
	{
137 138 139 140 141 142 143 144 145
		if (m_tobj.type != UPnPDirObject::Type::UNKNOWN &&
		    tag_type == TAG_NUM_OF_ITEM_TYPES) {
			tag_type = tag_table_lookup(upnp_tags, name);
			if (tag_type != TAG_NUM_OF_ITEM_TYPES)
				return;
		} else {
			assert(tag_type == TAG_NUM_OF_ITEM_TYPES);
		}

146 147 148 149
		switch (name[0]) {
		case 'c':
			if (!strcmp(name, "container")) {
				m_tobj.clear();
150
				m_tobj.type = UPnPDirObject::Type::CONTAINER;
151 152 153 154 155 156 157 158

				const char *id = GetAttribute(attrs, "id");
				if (id != nullptr)
					m_tobj.m_id = id;

				const char *pid = GetAttribute(attrs, "parentID");
				if (pid != nullptr)
					m_tobj.m_pid = pid;
159 160
			}
			break;
161

162 163 164
		case 'i':
			if (!strcmp(name, "item")) {
				m_tobj.clear();
165
				m_tobj.type = UPnPDirObject::Type::ITEM;
166 167 168 169 170 171 172 173

				const char *id = GetAttribute(attrs, "id");
				if (id != nullptr)
					m_tobj.m_id = id;

				const char *pid = GetAttribute(attrs, "parentID");
				if (pid != nullptr)
					m_tobj.m_pid = pid;
174 175
			}
			break;
176 177 178 179 180 181

		case 'r':
			if (!strcmp(name, "res")) {
				// <res protocolInfo="http-get:*:audio/mpeg:*" size="5171496"
				// bitrate="24576" duration="00:03:35" sampleFrequency="44100"
				// nrAudioChannels="2">
182

183 184 185
				const char *duration =
					GetAttribute(attrs, "duration");
				if (duration != nullptr)
186
					tag.SetTime(ParseDuration(duration));
187 188

				state = RES;
189 190
			}

191
			break;
192 193 194 195

		case 'u':
			if (strcmp(name, "upnp:class") == 0)
				state = CLASS;
196 197 198 199
		}
	}

	bool checkobjok() {
200
		if (m_tobj.m_id.empty() || m_tobj.m_pid.empty() ||
201
		    m_tobj.name.empty() ||
202 203
		    (m_tobj.type == UPnPDirObject::Type::ITEM &&
		     m_tobj.item_class == UPnPDirObject::ItemClass::UNKNOWN))
204
			return false;
205

206
		return true;
207 208 209 210
	}

	virtual void EndElement(const XML_Char *name)
	{
211 212 213 214 215 216 217 218 219 220 221 222 223
		if (tag_type != TAG_NUM_OF_ITEM_TYPES) {
			assert(m_tobj.type != UPnPDirObject::Type::UNKNOWN);

			tag.AddItem(tag_type, value.c_str());

			if (tag_type == TAG_TITLE)
				m_tobj.name = titleToPathElt(std::move(value));

			value.clear();
			tag_type = TAG_NUM_OF_ITEM_TYPES;
			return;
		}

224
		if ((!strcmp(name, "container") || !strcmp(name, "item")) &&
225 226
		    checkobjok()) {
			tag.Commit(m_tobj.tag);
227
			m_dir.objects.emplace_back(std::move(m_tobj));
228
		}
229

230
		state = NONE;
231 232 233 234
	}

	virtual void CharacterData(const XML_Char *s, int len)
	{
235 236
		if (tag_type != TAG_NUM_OF_ITEM_TYPES) {
			assert(m_tobj.type != UPnPDirObject::Type::UNKNOWN);
237

238
			value.append(s, len);
239 240 241
			return;
		}

242 243 244
		switch (state) {
		case NONE:
			break;
245

246 247
		case RES:
			m_tobj.url.assign(s, len);
248
			break;
249 250 251

		case CLASS:
			m_tobj.item_class = ParseItemClass(s, len);
252 253 254 255 256 257
			break;
		}
	}
};

bool
258
UPnPDirContent::parse(const char *input, Error &error)
259 260
{
	UPnPDirParser parser(*this);
261
	return parser.Parse(input, strlen(input), true, error);
262
}