/* ======================================================================== * Copyright (c) 2005-2020 The OPC Foundation, Inc. All rights reserved. * * OPC Foundation MIT License 1.00 * * Permission is hereby granted, free of charge, to any person * obtaining a copy of this software and associated documentation * files (the "Software"), to deal in the Software without * restriction, including without limitation the rights to use, * copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the * Software is furnished to do so, subject to the following * conditions: * * The above copyright notice and this permission notice shall be * included in all copies or substantial portions of the Software. * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES * OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT * HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, * WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR * OTHER DEALINGS IN THE SOFTWARE. * * The complete license agreement can be found here: * http://opcfoundation.org/License/MIT/1.00/ * ======================================================================*/ using Mono.Options; using Opc.Ua; using Opc.Ua.Configuration; using Opc.Ua.Server; using Serilog; using System; using System.Collections.Generic; using System.Linq; using System.Security.Cryptography; using System.Text; using System.Threading; using System.Threading.Tasks; namespace Quickstarts.ReferenceServer { /// /// A dialog which asks for user input. /// public class ApplicationMessageDlg : IApplicationMessageDlg { private string m_message = string.Empty; private bool m_ask = false; public override void Message(string text, bool ask) { m_message = text; m_ask = ask; } public override async Task ShowAsync() { if (m_ask) { m_message += " (y/n, default y): "; Console.Write(m_message); } else { Console.WriteLine(m_message); } if (m_ask) { try { ConsoleKeyInfo result = Console.ReadKey(); Console.WriteLine(); return await Task.FromResult((result.KeyChar == 'y') || (result.KeyChar == 'Y') || (result.KeyChar == '\r')).ConfigureAwait(false); } catch { // intentionally fall through } } return await Task.FromResult(true).ConfigureAwait(false); } } /// /// The error code why the server exited. /// public enum ExitCode : int { Ok = 0, ErrorServerNotStarted = 0x80, ErrorServerRunning = 0x81, ErrorServerException = 0x82, ErrorInvalidCommandLine = 0x100 }; /// /// The program. /// public static class Program { public static async Task Main(string[] args) { Console.WriteLine("{0} OPC UA Reference Server", Utils.IsRunningOnMono() ? "Mono" : ".Net Core"); // command line options bool showHelp = false; bool autoAccept = false; bool console = false; string password = null; Mono.Options.OptionSet options = new Mono.Options.OptionSet { { "h|help", "show this message and exit", h => showHelp = h != null }, { "a|autoaccept", "auto accept certificates (for testing only)", a => autoAccept = a != null }, { "c|console", "log trace to console", c => console = c != null }, { "p|password=", "optional password for private key", (string p) => password = p } }; try { IList extraArgs = options.Parse(args); foreach (string extraArg in extraArgs) { Console.WriteLine("Error: Unknown option: {0}", extraArg); showHelp = true; } } catch (OptionException e) { Console.WriteLine(e.Message); showHelp = true; } if (showHelp) { Console.WriteLine(Utils.IsRunningOnMono() ? "Usage: mono MonoReferenceServer.exe [OPTIONS]" : "Usage: dotnet ConsoleReferenceServer.dll [OPTIONS]"); Console.WriteLine(); Console.WriteLine("Options:"); options.WriteOptionDescriptions(Console.Out); return (int)ExitCode.ErrorInvalidCommandLine; } var server = new MyRefServer() { AutoAccept = autoAccept, LogConsole = console, Password = password }; await server.Run().ConfigureAwait(false); return (int)server.ExitCode; } } public class MyRefServer { private ReferenceServer m_server; private Task m_status; private DateTime m_lastEventTime; public bool LogConsole { get; set; } = false; public bool AutoAccept { get; set; } = false; public string Password { get; set; } = null; public ExitCode ExitCode { get; private set; } public async Task Run() { try { ExitCode = ExitCode.ErrorServerNotStarted; await StartConsoleReferenceServerAsync().ConfigureAwait(false); Console.WriteLine("Server started. Press Ctrl-C to exit..."); ExitCode = ExitCode.ErrorServerRunning; } catch (Exception ex) { Console.WriteLine("Exception: {0}", ex.Message); ExitCode = ExitCode.ErrorServerException; return; } var quitEvent = new ManualResetEvent(false); try { Console.CancelKeyPress += (sender, eArgs) => { quitEvent.Set(); eArgs.Cancel = true; }; } catch { } // wait for timeout or Ctrl-C quitEvent.WaitOne(); if (m_server != null) { Console.WriteLine("Server stopped. Waiting for exit..."); using (ReferenceServer server = m_server) { // Stop status thread m_server = null; m_status.Wait(); // Stop server and dispose server.Stop(); } } ExitCode = ExitCode.Ok; } private void CertificateValidator_CertificateValidation(CertificateValidator validator, CertificateValidationEventArgs e) { if (e.Error.StatusCode == StatusCodes.BadCertificateUntrusted) { if (AutoAccept) { if (!LogConsole) { Console.WriteLine("Accepted Certificate: {0}", e.Certificate.Subject); } Utils.Trace(Utils.TraceMasks.Security, "Accepted Certificate: {0}", e.Certificate.Subject); e.Accept = true; return; } } if (!LogConsole) { Console.WriteLine("Rejected Certificate: {0} {1}", e.Error, e.Certificate.Subject); } Utils.Trace(Utils.TraceMasks.Security, "Rejected Certificate: {0} {1}", e.Error, e.Certificate.Subject); } private async Task StartConsoleReferenceServerAsync() { ApplicationInstance.MessageDlg = new ApplicationMessageDlg(); CertificatePasswordProvider PasswordProvider = new CertificatePasswordProvider(Password); ApplicationInstance application = new ApplicationInstance { ApplicationName = "Quickstart Reference Server", ApplicationType = ApplicationType.Server, ConfigSectionName = Utils.IsRunningOnMono() ? "Quickstarts.MonoReferenceServer" : "Quickstarts.ReferenceServer", CertificatePasswordProvider = PasswordProvider }; // load the application configuration. ApplicationConfiguration config = await application.LoadApplicationConfiguration(false).ConfigureAwait(false); var loggerConfiguration = new Serilog.LoggerConfiguration(); if (LogConsole) { loggerConfiguration.WriteTo.Console(restrictedToMinimumLevel: Serilog.Events.LogEventLevel.Warning); } #if DEBUG else { loggerConfiguration.WriteTo.Debug(restrictedToMinimumLevel: Serilog.Events.LogEventLevel.Warning); } #endif SerilogTraceLogger.Create(loggerConfiguration, config); // check the application certificate. bool haveAppCertificate = await application.CheckApplicationInstanceCertificate( false, CertificateFactory.DefaultKeySize, CertificateFactory.DefaultLifeTime).ConfigureAwait(false); if (!haveAppCertificate) { throw new Exception("Application instance certificate invalid!"); } if (!config.SecurityConfiguration.AutoAcceptUntrustedCertificates) { config.CertificateValidator.CertificateValidation += new CertificateValidationEventHandler(CertificateValidator_CertificateValidation); } // start the server. m_server = new ReferenceServer(); await application.Start(m_server).ConfigureAwait(false); // print endpoint info var endpoints = application.Server.GetEndpoints().Select(e => e.EndpointUrl).Distinct(); foreach (var endpoint in endpoints) { Console.WriteLine(endpoint); } // start the status thread m_status = Task.Run(new Action(StatusThreadAsync)); // print notification on session events m_server.CurrentInstance.SessionManager.SessionActivated += EventStatus; m_server.CurrentInstance.SessionManager.SessionClosing += EventStatus; m_server.CurrentInstance.SessionManager.SessionCreated += EventStatus; } private void EventStatus(Session session, SessionEventReason reason) { m_lastEventTime = DateTime.UtcNow; PrintSessionStatus(session, reason.ToString()); } private void PrintSessionStatus(Session session, string reason, bool lastContact = false) { lock (session.DiagnosticsLock) { StringBuilder item = new StringBuilder(); item.AppendFormat("{0,9}:{1,20}:", reason, session.SessionDiagnostics.SessionName); if (lastContact) { item.AppendFormat("Last Event:{0:HH:mm:ss}", session.SessionDiagnostics.ClientLastContactTime.ToLocalTime()); } else { if (session.Identity != null) { item.AppendFormat(":{0,20}", session.Identity.DisplayName); } item.AppendFormat(":{0}", session.Id); } Console.WriteLine(item.ToString()); } } private async void StatusThreadAsync() { while (m_server != null) { if (DateTime.UtcNow - m_lastEventTime > TimeSpan.FromMilliseconds(6000)) { IList sessions = m_server.CurrentInstance.SessionManager.GetSessions(); for (int ii = 0; ii < sessions.Count; ii++) { Session session = sessions[ii]; PrintSessionStatus(session, "-Status-", true); } m_lastEventTime = DateTime.UtcNow; } await Task.Delay(1000).ConfigureAwait(false); } } } }