do not fully trust viewers about parcels, plus cosmetics

This commit is contained in:
UbitUmarov
2026-02-23 20:52:34 +00:00
parent b42e8dc0ad
commit b41033a8f1
4 changed files with 105 additions and 199 deletions

View File

@@ -95,7 +95,7 @@ namespace osWebRtcVoice
} }
// Note that the session_id is a long number in the JSON so we convert the string. // Note that the session_id is a long number in the JSON so we convert the string.
public string sessionId { public string sessionId {
get { return m_message.ContainsKey("session_id") ? OSDToLong(m_message["session_id"]).ToString() : String.Empty; } get { return m_message.TryGetValue("session_id", out OSD tmposd) ? OSDToLong(tmposd).ToString() : string.Empty; }
set { m_message["session_id"] = long.Parse(value); } set { m_message["session_id"] = long.Parse(value); }
} }
public bool hasSessionId { get { return m_message.ContainsKey("session_id"); } } public bool hasSessionId { get { return m_message.ContainsKey("session_id"); } }
@@ -134,7 +134,7 @@ namespace osWebRtcVoice
// and one fetches it with .AsInteger(), it will return the first 4 bytes as an integer // and one fetches it with .AsInteger(), it will return the first 4 bytes as an integer
// and not the long value. So this function looks at the type of the OSD object and // and not the long value. So this function looks at the type of the OSD object and
// extracts the number appropriately. // extracts the number appropriately.
public long OSDToLong(OSD pIn) public static long OSDToLong(OSD pIn)
{ {
long ret = 0; long ret = 0;
switch (pIn.Type) switch (pIn.Type)
@@ -210,7 +210,7 @@ namespace osWebRtcVoice
} }
// Return the "data" portion of the response as an OSDMap or null if there is none // Return the "data" portion of the response as an OSDMap or null if there is none
public OSDMap dataSection { get { return m_message.ContainsKey("data") ? (m_message["data"] as OSDMap) : null; } } public OSDMap dataSection { get { return m_message.TryGetOSDMap("data", out OSDMap osdm) ? osdm : null; } }
// Check if a successful response code is in the response // Check if a successful response code is in the response
public virtual bool isSuccess { get { return CheckReturnCode("success"); } } public virtual bool isSuccess { get { return CheckReturnCode("success"); } }

View File

@@ -76,7 +76,6 @@ namespace osWebRtcVoice
m_MessageDetails = moduleConfig.GetBoolean("MessageDetails", false); m_MessageDetails = moduleConfig.GetBoolean("MessageDetails", false);
} }
} }
} }
// Create a local viewer session. This gets a local viewer session ID that is // Create a local viewer session. This gets a local viewer session ID that is

View File

