Device.cxx 3.17 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
 * 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 "Device.hxx"
#include "Util.hxx"
22
#include "lib/expat/ExpatParser.hxx"
23 24 25

#include <string.h>

26 27
/* this destructor exists here just so it won't get inlined */
UPnPDevice::~UPnPDevice() noexcept = default;
28

29 30 31 32 33 34
/**
 * An XML parser which constructs an UPnP device object from the
 * device descriptor.
 */
class UPnPDeviceParser final : public CommonExpatParser {
	UPnPDevice &m_device;
35 36 37

	std::string *value;

38 39 40
	UPnPService m_tservice;

public:
Max Kellermann's avatar
Max Kellermann committed
41
	explicit UPnPDeviceParser(UPnPDevice& device)
42 43
		:m_device(device),
		 value(nullptr) {}
44 45

protected:
46
	void StartElement(const XML_Char *name, const XML_Char **) override {
47 48
		value = nullptr;

49
		switch (name[0]) {
50
		case 'c':
51 52
			if (strcmp(name, "controlURL") == 0)
				value = &m_tservice.controlURL;
53 54
			break;
		case 'd':
55 56
			if (strcmp(name, "deviceType") == 0)
				value = &m_device.deviceType;
57 58
			break;
		case 'f':
59 60
			if (strcmp(name, "friendlyName") == 0)
				value = &m_device.friendlyName;
61 62
			break;
		case 'm':
63 64 65 66
			if (strcmp(name, "manufacturer") == 0)
				value = &m_device.manufacturer;
			else if (strcmp(name, "modelName") == 0)
				value = &m_device.modelName;
67 68
			break;
		case 's':
69 70
			if (strcmp(name, "serviceType") == 0)
				value = &m_tservice.serviceType;
71 72
			break;
		case 'U':
73 74 75 76
			if (strcmp(name, "UDN") == 0)
				value = &m_device.UDN;
			else if (strcmp(name, "URLBase") == 0)
				value = &m_device.URLBase;
77 78 79
			break;
		}
	}
80

81
	void EndElement(const XML_Char *name) override {
82 83 84 85 86
		if (value != nullptr) {
			trimstring(*value);
			value = nullptr;
		} else if (!strcmp(name, "service")) {
			m_device.services.emplace_back(std::move(m_tservice));
87
			m_tservice = {};
88 89 90
		}
	}

91
	void CharacterData(const XML_Char *s, int len) override {
92 93 94
		if (value != nullptr)
			value->append(s, len);
	}
95 96
};

97 98
void
UPnPDevice::Parse(const std::string &url, const char *description)
99
{
100 101
	{
		UPnPDeviceParser mparser(*this);
102
		mparser.Parse(description, strlen(description), true);
103 104 105 106 107 108 109 110 111 112 113 114
	}

	if (URLBase.empty()) {
		// The standard says that if the URLBase value is empty, we should use
		// the url the description was retrieved from. However this is
		// sometimes something like http://host/desc.xml, sometimes something
		// like http://host/

		if (url.size() < 8) {
			// ???
			URLBase = url;
		} else {
115
			auto hostslash = url.find('/', 7);
116 117 118 119 120 121 122 123
			if (hostslash == std::string::npos || hostslash == url.size()-1) {
				URLBase = url;
			} else {
				URLBase = path_getfather(url);
			}
		}
	}
}