Change policy about how we handle generated files
[amule.git] / src / UPnPBase.cpp
blob46ac7451e6cebbeb90217ea4d37cb8b9b3717759
1 //
2 // This file is part of the aMule Project.
3 //
4 // Copyright (c) 2004-2011 Marcelo Roberto Jimenez ( phoenix@amule.org )
5 // Copyright (c) 2006-2011 aMule Team ( admin@amule.org / http://www.amule.org )
6 //
7 // Any parts of this program derived from the xMule, lMule or eMule project,
8 // or contributed by third-party developers are copyrighted by their
9 // respective authors.
11 // This program is free software; you can redistribute it and/or modify
12 // it under the terms of the GNU General Public License as published by
13 // the Free Software Foundation; either version 2 of the License, or
14 // (at your option) any later version.
16 // This program is distributed in the hope that it will be useful,
17 // but WITHOUT ANY WARRANTY; without even the implied warranty of
18 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
19 // GNU General Public License for more details.
21 // You should have received a copy of the GNU General Public License
22 // along with this program; if not, write to the Free Software
23 // Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
26 #ifdef HAVE_CONFIG_H
27 # include "config.h" // Needed for ENABLE_UPNP
28 #endif
30 #ifdef ENABLE_UPNP
32 // check for broken Debian-hacked libUPnP
33 #include <upnp.h>
34 #ifdef STRING_H // defined in UpnpString.h Yes, I would have liked UPNPSTRING_H much better.
35 #define BROKEN_DEBIAN_LIBUPNP
36 #endif
38 #include "UPnPBase.h"
40 #include <algorithm> // For transform()
42 #ifdef BROKEN_DEBIAN_LIBUPNP
43 #define GET_UPNP_STRING(a) UpnpString_get_String(a)
44 #else
45 #define GET_UPNP_STRING(a) (a)
46 #endif
48 std::string stdEmptyString;
50 const char s_argument[] = "argument";
51 const char s_argumentList[] = "argumentList";
52 const char s_action[] = "action";
53 const char s_actionList[] = "actionList";
54 const char s_allowedValue[] = "allowedValue";
55 const char s_allowedValueList[] = "allowedValueList";
56 const char s_stateVariable[] = "stateVariable";
57 const char s_serviceStateTable[] = "serviceStateTable";
58 const char s_service[] = "service";
59 const char s_serviceList[] = "serviceList";
60 const char s_device[] = "device";
61 const char s_deviceList[] = "deviceList";
63 /**
64 * Case insensitive std::string comparison
66 static bool stdStringIsEqualCI(const std::string &s1, const std::string &s2)
68 std::string ns1(s1);
69 std::string ns2(s2);
70 std::transform(ns1.begin(), ns1.end(), ns1.begin(), tolower);
71 std::transform(ns2.begin(), ns2.end(), ns2.begin(), tolower);
72 return ns1 == ns2;
76 CUPnPPortMapping::CUPnPPortMapping(
77 int port,
78 const std::string &protocol,
79 bool enabled,
80 const std::string &description)
82 m_port(),
83 m_protocol(protocol),
84 m_enabled(enabled ? "1" : "0"),
85 m_description(description),
86 m_key()
88 std::ostringstream oss;
89 oss << port;
90 m_port = oss.str();
91 m_key = m_protocol + m_port;
94 namespace UPnP {
96 static const std::string ROOT_DEVICE("upnp:rootdevice");
98 namespace Device {
99 static const std::string IGW("urn:schemas-upnp-org:device:InternetGatewayDevice:1");
100 static const std::string WAN("urn:schemas-upnp-org:device:WANDevice:1");
101 static const std::string WAN_Connection("urn:schemas-upnp-org:device:WANConnectionDevice:1");
102 static const std::string LAN("urn:schemas-upnp-org:device:LANDevice:1");
105 namespace Service {
106 static const std::string Layer3_Forwarding("urn:schemas-upnp-org:service:Layer3Forwarding:1");
107 static const std::string WAN_Common_Interface_Config("urn:schemas-upnp-org:service:WANCommonInterfaceConfig:1");
108 static const std::string WAN_IP_Connection("urn:schemas-upnp-org:service:WANIPConnection:1");
109 static const std::string WAN_PPP_Connection("urn:schemas-upnp-org:service:WANPPPConnection:1");
112 static std::string ProcessErrorMessage(
113 const std::string &messsage,
114 int errorCode,
115 const DOMString errorString,
116 IXML_Document *doc)
118 std::ostringstream msg;
119 if (errorString == NULL || *errorString == 0) {
120 errorString = "Not available";
122 if (errorCode > 0) {
123 msg << "Error: " <<
124 messsage <<
125 ": Error code :'";
126 if (doc) {
127 CUPnPError e(doc);
128 msg << e.getErrorCode() <<
129 "', Error description :'" <<
130 e.getErrorDescription() <<
131 "'.";
132 } else {
133 msg << errorCode <<
134 "', Error description :'" <<
135 errorString <<
136 "'.";
138 AddDebugLogLineN(logUPnP, msg);
139 } else {
140 msg << "Error: " <<
141 messsage <<
142 ": UPnP SDK error: " <<
143 UpnpGetErrorMessage(errorCode) <<
144 " (" << errorCode << ").";
145 AddDebugLogLineN(logUPnP, msg);
148 return msg.str();
152 static void ProcessActionResponse(
153 IXML_Document *RespDoc,
154 const std::string &actionName)
156 std::ostringstream msg;
157 msg << "Response: ";
158 IXML_Element *root = IXML::Document::GetRootElement(RespDoc);
159 IXML_Element *child = IXML::Element::GetFirstChild(root);
160 if (child) {
161 while (child) {
162 const DOMString childTag = IXML::Element::GetTag(child);
163 std::string childValue = IXML::Element::GetTextValue(child);
164 msg << "\n " <<
165 childTag << "='" <<
166 childValue << "'";
167 child = IXML::Element::GetNextSibling(child);
169 } else {
170 msg << "\n Empty response for action '" <<
171 actionName << "'.";
173 AddDebugLogLineN(logUPnP, msg);
176 } /* namespace UPnP */
179 namespace IXML {
182 * \brief Returns the root node of a given document.
184 IXML_Element *Document::GetRootElement(IXML_Document *doc)
186 return reinterpret_cast<IXML_Element *>(ixmlNode_getFirstChild(&doc->n));
190 * \brief Frees the given document.
192 * \note Any nodes extracted via any other interface function will become
193 * invalid after this call unless explicitly cloned.
195 inline void Document::Free(IXML_Document *doc)
197 ixmlDocument_free(doc);
200 namespace Element {
203 * \brief Returns the first child of a given element.
205 IXML_Element *GetFirstChild(IXML_Element *parent)
207 return reinterpret_cast<IXML_Element *>(ixmlNode_getFirstChild(&parent->n));
213 * \brief Returns the next sibling of a given child.
215 IXML_Element *GetNextSibling(IXML_Element *child)
217 return reinterpret_cast<IXML_Element *>(ixmlNode_getNextSibling(&child->n));
222 * \brief Returns the element tag (name)
224 const DOMString GetTag(IXML_Element *element)
226 return ixmlNode_getNodeName(&element->n);
231 * \brief Returns the TEXT node value of the current node.
233 const std::string GetTextValue(IXML_Element *element)
235 if (!element) {
236 return stdEmptyString;
238 IXML_Node *text = ixmlNode_getFirstChild(&element->n);
239 const DOMString s = ixmlNode_getNodeValue(text);
240 std::string ret;
241 if (s) {
242 ret = s;
245 return ret;
250 * \brief Returns the TEXT node value of the first child matching tag.
252 const std::string GetChildValueByTag(IXML_Element *element, const DOMString tag)
254 return GetTextValue(GetFirstChildByTag(element, tag));
259 * \brief Returns the first child element that matches the requested tag or
260 * NULL if not found.
262 IXML_Element *GetFirstChildByTag(IXML_Element *element, const DOMString tag)
264 if (!element || !tag) {
265 return NULL;
268 IXML_Node *child = ixmlNode_getFirstChild(&element->n);
269 const DOMString childTag = ixmlNode_getNodeName(child);
270 while(child && childTag && strcmp(tag, childTag)) {
271 child = ixmlNode_getNextSibling(child);
272 childTag = ixmlNode_getNodeName(child);
275 return reinterpret_cast<IXML_Element *>(child);
280 * \brief Returns the next sibling element that matches the requested tag. Should be
281 * used with the return value of GetFirstChildByTag().
283 IXML_Element *GetNextSiblingByTag(IXML_Element *element, const DOMString tag)
285 if (!element || !tag) {
286 return NULL;
289 IXML_Node *child = &element->n;
290 const DOMString childTag = NULL;
291 do {
292 child = ixmlNode_getNextSibling(child);
293 childTag = ixmlNode_getNodeName(child);
294 } while(child && childTag && strcmp(tag, childTag));
296 return reinterpret_cast<IXML_Element *>(child);
300 const std::string GetAttributeByTag(IXML_Element *element, const DOMString tag)
302 IXML_NamedNodeMap *NamedNodeMap = ixmlNode_getAttributes(&element->n);
303 IXML_Node *attribute = ixmlNamedNodeMap_getNamedItem(NamedNodeMap, tag);
304 const DOMString s = ixmlNode_getNodeValue(attribute);
305 std::string ret;
306 if (s) {
307 ret = s;
309 ixmlNamedNodeMap_free(NamedNodeMap);
311 return ret;
314 } /* namespace Element */
316 } /* namespace IXML */
319 CUPnPError::CUPnPError(IXML_Document *errorDoc)
321 m_root (IXML::Document::GetRootElement(errorDoc)),
322 m_ErrorCode (IXML::Element::GetChildValueByTag(m_root, "errorCode")),
323 m_ErrorDescription(IXML::Element::GetChildValueByTag(m_root, "errorDescription"))
328 CUPnPArgument::CUPnPArgument(
329 const CUPnPControlPoint &WXUNUSED(upnpControlPoint),
330 IXML_Element *argument,
331 const std::string &WXUNUSED(SCPDURL))
333 m_name (IXML::Element::GetChildValueByTag(argument, "name")),
334 m_direction (IXML::Element::GetChildValueByTag(argument, "direction")),
335 m_retval (IXML::Element::GetFirstChildByTag(argument, "retval")),
336 m_relatedStateVariable(IXML::Element::GetChildValueByTag(argument, "relatedStateVariable"))
338 std::ostringstream msg;
339 msg << "\n Argument:" <<
340 "\n name: " << m_name <<
341 "\n direction: " << m_direction <<
342 "\n retval: " << m_retval <<
343 "\n relatedStateVariable: " << m_relatedStateVariable;
344 AddDebugLogLineN(logUPnP, msg);
348 CUPnPAction::CUPnPAction(
349 const CUPnPControlPoint &upnpControlPoint,
350 IXML_Element *action,
351 const std::string &SCPDURL)
353 m_ArgumentList(upnpControlPoint, action, SCPDURL),
354 m_name(IXML::Element::GetChildValueByTag(action, "name"))
356 std::ostringstream msg;
357 msg << "\n Action:" <<
358 "\n name: " << m_name;
359 AddDebugLogLineN(logUPnP, msg);
363 CUPnPAllowedValue::CUPnPAllowedValue(
364 const CUPnPControlPoint &WXUNUSED(upnpControlPoint),
365 IXML_Element *allowedValue,
366 const std::string &WXUNUSED(SCPDURL))
368 m_allowedValue(IXML::Element::GetTextValue(allowedValue))
370 std::ostringstream msg;
371 msg << "\n AllowedValue:" <<
372 "\n allowedValue: " << m_allowedValue;
373 AddDebugLogLineN(logUPnP, msg);
377 CUPnPStateVariable::CUPnPStateVariable(
378 const CUPnPControlPoint &upnpControlPoint,
379 IXML_Element *stateVariable,
380 const std::string &SCPDURL)
382 m_AllowedValueList(upnpControlPoint, stateVariable, SCPDURL),
383 m_name (IXML::Element::GetChildValueByTag(stateVariable, "name")),
384 m_dataType (IXML::Element::GetChildValueByTag(stateVariable, "dataType")),
385 m_defaultValue(IXML::Element::GetChildValueByTag(stateVariable, "defaultValue")),
386 m_sendEvents (IXML::Element::GetAttributeByTag (stateVariable, "sendEvents"))
388 std::ostringstream msg;
389 msg << "\n StateVariable:" <<
390 "\n name: " << m_name <<
391 "\n dataType: " << m_dataType <<
392 "\n defaultValue: " << m_defaultValue <<
393 "\n sendEvents: " << m_sendEvents;
394 AddDebugLogLineN(logUPnP, msg);
398 CUPnPSCPD::CUPnPSCPD(
399 const CUPnPControlPoint &upnpControlPoint,
400 IXML_Element *scpd,
401 const std::string &SCPDURL)
403 m_ActionList(upnpControlPoint, scpd, SCPDURL),
404 m_ServiceStateTable(upnpControlPoint, scpd, SCPDURL),
405 m_SCPDURL(SCPDURL)
410 CUPnPArgumentValue::CUPnPArgumentValue()
412 m_argument(),
413 m_value()
418 CUPnPArgumentValue::CUPnPArgumentValue(
419 const std::string &argument, const std::string &value)
421 m_argument(argument),
422 m_value(value)
427 CUPnPService::CUPnPService(
428 const CUPnPControlPoint &upnpControlPoint,
429 IXML_Element *service,
430 const std::string &URLBase)
432 m_UPnPControlPoint(upnpControlPoint),
433 m_serviceType(IXML::Element::GetChildValueByTag(service, "serviceType")),
434 m_serviceId (IXML::Element::GetChildValueByTag(service, "serviceId")),
435 m_SCPDURL (IXML::Element::GetChildValueByTag(service, "SCPDURL")),
436 m_controlURL (IXML::Element::GetChildValueByTag(service, "controlURL")),
437 m_eventSubURL(IXML::Element::GetChildValueByTag(service, "eventSubURL")),
438 m_timeout(1801),
439 m_SCPD(nullptr)
441 std::ostringstream msg;
442 int errcode;
444 memset(m_SID, 0 , sizeof(Upnp_SID));
446 std::vector<char> vscpdURL(URLBase.length() + m_SCPDURL.length() + 1);
447 char *scpdURL = &vscpdURL[0];
448 errcode = UpnpResolveURL(
449 URLBase.c_str(),
450 m_SCPDURL.c_str(),
451 scpdURL);
452 if( errcode != UPNP_E_SUCCESS ) {
453 msg << "Error generating scpdURL from " <<
454 "|" << URLBase << "|" <<
455 m_SCPDURL << "|.";
456 AddDebugLogLineN(logUPnP, msg);
457 } else {
458 m_absSCPDURL = scpdURL;
461 std::vector<char> vcontrolURL(
462 URLBase.length() + m_controlURL.length() + 1);
463 char *controlURL = &vcontrolURL[0];
464 errcode = UpnpResolveURL(
465 URLBase.c_str(),
466 m_controlURL.c_str(),
467 controlURL);
468 if( errcode != UPNP_E_SUCCESS ) {
469 msg << "Error generating controlURL from " <<
470 "|" << URLBase << "|" <<
471 m_controlURL << "|.";
472 AddDebugLogLineN(logUPnP, msg);
473 } else {
474 m_absControlURL = controlURL;
477 std::vector<char> veventURL(
478 URLBase.length() + m_eventSubURL.length() + 1);
479 char *eventURL = &veventURL[0];
480 errcode = UpnpResolveURL(
481 URLBase.c_str(),
482 m_eventSubURL.c_str(),
483 eventURL);
484 if( errcode != UPNP_E_SUCCESS ) {
485 msg << "Error generating eventURL from " <<
486 "|" << URLBase << "|" <<
487 m_eventSubURL << "|.";
488 AddDebugLogLineN(logUPnP, msg);
489 } else {
490 m_absEventSubURL = eventURL;
493 msg << "\n Service:" <<
494 "\n serviceType: " << m_serviceType <<
495 "\n serviceId: " << m_serviceId <<
496 "\n SCPDURL: " << m_SCPDURL <<
497 "\n absSCPDURL: " << m_absSCPDURL <<
498 "\n controlURL: " << m_controlURL <<
499 "\n absControlURL: " << m_absControlURL <<
500 "\n eventSubURL: " << m_eventSubURL <<
501 "\n absEventSubURL: " << m_absEventSubURL;
502 AddDebugLogLineN(logUPnP, msg);
504 if (m_serviceType == UPnP::Service::WAN_IP_Connection ||
505 m_serviceType == UPnP::Service::WAN_PPP_Connection) {
506 #if 0
507 m_serviceType == UPnP::Service::WAN_PPP_Connection ||
508 m_serviceType == UPnP::Service::WAN_Common_Interface_Config ||
509 m_serviceType == UPnP::Service::Layer3_Forwarding) {
510 #endif
511 #if 0
512 //#warning Delete this code on release.
513 if (!upnpControlPoint.WanServiceDetected()) {
514 // This condition can be used to suspend the parse
515 // of the XML tree.
516 #endif
517 //#warning Delete this code when m_WanService is no longer used.
518 const_cast<CUPnPControlPoint &>(upnpControlPoint).SetWanService(this);
519 // Log it
520 msg.str("");
521 msg << "WAN Service Detected: '" <<
522 m_serviceType << "'.";
523 AddDebugLogLineC(logUPnP, msg);
524 // Subscribe
525 const_cast<CUPnPControlPoint &>(upnpControlPoint).Subscribe(*this);
526 #if 0
527 //#warning Delete this code on release.
528 } else {
529 msg.str("");
530 msg << "WAN service detected again: '" <<
531 m_serviceType <<
532 "'. Will only use the first instance.";
533 AddDebugLogLineC(logUPnP, msg);
535 #endif
536 } else {
537 msg.str("");
538 msg << "Uninteresting service detected: '" <<
539 m_serviceType << "'. Ignoring.";
540 AddDebugLogLineC(logUPnP, msg);
545 CUPnPService::~CUPnPService()
550 bool CUPnPService::Execute(
551 const std::string &ActionName,
552 const std::vector<CUPnPArgumentValue> &ArgValue) const
554 std::ostringstream msg;
555 if (m_SCPD.get() == nullptr) {
556 msg << "Service without SCPD Document, cannot execute action '" << ActionName <<
557 "' for service '" << GetServiceType() << "'.";
558 AddDebugLogLineN(logUPnP, msg);
559 return false;
561 std::ostringstream msgAction("Sending action ");
562 // Check for correct action name
563 ActionList::const_iterator itAction =
564 m_SCPD->GetActionList().find(ActionName);
565 if (itAction == m_SCPD->GetActionList().end()) {
566 msg << "Invalid action name '" << ActionName <<
567 "' for service '" << GetServiceType() << "'.";
568 AddDebugLogLineN(logUPnP, msg);
569 return false;
571 msgAction << ActionName << "(";
572 bool firstTime = true;
573 // Check for correct Argument/Value pairs
574 const CUPnPAction &action = *(itAction->second);
575 for (unsigned int i = 0; i < ArgValue.size(); ++i) {
576 ArgumentList::const_iterator itArg =
577 action.GetArgumentList().find(ArgValue[i].GetArgument());
578 if (itArg == action.GetArgumentList().end()) {
579 msg << "Invalid argument name '" << ArgValue[i].GetArgument() <<
580 "' for action '" << action.GetName() <<
581 "' for service '" << GetServiceType() << "'.";
582 AddDebugLogLineN(logUPnP, msg);
583 return false;
585 const CUPnPArgument &argument = *(itArg->second);
586 if (tolower(argument.GetDirection()[0]) != 'i' ||
587 tolower(argument.GetDirection()[1]) != 'n') {
588 msg << "Invalid direction for argument '" <<
589 ArgValue[i].GetArgument() <<
590 "' for action '" << action.GetName() <<
591 "' for service '" << GetServiceType() << "'.";
592 AddDebugLogLineN(logUPnP, msg);
593 return false;
595 const std::string relatedStateVariableName =
596 argument.GetRelatedStateVariable();
597 if (!relatedStateVariableName.empty()) {
598 ServiceStateTable::const_iterator itSVT =
599 m_SCPD->GetServiceStateTable().
600 find(relatedStateVariableName);
601 if (itSVT == m_SCPD->GetServiceStateTable().end()) {
602 msg << "Inconsistent Service State Table, did not find '" <<
603 relatedStateVariableName <<
604 "' for argument '" << argument.GetName() <<
605 "' for action '" << action.GetName() <<
606 "' for service '" << GetServiceType() << "'.";
607 AddDebugLogLineN(logUPnP, msg);
608 return false;
610 const CUPnPStateVariable &stateVariable = *(itSVT->second);
611 if ( !stateVariable.GetAllowedValueList().empty() &&
612 stateVariable.GetAllowedValueList().find(ArgValue[i].GetValue()) ==
613 stateVariable.GetAllowedValueList().end()) {
614 msg << "Value not allowed '" << ArgValue[i].GetValue() <<
615 "' for state variable '" << relatedStateVariableName <<
616 "' for argument '" << argument.GetName() <<
617 "' for action '" << action.GetName() <<
618 "' for service '" << GetServiceType() << "'.";
619 AddDebugLogLineN(logUPnP, msg);
620 return false;
623 if (firstTime) {
624 firstTime = false;
625 } else {
626 msgAction << ", ";
628 msgAction <<
629 ArgValue[i].GetArgument() <<
630 "='" <<
631 ArgValue[i].GetValue() <<
632 "'";
634 msgAction << ")";
635 AddDebugLogLineN(logUPnP, msgAction);
636 // Everything is ok, make the action
637 IXML_Document *ActionDoc = NULL;
638 if (!ArgValue.empty()) {
639 for (unsigned int i = 0; i < ArgValue.size(); ++i) {
640 int ret = UpnpAddToAction(
641 &ActionDoc,
642 action.GetName().c_str(),
643 GetServiceType().c_str(),
644 ArgValue[i].GetArgument().c_str(),
645 ArgValue[i].GetValue().c_str());
646 if (ret != UPNP_E_SUCCESS) {
647 UPnP::ProcessErrorMessage(
648 "UpnpAddToAction", ret, NULL, NULL);
649 return false;
652 } else {
653 ActionDoc = UpnpMakeAction(
654 action.GetName().c_str(),
655 GetServiceType().c_str(),
656 0, NULL);
657 if (!ActionDoc) {
658 msg << "Error: UpnpMakeAction returned NULL.";
659 AddDebugLogLineN(logUPnP, msg);
660 return false;
663 #if 0
664 // Send the action asynchronously
665 UpnpSendActionAsync(
666 m_UPnPControlPoint.GetUPnPClientHandle(),
667 GetAbsControlURL().c_str(),
668 GetServiceType().c_str(),
669 NULL, ActionDoc,
670 static_cast<Upnp_FunPtr>(&CUPnPControlPoint::Callback),
671 NULL);
672 return true;
673 #endif
675 // Send the action synchronously
676 IXML_Document *RespDoc = NULL;
677 int ret = UpnpSendAction(
678 m_UPnPControlPoint.GetUPnPClientHandle(),
679 GetAbsControlURL().c_str(),
680 GetServiceType().c_str(),
681 NULL, ActionDoc, &RespDoc);
682 if (ret != UPNP_E_SUCCESS) {
683 UPnP::ProcessErrorMessage(
684 "UpnpSendAction", ret, NULL, RespDoc);
685 IXML::Document::Free(ActionDoc);
686 IXML::Document::Free(RespDoc);
687 return false;
689 IXML::Document::Free(ActionDoc);
691 // Check the response document
692 UPnP::ProcessActionResponse(RespDoc, action.GetName());
694 // Free the response document
695 IXML::Document::Free(RespDoc);
697 return true;
701 const std::string CUPnPService::GetStateVariable(
702 const std::string &stateVariableName) const
704 std::ostringstream msg;
705 DOMString StVarVal;
706 int ret = UpnpGetServiceVarStatus(
707 m_UPnPControlPoint.GetUPnPClientHandle(),
708 GetAbsControlURL().c_str(),
709 stateVariableName.c_str(),
710 &StVarVal);
711 if (ret != UPNP_E_SUCCESS) {
712 msg << "GetStateVariable(\"" <<
713 stateVariableName <<
714 "\"): in a call to UpnpGetServiceVarStatus";
715 UPnP::ProcessErrorMessage(
716 msg.str(), ret, StVarVal, NULL);
717 return stdEmptyString;
719 msg << "GetStateVariable: " <<
720 stateVariableName <<
721 "='" <<
722 StVarVal <<
723 "'.";
724 AddDebugLogLineN(logUPnP, msg);
725 return StVarVal;
729 CUPnPDevice::CUPnPDevice(
730 const CUPnPControlPoint &upnpControlPoint,
731 IXML_Element *device,
732 const std::string &URLBase)
734 m_DeviceList(upnpControlPoint, device, URLBase),
735 m_ServiceList(upnpControlPoint, device, URLBase),
736 m_deviceType (IXML::Element::GetChildValueByTag(device, "deviceType")),
737 m_friendlyName (IXML::Element::GetChildValueByTag(device, "friendlyName")),
738 m_manufacturer (IXML::Element::GetChildValueByTag(device, "manufacturer")),
739 m_manufacturerURL (IXML::Element::GetChildValueByTag(device, "manufacturerURL")),
740 m_modelDescription (IXML::Element::GetChildValueByTag(device, "modelDescription")),
741 m_modelName (IXML::Element::GetChildValueByTag(device, "modelName")),
742 m_modelNumber (IXML::Element::GetChildValueByTag(device, "modelNumber")),
743 m_modelURL (IXML::Element::GetChildValueByTag(device, "modelURL")),
744 m_serialNumber (IXML::Element::GetChildValueByTag(device, "serialNumber")),
745 m_UDN (IXML::Element::GetChildValueByTag(device, "UDN")),
746 m_UPC (IXML::Element::GetChildValueByTag(device, "UPC")),
747 m_presentationURL (IXML::Element::GetChildValueByTag(device, "presentationURL"))
749 std::ostringstream msg;
750 int presURLlen = strlen(URLBase.c_str()) +
751 strlen(m_presentationURL.c_str()) + 2;
752 std::vector<char> vpresURL(presURLlen);
753 char* presURL = &vpresURL[0];
754 int errcode = UpnpResolveURL(
755 URLBase.c_str(),
756 m_presentationURL.c_str(),
757 presURL);
758 if (errcode != UPNP_E_SUCCESS) {
759 msg << "Error generating presentationURL from " <<
760 "|" << URLBase << "|" <<
761 m_presentationURL << "|.";
762 AddDebugLogLineN(logUPnP, msg);
763 } else {
764 m_presentationURL = presURL;
767 msg.str("");
768 msg << "\n Device: " <<
769 "\n friendlyName: " << m_friendlyName <<
770 "\n deviceType: " << m_deviceType <<
771 "\n manufacturer: " << m_manufacturer <<
772 "\n manufacturerURL: " << m_manufacturerURL <<
773 "\n modelDescription: " << m_modelDescription <<
774 "\n modelName: " << m_modelName <<
775 "\n modelNumber: " << m_modelNumber <<
776 "\n modelURL: " << m_modelURL <<
777 "\n serialNumber: " << m_serialNumber <<
778 "\n UDN: " << m_UDN <<
779 "\n UPC: " << m_UPC <<
780 "\n presentationURL: " << m_presentationURL;
781 AddDebugLogLineN(logUPnP, msg);
785 CUPnPRootDevice::CUPnPRootDevice(
786 const CUPnPControlPoint &upnpControlPoint,
787 IXML_Element *rootDevice,
788 const std::string &OriginalURLBase,
789 const std::string &FixedURLBase,
790 const char *location,
791 int expires)
793 CUPnPDevice(upnpControlPoint, rootDevice, FixedURLBase),
794 m_URLBase(OriginalURLBase),
795 m_location(location),
796 m_expires(expires)
798 std::ostringstream msg;
799 msg <<
800 "\n Root Device: " <<
801 "\n URLBase: " << m_URLBase <<
802 "\n Fixed URLBase: " << FixedURLBase <<
803 "\n location: " << m_location <<
804 "\n expires: " << m_expires;
805 AddDebugLogLineN(logUPnP, msg);
809 CUPnPControlPoint *CUPnPControlPoint::s_CtrlPoint = NULL;
812 CUPnPControlPoint::CUPnPControlPoint(unsigned short udpPort)
814 m_UPnPClientHandle(),
815 m_RootDeviceMap(),
816 m_ServiceMap(),
817 m_ActivePortMappingsMap(),
818 m_RootDeviceListMutex(),
819 m_IGWDeviceDetected(false),
820 m_WanService(NULL)
822 // Pointer to self
823 s_CtrlPoint = this;
824 // Null string at first
825 std::ostringstream msg;
827 // Start UPnP
828 int ret;
829 char *ipAddress = NULL;
830 unsigned short port = 0;
831 ret = UpnpInit(ipAddress, udpPort);
832 if (ret != UPNP_E_SUCCESS) {
833 msg << "error(UpnpInit): Error code ";
834 goto error;
836 port = UpnpGetServerPort();
837 ipAddress = UpnpGetServerIpAddress();
838 msg << "bound to " << ipAddress << ":" <<
839 port << ".";
840 AddDebugLogLineN(logUPnP, msg);
841 msg.str("");
842 ret = UpnpRegisterClient(
843 static_cast<Upnp_FunPtr>(&CUPnPControlPoint::Callback),
844 &m_UPnPClientHandle,
845 &m_UPnPClientHandle);
846 if (ret != UPNP_E_SUCCESS) {
847 msg << "error(UpnpRegisterClient): Error registering callback: ";
848 goto error;
851 // We could ask for just the right device here. If the root device
852 // contains the device we want, it will respond with the full XML doc,
853 // including the root device and every sub-device it has.
855 // But let's find out what we have in our network by calling UPnP::ROOT_DEVICE.
857 // We should not search twice, because this will produce two
858 // UPNP_DISCOVERY_SEARCH_TIMEOUT events, and we might end with problems
859 // on the mutex.
860 ret = UpnpSearchAsync(m_UPnPClientHandle, 3, UPnP::ROOT_DEVICE.c_str(), NULL);
861 //ret = UpnpSearchAsync(m_UPnPClientHandle, 3, UPnP::Device::IGW.c_str(), this);
862 //ret = UpnpSearchAsync(m_UPnPClientHandle, 3, UPnP::Device::LAN.c_str(), this);
863 //ret = UpnpSearchAsync(m_UPnPClientHandle, 3, UPnP::Device::WAN_Connection.c_str(), this);
864 if (ret != UPNP_E_SUCCESS) {
865 msg << "error(UpnpSearchAsync): Error sending search request: ";
866 goto error;
869 // Wait for the UPnP initialization to complete.
871 // Lock the search timeout mutex
872 m_WaitForSearchTimeoutMutex.Lock();
874 // Lock it again, so that we block. Unlocking will only happen
875 // when the UPNP_DISCOVERY_SEARCH_TIMEOUT event occurs at the
876 // callback.
877 CUPnPMutexLocker lock(m_WaitForSearchTimeoutMutex);
879 return;
881 // Error processing
882 error:
883 UpnpFinish();
884 msg << ret << ": " << UpnpGetErrorMessage(ret) << ".";
885 throw CUPnPException(msg);
889 CUPnPControlPoint::~CUPnPControlPoint()
891 for( RootDeviceMap::iterator it = m_RootDeviceMap.begin();
892 it != m_RootDeviceMap.end();
893 ++it) {
894 delete it->second;
896 // Remove all first
897 // RemoveAll();
898 UpnpUnRegisterClient(m_UPnPClientHandle);
899 UpnpFinish();
903 bool CUPnPControlPoint::AddPortMappings(
904 std::vector<CUPnPPortMapping> &upnpPortMapping)
906 std::ostringstream msg;
907 if (!WanServiceDetected()) {
908 msg << "UPnP Error: "
909 "CUPnPControlPoint::AddPortMapping: "
910 "WAN Service not detected.";
911 AddDebugLogLineC(logUPnP, msg);
912 return false;
915 int n = upnpPortMapping.size();
916 bool ok = false;
918 // Check the number of port mappings before
919 std::istringstream PortMappingNumberOfEntries(
920 m_WanService->GetStateVariable(
921 "PortMappingNumberOfEntries"));
922 unsigned long oldNumberOfEntries;
923 PortMappingNumberOfEntries >> oldNumberOfEntries;
925 // Add the enabled port mappings
926 for (int i = 0; i < n; ++i) {
927 if (upnpPortMapping[i].getEnabled() == "1") {
928 // Add the mapping to the control point
929 // active mappings list
930 m_ActivePortMappingsMap[upnpPortMapping[i].getKey()] =
931 upnpPortMapping[i];
933 // Add the port mapping
934 PrivateAddPortMapping(upnpPortMapping[i]);
938 // Test some variables, this is deprecated, might not work
939 // with some routers
940 m_WanService->GetStateVariable("ConnectionType");
941 m_WanService->GetStateVariable("PossibleConnectionTypes");
942 m_WanService->GetStateVariable("ConnectionStatus");
943 m_WanService->GetStateVariable("Uptime");
944 m_WanService->GetStateVariable("LastConnectionError");
945 m_WanService->GetStateVariable("RSIPAvailable");
946 m_WanService->GetStateVariable("NATEnabled");
947 m_WanService->GetStateVariable("ExternalIPAddress");
948 m_WanService->GetStateVariable("PortMappingNumberOfEntries");
949 m_WanService->GetStateVariable("PortMappingLeaseDuration");
951 // Just for testing
952 std::vector<CUPnPArgumentValue> argval;
953 argval.resize(0);
954 m_WanService->Execute("GetStatusInfo", argval);
956 #if 0
957 // These do not work. Their value must be requested for a
958 // specific port mapping.
959 m_WanService->GetStateVariable("PortMappingEnabled");
960 m_WanService->GetStateVariable("RemoteHost");
961 m_WanService->GetStateVariable("ExternalPort");
962 m_WanService->GetStateVariable("InternalPort");
963 m_WanService->GetStateVariable("PortMappingProtocol");
964 m_WanService->GetStateVariable("InternalClient");
965 m_WanService->GetStateVariable("PortMappingDescription");
966 #endif
968 // Debug only
969 msg.str("");
970 msg << "CUPnPControlPoint::AddPortMappings: "
971 "m_ActivePortMappingsMap.size() == " <<
972 m_ActivePortMappingsMap.size();
973 AddDebugLogLineN(logUPnP, msg);
975 // Not very good, must find a better test
976 PortMappingNumberOfEntries.str(
977 m_WanService->GetStateVariable(
978 "PortMappingNumberOfEntries"));
979 unsigned long newNumberOfEntries;
980 PortMappingNumberOfEntries >> newNumberOfEntries;
981 ok = newNumberOfEntries - oldNumberOfEntries == 4;
983 return ok;
987 void CUPnPControlPoint::RefreshPortMappings()
989 for ( PortMappingMap::iterator it = m_ActivePortMappingsMap.begin();
990 it != m_ActivePortMappingsMap.end();
991 ++it) {
992 PrivateAddPortMapping(it->second);
995 // For testing
996 m_WanService->GetStateVariable("PortMappingNumberOfEntries");
1000 bool CUPnPControlPoint::PrivateAddPortMapping(
1001 CUPnPPortMapping &upnpPortMapping)
1003 // Get an IP address. The UPnP server one must do.
1004 std::string ipAddress(UpnpGetServerIpAddress());
1006 // Start building the action
1007 std::string actionName("AddPortMapping");
1008 std::vector<CUPnPArgumentValue> argval(8);
1010 // Action parameters
1011 argval[0].SetArgument("NewRemoteHost");
1012 argval[0].SetValue("");
1013 argval[1].SetArgument("NewExternalPort");
1014 argval[1].SetValue(upnpPortMapping.getPort());
1015 argval[2].SetArgument("NewProtocol");
1016 argval[2].SetValue(upnpPortMapping.getProtocol());
1017 argval[3].SetArgument("NewInternalPort");
1018 argval[3].SetValue(upnpPortMapping.getPort());
1019 argval[4].SetArgument("NewInternalClient");
1020 argval[4].SetValue(ipAddress);
1021 argval[5].SetArgument("NewEnabled");
1022 argval[5].SetValue("1");
1023 argval[6].SetArgument("NewPortMappingDescription");
1024 argval[6].SetValue(upnpPortMapping.getDescription());
1025 argval[7].SetArgument("NewLeaseDuration");
1026 argval[7].SetValue("0");
1028 // Execute
1029 bool ret = true;
1030 for (ServiceMap::iterator it = m_ServiceMap.begin();
1031 it != m_ServiceMap.end(); ++it) {
1032 ret &= it->second->Execute(actionName, argval);
1035 return ret;
1039 bool CUPnPControlPoint::DeletePortMappings(
1040 std::vector<CUPnPPortMapping> &upnpPortMapping)
1042 std::ostringstream msg;
1043 if (!WanServiceDetected()) {
1044 msg << "UPnP Error: "
1045 "CUPnPControlPoint::DeletePortMapping: "
1046 "WAN Service not detected.";
1047 AddDebugLogLineC(logUPnP, msg);
1048 return false;
1051 int n = upnpPortMapping.size();
1052 bool ok = false;
1054 // Check the number of port mappings before
1055 std::istringstream PortMappingNumberOfEntries(
1056 m_WanService->GetStateVariable(
1057 "PortMappingNumberOfEntries"));
1058 unsigned long oldNumberOfEntries;
1059 PortMappingNumberOfEntries >> oldNumberOfEntries;
1061 // Delete the enabled port mappings
1062 for (int i = 0; i < n; ++i) {
1063 if (upnpPortMapping[i].getEnabled() == "1") {
1064 // Delete the mapping from the control point
1065 // active mappings list
1066 PortMappingMap::iterator it =
1067 m_ActivePortMappingsMap.find(
1068 upnpPortMapping[i].getKey());
1069 if (it != m_ActivePortMappingsMap.end()) {
1070 m_ActivePortMappingsMap.erase(it);
1071 } else {
1072 msg << "UPnP Error: "
1073 "CUPnPControlPoint::DeletePortMapping: "
1074 "Mapping was not found in the active "
1075 "mapping map.";
1076 AddDebugLogLineC(logUPnP, msg);
1079 // Delete the port mapping
1080 PrivateDeletePortMapping(upnpPortMapping[i]);
1084 // Debug only
1085 msg.str("");
1086 msg << "CUPnPControlPoint::DeletePortMappings: "
1087 "m_ActivePortMappingsMap.size() == " <<
1088 m_ActivePortMappingsMap.size();
1089 AddDebugLogLineN(logUPnP, msg);
1091 // Not very good, must find a better test
1092 PortMappingNumberOfEntries.str(
1093 m_WanService->GetStateVariable(
1094 "PortMappingNumberOfEntries"));
1095 unsigned long newNumberOfEntries;
1096 PortMappingNumberOfEntries >> newNumberOfEntries;
1097 ok = oldNumberOfEntries - newNumberOfEntries == 4;
1099 return ok;
1103 bool CUPnPControlPoint::PrivateDeletePortMapping(
1104 CUPnPPortMapping &upnpPortMapping)
1106 // Start building the action
1107 std::string actionName("DeletePortMapping");
1108 std::vector<CUPnPArgumentValue> argval(3);
1110 // Action parameters
1111 argval[0].SetArgument("NewRemoteHost");
1112 argval[0].SetValue("");
1113 argval[1].SetArgument("NewExternalPort");
1114 argval[1].SetValue(upnpPortMapping.getPort());
1115 argval[2].SetArgument("NewProtocol");
1116 argval[2].SetValue(upnpPortMapping.getProtocol());
1118 // Execute
1119 bool ret = true;
1120 for (ServiceMap::iterator it = m_ServiceMap.begin();
1121 it != m_ServiceMap.end(); ++it) {
1122 ret &= it->second->Execute(actionName, argval);
1125 return ret;
1129 // This function is static
1130 #if UPNP_VERSION >= 10800
1131 int CUPnPControlPoint::Callback(Upnp_EventType_e EventType, const void *Event, void * /*Cookie*/)
1132 #else
1133 int CUPnPControlPoint::Callback(Upnp_EventType EventType, void *Event, void * /*Cookie*/)
1134 #endif
1136 std::ostringstream msg;
1137 std::ostringstream msg2;
1138 // Somehow, this is unreliable. UPNP_DISCOVERY_ADVERTISEMENT_ALIVE events
1139 // happen with a wrong cookie and... boom!
1140 // CUPnPControlPoint *upnpCP = static_cast<CUPnPControlPoint *>(Cookie);
1141 CUPnPControlPoint *upnpCP = CUPnPControlPoint::s_CtrlPoint;
1143 //fprintf(stderr, "Callback: %d, Cookie: %p\n", EventType, Cookie);
1144 switch (EventType) {
1145 case UPNP_DISCOVERY_ADVERTISEMENT_ALIVE:
1146 //fprintf(stderr, "Callback: UPNP_DISCOVERY_ADVERTISEMENT_ALIVE\n");
1147 msg << "error(UPNP_DISCOVERY_ADVERTISEMENT_ALIVE): ";
1148 msg2<< "UPNP_DISCOVERY_ADVERTISEMENT_ALIVE: ";
1149 goto upnpDiscovery;
1150 case UPNP_DISCOVERY_SEARCH_RESULT: {
1151 //fprintf(stderr, "Callback: UPNP_DISCOVERY_SEARCH_RESULT\n");
1152 msg << "error(UPNP_DISCOVERY_SEARCH_RESULT): ";
1153 msg2<< "UPNP_DISCOVERY_SEARCH_RESULT: ";
1154 // UPnP Discovery
1155 upnpDiscovery:
1156 #if UPNP_VERSION >= 10800
1157 UpnpDiscovery *d_event = (UpnpDiscovery *)Event;
1158 #else
1159 struct Upnp_Discovery *d_event = (struct Upnp_Discovery *)Event;
1160 #endif
1161 IXML_Document *doc = NULL;
1162 #if UPNP_VERSION >= 10800
1163 int errCode = UpnpDiscovery_get_ErrCode(d_event);
1164 if (errCode != UPNP_E_SUCCESS) {
1165 msg << UpnpGetErrorMessage(errCode) << ".";
1166 #else
1167 int ret;
1168 if (d_event->ErrCode != UPNP_E_SUCCESS) {
1169 msg << UpnpGetErrorMessage(d_event->ErrCode) << ".";
1170 #endif
1171 AddDebugLogLineC(logUPnP, msg);
1173 // Get the XML tree device description in doc
1174 #if UPNP_VERSION >= 10800
1175 const char *location = UpnpDiscovery_get_Location_cstr(d_event);
1176 int ret = UpnpDownloadXmlDoc(location, &doc);
1177 #else
1178 ret = UpnpDownloadXmlDoc(d_event->Location, &doc);
1179 #endif
1180 if (ret != UPNP_E_SUCCESS) {
1181 msg << "Error retrieving device description from " <<
1182 #if UPNP_VERSION >= 10800
1183 location << ": " <<
1184 #else
1185 d_event->Location << ": " <<
1186 #endif
1187 UpnpGetErrorMessage(ret) <<
1188 "(" << ret << ").";
1189 AddDebugLogLineC(logUPnP, msg);
1190 } else {
1191 msg2 << "Retrieving device description from " <<
1192 #if UPNP_VERSION >= 10800
1193 location << ".";
1194 #else
1195 d_event->Location << ".";
1196 #endif
1197 AddDebugLogLineN(logUPnP, msg2);
1199 if (doc) {
1200 // Get the root node
1201 IXML_Element *root = IXML::Document::GetRootElement(doc);
1202 // Extract the URLBase
1203 const std::string urlBase = IXML::Element::GetChildValueByTag(root, "URLBase");
1204 // Get the root device
1205 IXML_Element *rootDevice = IXML::Element::GetFirstChildByTag(root, "device");
1206 // Extract the deviceType
1207 std::string devType(IXML::Element::GetChildValueByTag(rootDevice, "deviceType"));
1208 // Only add device if it is an InternetGatewayDevice
1209 if (stdStringIsEqualCI(devType, UPnP::Device::IGW)) {
1210 // This condition can be used to auto-detect
1211 // the UPnP device we are interested in.
1212 // Obs.: Don't block the entry here on this
1213 // condition! There may be more than one device,
1214 // and the first that enters may not be the one
1215 // we are interested in!
1216 upnpCP->SetIGWDeviceDetected(true);
1217 // Log it if not UPNP_DISCOVERY_ADVERTISEMENT_ALIVE,
1218 // we don't want to spam our logs.
1219 if (EventType != UPNP_DISCOVERY_ADVERTISEMENT_ALIVE) {
1220 msg.str("Internet Gateway Device Detected.");
1221 AddDebugLogLineC(logUPnP, msg);
1223 // Add the root device to our list
1224 #if UPNP_VERSION >= 10800
1225 int expires = UpnpDiscovery_get_Expires(d_event);
1226 upnpCP->AddRootDevice(rootDevice, urlBase,
1227 location, expires);
1228 #else
1229 upnpCP->AddRootDevice(rootDevice, urlBase,
1230 d_event->Location, d_event->Expires);
1231 #endif
1233 // Free the XML doc tree
1234 IXML::Document::Free(doc);
1236 break;
1238 case UPNP_DISCOVERY_SEARCH_TIMEOUT: {
1239 //fprintf(stderr, "Callback: UPNP_DISCOVERY_SEARCH_TIMEOUT\n");
1240 // Search timeout
1241 msg << "UPNP_DISCOVERY_SEARCH_TIMEOUT.";
1242 AddDebugLogLineN(logUPnP, msg);
1244 // Unlock the search timeout mutex
1245 upnpCP->m_WaitForSearchTimeoutMutex.Unlock();
1247 break;
1249 case UPNP_DISCOVERY_ADVERTISEMENT_BYEBYE: {
1250 //fprintf(stderr, "Callback: UPNP_DISCOVERY_ADVERTISEMENT_BYEBYE\n");
1251 // UPnP Device Removed
1252 #if UPNP_VERSION >= 10800
1253 UpnpDiscovery *dab_event = (UpnpDiscovery *)Event;
1254 int errCode = UpnpDiscovery_get_ErrCode(dab_event);
1255 if (errCode != UPNP_E_SUCCESS) {
1256 #else
1257 struct Upnp_Discovery *dab_event = (struct Upnp_Discovery *)Event;
1258 if (dab_event->ErrCode != UPNP_E_SUCCESS) {
1259 #endif
1260 msg << "error(UPNP_DISCOVERY_ADVERTISEMENT_BYEBYE): " <<
1261 #if UPNP_VERSION >= 10800
1262 UpnpGetErrorMessage(errCode) <<
1263 #else
1264 UpnpGetErrorMessage(dab_event->ErrCode) <<
1265 #endif
1266 ".";
1267 AddDebugLogLineC(logUPnP, msg);
1269 #if UPNP_VERSION >= 10800
1270 std::string devType = UpnpDiscovery_get_DeviceType_cstr(dab_event);
1271 #else
1272 std::string devType = dab_event->DeviceType;
1273 #endif
1274 // Check for an InternetGatewayDevice and removes it from the list
1276 std::transform(devType.begin(), devType.end(), devType.begin(), tolower);
1278 if (stdStringIsEqualCI(devType, UPnP::Device::IGW)) {
1279 #if UPNP_VERSION >= 10800
1280 const char *deviceID =
1281 UpnpDiscovery_get_DeviceID_cstr(dab_event);
1282 upnpCP->RemoveRootDevice(deviceID);
1283 #else
1284 upnpCP->RemoveRootDevice(dab_event->DeviceId);
1285 #endif
1287 break;
1289 case UPNP_EVENT_RECEIVED: {
1290 //fprintf(stderr, "Callback: UPNP_EVENT_RECEIVED\n");
1291 // Event reveived
1292 #if UPNP_VERSION >= 10800
1293 UpnpEvent *e_event = (UpnpEvent *)Event;
1294 int eventKey = UpnpEvent_get_EventKey(e_event);
1295 IXML_Document *changedVariables =
1296 UpnpEvent_get_ChangedVariables(e_event);
1297 const std::string sid = UpnpEvent_get_SID_cstr(e_event);
1298 #else
1299 struct Upnp_Event *e_event = (struct Upnp_Event *)Event;
1300 const std::string Sid = e_event->Sid;
1301 #endif
1302 // Parses the event
1303 #if UPNP_VERSION >= 10800
1304 upnpCP->OnEventReceived(sid, eventKey, changedVariables);
1305 #else
1306 upnpCP->OnEventReceived(Sid, e_event->EventKey, e_event->ChangedVariables);
1307 #endif
1308 break;
1310 case UPNP_EVENT_SUBSCRIBE_COMPLETE:
1311 //fprintf(stderr, "Callback: UPNP_EVENT_SUBSCRIBE_COMPLETE\n");
1312 msg << "error(UPNP_EVENT_SUBSCRIBE_COMPLETE): ";
1313 goto upnpEventRenewalComplete;
1314 case UPNP_EVENT_UNSUBSCRIBE_COMPLETE:
1315 //fprintf(stderr, "Callback: UPNP_EVENT_UNSUBSCRIBE_COMPLETE\n");
1316 msg << "error(UPNP_EVENT_UNSUBSCRIBE_COMPLETE): ";
1317 goto upnpEventRenewalComplete;
1318 case UPNP_EVENT_RENEWAL_COMPLETE: {
1319 //fprintf(stderr, "Callback: UPNP_EVENT_RENEWAL_COMPLETE\n");
1320 msg << "error(UPNP_EVENT_RENEWAL_COMPLETE): ";
1321 upnpEventRenewalComplete:
1322 #if UPNP_VERSION >= 10800
1323 UpnpEventSubscribe *es_event = (UpnpEventSubscribe *)Event;
1324 int errCode = UpnpEventSubscribe_get_ErrCode(es_event);
1325 if (errCode != UPNP_E_SUCCESS) {
1326 #else
1327 struct Upnp_Event_Subscribe *es_event =
1328 (struct Upnp_Event_Subscribe *)Event;
1329 if (es_event->ErrCode != UPNP_E_SUCCESS) {
1330 #endif
1331 msg << "Error in Event Subscribe Callback";
1332 #if UPNP_VERSION >= 10800
1333 UPnP::ProcessErrorMessage(msg.str(), errCode, NULL, NULL);
1334 #else
1335 UPnP::ProcessErrorMessage(
1336 msg.str(), es_event->ErrCode, NULL, NULL);
1337 #endif
1338 } else {
1339 #if 0
1340 #if UPNP_VERSION >= 10800
1342 const UpnpString *publisherUrl =
1343 UpnpEventSubscribe_get_PublisherUrl(es_event);
1344 const char *sid = UpnpEvent_get_SID_cstr(es_event);
1345 int timeOut = UpnpEvent_get_TimeOut(es_event);
1346 TvCtrlPointHandleSubscribeUpdate(
1347 publisherUrl, sid, timeOut);
1348 #else
1349 TvCtrlPointHandleSubscribeUpdate(
1350 GET_UPNP_STRING(es_event->PublisherUrl),
1351 es_event->Sid,
1352 es_event->TimeOut );
1353 #endif
1354 #endif
1356 break;
1358 case UPNP_EVENT_AUTORENEWAL_FAILED:
1359 //fprintf(stderr, "Callback: UPNP_EVENT_AUTORENEWAL_FAILED\n");
1360 msg << "error(UPNP_EVENT_AUTORENEWAL_FAILED): ";
1361 msg2 << "UPNP_EVENT_AUTORENEWAL_FAILED: ";
1362 goto upnpEventSubscriptionExpired;
1363 case UPNP_EVENT_SUBSCRIPTION_EXPIRED: {
1364 //fprintf(stderr, "Callback: UPNP_EVENT_SUBSCRIPTION_EXPIRED\n");
1365 msg << "error(UPNP_EVENT_SUBSCRIPTION_EXPIRED): ";
1366 msg2 << "UPNP_EVENT_SUBSCRIPTION_EXPIRED: ";
1367 upnpEventSubscriptionExpired:
1368 #if UPNP_VERSION >= 10800
1369 UpnpEventSubscribe *es_event = (UpnpEventSubscribe *)Event;
1370 #else
1371 struct Upnp_Event_Subscribe *es_event =
1372 (struct Upnp_Event_Subscribe *)Event;
1373 #endif
1374 Upnp_SID newSID;
1375 memset(newSID, 0, sizeof(Upnp_SID));
1376 int TimeOut = 1801;
1377 #if UPNP_VERSION >= 10800
1378 const char *publisherUrl =
1379 UpnpEventSubscribe_get_PublisherUrl_cstr(es_event);
1380 #endif
1381 int ret = UpnpSubscribe(
1382 upnpCP->m_UPnPClientHandle,
1383 #if UPNP_VERSION >= 10800
1384 publisherUrl,
1385 #else
1386 GET_UPNP_STRING(es_event->PublisherUrl),
1387 #endif
1388 &TimeOut,
1389 newSID);
1390 if (ret != UPNP_E_SUCCESS) {
1391 msg << "Error Subscribing to EventURL";
1392 #if UPNP_VERSION >= 10800
1393 int errCode = UpnpEventSubscribe_get_ErrCode(es_event);
1394 #endif
1395 UPnP::ProcessErrorMessage(
1396 #if UPNP_VERSION >= 10800
1397 msg.str(), errCode, NULL, NULL);
1398 #else
1399 msg.str(), es_event->ErrCode, NULL, NULL);
1400 #endif
1401 } else {
1402 ServiceMap::iterator it =
1403 #if UPNP_VERSION >= 10800
1404 upnpCP->m_ServiceMap.find(publisherUrl);
1405 #else
1406 upnpCP->m_ServiceMap.find(GET_UPNP_STRING(es_event->PublisherUrl));
1407 #endif
1408 if (it != upnpCP->m_ServiceMap.end()) {
1409 CUPnPService &service = *(it->second);
1410 service.SetTimeout(TimeOut);
1411 service.SetSID(newSID);
1412 msg2 << "Re-subscribed to EventURL '" <<
1413 #if UPNP_VERSION >= 10800
1414 publisherUrl <<
1415 #else
1416 GET_UPNP_STRING(es_event->PublisherUrl) <<
1417 #endif
1418 "' with SID == '" <<
1419 newSID << "'.";
1420 AddDebugLogLineC(logUPnP, msg2);
1421 // In principle, we should test to see if the
1422 // service is the same. But here we only have one
1423 // service, so...
1424 upnpCP->RefreshPortMappings();
1425 } else {
1426 msg << "Error: did not find service " <<
1427 newSID << " in the service map.";
1428 AddDebugLogLineC(logUPnP, msg);
1431 break;
1433 case UPNP_CONTROL_ACTION_COMPLETE: {
1434 //fprintf(stderr, "Callback: UPNP_CONTROL_ACTION_COMPLETE\n");
1435 // This is here if we choose to do this asynchronously
1436 #if UPNP_VERSION >= 10800
1437 UpnpActionComplete *a_event = (UpnpActionComplete *)Event;
1438 int errCode = UpnpActionComplete_get_ErrCode(a_event);
1439 IXML_Document *actionResult =
1440 UpnpActionComplete_get_ActionResult(a_event);
1441 if (errCode != UPNP_E_SUCCESS) {
1442 #else
1443 struct Upnp_Action_Complete *a_event =
1444 (struct Upnp_Action_Complete *)Event;
1445 if (a_event->ErrCode != UPNP_E_SUCCESS) {
1446 #endif
1447 UPnP::ProcessErrorMessage(
1448 "UpnpSendActionAsync",
1449 #if UPNP_VERSION >= 10800
1450 errCode, NULL,
1451 actionResult);
1452 #else
1453 a_event->ErrCode, NULL,
1454 a_event->ActionResult);
1455 #endif
1456 } else {
1457 // Check the response document
1458 UPnP::ProcessActionResponse(
1459 #if UPNP_VERSION >= 10800
1460 actionResult,
1461 #else
1462 a_event->ActionResult,
1463 #endif
1464 "<UpnpSendActionAsync>");
1466 /* No need for any processing here, just print out results.
1467 * Service state table updates are handled by events.
1469 break;
1471 case UPNP_CONTROL_GET_VAR_COMPLETE: {
1472 //fprintf(stderr, "Callback: UPNP_CONTROL_GET_VAR_COMPLETE\n");
1473 msg << "error(UPNP_CONTROL_GET_VAR_COMPLETE): ";
1474 #if UPNP_VERSION >= 10800
1475 UpnpStateVarComplete *sv_event = (UpnpStateVarComplete *)Event;
1476 int errCode = UpnpStateVarComplete_get_ErrCode(sv_event);
1477 if (errCode != UPNP_E_SUCCESS) {
1478 #else
1479 struct Upnp_State_Var_Complete *sv_event =
1480 (struct Upnp_State_Var_Complete *)Event;
1481 if (sv_event->ErrCode != UPNP_E_SUCCESS) {
1482 #endif
1483 msg << "m_UpnpGetServiceVarStatusAsync";
1484 UPnP::ProcessErrorMessage(
1485 #if UPNP_VERSION >= 10800
1486 msg.str(), errCode, NULL, NULL);
1487 #else
1488 msg.str(), sv_event->ErrCode, NULL, NULL);
1489 #endif
1490 } else {
1491 #if 0
1492 // Warning: The use of UpnpGetServiceVarStatus and
1493 // UpnpGetServiceVarStatusAsync is deprecated by the
1494 // UPnP forum.
1495 #if UPNP_VERSION >= 10800
1496 const char *ctrlUrl =
1497 UpnpStateVarComplete_get_CtrlUrl(sv_event);
1498 const char *stateVarName =
1499 UpnpStateVarComplete_get_StateVarName(sv_event);
1500 const DOMString currentVal =
1501 UpnpStateVarComplete_get_CurrentVal(sv_event);
1502 TvCtrlPointHandleGetVar(
1503 ctrlUrl, stateVarName, currentVal);
1504 #else
1505 TvCtrlPointHandleGetVar(
1506 sv_event->CtrlUrl,
1507 sv_event->StateVarName,
1508 sv_event->CurrentVal );
1509 #endif
1510 #endif
1512 break;
1514 // ignore these cases, since this is not a device
1515 case UPNP_CONTROL_GET_VAR_REQUEST:
1516 //fprintf(stderr, "Callback: UPNP_CONTROL_GET_VAR_REQUEST\n");
1517 msg << "error(UPNP_CONTROL_GET_VAR_REQUEST): ";
1518 goto eventSubscriptionRequest;
1519 case UPNP_CONTROL_ACTION_REQUEST:
1520 //fprintf(stderr, "Callback: UPNP_CONTROL_ACTION_REQUEST\n");
1521 msg << "error(UPNP_CONTROL_ACTION_REQUEST): ";
1522 goto eventSubscriptionRequest;
1523 case UPNP_EVENT_SUBSCRIPTION_REQUEST:
1524 //fprintf(stderr, "Callback: UPNP_EVENT_SUBSCRIPTION_REQUEST\n");
1525 msg << "error(UPNP_EVENT_SUBSCRIPTION_REQUEST): ";
1526 eventSubscriptionRequest:
1527 msg << "This is not a UPnP Device, this is a UPnP Control Point, event ignored.";
1528 AddDebugLogLineC(logUPnP, msg);
1529 break;
1530 default:
1531 // Humm, this is not good, we forgot to handle something...
1532 fprintf(stderr,
1533 "Callback: default... Unknown event:'%d', not good.\n",
1534 EventType);
1535 msg << "error(UPnP::Callback): Event not handled:'" <<
1536 EventType << "'.";
1537 fprintf(stderr, "%s\n", msg.str().c_str());
1538 AddDebugLogLineC(logUPnP, msg);
1539 // Better not throw in the callback. Who would catch it?
1540 //throw CUPnPException(msg);
1541 break;
1544 return 0;
1548 void CUPnPControlPoint::OnEventReceived(
1549 const std::string &Sid,
1550 int EventKey,
1551 IXML_Document *ChangedVariablesDoc)
1553 std::ostringstream msg;
1554 msg << "UPNP_EVENT_RECEIVED:" <<
1555 "\n SID: " << Sid <<
1556 "\n Key: " << EventKey <<
1557 "\n Property list:";
1558 IXML_Element *root = IXML::Document::GetRootElement(ChangedVariablesDoc);
1559 IXML_Element *child = IXML::Element::GetFirstChild(root);
1560 if (child) {
1561 while (child) {
1562 IXML_Element *child2 = IXML::Element::GetFirstChild(child);
1563 const DOMString childTag = IXML::Element::GetTag(child2);
1564 std::string childValue = IXML::Element::GetTextValue(child2);
1565 msg << "\n " <<
1566 childTag << "='" <<
1567 childValue << "'";
1568 child = IXML::Element::GetNextSibling(child);
1570 } else {
1571 msg << "\n Empty property list.";
1573 AddDebugLogLineC(logUPnP, msg);
1577 void CUPnPControlPoint::AddRootDevice(
1578 IXML_Element *rootDevice, const std::string &urlBase,
1579 const char *location, int expires)
1581 // Lock the Root Device List
1582 CUPnPMutexLocker lock(m_RootDeviceListMutex);
1584 // Root node's URLBase
1585 std::string OriginalURLBase(urlBase);
1586 std::string FixedURLBase(OriginalURLBase.empty() ?
1587 location :
1588 OriginalURLBase);
1590 // Get the UDN (Unique Device Name)
1591 std::string UDN(IXML::Element::GetChildValueByTag(rootDevice, "UDN"));
1592 RootDeviceMap::iterator it = m_RootDeviceMap.find(UDN);
1593 bool alreadyAdded = it != m_RootDeviceMap.end();
1594 if (alreadyAdded) {
1595 // Just set the expires field
1596 it->second->SetExpires(expires);
1597 } else {
1598 // Add a new root device to the root device list
1599 CUPnPRootDevice *upnpRootDevice = new CUPnPRootDevice(
1600 *this, rootDevice,
1601 OriginalURLBase, FixedURLBase,
1602 location, expires);
1603 m_RootDeviceMap[upnpRootDevice->GetUDN()] = upnpRootDevice;
1608 void CUPnPControlPoint::RemoveRootDevice(const char *udn)
1610 // Lock the Root Device List
1611 CUPnPMutexLocker lock(m_RootDeviceListMutex);
1613 // Remove
1614 std::string UDN(udn);
1615 RootDeviceMap::iterator it = m_RootDeviceMap.find(UDN);
1616 if (it != m_RootDeviceMap.end()) {
1617 delete it->second;
1618 m_RootDeviceMap.erase(UDN);
1623 void CUPnPControlPoint::Subscribe(CUPnPService &service)
1625 std::ostringstream msg;
1627 IXML_Document *scpdDoc = NULL;
1628 int errcode = UpnpDownloadXmlDoc(
1629 service.GetAbsSCPDURL().c_str(), &scpdDoc);
1630 if (errcode == UPNP_E_SUCCESS) {
1631 // Get the root node of this service (the SCPD Document)
1632 IXML_Element *scpdRoot = IXML::Document::GetRootElement(scpdDoc);
1633 CUPnPSCPD *scpd = new CUPnPSCPD(*this, scpdRoot, service.GetAbsSCPDURL());
1634 service.SetSCPD(scpd);
1635 IXML::Document::Free(scpdDoc);
1636 m_ServiceMap[service.GetAbsEventSubURL()] = &service;
1637 msg << "Successfully retrieved SCPD Document for service " <<
1638 service.GetServiceType() << ", absEventSubURL: " <<
1639 service.GetAbsEventSubURL() << ".";
1640 AddDebugLogLineC(logUPnP, msg);
1641 msg.str("");
1643 // Now try to subscribe to this service. If the subscription
1644 // is not successfull, we will not be notified about events,
1645 // but it may be possible to use the service anyway.
1646 errcode = UpnpSubscribe(m_UPnPClientHandle,
1647 service.GetAbsEventSubURL().c_str(),
1648 service.GetTimeoutAddr(),
1649 service.GetSID());
1650 if (errcode == UPNP_E_SUCCESS) {
1651 msg << "Successfully subscribed to service " <<
1652 service.GetServiceType() << ", absEventSubURL: " <<
1653 service.GetAbsEventSubURL() << ".";
1654 AddDebugLogLineC(logUPnP, msg);
1655 } else {
1656 msg << "Error subscribing to service " <<
1657 service.GetServiceType() << ", absEventSubURL: " <<
1658 service.GetAbsEventSubURL() << ", error: " <<
1659 UpnpGetErrorMessage(errcode) << ".";
1660 goto error;
1662 } else {
1663 msg << "Error getting SCPD Document from " <<
1664 service.GetAbsSCPDURL() << ".";
1665 AddDebugLogLineC(logUPnP, msg);
1668 return;
1670 // Error processing
1671 error:
1672 AddDebugLogLineC(logUPnP, msg);
1676 void CUPnPControlPoint::Unsubscribe(CUPnPService &service)
1678 ServiceMap::iterator it = m_ServiceMap.find(service.GetAbsEventSubURL());
1679 if (it != m_ServiceMap.end()) {
1680 m_ServiceMap.erase(it);
1681 UpnpUnSubscribe(m_UPnPClientHandle, service.GetSID());
1685 #endif /* ENABLE_UPNP */