38template <
typename ClientType>
96 uint64_t now = millis();
121 uint64_t
end = millis() + 2000;
122 while (millis() <
end) {
148 uint64_t now = millis();
152 if (count > 0) rc =
true;
161 if (count > 0) rc =
true;
190 if (schedule !=
nullptr) {
214 if (
p_device_info ==
nullptr || requestPath ==
nullptr)
return nullptr;
225 std::function<
size_t(Print&,
void*)> changeWriter,
226 void* ref)
override {
232 "addChange: No service info available for %s",
236 mgr.addChange(serviceInfo, changeWriter, ref);
271 auto start = millis();
279 bool is_attribute =
false;
280 bool is_action =
false;
283 Client& client = server->
client();
286 size_t len = client.readBytes(buffer,
sizeof(buffer));
288 xp.
write((
const uint8_t*)buffer, len);
290 while (xp.
parse(outNodeName, outPath, outText, outAttributes)) {
292 const char* argName = outNodeName.
c_str();
304 if (outNodeName.
equals(
"s:Envelope") || outNodeName.
equals(
"Body")) {
307 if (outNodeName.
equals(
"s:Body")) {
325 Print& out,
const char* replyName,
const char* serviceId,
326 std::function<
size_t(Print&,
void*)> valuesWriter =
nullptr,
327 void* ref =
nullptr) {
333 "xmlns:s=\"http://schemas.xmlsoap.org/soap/envelope/\"\n");
336 xp.
printf(
"<u:%s xmlns:u=\"urn:schemas-upnp-org:service:%s:1\">\n",
337 replyName, serviceId);
342 result += valuesWriter(out, ref);
345 result += xp.
printf(
"</u:%s>\n", replyName);
355 const char* sink =
"") {
357 return printReplyXML(out,
"GetProtocolInfoResponse",
"ConnectionManager",
358 [source, sink](Print& o,
void* ref) ->
size_t {
361 written += o.print(
"<Source>");
362 written += o.print(
StrView(source).c_str());
363 written += o.print(
"</Source>");
364 written += o.print(
"<Sink>");
365 written += o.print(
StrView(sink).c_str());
366 written += o.print(
"</Sink>");
375 [ids](Print& o,
void* ref) ->
size_t {
380 written += o.print(
"<CurrentConnectionIDs>");
381 written += o.print(
StrView(ids ? ids :
"0").c_str());
382 written += o.print(
"</CurrentConnectionIDs>");
388 const char* protocolInfo,
389 const char* connectionID,
390 const char* direction) {
393 out,
"GetCurrentConnectionInfoResponse",
"ConnectionManager",
394 [protocolInfo, connectionID, direction](Print& o,
void* ref) ->
size_t {
399 written += o.print(
"<RcsID>0</RcsID>");
400 written += o.print(
"<AVTransportID>0</AVTransportID>");
401 written += o.print(
"<ProtocolInfo>");
402 written += o.print(
StrView(protocolInfo).c_str());
403 written += o.print(
"</ProtocolInfo>");
404 written += o.print(
"<PeerConnectionManager></PeerConnectionManager>");
405 written += o.print(
"<PeerConnectionID>");
406 written += o.print(
StrView(connectionID).c_str());
407 written += o.print(
"</PeerConnectionID>");
408 written += o.print(
"<Direction>");
409 written += o.print(
StrView(direction).c_str());
410 written += o.print(
"</Direction>");
411 written += o.print(
"<Status>OK</Status>");
419 bool& is_subscribe) {
422 is_subscribe =
false;
433 server->
replyError(501,
"Unsupported Method");
445 "Subscriptions: active=%d pending=%d / Scheduler: size=%d "
452 (
unsigned)ESP.getFreeHeap(), (
unsigned)ESP.getFreePsram());
501 static uint64_t last_mem_log = 0;
502 const uint64_t MEM_LOG_INTERVAL_MS = 10000;
503 uint64_t now = millis();
504 if (runImmediately ||
505 (uint64_t)(now - last_mem_log) >= MEM_LOG_INTERVAL_MS) {
532 postAlive1->
time = millis() + 100;
541 bool has_error =
false;
573 "Service validation failed - missing URLs/callbacks");
593 if (!
StrView(device_path).isEmpty()) {
617 "Skipping service registration due to validation failure");
637 assert(device_xml !=
nullptr);
638 if (device_xml !=
nullptr) {
644 [](Print& out,
void* ref) ->
size_t {
647 200,
"SUCCESS", device_xml);
659 assert(device !=
nullptr);
661 assert(svc !=
nullptr);
664 return device->getSubscriptionMgr().processSubscribeRequest(*server, *svc);
672 assert(device !=
nullptr);
674 assert(svc !=
nullptr);
677 return device->getSubscriptionMgr().processUnsubscribeRequest(*server,
Represents a request to invoke a remote DLNA service action.
Definition: Action.h:104
void addArgument(Argument arg)
Definition: Action.h:113
void setAction(const char *act)
Definition: Action.h:155
const char * getAction()
Definition: Action.h:157
Device Attributes and generation of XML using urn:schemas-upnp-org:device-1-0. We could just return a...
Definition: DLNADeviceInfo.h:25
Vector< DLNAServiceInfo > & getServices()
Provides all service definitions.
Definition: DLNADeviceInfo.h:203
Icon getIcon(int idx=0)
Provides the item at indix.
Definition: DLNADeviceInfo.h:228
DLNAServiceInfo & getService(const char *id)
Finds a service definition by name.
Definition: DLNADeviceInfo.h:181
void setSubscriptionActive(bool flag)
Definition: DLNADeviceInfo.h:248
const char * getUDN()
Provide the udn uuid.
Definition: DLNADeviceInfo.h:88
const char * getBaseURL()
Provides the base url.
Definition: DLNADeviceInfo.h:118
Url & getDeviceURL()
This method returns base url/device.xml.
Definition: DLNADeviceInfo.h:129
virtual void setupServices(IHttpServer &server, IUDPService &udp)
to be implemented by subclasses
Definition: DLNADeviceInfo.h:278
DLNAServiceInfo & getServiceByAbbrev(const char *abbrev)
Finds a service definition by name.
Definition: DLNADeviceInfo.h:191
const char * getDeviceType()
Definition: DLNADeviceInfo.h:82
Translates DLNA UDP Requests to Schedule so that we can schedule a reply.
Definition: DLNADeviceRequestParser.h:15
Schedule * parse(DLNADeviceInfo &device, RequestData &req)
Definition: DLNADeviceRequestParser.h:20
void addMSearchST(const char *accept)
Definition: DLNADeviceRequestParser.h:18
Setup of a Basic DLNA Device service. The device registers itself to the network and answers to the D...
Definition: DLNADevice.h:39
void * reference
Definition: DLNADevice.h:471
void setSchedulerIntervalMs(uint32_t ms)
Set scheduler interval in milliseconds.
Definition: DLNADevice.h:480
uint64_t next_subscriptions_timeout_ms
Definition: DLNADevice.h:470
int loopUDPMessages()
Process incoming UDP and execute scheduled replies.
Definition: DLNADevice.h:184
void * getReference() override
Gets the reference pointer.
Definition: DLNADevice.h:441
bool is_active
Definition: DLNADevice.h:457
void setDeviceInfo(DLNADeviceInfo &device)
Definition: DLNADevice.h:473
static bool handleUnsubscribe(IHttpServer *server, const char *requestPath, HttpRequestHandlerLine *hl)
Handle UNSUBSCRIBE requests.
Definition: DLNADevice.h:668
uint64_t subscriptions_interval_ms
Definition: DLNADevice.h:467
void end() override
Stops the processing and releases the resources.
Definition: DLNADevice.h:111
void setReference(void *ref) override
Sets a reference pointer that can be used to associate application.
Definition: DLNADevice.h:438
void setSubscriptionsIntervalMs(uint32_t ms)
Set subscription publish interval in milliseconds.
Definition: DLNADevice.h:488
DLNAServiceInfo * getServiceByEventPath(const char *requestPath) override
Definition: DLNADevice.h:212
static size_t printReplyXML(Print &out, const char *replyName, const char *serviceId, std::function< size_t(Print &, void *)> valuesWriter=nullptr, void *ref=nullptr)
Builds a standard SOAP reply envelope.
Definition: DLNADevice.h:324
uint32_t getSchedulerIntervalMs() const
Get scheduler interval in milliseconds.
Definition: DLNADevice.h:483
DLNAServiceInfo & getServiceByAbbrev(const char *abbrev) override
Get Service by subscription ns abbrev.
Definition: DLNADevice.h:205
bool setupScheduler()
Schedule PostAlive messages.
Definition: DLNADevice.h:524
uint64_t next_scheduler_timeout_ms
Definition: DLNADevice.h:469
IUDPService * p_udp
Definition: DLNADevice.h:464
void logScheduledStatus(bool runImmediately=false)
Definition: DLNADevice.h:500
bool begin(DLNADeviceInfo &device, IUDPService &udp, IHttpServer &server) override
start the
Definition: DLNADevice.h:46
int loopPublishSubscriptions()
Publish pending subscription notifications.
Definition: DLNADevice.h:172
static size_t replyGetCurrentConnectionInfo(Print &out, const char *protocolInfo, const char *connectionID, const char *direction)
Definition: DLNADevice.h:387
IHttpServer * p_server
Definition: DLNADevice.h:465
Scheduler scheduler
Definition: DLNADevice.h:460
uint32_t post_alive_repeat_ms
Definition: DLNADevice.h:459
static bool handleSubscription(IHttpServer *server, const char *requestPath, HttpRequestHandlerLine *hl, bool &is_subscribe)
Static handler for SUBSCRIBE/UNSUBSCRIBE requests on service event URLs.
Definition: DLNADevice.h:417
static bool handleSubscribe(IHttpServer *server, const char *requestPath, HttpRequestHandlerLine *hl)
Handle SUBSCRIBE requests.
Definition: DLNADevice.h:655
DLNAServiceInfo & getService(const char *id) override
Provide addess to the service information.
Definition: DLNADevice.h:200
bool loop(int loopAction=RUN_ALL) override
Definition: DLNADevice.h:134
DLNADeviceInfo * p_device_info
Definition: DLNADevice.h:463
bool validateServiceInfo(const DLNAServiceInfo &service)
Validates that all required service URLs and callbacks are defined.
Definition: DLNADevice.h:539
void setPostAliveRepeatMs(uint32_t ms) override
Definition: DLNADevice.h:250
DLNADeviceRequestParser parser
Definition: DLNADevice.h:462
static size_t replyGetProtocolInfo(Print &out, const char *source="", const char *sink="")
Definition: DLNADevice.h:354
SubscriptionMgrDevice< ClientType > subscription_mgr
Definition: DLNADevice.h:461
void logStatus()
Definition: DLNADevice.h:443
virtual bool setupDLNAServer(IHttpServer &srv)
set up Web Server to handle Service Addresses
Definition: DLNADevice.h:580
static void parseActionRequest(IHttpServer *server, const char *requestPath, HttpRequestHandlerLine *hl, ActionRequest &action)
Parses the SOAP content of a DLNA action request.
Definition: DLNADevice.h:267
uint32_t getSubscriptionsIntervalMs() const
Get subscription publish interval in milliseconds.
Definition: DLNADevice.h:493
bool setupParser()
Definition: DLNADevice.h:514
DLNADeviceInfo & getDeviceInfo() override
Provides the device.
Definition: DLNADevice.h:240
bool isSchedulerActive() override
Checks if the scheduler is active.
Definition: DLNADevice.h:246
bool loopServer() override
Process http server loop.
Definition: DLNADevice.h:175
ISubscriptionMgrDevice & getSubscriptionMgr() override
Get subscription manager for event handling.
Definition: DLNADevice.h:106
static size_t replyGetCurrentConnectionIDs(Print &out, const char *ids)
Definition: DLNADevice.h:371
void setSubscriptionsActive(bool flag) override
Enable or disable subscription notifications: call before begin.
Definition: DLNADevice.h:253
uint64_t scheduler_interval_ms
Definition: DLNADevice.h:466
static void deviceXMLCallback(IHttpServer *server, const char *requestPath, HttpRequestHandlerLine *hl)
callback to provide device XML
Definition: DLNADevice.h:632
bool isSubscriptionsActive() const override
Check if subscription notifications are active.
Definition: DLNADevice.h:262
void setSchedulerActive(bool flag) override
We can activate/deactivate the scheduler.
Definition: DLNADevice.h:243
DLNADevice()
Definition: DLNADevice.h:41
bool is_subscriptions_active
Definition: DLNADevice.h:458
void addChange(const char *serviceAbbrev, std::function< size_t(Print &, void *)> changeWriter, void *ref) override
Record a state variable change for subscription notifications.
Definition: DLNADevice.h:224
Attributes needed for the DLNA Service Definition.
Definition: DLNAServiceInfo.h:18
http_callback control_cb
Definition: DLNAServiceInfo.h:44
http_callback scp_cb
Definition: DLNAServiceInfo.h:43
Str event_sub_url
Definition: DLNAServiceInfo.h:41
Str scpd_url
Definition: DLNAServiceInfo.h:39
http_callback event_sub_cb
Definition: DLNAServiceInfo.h:45
Str control_url
Definition: DLNAServiceInfo.h:40
Used to register and process callbacks.
Definition: HttpRequestHandlerLine.h:15
void ** context
Definition: HttpRequestHandlerLine.h:40
Abstract interface for DLNA device functionality.
Definition: IDevice.h:30
Abstract interface for HTTP server functionality.
Definition: IHttpServer.h:30
virtual bool begin()=0
Start the HTTP server.
virtual void * getReference()=0
Get user reference pointer.
virtual bool doLoop()=0
Process server loop.
virtual void replyNotFound()=0
Send 404 Not Found response.
virtual void end()=0
Stop the HTTP server.
virtual void on(const char *url, TinyMethodID method, web_callback_fn fn, void *ctx[]=nullptr, int ctxCount=0)=0
Register callback for HTTP request with context.
virtual void setReference(void *reference)=0
Set user reference pointer.
virtual Client & client()=0
Get reference to current client.
virtual void replyError(int err, const char *msg="Internal Server Error")=0
Send error response with status code.
virtual HttpRequestHeader & requestHeader()=0
Get reference to request header.
virtual void reply(const char *contentType, Stream &inputStream, int size, int status=200, const char *msg=SUCCESS)=0
Send response from input stream with known size.
virtual void rewrite(const char *from, const char *to)=0
Add URL rewrite rule.
Abstract interface for UPnP event subscription management.
Definition: ISubscriptionMgrDevice.h:27
virtual void setSubscriptionsActive(bool flag)=0
Enable/disable subscription processing.
virtual void end()=0
Cleanup and stop subscription manager.
Abstract Interface for UDP API.
Definition: IUDPService.h:33
virtual RequestData receive()=0
Receive incoming UDP data and peer information.
virtual bool begin(int port)=0
Initialize UDP service on specified port.
Information about the icon.
Definition: Icon.h:10
const char * icon_url
Definition: Icon.h:16
int icon_size
Definition: Icon.h:18
const char * mime
Definition: Icon.h:12
uint8_t * icon_data
Definition: Icon.h:17
Send out PostAlive messages: Repeated every 5 seconds.
Definition: Schedule.h:218
Send out ByeBye message.
Definition: Schedule.h:290
Scheduler which processes all due Schedules (to send out UDP replies)
Definition: Scheduler.h:15
int execute(IUDPService &udp)
Execute all due schedules.
Definition: Scheduler.h:30
void setActive(bool flag)
Definition: Scheduler.h:81
bool isActive()
Definition: Scheduler.h:83
int size()
Number of queued schedules.
Definition: Scheduler.h:79
void add(Schedule *schedule)
Add a schedule to the scheduler.
Definition: Scheduler.h:18
A simple wrapper to provide string functions on char*. If the underlying char* is a const we do not a...
Definition: StrView.h:18
virtual bool equals(const char *str)
checks if the string equals indicated parameter string
Definition: StrView.h:177
virtual bool contains(const char *str)
checks if the string contains a substring
Definition: StrView.h:284
Heap-backed string utility used throughout tiny_dlna.
Definition: Str.h:27
bool equals(const char *other) const
Exact string equality with C-string.
Definition: Str.h:167
const char * c_str() const
C-string pointer to internal buffer.
Definition: Str.h:88
Manages UPnP event subscriptions and queued notifications for a DLNA device.
Definition: SubscriptionMgrDevice.h:90
URL parser which breaks a full url string up into its individual parts.
Definition: Url.h:18
const char * path()
Definition: Url.h:40
Lightweight wrapper around std::vector with Arduino-friendly helpers and a pluggable allocator.
Definition: Vector.h:39
Helper class that implements a Print interface to accumulate XML data and then parse it using XMLPars...
Definition: XMLParserPrint.h:16
size_t write(uint8_t ch) override
Writes a single byte to the buffer (Print interface)
Definition: XMLParserPrint.h:32
void setExpandEncoded(bool flag)
Forwards expand-entities setting to the underlying XMLParser.
Definition: XMLParserPrint.h:48
void end()
Resets the internal buffer.
Definition: XMLParserPrint.h:87
bool parse(Str &outNodeName, Vector< Str > &outPath, Str &outText, Str &outAttributes)
Parses the accumulated XML data and returns results via output parameters.
Definition: XMLParserPrint.h:59
#define DLNA_MAX_URL_LEN
app-wide max URL length
Definition: dlna_config.h:45
#define XML_PARSER_BUFFER_SIZE
Define XML parse buffer size.
Definition: dlna_config.h:25
#define DLNA_LOOP_DELAY_MS
Define delay in ms for main DLNA loop.
Definition: dlna_config.h:5
#define DLNA_RUN_SCHEDULER_EVERY_MS
Define scheduler run interval in ms.
Definition: dlna_config.h:10
#define DLNA_RUN_SUBSCRIPTIONS_EVERY_MS
Define subscription publish interval in ms.
Definition: dlna_config.h:15
Definition: Allocator.h:13
@ RUN_UDP
Definition: IDevice.h:18
@ RUN_SERVER
Definition: IDevice.h:18
@ RUN_ALL
Definition: IDevice.h:18
@ RUN_SUBSCRIPTIONS
Definition: IDevice.h:18
@ T_UNSUBSCRIBE
Definition: HttpHeader.h:46
@ T_SUBSCRIBE
Definition: HttpHeader.h:47
@ T_GET
Definition: HttpHeader.h:37
@ T_POST
Definition: HttpHeader.h:39
Provides information of the received UDP which consists of the (xml) data and the peer address and po...
Definition: IUDPService.h:22
An individual Schedule (to send out UDP messages)
Definition: Schedule.h:18
uint64_t time
Definition: Schedule.h:21
uint32_t repeat_ms
Definition: Schedule.h:23
Functions to efficiently output XML. XML data contains a lot of redundancy so it is more memory effic...
Definition: XMLPrinter.h:56
size_t printNodeBegin(const char *node, const char *attributes=nullptr, const char *ns=nullptr)
Prints the beginning of an XML node.
Definition: XMLPrinter.h:314
size_t printf(const char *fmt,...)
printf-style helper that formats into an internal buffer and writes to the configured Print output.
Definition: XMLPrinter.h:248
size_t printNodeEnd(const char *node, const char *ns=nullptr)
Prints the end of an XML node.
Definition: XMLPrinter.h:352