Create a list of mDNS services, dynamically

This commit is contained in:
Santiago Lo Coco 2024-11-03 14:28:53 +01:00
parent 5c4ada28f9
commit e2b00eaf28
4 changed files with 91 additions and 30 deletions

View File

@ -11,6 +11,7 @@ public class EndpointLoader : MonoBehaviour
public WebView webView1; public WebView webView1;
public WebView webView2; public WebView webView2;
public ServiceDiscovery serviceDiscovery; public ServiceDiscovery serviceDiscovery;
public ServicesListPopulator servicesListPopulator;
private bool triedMulticast = false; private bool triedMulticast = false;
private string apiUrl = "http://windows.loca:5000/api/endpoints"; // Typo on purpose private string apiUrl = "http://windows.loca:5000/api/endpoints"; // Typo on purpose
@ -18,6 +19,7 @@ public class EndpointLoader : MonoBehaviour
private const string defaultEndpoint2 = "http://windows.local:8200/mystream/"; private const string defaultEndpoint2 = "http://windows.local:8200/mystream/";
private bool defaultEndpoint1Loaded = false; private bool defaultEndpoint1Loaded = false;
private bool defaultEndpoint2Loaded = false; private bool defaultEndpoint2Loaded = false;
private List<MdnsService> availableServices = new List<MdnsService>();
private void Start() private void Start()
{ {
@ -108,14 +110,29 @@ public class EndpointLoader : MonoBehaviour
Debug.Log("Starting multicast discovery for endpoints"); Debug.Log("Starting multicast discovery for endpoints");
triedMulticast = true; triedMulticast = true;
serviceDiscovery.StartListening((ipAddress, port) => serviceDiscovery.StartListening((service) =>
{ {
Debug.Log($"Received multicast message: {ipAddress}:{port}"); Debug.Log($"Received multicast message: {service.Host}");
apiUrl = $"http://{ipAddress}:{port}/api/endpoints"; availableServices.Add(service);
StartCoroutine(LoadEndpoints()); AddServiceToTable(service);
}); });
} }
private void AddServiceToTable(MdnsService service)
{
Debug.Log($"Adding service to table: {service.Host}");
servicesListPopulator.AddItemFromService(service, () =>
{
ChangeApiUrl($"http://{service.Host}:{service.Port}{service.Path}");
});
}
public void ChangeApiUrl(string newUrl)
{
apiUrl = newUrl;
StartCoroutine(LoadEndpoints());
}
public void ReloadEndpoints() public void ReloadEndpoints()
{ {
triedMulticast = false; triedMulticast = false;

View File

@ -10,20 +10,19 @@ using UnityEngine;
public class ServiceDiscovery : MonoBehaviour public class ServiceDiscovery : MonoBehaviour
{ {
private UdpClient udpClient; private UdpClient udpClient;
private Action<string, string> action; private Action<MdnsService> action;
private string receivedIp; private string receivedIp;
private string receivedPort; private string receivedPort;
private string receivedPath; private string receivedPath;
private string receivedHost; private string receivedHost;
private bool messageReceived = false;
private const string multicastAddress = "224.0.0.251"; private const string multicastAddress = "224.0.0.251";
private const int multicastPort = 5353; private const int multicastPort = 5353;
public List<MdnsService> discoveredServices = new List<MdnsService>(); public List<MdnsService> discoveredServices = new List<MdnsService>();
public void StartListening(Action<string, string> action) public void StartListening(Action<MdnsService> action)
{ {
try try
{ {
@ -107,21 +106,22 @@ public class ServiceDiscovery : MonoBehaviour
byte[] receivedBytes = udpClient.EndReceive(result, ref remoteEndPoint); byte[] receivedBytes = udpClient.EndReceive(result, ref remoteEndPoint);
ushort flags = BitConverter.ToUInt16(new byte[] { receivedBytes[3], receivedBytes[2] }, 0); ushort flags = BitConverter.ToUInt16(new byte[] { receivedBytes[3], receivedBytes[2] }, 0);
Debug.Log($"Flags: {flags:X2}"); // Debug.Log($"Flags: {flags:X2}");
if (flags == 0x0100) // Standard query if (flags == 0x0100) // Standard query
{ {
Debug.Log("Ignoring non-response packet"); // Debug.Log("Ignoring non-response packet");
udpClient?.BeginReceive(OnReceive, null); udpClient?.BeginReceive(OnReceive, null);
return; return;
} }
Debug.Log($"Received message: {receivedBytes} from {remoteEndPoint}"); // Debug.Log($"Received message: {receivedBytes} from {remoteEndPoint}");
ParseMdnsResponse(receivedBytes); ParseMdnsResponse(receivedBytes);
if (receivedIp != null && receivedPort != null) if (receivedIp != null && receivedPort != null)
{ {
// messageReceived = true;
// StopListening(); // StopListening();
MdnsService currentService = new MdnsService(receivedIp, int.Parse(receivedPort), receivedPath, receivedHost);
/*
MdnsService currentService = new MdnsService MdnsService currentService = new MdnsService
{ {
IpAddress = receivedIp, IpAddress = receivedIp,
@ -129,21 +129,17 @@ public class ServiceDiscovery : MonoBehaviour
Path = receivedPath, Path = receivedPath,
Host = receivedHost Host = receivedHost
}; };
*/
discoveredServices.Add(currentService); discoveredServices.Add(currentService);
Debug.Log($"Added service: {currentService.IpAddress}:{currentService.Port}, " + Debug.Log($"Added service: http://{currentService.Host}:{currentService.Port}{currentService.Path}");
$"{currentService.Path}, {currentService.Host}");
receivedIp = null; receivedIp = null;
receivedPort = null; receivedPort = null;
receivedPath = null; receivedPath = null;
receivedHost = null; receivedHost = null;
// receivedIp = receivedHost = receivedPort = receivedPath = null;
// action?.Invoke(currentService);
} }
/*
else
{
udpClient?.BeginReceive(OnReceive, null);
}
*/
udpClient?.BeginReceive(OnReceive, null); udpClient?.BeginReceive(OnReceive, null);
} }
catch (Exception ex) catch (Exception ex)
@ -159,7 +155,7 @@ public class ServiceDiscovery : MonoBehaviour
ushort answerRRs = (ushort)IPAddress.NetworkToHostOrder(BitConverter.ToInt16(data, 6)); ushort answerRRs = (ushort)IPAddress.NetworkToHostOrder(BitConverter.ToInt16(data, 6));
ushort additionalRRs = (ushort)IPAddress.NetworkToHostOrder(BitConverter.ToInt16(data, 10)); ushort additionalRRs = (ushort)IPAddress.NetworkToHostOrder(BitConverter.ToInt16(data, 10));
Debug.Log($"Questions: {questions}, Answer RRs: {answerRRs}, Additional RRs: {additionalRRs}"); // Debug.Log($"Questions: {questions}, Answer RRs: {answerRRs}, Additional RRs: {additionalRRs}");
for (int i = 0; i < questions; i++) for (int i = 0; i < questions; i++)
{ {
@ -219,7 +215,7 @@ public class ServiceDiscovery : MonoBehaviour
if (txtData.Contains("path")) if (txtData.Contains("path"))
{ {
receivedPath = txtData.Split('=')[1]; receivedPath = txtData.Split('=')[1];
Debug.Log($"Received path: {receivedPath}"); // Debug.Log($"Received path: {receivedPath}");
} }
} }
else if (recordType == 47) // NSEC Record else if (recordType == 47) // NSEC Record
@ -277,13 +273,14 @@ public class ServiceDiscovery : MonoBehaviour
private void Update() private void Update()
{ {
if (messageReceived) if (discoveredServices.Count > 0)
{ {
Debug.Log($"Invoking action with: {receivedIp}:{receivedPort}"); foreach (MdnsService service in discoveredServices)
action?.Invoke(receivedIp, receivedPort); {
messageReceived = false; Debug.Log($"Invoking action with: {service}");
receivedIp = null; action?.Invoke(service);
receivedPort = null; }
discoveredServices.Clear();
} }
} }
@ -299,12 +296,12 @@ public class ServiceDiscovery : MonoBehaviour
udpClient = null; udpClient = null;
} }
public class MdnsService /* public class MdnsService
{ {
public string IpAddress { get; set; } public string IpAddress { get; set; }
public int Port { get; set; } public int Port { get; set; }
public string Path { get; set; } public string Path { get; set; }
public string Host { get; set; } public string Host { get; set; }
} }
} */}

View File

@ -0,0 +1,36 @@
using TMPro;
using System;
using System.Collections;
using UnityEngine;
using Microsoft.MixedReality.Toolkit.UI;
using Microsoft.MixedReality.Toolkit.Utilities;
public class ServicesListPopulator : MonoBehaviour
{
[SerializeField]
private ScrollingObjectCollection scrollView;
[SerializeField]
private GameObject dynamicItem;
[SerializeField]
private GridObjectCollection gridObjectCollection;
public void AddItemFromService(MdnsService service, Action action)
{
GameObject itemInstance = Instantiate(dynamicItem, gridObjectCollection.transform);
itemInstance.SetActive(true);
Debug.Log($"Adding service to table: {service}");
TextMeshPro[] textMeshes = itemInstance.GetComponentsInChildren<TextMeshPro>();
textMeshes[0].text = service.Host + ":" + service.Port + service.Path;
textMeshes[1].text = service.IpAddress;
itemInstance.GetComponentInChildren<Interactable>().OnClick.AddListener(() =>
{
Debug.Log($"Clicked on service: {service.Host}");
action.Invoke();
});
gridObjectCollection.UpdateCollection();
}
}

View File

@ -0,0 +1,11 @@
fileFormatVersion: 2
guid: be30c5115f8bf5c4bbb25e5fbb5e9c8a
MonoImporter:
externalObjects: {}
serializedVersion: 2
defaultReferences: []
executionOrder: 0
icon: {instanceID: 0}
userData:
assetBundleName:
assetBundleVariant: