213 lines
6.4 KiB
C++
213 lines
6.4 KiB
C++
|
/*
|
||
|
==============================================================================
|
||
|
|
||
|
This file is part of the JUCE library.
|
||
|
Copyright (c) 2020 - Raw Material Software Limited
|
||
|
|
||
|
JUCE is an open source library subject to commercial or open-source
|
||
|
licensing.
|
||
|
|
||
|
The code included in this file is provided under the terms of the ISC license
|
||
|
http://www.isc.org/downloads/software-support-policy/isc-license. Permission
|
||
|
To use, copy, modify, and/or distribute this software for any purpose with or
|
||
|
without fee is hereby granted provided that the above copyright notice and
|
||
|
this permission notice appear in all copies.
|
||
|
|
||
|
JUCE IS PROVIDED "AS IS" WITHOUT ANY WARRANTY, AND ALL WARRANTIES, WHETHER
|
||
|
EXPRESSED OR IMPLIED, INCLUDING MERCHANTABILITY AND FITNESS FOR PURPOSE, ARE
|
||
|
DISCLAIMED.
|
||
|
|
||
|
==============================================================================
|
||
|
*/
|
||
|
|
||
|
namespace juce
|
||
|
{
|
||
|
|
||
|
#if JUCE_ANDROID
|
||
|
extern void acquireMulticastLock();
|
||
|
extern void releaseMulticastLock();
|
||
|
#endif
|
||
|
|
||
|
NetworkServiceDiscovery::Advertiser::Advertiser (const String& serviceTypeUID,
|
||
|
const String& serviceDescription,
|
||
|
int broadcastPortToUse, int connectionPort,
|
||
|
RelativeTime minTimeBetweenBroadcasts)
|
||
|
: Thread ("Discovery_broadcast"),
|
||
|
message (serviceTypeUID), broadcastPort (broadcastPortToUse),
|
||
|
minInterval (minTimeBetweenBroadcasts)
|
||
|
{
|
||
|
message.setAttribute ("id", Uuid().toString());
|
||
|
message.setAttribute ("name", serviceDescription);
|
||
|
message.setAttribute ("address", String());
|
||
|
message.setAttribute ("port", connectionPort);
|
||
|
|
||
|
startThread (2);
|
||
|
}
|
||
|
|
||
|
NetworkServiceDiscovery::Advertiser::~Advertiser()
|
||
|
{
|
||
|
stopThread (2000);
|
||
|
socket.shutdown();
|
||
|
}
|
||
|
|
||
|
void NetworkServiceDiscovery::Advertiser::run()
|
||
|
{
|
||
|
if (! socket.bindToPort (0))
|
||
|
{
|
||
|
jassertfalse;
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
while (! threadShouldExit())
|
||
|
{
|
||
|
sendBroadcast();
|
||
|
wait ((int) minInterval.inMilliseconds());
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void NetworkServiceDiscovery::Advertiser::sendBroadcast()
|
||
|
{
|
||
|
static IPAddress local = IPAddress::local();
|
||
|
|
||
|
for (auto& address : IPAddress::getAllAddresses())
|
||
|
{
|
||
|
if (address == local)
|
||
|
continue;
|
||
|
|
||
|
message.setAttribute ("address", address.toString());
|
||
|
|
||
|
auto broadcastAddress = IPAddress::getInterfaceBroadcastAddress (address);
|
||
|
auto data = message.toString (XmlElement::TextFormat().singleLine().withoutHeader());
|
||
|
|
||
|
socket.write (broadcastAddress.toString(), broadcastPort, data.toRawUTF8(), (int) data.getNumBytesAsUTF8());
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//==============================================================================
|
||
|
NetworkServiceDiscovery::AvailableServiceList::AvailableServiceList (const String& serviceType, int broadcastPort)
|
||
|
: Thread ("Discovery_listen"), serviceTypeUID (serviceType)
|
||
|
{
|
||
|
#if JUCE_ANDROID
|
||
|
acquireMulticastLock();
|
||
|
#endif
|
||
|
|
||
|
socket.bindToPort (broadcastPort);
|
||
|
startThread (2);
|
||
|
}
|
||
|
|
||
|
NetworkServiceDiscovery::AvailableServiceList::~AvailableServiceList()
|
||
|
{
|
||
|
socket.shutdown();
|
||
|
stopThread (2000);
|
||
|
|
||
|
#if JUCE_ANDROID
|
||
|
releaseMulticastLock();
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
void NetworkServiceDiscovery::AvailableServiceList::run()
|
||
|
{
|
||
|
while (! threadShouldExit())
|
||
|
{
|
||
|
if (socket.waitUntilReady (true, 200) == 1)
|
||
|
{
|
||
|
char buffer[1024];
|
||
|
auto bytesRead = socket.read (buffer, sizeof (buffer) - 1, false);
|
||
|
|
||
|
if (bytesRead > 10)
|
||
|
if (auto xml = parseXML (String (CharPointer_UTF8 (buffer),
|
||
|
CharPointer_UTF8 (buffer + bytesRead))))
|
||
|
if (xml->hasTagName (serviceTypeUID))
|
||
|
handleMessage (*xml);
|
||
|
}
|
||
|
|
||
|
removeTimedOutServices();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
std::vector<NetworkServiceDiscovery::Service> NetworkServiceDiscovery::AvailableServiceList::getServices() const
|
||
|
{
|
||
|
const ScopedLock sl (listLock);
|
||
|
auto listCopy = services;
|
||
|
return listCopy;
|
||
|
}
|
||
|
|
||
|
void NetworkServiceDiscovery::AvailableServiceList::handleAsyncUpdate()
|
||
|
{
|
||
|
if (onChange != nullptr)
|
||
|
onChange();
|
||
|
}
|
||
|
|
||
|
void NetworkServiceDiscovery::AvailableServiceList::handleMessage (const XmlElement& xml)
|
||
|
{
|
||
|
Service service;
|
||
|
service.instanceID = xml.getStringAttribute ("id");
|
||
|
|
||
|
if (service.instanceID.trim().isNotEmpty())
|
||
|
{
|
||
|
service.description = xml.getStringAttribute ("name");
|
||
|
service.address = IPAddress (xml.getStringAttribute ("address"));
|
||
|
service.port = xml.getIntAttribute ("port");
|
||
|
service.lastSeen = Time::getCurrentTime();
|
||
|
|
||
|
handleMessage (service);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void sortServiceList (std::vector<NetworkServiceDiscovery::Service>& services)
|
||
|
{
|
||
|
auto compareServices = [] (const NetworkServiceDiscovery::Service& s1,
|
||
|
const NetworkServiceDiscovery::Service& s2)
|
||
|
{
|
||
|
return s1.instanceID < s2.instanceID;
|
||
|
};
|
||
|
|
||
|
std::sort (services.begin(), services.end(), compareServices);
|
||
|
}
|
||
|
|
||
|
void NetworkServiceDiscovery::AvailableServiceList::handleMessage (const Service& service)
|
||
|
{
|
||
|
const ScopedLock sl (listLock);
|
||
|
|
||
|
for (auto& s : services)
|
||
|
{
|
||
|
if (s.instanceID == service.instanceID)
|
||
|
{
|
||
|
if (s.description != service.description
|
||
|
|| s.address != service.address
|
||
|
|| s.port != service.port)
|
||
|
{
|
||
|
s = service;
|
||
|
triggerAsyncUpdate();
|
||
|
}
|
||
|
|
||
|
s.lastSeen = service.lastSeen;
|
||
|
return;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
services.push_back (service);
|
||
|
sortServiceList (services);
|
||
|
triggerAsyncUpdate();
|
||
|
}
|
||
|
|
||
|
void NetworkServiceDiscovery::AvailableServiceList::removeTimedOutServices()
|
||
|
{
|
||
|
const double timeoutSeconds = 5.0;
|
||
|
auto oldestAllowedTime = Time::getCurrentTime() - RelativeTime::seconds (timeoutSeconds);
|
||
|
|
||
|
const ScopedLock sl (listLock);
|
||
|
|
||
|
auto oldEnd = std::end (services);
|
||
|
auto newEnd = std::remove_if (std::begin (services), oldEnd,
|
||
|
[=] (const Service& s) { return s.lastSeen < oldestAllowedTime; });
|
||
|
|
||
|
if (newEnd != oldEnd)
|
||
|
{
|
||
|
services.erase (newEnd, oldEnd);
|
||
|
triggerAsyncUpdate();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
} // namespace juce
|