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 { /// /// Manages the hosting services. /// [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 m_HostingServiceInfos; [FormerlySerializedAs("m_settings")] [SerializeField] AddressableAssetSettings m_Settings; [FormerlySerializedAs("m_nextInstanceId")] [SerializeField] int m_NextInstanceId; [FormerlySerializedAs("m_registeredServiceTypeRefs")] [SerializeField] List m_RegisteredServiceTypeRefs; readonly Type[] m_BuiltinServiceTypes = { typeof(HttpHostingService) }; Dictionary m_HostingServiceInfoMap; ILogger m_Logger; List m_RegisteredServiceTypes; [SerializeField] int m_PingTimeoutInMilliseconds = 5000; /// /// Timeout in milliseconds for filtering ip addresses for the hosting service /// internal int PingTimeoutInMilliseconds { get { return m_PingTimeoutInMilliseconds; } set { m_PingTimeoutInMilliseconds = value; } } /// /// Key/Value pairs valid for profile variable substitution /// public Dictionary 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}"; } /// /// Direct logging output of all managed services /// public ILogger Logger { get { return m_Logger; } set { m_Logger = value ?? Debug.unityLogger; foreach (var svc in HostingServices) svc.Logger = m_Logger; } } /// /// Static method for use in starting up the HostingServicesManager in batch mode. /// /// 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(); } /// /// Static method for use in starting up the HostingServicesManager in batch mode. This method /// without parameters will find and use the default object. /// public static void BatchMode() { BatchMode(AddressableAssetSettingsDefaultObject.Settings); } /// /// Indicates whether or not this HostingServiceManager is initialized /// public bool IsInitialized { get { return m_Settings != null; } } /// /// Return an enumerable list of all configured objects /// public ICollection HostingServices { get { return m_HostingServiceInfoMap.Keys; } } /// /// Get an array of all types that have been used by the manager, or are known /// built-in types available for use. /// /// public Type[] RegisteredServiceTypes { get { if (m_RegisteredServiceTypes.Count == 0) m_RegisteredServiceTypes.AddRange(m_BuiltinServiceTypes); return m_RegisteredServiceTypes.ToArray(); } } /// /// The id value that will be assigned to the next add to the manager. /// public int NextInstanceId { get { return m_NextInstanceId; } } /// /// Create a new /// public HostingServicesManager() { GlobalProfileVariables = new Dictionary(); m_HostingServiceInfos = new List(); m_HostingServiceInfoMap = new Dictionary(); m_RegisteredServiceTypes = new List(); m_RegisteredServiceTypeRefs = new List(); m_Logger = Debug.unityLogger; } /// /// Initialize manager with the given object. /// /// public void Initialize(AddressableAssetSettings settings) { if (IsInitialized) return; m_Settings = settings; RefreshGlobalProfileVariables(); } /// /// Calls on all managed instances /// where is true /// public void StopAllServices() { foreach (var svc in HostingServices) { try { if (svc.IsHostingServiceRunning) svc.StopHostingService(); } catch (Exception e) { m_Logger.LogFormat(LogType.Error, e.Message); } } } /// /// Calls on all managed instances /// where is false /// public void StartAllServices() { foreach (var svc in HostingServices) { try { if (!svc.IsHostingServiceRunning) svc.StartHostingService(); } catch (Exception e) { m_Logger.LogFormat(LogType.Error, e.Message); } } } /// /// Add a new hosting service instance of the given type. The must implement the /// interface, or an is thrown. /// /// A object for the service. Must implement /// A descriptive name for the new service instance. /// 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; } /// /// Stops the given , unregisters callbacks, and removes it from management. This /// function does nothing if the service is not being managed by this /// /// 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); } /// /// Should be called by parent instance Awake method /// internal void OnAwake() { RefreshGlobalProfileVariables(); } /// /// Should be called by parent instance OnEnable method /// 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(); } /// /// Should be called by parent instance OnDisable method /// 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 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)); } } /// Ensure object is ready for serialization, and calls methods /// on all managed instances /// 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)); } /// Ensure object is ready for serialization, and calls methods /// on all managed instances /// public void OnAfterDeserialize() { m_HostingServiceInfoMap = new Dictionary(); 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(); foreach (var typeRef in m_RegisteredServiceTypeRefs) { var type = Type.GetType(typeRef, false); if (type == null) continue; m_RegisteredServiceTypes.Add(type); } } /// /// Refresh values in the global profile variables table. /// 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(); 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 FilterValidIPAddresses(List ipAddresses) { List validIpList = new List(); 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; } } }