using System; using System.Collections.Generic; using Unity.MLAgents.Policies; using Unity.MLAgents.Sensors; using UnityEngine; using UnityEngine.Analytics; #if UNITY_EDITOR using UnityEditor; using UnityEditor.Analytics; #endif namespace Unity.MLAgents.Analytics { internal class TrainingAnalytics { const string k_VendorKey = "unity.ml-agents"; const string k_RemotePolicyInitializedEventName = "ml_agents_remote_policy_initialized"; /// /// Whether or not we've registered this particular event yet /// static bool s_EventsRegistered = false; /// /// Hourly limit for this event name /// const int k_MaxEventsPerHour = 1000; /// /// Maximum number of items in this event. /// const int k_MaxNumberOfElements = 1000; /// /// Behaviors that we've already sent events for. /// private static HashSet s_SentRemotePolicyInitialized; private static Guid s_TrainingSessionGuid; // These are set when the RpcCommunicator connects private static string s_TrainerPackageVersion = ""; private static string s_TrainerCommunicationVersion = ""; static bool EnableAnalytics() { if (s_EventsRegistered) { return true; } #if UNITY_EDITOR AnalyticsResult result = EditorAnalytics.RegisterEventWithLimit(k_RemotePolicyInitializedEventName, k_MaxEventsPerHour, k_MaxNumberOfElements, k_VendorKey); #else AnalyticsResult result = AnalyticsResult.UnsupportedPlatform; #endif if (result != AnalyticsResult.Ok) { return false; } s_EventsRegistered = true; if (s_SentRemotePolicyInitialized == null) { s_SentRemotePolicyInitialized = new HashSet(); s_TrainingSessionGuid = Guid.NewGuid(); } return s_EventsRegistered; } /// /// Cache information about the trainer when it becomes available in the RpcCommunicator. /// /// /// public static void SetTrainerInformation(string packageVersion, string communicationVersion) { s_TrainerPackageVersion = packageVersion; s_TrainerCommunicationVersion = communicationVersion; } public static bool IsAnalyticsEnabled() { #if UNITY_EDITOR return EditorAnalytics.enabled; #else return false; #endif } public static void RemotePolicyInitialized( string fullyQualifiedBehaviorName, IList sensors, BrainParameters brainParameters ) { if (!IsAnalyticsEnabled()) return; if (!EnableAnalytics()) return; // Extract base behavior name (no team ID) var behaviorName = ParseBehaviorName(fullyQualifiedBehaviorName); var added = s_SentRemotePolicyInitialized.Add(behaviorName); if (!added) { // We previously added this model. Exit so we don't resend. return; } var data = GetEventForRemotePolicy(behaviorName, sensors, brainParameters); // Note - to debug, use JsonUtility.ToJson on the event. // Debug.Log( // $"Would send event {k_RemotePolicyInitializedEventName} with body {JsonUtility.ToJson(data, true)}" // ); #if UNITY_EDITOR if (AnalyticsUtils.s_SendEditorAnalytics) { EditorAnalytics.SendEventWithLimit(k_RemotePolicyInitializedEventName, data); } #else return; #endif } internal static string ParseBehaviorName(string fullyQualifiedBehaviorName) { var lastQuestionIndex = fullyQualifiedBehaviorName.LastIndexOf("?"); if (lastQuestionIndex < 0) { // Nothing to remove return fullyQualifiedBehaviorName; } return fullyQualifiedBehaviorName.Substring(0, lastQuestionIndex); } static RemotePolicyInitializedEvent GetEventForRemotePolicy( string behaviorName, IList sensors, BrainParameters brainParameters) { var remotePolicyEvent = new RemotePolicyInitializedEvent(); // Hash the behavior name so that there's no concern about PII or "secret" data being leaked. remotePolicyEvent.BehaviorName = AnalyticsUtils.Hash(behaviorName); remotePolicyEvent.TrainingSessionGuid = s_TrainingSessionGuid.ToString(); remotePolicyEvent.ActionSpec = EventActionSpec.FromBrainParameters(brainParameters); remotePolicyEvent.ObservationSpecs = new List(sensors.Count); foreach (var sensor in sensors) { remotePolicyEvent.ObservationSpecs.Add(EventObservationSpec.FromSensor(sensor)); } remotePolicyEvent.MLAgentsEnvsVersion = s_TrainerPackageVersion; remotePolicyEvent.TrainerCommunicationVersion = s_TrainerCommunicationVersion; return remotePolicyEvent; } } }