// Copyright (C) Stichting Deltares 2017. All rights reserved. // // This file is part of Ringtoets. // // Ringtoets is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // // You should have received a copy of the GNU General Public License // along with this program. If not, see . // // All names, logos, and references to "Deltares" are registered trademarks of // Stichting Deltares and remain full property of Stichting Deltares at all times. // All rights reserved. using System; using System.Collections.Generic; using System.Configuration; using System.Diagnostics; using System.DirectoryServices.AccountManagement; using System.Globalization; using System.IO; using System.Linq; using System.Reflection; using System.Text.RegularExpressions; using System.Threading; using System.Windows; using System.Windows.Controls.Primitives; using System.Windows.Forms; using Application.Ringtoets.Migration; using Application.Ringtoets.Storage; using Core.Common.Gui; using Core.Common.Gui.Appenders; using Core.Common.Gui.Forms.MainWindow; using Core.Common.Gui.Settings; using Core.Common.Utils; using Core.Common.Utils.Settings; using Core.Plugins.Chart; using Core.Plugins.CommonTools; using Core.Plugins.Map; using Core.Plugins.ProjectExplorer; using log4net; using log4net.Appender; using Ringtoets.ClosingStructures.Plugin; using Ringtoets.DuneErosion.Plugin; using Ringtoets.GrassCoverErosionInwards.Plugin; using Ringtoets.GrassCoverErosionOutwards.Plugin; using Ringtoets.HeightStructures.Plugin; using Ringtoets.Integration.Data; using Ringtoets.Integration.Forms; using Ringtoets.Integration.Plugin; using Ringtoets.MacroStabilityInwards.Plugin; using Ringtoets.Piping.Plugin; using Ringtoets.StabilityPointStructures.Plugin; using Ringtoets.StabilityStoneCover.Plugin; using Ringtoets.WaveImpactAsphaltCover.Plugin; using CoreCommonGuiResources = Core.Common.Gui.Properties.Resources; using MessageBox = System.Windows.MessageBox; #if INCLUDE_DEMOPROJECT using Demo.Ringtoets.GUIs; #endif namespace Application.Ringtoets { /// /// Interaction logic for App.xaml. /// public partial class App { // Start application after this process will exit (used during restart) private const string argumentWaitForProcess = "--wait-for-process="; private const int numberOfDaysToKeepLogFiles = 30; private static readonly ILog log = LogManager.GetLogger(typeof(App)); private static GuiCore gui; private static int waitForProcessId = -1; private static string fileToOpen = string.Empty; private static Mutex singleInstanceMutex; static App() { SettingsHelper.Instance = new RingtoetsSettingsHelper(); SetLanguage(); string userDisplay = UserDisplay(); log.Info(string.Format(CoreCommonGuiResources.App_Starting_Ringtoets_version_0_by_user_0, SettingsHelper.Instance.ApplicationVersion, userDisplay)); } private delegate void ExceptionDelegate(Exception exception); protected override void OnExit(ExitEventArgs e) { singleInstanceMutex?.ReleaseMutex(); base.OnExit(e); } /// /// Runs the main Ringtoets application. /// private static void RunRingtoets() { string loaderDirectory = Path.GetDirectoryName(Assembly.GetExecutingAssembly().Location); if (loaderDirectory != null) { Environment.CurrentDirectory = loaderDirectory; } System.Windows.Forms.Application.SetUnhandledExceptionMode(UnhandledExceptionMode.CatchException, true); // handle exception from UI thread System.Windows.Forms.Application.ThreadException += Application_ThreadException; // handle exception from all threads except UI AppDomain.CurrentDomain.UnhandledException += AppDomain_UnhandledException; gui.Run(fileToOpen); // Ringtoets started, clean-up all possible memory GC.Collect(); GC.WaitForPendingFinalizers(); } private void App_Startup(object sender, StartupEventArgs e) { ParseArguments(e.Args); WaitForPreviousInstanceToExit(); if (ShutdownIfNotFirstInstance()) { return; } DeleteOldLogFiles(); Resources.Add(SystemParameters.MenuPopupAnimationKey, PopupAnimation.None); var settings = new GuiCoreSettings { SupportEmailAddress = "www.helpdeskwater.nl", SupportPhoneNumber = "+31 (0)88-797 7102", MainWindowTitle = "Ringtoets - St. Louis (EAP)", ManualFilePath = "..\\Gebruikershandleiding Ringtoets 17.1.1.pdf" }; var mainWindow = new MainWindow(); var projectMigrator = new RingtoetsProjectMigrator(new DialogBasedInquiryHelper(mainWindow)); gui = new GuiCore(mainWindow, new StorageSqLite(), projectMigrator, new RingtoetsProjectFactory(), settings) { Plugins = { new ProjectExplorerPlugin(), new CommonToolsPlugin(), new RingtoetsPlugin(), new ClosingStructuresPlugin(), new StabilityPointStructuresPlugin(), new WaveImpactAsphaltCoverPlugin(), new GrassCoverErosionInwardsPlugin(), new GrassCoverErosionOutwardsPlugin(), new PipingPlugin(), new HeightStructuresPlugin(), new StabilityStoneCoverPlugin(), new DuneErosionPlugin(), new MacroStabilityInwardsPlugin(), new ChartPlugin(), new MapPlugin() #if INCLUDE_DEMOPROJECT , new DemoProjectPlugin() #endif } }; RunRingtoets(); mainWindow.Show(); } private static bool ParseFileArgument(string potentialPath) { if (potentialPath.Length > 0) { try { IOUtils.ValidateFilePath(potentialPath); fileToOpen = potentialPath; return true; } catch (ArgumentException) {} } return false; } /// /// app.config has been configured to use /// to write log files to the ringtoets user data folder. This method deletes the old log files /// that have been written there. /// private void DeleteOldLogFiles() { try { IOUtils.DeleteOldFiles(GetLogFileDirectory(), "*.log", numberOfDaysToKeepLogFiles); } catch (Exception e) { if (e is ArgumentException || e is IOException) { return; } throw; } } private bool ShutdownIfNotFirstInstance() { var hasMutex = false; try { if (!Debugger.IsAttached) { if (!AcquireSingleInstancePerUserMutex()) { MessageBox.Show(CoreCommonGuiResources.App_ShutdownIfNotFirstInstance_Cannot_start_multiple_instances_of_Ringtoets_Please_close_the_other_instance_first); Shutdown(1); return true; //done here } hasMutex = true; } } finally { if (!hasMutex) { singleInstanceMutex = null; } } return false; } /// /// If variable waitForProcessId > -1, the application will hold until /// the process with that ID has exited. /// private static void WaitForPreviousInstanceToExit() { // Wait until previous version of Ringtoets has exited if (waitForProcessId == -1) { return; } try { Process process = Process.GetProcessById(waitForProcessId); process.WaitForExit(); } catch { //Ignored, because the process may already be closed } } private static bool AcquireSingleInstancePerUserMutex() { var createdNew = false; try { //include the user name in the (global) mutex to ensure we limit only the number of instances per //user, not per system (essential on for example Citrix systems). //include the application name in the mutex to ensure we are allowed to start for example 'Sobek' //and 'Morphan' side by side. string applicationName = ConfigurationManager.AppSettings.AllKeys.Contains("applicationName") ? ConfigurationManager.AppSettings["applicationName"] : string.Empty; string mutexName = string.Format("Ringtoets-single-instance-mutex-{0}-{1}", Environment.UserName, applicationName); singleInstanceMutex = new Mutex(true, mutexName, out createdNew); } catch (AbandonedMutexException) {} //might throw an abandoned mutex exception if the previous DS instance forcefully exited. return createdNew; } private static void AppDomain_UnhandledException(object sender, UnhandledExceptionEventArgs e) { Exception exception = e.ExceptionObject as Exception ?? new Exception(CoreCommonGuiResources.App_Unhandled_exception); HandleExceptionOnMainThread(exception); } private static void Application_ThreadException(object sender, ThreadExceptionEventArgs e) { HandleExceptionOnMainThread(e.Exception); } private static void HandleExceptionOnMainThread(Exception exception) { var control = (Control) gui.MainWindow.PropertyGrid; if (control != null && control.InvokeRequired) { // Invoke executes a delegate on the thread that owns _MainForms's underlying window handle. control.Invoke(new ExceptionDelegate(HandleException), exception); } else { HandleException(exception); } } private static void HandleException(Exception exception) { log.Error(CoreCommonGuiResources.App_Unhandled_exception, exception); if (gui?.MainWindow != null) { using (var exceptionDialog = new ExceptionDialog(gui.MainWindow, gui, exception) { OpenLogClicked = () => { gui.ApplicationCommands?.OpenLogFileExternal(); } }) { if (exceptionDialog.ShowDialog() == DialogResult.OK) { Restart(); return; } } } Environment.Exit(1); } private static void Restart() { Process.Start(typeof(App).Assembly.Location, argumentWaitForProcess + Process.GetCurrentProcess().Id); Environment.Exit(1); } /// /// Parses the process' start-up parameters. /// /// List of start-up parameters. private static void ParseArguments(IEnumerable arguments) { var argumentWaitForProcessRegex = new Regex("^" + argumentWaitForProcess + @"(?\d+)$", RegexOptions.IgnoreCase); foreach (string arg in arguments) { Match match = argumentWaitForProcessRegex.Match(arg); if (match.Success) { int pid = int.Parse(match.Groups["processId"].Value); if (pid > 0) { waitForProcessId = pid; break; } } if (ParseFileArgument(arg)) { break; } } } private static void SetLanguage() { string language = ConfigurationManager.AppSettings["language"]; if (language != null) { var localMachineDateTimeFormat = (DateTimeFormatInfo) Thread.CurrentThread.CurrentCulture.DateTimeFormat.Clone(); localMachineDateTimeFormat.DayNames = CultureInfo.InvariantCulture.DateTimeFormat.DayNames; localMachineDateTimeFormat.MonthNames = CultureInfo.InvariantCulture.DateTimeFormat.MonthNames; localMachineDateTimeFormat.AbbreviatedDayNames = CultureInfo.InvariantCulture.DateTimeFormat.AbbreviatedDayNames; localMachineDateTimeFormat.AbbreviatedMonthGenitiveNames = CultureInfo.InvariantCulture.DateTimeFormat.AbbreviatedMonthGenitiveNames; localMachineDateTimeFormat.AbbreviatedMonthNames = CultureInfo.InvariantCulture.DateTimeFormat.AbbreviatedMonthNames; var cultureInfo = new CultureInfo(language) { NumberFormat = Thread.CurrentThread.CurrentCulture.NumberFormat, DateTimeFormat = localMachineDateTimeFormat }; Thread.CurrentThread.CurrentUICulture = cultureInfo; Thread.CurrentThread.CurrentCulture = cultureInfo; } } private static string UserDisplay() { try { return string.Format("{0} ({1})", UserPrincipal.Current.DisplayName, UserPrincipal.Current.SamAccountName); } catch (SystemException) { // Cannot only catch specified exceptions, as there are some hidden exception // that can be thrown when calling UserPrincipal.Current. return Environment.UserName; } } private static string GetLogFileDirectory() { FileAppender fileAppender = LogManager.GetAllRepositories() .SelectMany(r => r.GetAppenders()) .OfType() .FirstOrDefault(); return string.IsNullOrWhiteSpace(fileAppender?.File) ? string.Empty : Path.GetDirectoryName(fileAppender.File); } } }