using Grpc.Core; using System.IO; using System.Threading; using System.Threading.Tasks; #if UNITY_EDITOR using UnityEditor; #endif using UnityEngine; using MLAgents.CommunicatorObjects; namespace MLAgents { /// Responsible for communication with External using gRPC. public class RPCCommunicator : Communicator { /// If true, the communication is active. bool m_isOpen; /// The Unity to External client. UnityToExternal.UnityToExternalClient m_client; /// The communicator parameters sent at construction CommunicatorParameters m_communicatorParameters; /// /// Initializes a new instance of the RPCCommunicator class. /// /// Communicator parameters. public RPCCommunicator(CommunicatorParameters communicatorParameters) { m_communicatorParameters = communicatorParameters; } /// /// Initialize the communicator by sending the first UnityOutput and receiving the /// first UnityInput. The second UnityInput is stored in the unityInput argument. /// /// The first Unity Input. /// The first Unity Output. /// The second Unity input. public UnityInput Initialize(UnityOutput unityOutput, out UnityInput unityInput) { m_isOpen = true; var channel = new Channel( "localhost:"+m_communicatorParameters.port, ChannelCredentials.Insecure); m_client = new UnityToExternal.UnityToExternalClient(channel); var result = m_client.Exchange(WrapMessage(unityOutput, 200)); unityInput = m_client.Exchange(WrapMessage(null, 200)).UnityInput; #if UNITY_EDITOR EditorApplication.playModeStateChanged += HandleOnPlayModeChanged; #endif return result.UnityInput; } /// /// Close the communicator gracefully on both sides of the communication. /// public void Close() { if (!m_isOpen) { return; } try { m_client.Exchange(WrapMessage(null, 400)); m_isOpen = false; } catch { return; } } /// /// Send a UnityOutput and receives a UnityInput. /// /// The next UnityInput. /// The UnityOutput to be sent. public UnityInput Exchange(UnityOutput unityOutput) { if (!m_isOpen) { return null; } try { var message = m_client.Exchange(WrapMessage(unityOutput, 200)); if (message.Header.Status == 200) { return message.UnityInput; } else { m_isOpen = false; return null; } } catch { m_isOpen = false; return null; } } /// /// Wraps the UnityOuptut into a message with the appropriate status. /// /// The UnityMessage corresponding. /// The UnityOutput to be wrapped. /// The status of the message. private static UnityMessage WrapMessage(UnityOutput content, int status) { return new UnityMessage { Header = new Header { Status = status }, UnityOutput = content }; } /// /// When the Unity application quits, the communicator must be closed /// private void OnApplicationQuit() { Close(); } #if UNITY_EDITOR /// /// When the editor exits, the communicator must be closed /// /// State. private void HandleOnPlayModeChanged(PlayModeStateChange state) { // This method is run whenever the playmode state is changed. if (state==PlayModeStateChange.ExitingPlayMode) { Close(); } } #endif } }