@@ -56,7 +56,7 @@ namespace osWebRtcVoice
/// This module provides the WebRTC voice interface for viewer clients.. /// This module provides the WebRTC voice interface for viewer clients..
/// ///
/// In particular, it provides the following capabilities: /// In particular, it provides the following capabilities:
/// ProvisionVoiceAccountRequest, VoiceSignalingRequest, and ParcelVoiceInfoRequest. /// ProvisionVoiceAccountRequest, VoiceSignalingRequest and limited ChatSessionRequest
/// which are the user interface to the voice service. /// which are the user interface to the voice service.
/// ///
/// Initially, when the user connects to the region, the region feature "VoiceServiceType" is /// Initially, when the user connects to the region, the region feature "VoiceServiceType" is
@@ -70,14 +70,12 @@ namespace osWebRtcVoice
private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType); private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType);
private static readonly string logHeader = "[REGION WEBRTC VOICE]"; private static readonly string logHeader = "[REGION WEBRTC VOICE]";
private static byte[] llsdUndefAnswerBytes = Util.UTF8.GetBytes("<llsd><undef /></llsd>");
private bool _MessageDetails = false; private bool _MessageDetails = false;
// Control info // Control info
private static bool m_Enabled = false; private static bool m_Enabled = false;
private readonly Dictionary<string, string> m_UUIDName = new();
private Dictionary<string, string> m_ParcelAddress = new();
private IConfig m_Config; private IConfig m_Config;
// ISharedRegionModule.Initialize // ISharedRegionModule.Initialize
@@ -149,7 +147,7 @@ namespace osWebRtcVoice
// everytime OpenSim hands out capabilities to a client // everytime OpenSim hands out capabilities to a client
// (login, region crossing). We contribute three capabilities to // (login, region crossing). We contribute three capabilities to
// the set of capabilities handed back to the client: // the set of capabilities handed back to the client:
// ProvisionVoiceAccountRequest, VoiceSignalingRequest, and ParcelVoiceInfoRequest. // ProvisionVoiceAccountRequest, VoiceSignalingRequest and limited ChatSessionRequest
// //
// ProvisionVoiceAccountRequest allows the client to obtain // ProvisionVoiceAccountRequest allows the client to obtain
// voice communication information the the avater. // voice communication information the the avater.
@@ -165,7 +163,7 @@ namespace osWebRtcVoice
public void OnRegisterCaps(Scene scene, UUID agentID, Caps caps) public void OnRegisterCaps(Scene scene, UUID agentID, Caps caps)
{ {
m_log.Debug( m_log.Debug(
$"{logHeader}: OnRegisterCaps() called with agentID {agentID} caps {caps} in scene {scene.RegionInfo.RegionName}"); $"{logHeader}: OnRegisterCaps called with agentID {agentID} caps {caps} in scene {scene.Name}");
caps.RegisterSimpleHandler("ProvisionVoiceAccountRequest", caps.RegisterSimpleHandler("ProvisionVoiceAccountRequest",
new SimpleStreamHandler("/" + UUID.Random(), (IOSHttpRequest httpRequest, IOSHttpResponse httpResponse) => new SimpleStreamHandler("/" + UUID.Random(), (IOSHttpRequest httpRequest, IOSHttpResponse httpResponse) =>
@@ -202,7 +200,7 @@ namespace osWebRtcVoice
IWebRtcVoiceService voiceService = scene.RequestModuleInterface<IWebRtcVoiceService>(); IWebRtcVoiceService voiceService = scene.RequestModuleInterface<IWebRtcVoiceService>();
if (voiceService is null) if (voiceService is null)
{ {
m_log.Error($"{logHeader}[ProvisionVoice]: no voice service not loaded"); m_log.Error($"{logHeader}[ProvisionVoice]: voice service not loaded");
response.StatusCode = (int)HttpStatusCode.NotFound; response.StatusCode = (int)HttpStatusCode.NotFound;
return; return;
} }
@@ -229,7 +227,7 @@ namespace osWebRtcVoice
if (vstosd is OSDString vst && !((string)vst).Equals("webrtc", StringComparison.OrdinalIgnoreCase)) if (vstosd is OSDString vst && !((string)vst).Equals("webrtc", StringComparison.OrdinalIgnoreCase))
{ {
m_log.Warn($"{logHeader}[ProvisionVoice]: voice_server_type is not 'webrtc'. Request: {map}"); m_log.Warn($"{logHeader}[ProvisionVoice]: voice_server_type is not 'webrtc'. Request: {map}");
response.RawBuffer = Util.UTF8.GetBytes("<llsd><undef /></llsd>"); response.RawBuffer = llsdUndefAnswerBytes;
response.StatusCode = (int)HttpStatusCode.OK; response.StatusCode = (int)HttpStatusCode.OK;
return; return;
} }
@@ -237,6 +235,76 @@ namespace osWebRtcVoice
if (_MessageDetails) m_log.DebugFormat($"{logHeader}[ProvisionVoice]: request: {map}"); if (_MessageDetails) m_log.DebugFormat($"{logHeader}[ProvisionVoice]: request: {map}");
if (map.TryGetString("channel_type", out string channelType))
{
//do fully not trust viewers voice parcel requests
if (channelType == "local")
{
if (!scene.RegionInfo.EstateSettings.AllowVoice)
{
m_log.Debug($"{logHeader}[ProvisionVoice]:region \"{scene.Name}\": voice not enabled in estate settings");
response.RawBuffer = llsdUndefAnswerBytes;
response.StatusCode = (int)HttpStatusCode.NotImplemented;
return;
}
if (scene.LandChannel == null)
{
m_log.Error($"{logHeader}[ProvisionVoice] region \"{scene.Name}\" land data not yet available");
response.RawBuffer = llsdUndefAnswerBytes;
response.StatusCode = (int)HttpStatusCode.NotImplemented;
return;
}
if(!scene.TryGetScenePresence(agentID, out ScenePresence sp))
{
m_log.Debug($"{logHeader}[ProvisionVoice]:avatar not found");
response.RawBuffer = llsdUndefAnswerBytes;
response.StatusCode = (int)HttpStatusCode.NotFound;
return;
}
if(map.TryGetInt("parcel_local_id", out int parcelID))
{
ILandObject parcel = scene.LandChannel.GetLandObject(parcelID);
if (parcel == null)
{
response.RawBuffer = llsdUndefAnswerBytes;
response.StatusCode = (int)HttpStatusCode.NotFound;
return;
}
LandData land = parcel.LandData;
if (land == null)
{
response.RawBuffer = llsdUndefAnswerBytes;
response.StatusCode = (int)HttpStatusCode.NotFound;
return;
}
if (!scene.RegionInfo.EstateSettings.TaxFree && (land.Flags & (uint)ParcelFlags.AllowVoiceChat) == 0)
{
m_log.Debug($"{logHeader}[ProvisionVoice]:parcel voice not allowed");
response.RawBuffer = llsdUndefAnswerBytes;
response.StatusCode = (int)HttpStatusCode.Forbidden;
return;
}
if ((land.Flags & (uint)ParcelFlags.UseEstateVoiceChan) != 0)
{
map.Remove("parcel_local_id"); // estate channel
}
else if(parcel.IsRestrictedFromLand(agentID) || parcel.IsBannedFromLand(agentID))
{
// check Z distance?
m_log.Debug($"{logHeader}[ProvisionVoice]:agent not allowed on parcel");
response.RawBuffer = llsdUndefAnswerBytes;
response.StatusCode = (int)HttpStatusCode.Forbidden;
return;
}
}
}
}
// The checks passed. Send the request to the voice service. // The checks passed. Send the request to the voice service.
OSDMap resp = voiceService.ProvisionVoiceAccountRequest(map, agentID, scene.RegionInfo.RegionID).Result; OSDMap resp = voiceService.ProvisionVoiceAccountRequest(map, agentID, scene.RegionInfo.RegionID).Result;
@@ -290,18 +358,20 @@ namespace osWebRtcVoice
{ {
if (vstosd is OSDString vst && !((string)vst).Equals("webrtc", StringComparison.OrdinalIgnoreCase)) if (vstosd is OSDString vst && !((string)vst).Equals("webrtc", StringComparison.OrdinalIgnoreCase))
{ {
response.RawBuffer = Util.UTF8.GetBytes("<llsd><undef /></llsd>"); response.RawBuffer = llsdUndefAnswerBytes;
response.StatusCode = (int)HttpStatusCode.OK;
return; return;
} }
} }
OSDMap resp = voiceService.VoiceSignalingRequest(map, agentID, scene.RegionInfo.RegionID).Result; OSDMap resp = voiceService.VoiceSignalingRequest(map, agentID, scene.RegionInfo.RegionID).Result;
if (_MessageDetails) m_log.Debug($"{logHeader}[VoiceSignalingRequest]: Response: {resp}"); if (_MessageDetails) m_log.Debug($"{logHeader}[VoiceSignalingRequest]: Response: {resp}");
// TODO: check for errors and package the response // TODO: check for errors and package the response
response.RawBuffer = llsdUndefAnswerBytes;
response.StatusCode = (int)HttpStatusCode.OK; response.StatusCode = (int)HttpStatusCode.OK;
response.RawBuffer = Util.UTF8.GetBytes("<llsd><undef /></llsd>");
return; return;
} }
@@ -401,161 +471,6 @@ namespace osWebRtcVoice
} }
} }
// NOTE NOTE!! This is code from the FreeSwitch module. It is not clear if this is correct for WebRtc.
/// <summary>
/// Callback for a client request for ParcelVoiceInfo
/// </summary>
/// <param name="scene">current scene object of the client</param>
/// <param name="request"></param>
/// <param name="path"></param>
/// <param name="param"></param>
/// <param name="agentID"></param>
/// <param name="caps"></param>
/// <returns></returns>
public void ParcelVoiceInfoRequest(IOSHttpRequest request, IOSHttpResponse response, UUID agentID, Scene scene)
{
if (request.HttpMethod != "POST")
{
response.StatusCode = (int)HttpStatusCode.NotFound;
return;
}
response.StatusCode = (int)HttpStatusCode.OK;
m_log.DebugFormat(
"{0}[PARCELVOICE]: ParcelVoiceInfoRequest() on {1} for {2}",
logHeader, scene.RegionInfo.RegionName, agentID);
ScenePresence avatar = scene.GetScenePresence(agentID);
if (avatar == null)
{
response.RawBuffer = Util.UTF8.GetBytes("<llsd>undef</llsd>");
return;
}
string avatarName = avatar.Name;
// - check whether we have a region channel in our cache
// - if not:
// create it and cache it
// - send it to the client
// - send channel_uri: as "sip:regionID@m_sipDomain"
try
{
string channelUri;
if (null == scene.LandChannel)
{
m_log.ErrorFormat("region \"{0}\": avatar \"{1}\": land data not yet available",
scene.RegionInfo.RegionName, avatarName);
response.RawBuffer = Util.UTF8.GetBytes("<llsd>undef</llsd>");
return;
}
// get channel_uri: check first whether estate
// settings allow voice, then whether parcel allows
// voice, if all do retrieve or obtain the parcel
// voice channel
LandData land = scene.GetLandData(avatar.AbsolutePosition);
// TODO: EstateSettings don't seem to get propagated...
if (!scene.RegionInfo.EstateSettings.AllowVoice)
{
m_log.DebugFormat("{0}[PARCELVOICE]: region \"{1}\": voice not enabled in estate settings",
logHeader, scene.RegionInfo.RegionName);
channelUri = String.Empty;
}
else
if (!scene.RegionInfo.EstateSettings.TaxFree && (land.Flags & (uint)ParcelFlags.AllowVoiceChat) == 0)
{
channelUri = String.Empty;
}
else
{
channelUri = ChannelUri(scene, land);
}
// fast foward encode
osUTF8 lsl = LLSDxmlEncode2.Start(512);
LLSDxmlEncode2.AddMap(lsl);
LLSDxmlEncode2.AddElem("parcel_local_id", land.LocalID, lsl);
LLSDxmlEncode2.AddElem("region_name", scene.Name, lsl);
LLSDxmlEncode2.AddMap("voice_credentials", lsl);
LLSDxmlEncode2.AddElem("channel_uri", channelUri, lsl);
//LLSDxmlEncode2.AddElem("channel_credentials", channel_credentials, lsl);
LLSDxmlEncode2.AddEndMap(lsl);
LLSDxmlEncode2.AddEndMap(lsl);
response.RawBuffer = LLSDxmlEncode2.EndToBytes(lsl);
}
catch (Exception e)
{
m_log.ErrorFormat("{0}[PARCELVOICE]: region \"{1}\": avatar \"{2}\": {3}, retry later",
logHeader, scene.RegionInfo.RegionName, avatarName, e.Message);
m_log.DebugFormat("{0}[PARCELVOICE]: region \"{1}\": avatar \"{2}\": {3} failed",
logHeader, scene.RegionInfo.RegionName, avatarName, e.ToString());
response.RawBuffer = Util.UTF8.GetBytes("<llsd>undef</llsd>");
}
}
// NOTE NOTE!! This is code from the FreeSwitch module. It is not clear if this is correct for WebRtc.
// Not sure what this Uri is for. Is this FreeSwitch specific?
// TODO: is this useful for WebRtc?
private string ChannelUri(Scene scene, LandData land)
{
string channelUri = null;
string landUUID;
string landName;
// Create parcel voice channel. If no parcel exists, then the voice channel ID is the same
// as the directory ID. Otherwise, it reflects the parcel's ID.
lock (m_ParcelAddress)
{
if (m_ParcelAddress.ContainsKey(land.GlobalID.ToString()))
{
m_log.DebugFormat("{0}: parcel id {1}: using sip address {2}",
logHeader, land.GlobalID, m_ParcelAddress[land.GlobalID.ToString()]);
return m_ParcelAddress[land.GlobalID.ToString()];
}
}
if (land.LocalID != 1 && (land.Flags & (uint)ParcelFlags.UseEstateVoiceChan) == 0)
{
landName = String.Format("{0}:{1}", scene.RegionInfo.RegionName, land.Name);
landUUID = land.GlobalID.ToString();
m_log.DebugFormat("{0}: Region:Parcel \"{1}\": parcel id {2}: using channel name {3}",
logHeader, landName, land.LocalID, landUUID);
}
else
{
landName = String.Format("{0}:{1}", scene.RegionInfo.RegionName, scene.RegionInfo.RegionName);
landUUID = scene.RegionInfo.RegionID.ToString();
m_log.DebugFormat("{0}: Region:Parcel \"{1}\": parcel id {2}: using channel name {3}",
logHeader, landName, land.LocalID, landUUID);
}
// slvoice handles the sip address differently if it begins with confctl, hiding it from the user in
// the friends list. however it also disables the personal speech indicators as well unless some
// siren14-3d codec magic happens. we dont have siren143d so we'll settle for the personal speech indicator.
channelUri = String.Format("sip:conf-{0}@{1}",
"x" + Convert.ToBase64String(Encoding.ASCII.GetBytes(landUUID)),
/*m_freeSwitchRealm*/ "webRTC");
lock (m_ParcelAddress)
{
if (!m_ParcelAddress.ContainsKey(land.GlobalID.ToString()))
{
m_ParcelAddress.Add(land.GlobalID.ToString(), channelUri);
}
}
return channelUri;
}
/// <summary> /// <summary>
/// Convert the LLSDXml body of the request to an OSDMap for easier handling. /// Convert the LLSDXml body of the request to an OSDMap for easier handling.
/// Also logs the request if message details is enabled. /// Also logs the request if message details is enabled.

