WuhuIslandTesting/Library/PackageCache/com.unity.addressables@1.21.12/Editor/HostingServices/HostingServicesManager.cs
2025-01-07 02:06:59 +01:00

596 lines
22 KiB
C#

using System;
using System.Collections.Generic;
using System.Linq;
using System.Net;
using System.Net.NetworkInformation;
using System.Net.Sockets;
using UnityEditor.AddressableAssets.Build.DataBuilders;
using UnityEditor.AddressableAssets.Settings;
using UnityEngine;
using UnityEngine.Serialization;
// ReSharper disable DelegateSubtraction
namespace UnityEditor.AddressableAssets.HostingServices
{
/// <summary>
/// Manages the hosting services.
/// </summary>
[Serializable]
public class HostingServicesManager : ISerializationCallbackReceiver
{
internal const string KPrivateIpAddressKey = "PrivateIpAddress";
internal string GetPrivateIpAddressKey(int id = 0)
{
if (id == 0)
return KPrivateIpAddressKey;
return $"{KPrivateIpAddressKey}_{id}";
}
[Serializable]
internal class HostingServiceInfo
{
[SerializeField]
internal string classRef;
[SerializeField]
internal KeyDataStore dataStore;
}
[FormerlySerializedAs("m_hostingServiceInfos")]
[SerializeField]
List<HostingServiceInfo> m_HostingServiceInfos;
[FormerlySerializedAs("m_settings")]
[SerializeField]
AddressableAssetSettings m_Settings;
[FormerlySerializedAs("m_nextInstanceId")]
[SerializeField]
int m_NextInstanceId;
[FormerlySerializedAs("m_registeredServiceTypeRefs")]
[SerializeField]
List<string> m_RegisteredServiceTypeRefs;
readonly Type[] m_BuiltinServiceTypes =
{
typeof(HttpHostingService)
};
Dictionary<IHostingService, HostingServiceInfo> m_HostingServiceInfoMap;
ILogger m_Logger;
List<Type> m_RegisteredServiceTypes;
[SerializeField]
int m_PingTimeoutInMilliseconds = 5000;
/// <summary>
/// Timeout in milliseconds for filtering ip addresses for the hosting service
/// </summary>
internal int PingTimeoutInMilliseconds
{
get { return m_PingTimeoutInMilliseconds; }
set { m_PingTimeoutInMilliseconds = value; }
}
/// <summary>
/// Key/Value pairs valid for profile variable substitution
/// </summary>
public Dictionary<string, string> GlobalProfileVariables { get; private set; }
internal static readonly string k_GlobalProfileVariablesCountKey = $"com.unity.addressables.{nameof(GlobalProfileVariables)}Count";
internal static string GetSessionStateKey(int id)
{
return $"com.unity.addressables.{nameof(GlobalProfileVariables)}{id}";
}
/// <summary>
/// Direct logging output of all managed services
/// </summary>
public ILogger Logger
{
get { return m_Logger; }
set
{
m_Logger = value ?? Debug.unityLogger;
foreach (var svc in HostingServices)
svc.Logger = m_Logger;
}
}
/// <summary>
/// Static method for use in starting up the HostingServicesManager in batch mode.
/// </summary>
/// <param name="settings"> </param>
public static void BatchMode(AddressableAssetSettings settings)
{
if (settings == null)
{
Debug.LogError("Could not load Addressable Assets settings - aborting.");
return;
}
var manager = settings.HostingServicesManager;
if (manager == null)
{
Debug.LogError("Could not load HostingServicesManager - aborting.");
return;
}
manager.StartAllServices();
}
/// <summary>
/// Static method for use in starting up the HostingServicesManager in batch mode. This method
/// without parameters will find and use the default <see cref="AddressableAssetSettings"/> object.
/// </summary>
public static void BatchMode()
{
BatchMode(AddressableAssetSettingsDefaultObject.Settings);
}
/// <summary>
/// Indicates whether or not this HostingServiceManager is initialized
/// </summary>
public bool IsInitialized
{
get { return m_Settings != null; }
}
/// <summary>
/// Return an enumerable list of all configured <see cref="IHostingService"/> objects
/// </summary>
public ICollection<IHostingService> HostingServices
{
get { return m_HostingServiceInfoMap.Keys; }
}
/// <summary>
/// Get an array of all <see cref="IHostingService"/> types that have been used by the manager, or are known
/// built-in types available for use.
/// </summary>
/// <returns></returns>
public Type[] RegisteredServiceTypes
{
get
{
if (m_RegisteredServiceTypes.Count == 0)
m_RegisteredServiceTypes.AddRange(m_BuiltinServiceTypes);
return m_RegisteredServiceTypes.ToArray();
}
}
/// <summary>
/// The id value that will be assigned to the next <see cref="IHostingService"/> add to the manager.
/// </summary>
public int NextInstanceId
{
get { return m_NextInstanceId; }
}
/// <summary>
/// Create a new <see cref="HostingServicesManager"/>
/// </summary>
public HostingServicesManager()
{
GlobalProfileVariables = new Dictionary<string, string>();
m_HostingServiceInfos = new List<HostingServiceInfo>();
m_HostingServiceInfoMap = new Dictionary<IHostingService, HostingServiceInfo>();
m_RegisteredServiceTypes = new List<Type>();
m_RegisteredServiceTypeRefs = new List<string>();
m_Logger = Debug.unityLogger;
}
/// <summary>
/// Initialize manager with the given <see cref="AddressableAssetSettings"/> object.
/// </summary>
/// <param name="settings"></param>
public void Initialize(AddressableAssetSettings settings)
{
if (IsInitialized) return;
m_Settings = settings;
RefreshGlobalProfileVariables();
}
/// <summary>
/// Calls <see cref="IHostingService.StopHostingService"/> on all managed <see cref="IHostingService"/> instances
/// where <see cref="IHostingService.IsHostingServiceRunning"/> is true
/// </summary>
public void StopAllServices()
{
foreach (var svc in HostingServices)
{
try
{
if (svc.IsHostingServiceRunning)
svc.StopHostingService();
}
catch (Exception e)
{
m_Logger.LogFormat(LogType.Error, e.Message);
}
}
}
/// <summary>
/// Calls <see cref="IHostingService.StartHostingService"/> on all managed <see cref="IHostingService"/> instances
/// where <see cref="IHostingService.IsHostingServiceRunning"/> is false
/// </summary>
public void StartAllServices()
{
foreach (var svc in HostingServices)
{
try
{
if (!svc.IsHostingServiceRunning)
svc.StartHostingService();
}
catch (Exception e)
{
m_Logger.LogFormat(LogType.Error, e.Message);
}
}
}
/// <summary>
/// Add a new hosting service instance of the given type. The <paramref name="serviceType"/> must implement the
/// <see cref="IHostingService"/> interface, or an <see cref="ArgumentException"/> is thrown.
/// </summary>
/// <param name="serviceType">A <see cref="Type"/> object for the service. Must implement <see cref="IHostingService"/></param>
/// <param name="name">A descriptive name for the new service instance.</param>
/// <returns></returns>
public IHostingService AddHostingService(Type serviceType, string name)
{
var svc = Activator.CreateInstance(serviceType) as IHostingService;
if (svc == null)
throw new ArgumentException("Provided type does not implement IHostingService", "serviceType");
if (!m_RegisteredServiceTypes.Contains(serviceType))
m_RegisteredServiceTypes.Add(serviceType);
var info = new HostingServiceInfo
{
classRef = TypeToClassRef(serviceType),
dataStore = new KeyDataStore()
};
svc.Logger = m_Logger;
svc.DescriptiveName = name;
svc.InstanceId = m_NextInstanceId;
svc.HostingServiceContentRoots.AddRange(GetAllContentRoots());
m_Settings.profileSettings.RegisterProfileStringEvaluationFunc(svc.EvaluateProfileString);
m_HostingServiceInfoMap.Add(svc, info);
m_Settings.SetDirty(AddressableAssetSettings.ModificationEvent.HostingServicesManagerModified, this, true, true);
AddressableAssetUtility.OpenAssetIfUsingVCIntegration(m_Settings);
m_NextInstanceId++;
return svc;
}
/// <summary>
/// Stops the given <see cref="IHostingService"/>, unregisters callbacks, and removes it from management. This
/// function does nothing if the service is not being managed by this <see cref="HostingServicesManager"/>
/// </summary>
/// <param name="svc"></param>
public void RemoveHostingService(IHostingService svc)
{
if (!m_HostingServiceInfoMap.ContainsKey(svc))
return;
svc.StopHostingService();
m_Settings.profileSettings.UnregisterProfileStringEvaluationFunc(svc.EvaluateProfileString);
m_HostingServiceInfoMap.Remove(svc);
m_Settings.SetDirty(AddressableAssetSettings.ModificationEvent.HostingServicesManagerModified, this, true, true);
AddressableAssetUtility.OpenAssetIfUsingVCIntegration(m_Settings);
}
/// <summary>
/// Should be called by parent <see cref="ScriptableObject"/> instance Awake method
/// </summary>
internal void OnAwake()
{
RefreshGlobalProfileVariables();
}
/// <summary>
/// Should be called by parent <see cref="ScriptableObject"/> instance OnEnable method
/// </summary>
public void OnEnable()
{
Debug.Assert(IsInitialized);
m_Settings.OnModification -= OnSettingsModification;
m_Settings.OnModification += OnSettingsModification;
m_Settings.profileSettings.RegisterProfileStringEvaluationFunc(EvaluateGlobalProfileVariableKey);
// GetAllContentRoots can return unpredictable results when there are no hosting services
if (HostingServices.Count > 0)
{
var contentRoots = GetAllContentRoots();
foreach (var svc in HostingServices)
{
svc.Logger = m_Logger;
m_Settings.profileSettings.RegisterProfileStringEvaluationFunc(svc.EvaluateProfileString);
var baseSvc = svc as BaseHostingService;
svc.HostingServiceContentRoots.Clear();
svc.HostingServiceContentRoots.AddRange(contentRoots);
baseSvc?.OnEnable();
}
}
LoadSessionStateKeysIfExists();
}
/// <summary>
/// Should be called by parent <see cref="ScriptableObject"/> instance OnDisable method
/// </summary>
public void OnDisable()
{
Debug.Assert(IsInitialized);
// ReSharper disable once DelegateSubtraction
m_Settings.OnModification -= OnSettingsModification;
m_Settings.profileSettings.UnregisterProfileStringEvaluationFunc(EvaluateGlobalProfileVariableKey);
foreach (var svc in HostingServices)
{
svc.Logger = null;
m_Settings.profileSettings.UnregisterProfileStringEvaluationFunc(svc.EvaluateProfileString);
(svc as BaseHostingService)?.OnDisable();
}
SaveSessionStateKeys();
}
internal void LoadSessionStateKeysIfExists()
{
int numKeys = SessionState.GetInt(k_GlobalProfileVariablesCountKey, 0);
if (numKeys > 0)
GlobalProfileVariables.Clear();
for (int i = 0; i < numKeys; i++)
{
string profileVar = SessionState.GetString(GetSessionStateKey(i), string.Empty);
if (!string.IsNullOrEmpty(profileVar))
GlobalProfileVariables.Add(GetPrivateIpAddressKey(i), profileVar);
}
}
internal void SaveSessionStateKeys()
{
int prevNumKeys = SessionState.GetInt(k_GlobalProfileVariablesCountKey, 0);
SessionState.SetInt(k_GlobalProfileVariablesCountKey, GlobalProfileVariables.Count);
int profileVarIdx = 0;
foreach (KeyValuePair<string, string> pair in GlobalProfileVariables)
{
SessionState.SetString(GetSessionStateKey(profileVarIdx), pair.Value);
profileVarIdx++;
}
EraseSessionStateKeys(profileVarIdx, prevNumKeys);
}
internal static void EraseSessionStateKeys()
{
int numKeys = SessionState.GetInt(k_GlobalProfileVariablesCountKey, 0);
EraseSessionStateKeys(0, numKeys);
SessionState.EraseInt(k_GlobalProfileVariablesCountKey);
}
static void EraseSessionStateKeys(int min, int max)
{
for (int i = min; i < max; i++)
{
SessionState.EraseString(GetSessionStateKey(i));
}
}
/// <summary> Ensure object is ready for serialization, and calls <see cref="IHostingService.OnBeforeSerialize"/> methods
/// on all managed <see cref="IHostingService"/> instances
/// </summary>
public void OnBeforeSerialize()
{
// https://docs.unity3d.com/ScriptReference/EditorWindow.OnInspectorUpdate.html
// Because the manager is a serialized field in the Addressables settings, this method is called
// at 10 frames per second when the settings are opened in the inspector...
// Be careful what you put in there...
m_HostingServiceInfos.Clear();
foreach (var svc in HostingServices)
{
var info = m_HostingServiceInfoMap[svc];
m_HostingServiceInfos.Add(info);
svc.OnBeforeSerialize(info.dataStore);
}
m_RegisteredServiceTypeRefs.Clear();
foreach (var type in m_RegisteredServiceTypes)
m_RegisteredServiceTypeRefs.Add(TypeToClassRef(type));
}
/// <summary> Ensure object is ready for serialization, and calls <see cref="IHostingService.OnBeforeSerialize"/> methods
/// on all managed <see cref="IHostingService"/> instances
/// </summary>
public void OnAfterDeserialize()
{
m_HostingServiceInfoMap = new Dictionary<IHostingService, HostingServiceInfo>();
foreach (var svcInfo in m_HostingServiceInfos)
{
IHostingService svc = CreateHostingServiceInstance(svcInfo.classRef);
if (svc == null) continue;
svc.OnAfterDeserialize(svcInfo.dataStore);
m_HostingServiceInfoMap.Add(svc, svcInfo);
}
m_RegisteredServiceTypes = new List<Type>();
foreach (var typeRef in m_RegisteredServiceTypeRefs)
{
var type = Type.GetType(typeRef, false);
if (type == null) continue;
m_RegisteredServiceTypes.Add(type);
}
}
/// <summary>
/// Refresh values in the global profile variables table.
/// </summary>
public void RefreshGlobalProfileVariables()
{
var vars = GlobalProfileVariables;
vars.Clear();
var ipAddressList = FilterValidIPAddresses(NetworkInterface.GetAllNetworkInterfaces()
.Where(n => n.NetworkInterfaceType != NetworkInterfaceType.Loopback && n.OperationalStatus == OperationalStatus.Up)
.SelectMany(n => n.GetIPProperties().UnicastAddresses)
.Where(a => a.Address.AddressFamily == AddressFamily.InterNetwork)
.Select(a => a.Address).ToList());
if (ipAddressList.Count > 0)
{
vars.Add(KPrivateIpAddressKey, ipAddressList[0].ToString());
if (ipAddressList.Count > 1)
{
for (var i = 1; i < ipAddressList.Count; i++)
vars.Add(KPrivateIpAddressKey + "_" + i, ipAddressList[i].ToString());
}
}
}
// Internal for unit tests
internal string EvaluateGlobalProfileVariableKey(string key)
{
string retVal;
GlobalProfileVariables.TryGetValue(key, out retVal);
return retVal;
}
void OnSettingsModification(AddressableAssetSettings s, AddressableAssetSettings.ModificationEvent evt, object obj)
{
switch (evt)
{
case AddressableAssetSettings.ModificationEvent.GroupAdded:
case AddressableAssetSettings.ModificationEvent.GroupRemoved:
case AddressableAssetSettings.ModificationEvent.GroupSchemaAdded:
case AddressableAssetSettings.ModificationEvent.GroupSchemaRemoved:
case AddressableAssetSettings.ModificationEvent.GroupSchemaModified:
case AddressableAssetSettings.ModificationEvent.ActiveProfileSet:
case AddressableAssetSettings.ModificationEvent.BuildSettingsChanged:
case AddressableAssetSettings.ModificationEvent.ProfileModified:
var profileRemoteBuildPath = m_Settings.profileSettings.GetValueByName(m_Settings.activeProfileId, AddressableAssetSettings.kRemoteBuildPath);
if (profileRemoteBuildPath != null && (profileRemoteBuildPath.Contains('[') || !CurrentContentRootsContain(profileRemoteBuildPath)))
ConfigureAllHostingServices();
break;
case AddressableAssetSettings.ModificationEvent.BatchModification:
ConfigureAllHostingServices();
break;
}
}
bool CurrentContentRootsContain(string root)
{
foreach (var svc in HostingServices)
{
if (!svc.HostingServiceContentRoots.Contains(root))
return false;
}
return true;
}
void ConfigureAllHostingServices()
{
if (HostingServices.Count > 0)
{
var contentRoots = GetAllContentRoots();
foreach (var svc in HostingServices)
{
svc.HostingServiceContentRoots.Clear();
svc.HostingServiceContentRoots.AddRange(contentRoots);
}
}
}
string[] GetAllContentRoots()
{
Debug.Assert(IsInitialized);
var contentRoots = new List<string>();
foreach (var group in m_Settings.groups)
{
if (group != null)
{
foreach (var schema in group.Schemas)
{
var configProvider = schema as IHostingServiceConfigurationProvider;
if (configProvider != null)
{
var groupRoot = configProvider.HostingServicesContentRoot;
if (groupRoot != null && !contentRoots.Contains(groupRoot))
contentRoots.Add(groupRoot);
}
}
}
}
return contentRoots.ToArray();
}
IHostingService CreateHostingServiceInstance(string classRef)
{
try
{
var objType = Type.GetType(classRef, true);
var svc = (IHostingService)Activator.CreateInstance(objType);
return svc;
}
catch (Exception e)
{
m_Logger.LogFormat(LogType.Error, "Could not create IHostingService from class ref '{0}'", classRef);
m_Logger.LogFormat(LogType.Error, e.Message);
}
return null;
}
static string TypeToClassRef(Type t)
{
return string.Format("{0}, {1}", t.FullName, t.Assembly.GetName().Name);
}
// For unit tests
internal AddressableAssetSettings Settings
{
get { return m_Settings; }
}
private List<IPAddress> FilterValidIPAddresses(List<IPAddress> ipAddresses)
{
List<IPAddress> validIpList = new List<IPAddress>();
if (PingTimeoutInMilliseconds < 0)
{
m_Logger.LogFormat(LogType.Error, "Cannot filter IP addresses. Timeout must be a non-negative integer.");
return validIpList;
}
foreach (IPAddress address in ipAddresses)
{
var sender = new System.Net.NetworkInformation.Ping();
var reply = sender.Send(address.ToString(), PingTimeoutInMilliseconds);
if (reply.Status == IPStatus.Success)
{
validIpList.Add(address);
}
}
return validIpList;
}
}
}