View File

@@ -26,12 +26,10 @@
*/ */
using System; using System;
using System.Linq;
using System.Collections.Generic; using System.Collections.Generic;
using System.Reflection; using System.Reflection;
using System.Threading.Tasks; using System.Threading.Tasks;
using OpenSim.Framework;
using OpenSim.Region.Framework.Scenes; using OpenSim.Region.Framework.Scenes;
using OpenSim.Region.Framework.Interfaces; using OpenSim.Region.Framework.Interfaces;
using OpenSim.Server.Base; using OpenSim.Server.Base;
@@ -85,49 +83,46 @@ namespace osWebRtcVoice
if (m_Enabled) if (m_Enabled)
{ {
// Get the DLLs for the two voice services // Get the DLLs for the two voice services
string spatialDllName = moduleConfig.GetString("SpatialVoiceService", String.Empty); string spatialDllName = moduleConfig.GetString("SpatialVoiceService", string.Empty);
string nonSpatialDllName = moduleConfig.GetString("NonSpatialVoiceService", String.Empty); string nonSpatialDllName = moduleConfig.GetString("NonSpatialVoiceService", string.Empty);
if (String.IsNullOrEmpty(spatialDllName) && String.IsNullOrEmpty(nonSpatialDllName)) if (string.IsNullOrEmpty(spatialDllName) && string.IsNullOrEmpty(nonSpatialDllName))
{ {
m_log.ErrorFormat("{0} No SpatialVoiceService or NonSpatialVoiceService specified in configuration", LogHeader); m_log.Error($"{LogHeader} No VoiceService specified in configuration");
m_Enabled = false; m_Enabled = false;
return;
} }
// Default non-spatial to spatial if not specified // Default non-spatial to spatial if not specified
if (String.IsNullOrEmpty(nonSpatialDllName)) if (string.IsNullOrEmpty(nonSpatialDllName))
{ {
m_log.DebugFormat("{0} nonSpatialDllName not specified. Defaulting to spatialDllName", LogHeader); m_log.Debug($"{LogHeader} nonSpatialDllName not specified. Defaulting to spatialDllName");
nonSpatialDllName = spatialDllName; nonSpatialDllName = spatialDllName;
} }
// Load the two voice services // Load the two voice services
m_log.DebugFormat("{0} Loading SpatialVoiceService from {1}", LogHeader, spatialDllName); m_log.Debug($"{LogHeader} Loading SpatialVoiceService from {spatialDllName}");
m_spatialVoiceService = ServerUtils.LoadPlugin<IWebRtcVoiceService>(spatialDllName, new object[] { m_Config }); m_spatialVoiceService = ServerUtils.LoadPlugin<IWebRtcVoiceService>(spatialDllName, [m_Config]);
if (m_spatialVoiceService is null) if (m_spatialVoiceService is null)
{ {
m_log.ErrorFormat("{0} Could not load SpatialVoiceService from {1}", LogHeader, spatialDllName); m_log.Error($"{LogHeader} Could not load SpatialVoiceService from {spatialDllName}, module disabled");
m_Enabled = false; m_Enabled = false;
return;
} }
m_log.DebugFormat("{0} Loading NonSpatialVoiceService from {1}", LogHeader, nonSpatialDllName); m_log.Debug($"{LogHeader} Loading NonSpatialVoiceService from {nonSpatialDllName}");
if (spatialDllName == nonSpatialDllName) if (spatialDllName != nonSpatialDllName)
{ {
m_log.DebugFormat("{0} NonSpatialVoiceService is same as SpatialVoiceService", LogHeader); m_nonSpatialVoiceService = ServerUtils.LoadPlugin<IWebRtcVoiceService>(nonSpatialDllName, [ m_Config ]);
m_nonSpatialVoiceService = m_spatialVoiceService;
}
else
{
m_nonSpatialVoiceService = ServerUtils.LoadPlugin<IWebRtcVoiceService>(nonSpatialDllName, new object[] { m_Config });
if (m_nonSpatialVoiceService is null) if (m_nonSpatialVoiceService is null)
{ {
m_log.ErrorFormat("{0} Could not load NonSpatialVoiceService from {1}", LogHeader, nonSpatialDllName); m_log.Error("{LogHeader} Could not load NonSpatialVoiceService from {nonSpatialDllName}");
m_Enabled = false; m_Enabled = false;
} }
} }
if (m_Enabled) if (m_Enabled)
{ {
m_log.InfoFormat("{0} WebRtcVoiceService enabled", LogHeader); m_log.Info($"{LogHeader} WebRtcVoiceService enabled");
} }
} }
} }
@@ -160,7 +155,7 @@ namespace osWebRtcVoice
{ {
if (m_Enabled) if (m_Enabled)
{ {
m_log.DebugFormat("{0} Adding WebRtcVoiceService to region {1}", LogHeader, scene.Name); m_log.Debug($"{LogHeader} Adding WebRtcVoiceService to region {scene.Name}");
scene.RegisterModuleInterface<IWebRtcVoiceService>(this); scene.RegisterModuleInterface<IWebRtcVoiceService>(this);
// TODO: figure out what events we care about // TODO: figure out what events we care about
@@ -218,21 +213,19 @@ namespace osWebRtcVoice
{ {
OSDMap response = null; OSDMap response = null;
IVoiceViewerSession vSession = null; IVoiceViewerSession vSession = null;
if (pRequest.ContainsKey("viewer_session")) if (pRequest.TryGetString("viewer_session", out string viewerSessionId))
{ {
// request has a viewer session. Use that to find the voice service // request has a viewer session. Use that to find the voice service
string viewerSessionId = pRequest["viewer_session"].AsString();
if (!VoiceViewerSession.TryGetViewerSession(viewerSessionId, out vSession)) if (!VoiceViewerSession.TryGetViewerSession(viewerSessionId, out vSession))
{ {
m_log.ErrorFormat("{0} ProvisionVoiceAccountRequest: viewer session {1} not found", LogHeader, viewerSessionId); m_log.Error($"{0} ProvisionVoiceAccountRequest: viewer session {viewerSessionId} not found");
} }
} }
else else
{ {
// the request does not have a viewer session. See if it's an initial request // the request does not have a viewer session. See if it's an initial request
if (pRequest.ContainsKey("channel_type")) if (pRequest.TryGetString("channel_type", out string channelType))
{ {
string channelType = pRequest["channel_type"].AsString();
if (channelType == "local") if (channelType == "local")
{ {
// TODO: check if this userId is making a new session (case that user is reconnecting) // TODO: check if this userId is making a new session (case that user is reconnecting)
@@ -248,7 +241,7 @@ namespace osWebRtcVoice
} }
else else
{ {
m_log.ErrorFormat("{0} ProvisionVoiceAccountRequest: no channel_type in request", LogHeader); m_log.Error($"{LogHeader} ProvisionVoiceAccountRequest: no channel_type in request");
} }
} }
if (vSession is not null) if (vSession is not null)
@@ -263,10 +256,9 @@ namespace osWebRtcVoice
{ {
OSDMap response = null; OSDMap response = null;
IVoiceViewerSession vSession = null; IVoiceViewerSession vSession = null;
if (pRequest.ContainsKey("viewer_session")) if (pRequest.TryGetString("viewer_session", out string viewerSessionId))
{ {
// request has a viewer session. Use that to find the voice service // request has a viewer session. Use that to find the voice service
string viewerSessionId = pRequest["viewer_session"].AsString();
if (VoiceViewerSession.TryGetViewerSession(viewerSessionId, out vSession)) if (VoiceViewerSession.TryGetViewerSession(viewerSessionId, out vSession))
{ {
response = await vSession.VoiceService.VoiceSignalingRequest(vSession, pRequest, pUserID, pSceneID); response = await vSession.VoiceService.VoiceSignalingRequest(vSession, pRequest, pUserID, pSceneID);