mirror of
https://github.com/finos/SymphonyElectron.git
synced 2025-01-07 06:33:58 -06:00
433 lines
23 KiB
C#
433 lines
23 KiB
C#
//css_dir ..\WixSharpToolset\;
|
|
//css_ref System.Core.dll;
|
|
//css_ref System.Windows.Forms.dll;
|
|
//css_ref Wix_bin\SDK\Microsoft.Deployment.WindowsInstaller.dll;
|
|
//css_ref WixSharp.UI.dll;
|
|
//css_imp WelcomeDialog.cs;
|
|
//css_imp WelcomeDialog.designer.cs;
|
|
//css_imp CloseDialog.cs;
|
|
//css_imp CloseDialog.designer.cs;
|
|
//css_imp ProgressDialog.cs
|
|
//css_imp ProgressDialog.Designer.cs
|
|
//css_imp ExitDialog.cs;
|
|
//css_imp ExitDialog.designer.cs;
|
|
//css_imp MaintenanceDialog.cs;
|
|
//css_imp MaintenanceDialog.designer.cs;
|
|
|
|
using WixSharp;
|
|
using WixSharp.Forms;
|
|
using Microsoft.Deployment.WindowsInstaller;
|
|
using Microsoft.Win32;
|
|
|
|
class Script
|
|
{
|
|
static public void Main(string[] args)
|
|
{
|
|
// The name "Symphony" is used in a lot of places, for paths, shortut names and installer filename, so define it once
|
|
var productName = "Symphony";
|
|
|
|
// Create a wixsharp project instance and assign the project name to it, and a hierarchy of all files to include
|
|
// Files are taken from multiple locations, and not all files in each location should be included, which is why
|
|
// the file list is rather long and explicit. At some point we might make the `dist` folder match exactly the
|
|
// desired contents of installation, and then we can simplify this bit.
|
|
var project = new ManagedProject(productName,
|
|
new Dir(@"%ProgramFiles%\" + productName,
|
|
new File(new Id("symphony_exe"), @"..\..\..\dist\win-unpacked\Symphony.exe",
|
|
// Create two shortcuts to the main Symphony.exe file, one on the desktop and one in the program menu
|
|
new FileShortcut(productName, @"%Desktop%") { IconFile = @"..\..\..\images\icon.ico" },
|
|
new FileShortcut(productName, @"%ProgramMenu%") { IconFile = @"..\..\..\images\icon.ico" }
|
|
),
|
|
new File(@"..\..\..\dist\win-unpacked\chrome_100_percent.pak"),
|
|
new File(@"..\..\..\dist\win-unpacked\chrome_200_percent.pak"),
|
|
new File(@"..\..\..\dist\win-unpacked\d3dcompiler_47.dll"),
|
|
new File(@"..\..\..\dist\win-unpacked\ffmpeg.dll"),
|
|
new File(@"..\..\..\dist\win-unpacked\icudtl.dat"),
|
|
new File(@"..\..\..\dist\win-unpacked\libEGL.dll"),
|
|
new File(@"..\..\..\dist\win-unpacked\libGLESv2.dll"),
|
|
new File(@"..\..\..\dist\win-unpacked\LICENSE.electron.txt"),
|
|
new File(@"..\..\..\dist\win-unpacked\LICENSES.chromium.html"),
|
|
new File(@"..\..\..\dist\win-unpacked\resources.pak"),
|
|
new File(@"..\..\..\dist\win-unpacked\snapshot_blob.bin"),
|
|
new File(@"..\..\..\dist\win-unpacked\v8_context_snapshot.bin"),
|
|
new File(@"..\..\..\dist\win-unpacked\vk_swiftshader.dll"),
|
|
new File(@"..\..\..\dist\win-unpacked\vk_swiftshader_icd.json"),
|
|
new File(@"..\..\..\dist\win-unpacked\vulkan-1.dll"),
|
|
new File(@"..\..\..\dist\win-unpacked\resources\app.asar.unpacked\node_modules\screen-share-indicator-frame\ScreenShareIndicatorFrame.exe"),
|
|
new File(@"..\..\..\dist\win-unpacked\resources\app.asar.unpacked\node_modules\screen-snippet\ScreenSnippet.exe"),
|
|
new Dir(@"config",
|
|
new Files(@"..\..\..\dist\win-unpacked\config\*.*")
|
|
),
|
|
new Dir(@"dictionaries",
|
|
new Files(@"..\..\..\dist\win-unpacked\dictionaries\*.*")
|
|
),
|
|
new Dir(@"library",
|
|
new File(@"..\..\..\library\dictionary"),
|
|
new File(@"..\..\..\library\indexvalidator-x64.exe"),
|
|
new File(@"..\..\..\library\libsymphonysearch-x64.dll"),
|
|
new File(@"..\..\..\library\lz4-win-x64.exe"),
|
|
new File(@"..\..\..\library\tar-win.exe")
|
|
),
|
|
new Dir(@"locales",
|
|
new Files(@"..\..\..\node_modules\electron\dist\locales\*.*")
|
|
),
|
|
new Dir(@"resources",
|
|
new DirFiles(@"..\..\..\dist\win-unpacked\resources\*.*"),
|
|
new Dir(@"app.asar.unpacked",
|
|
new Dir(@"node_modules",
|
|
new Dir(@"@felixrieseberg\spellchecker\build\Release",
|
|
new File(@"..\..\..\dist\win-unpacked\resources\app.asar.unpacked\node_modules\@felixrieseberg\spellchecker\build\Release\spellchecker.node")
|
|
),
|
|
new Dir(@"cld\build\Release",
|
|
new File(@"..\..\..\dist\win-unpacked\resources\app.asar.unpacked\node_modules\cld\build\Release\cld.node")
|
|
),
|
|
new Dir(@"diskusage\build\Release",
|
|
new File(@"..\..\..\dist\win-unpacked\resources\app.asar.unpacked\node_modules\diskusage\build\Release\diskusage.node")
|
|
),
|
|
new Dir(@"ffi-napi\build\Release",
|
|
new File(@"..\..\..\dist\win-unpacked\resources\app.asar.unpacked\node_modules\ffi-napi\build\Release\ffi_bindings.node")
|
|
),
|
|
new Dir(@"keyboard-layout\build\Release",
|
|
new File(@"..\..\..\dist\win-unpacked\resources\app.asar.unpacked\node_modules\keyboard-layout\build\Release\keyboard-layout-manager.node")
|
|
),
|
|
new Dir(@"ref-napi\build\Release",
|
|
new File(@"..\..\..\dist\win-unpacked\resources\app.asar.unpacked\node_modules\ref-napi\build\Release\binding.node")
|
|
),
|
|
new Dir(@"spawn-rx",
|
|
new Files(@"..\..\..\dist\win-unpacked\resources\app.asar.unpacked\node_modules\spawn-rx\*.*")
|
|
),
|
|
new Dir(@"swift-search\node_modules",
|
|
new Dir(@"ffi-napi\build\Release",
|
|
new File(@"..\..\..\dist\win-unpacked\resources\app.asar.unpacked\node_modules\swift-search\node_modules\ffi-napi\build\Release\ffi_bindings.node")
|
|
),
|
|
new Dir(@"ref-napi\build\Release",
|
|
new File(@"..\..\..\dist\win-unpacked\resources\app.asar.unpacked\node_modules\swift-search\node_modules\ref-napi\build\Release\binding.node")
|
|
)
|
|
)
|
|
)
|
|
)
|
|
),
|
|
new Dir(@"swiftshader",
|
|
new Files(@"..\..\..\dist\win-unpacked\swiftshader\*.*")
|
|
)
|
|
),
|
|
|
|
// Add a launch condition to require Windows Server 2008 or later
|
|
// The property values to compare against can be found here:
|
|
// https://docs.microsoft.com/en-us/windows/win32/msi/operating-system-property-values
|
|
new LaunchCondition("VersionNT>=600 AND WindowsBuild>=6001", "OS not supported"),
|
|
|
|
// Add registry entry used by protocol handler to launch symphony when opening symphony:// URIs
|
|
new RegValue(WixSharp.RegistryHive.ClassesRoot, productName + @"\shell\open\command", "", "\"[INSTALLDIR]Symphony.exe\" \"%1\"")
|
|
);
|
|
|
|
// The build script which calls the wix# builder, will be run from a command environment which has %SYMVER% set.
|
|
// So we just extract that version string, create a Version object from it, and pass it to out project definition.
|
|
var version = System.Environment.GetEnvironmentVariable("SYMVER");
|
|
project.Version = new System.Version(version);
|
|
|
|
// To get the correct behaviour with upgrading the product, the product GUID needs to be different for every build,
|
|
// but the UpgradeCode needs to stay the same. If we wanted to make a new major version and allow it to be installed
|
|
// side-by-side with the previous version, we would generate a new UpgradeCode for the new version onwards.
|
|
// More details can be found in this stackoverflow post:
|
|
// https://stackoverflow.com/a/26344742
|
|
project.GUID = new System.Guid("{4042AD1C-90E1-4032-B6B9-2BF6A4214096}");
|
|
project.ProductId = System.Guid.NewGuid();
|
|
project.UpgradeCode = new System.Guid("{36402281-8141-4797-8A90-07CFA75EFA55}");
|
|
|
|
// Allow any versions to be upgraded/downgraded freely
|
|
project.MajorUpgradeStrategy = MajorUpgradeStrategy.Default;
|
|
project.MajorUpgradeStrategy.RemoveExistingProductAfter = Step.InstallInitialize;
|
|
project.MajorUpgradeStrategy.UpgradeVersions.Minimum = "0.0.0";
|
|
project.MajorUpgradeStrategy.UpgradeVersions.Maximum = null; // No max version limit
|
|
project.MajorUpgradeStrategy.UpgradeVersions.IncludeMaximum = true;
|
|
project.MajorUpgradeStrategy.UpgradeVersions.IncludeMinimum = true;
|
|
project.MajorUpgradeStrategy.PreventDowngradingVersions.Minimum = "0.0.0";
|
|
project.MajorUpgradeStrategy.PreventDowngradingVersions.Maximum = "0.0.0";
|
|
project.MajorUpgradeStrategy.PreventDowngradingVersions.IncludeMaximum = true;
|
|
project.MajorUpgradeStrategy.PreventDowngradingVersions.IncludeMinimum = true;
|
|
|
|
// Declare all the custom properties we want to use, and assign them default values. It is possible to override
|
|
// these when running the installer, but if not specified, the defaults will be used.
|
|
project.Properties = new[]
|
|
{
|
|
new PublicProperty("ALLUSERS", "true"),
|
|
new PublicProperty("ALWAYS_ON_TOP", "DISABLED" ),
|
|
new PublicProperty("AUTO_LAUNCH_PATH", ""),
|
|
new PublicProperty("AUTO_START", "ENABLED"),
|
|
new PublicProperty("BRING_TO_FRONT", "DISABLED"),
|
|
new PublicProperty("CUSTOM_TITLE_BAR", "ENABLED"),
|
|
new PublicProperty("DEV_TOOLS_ENABLED", "true"),
|
|
new PublicProperty("FULL_SCREEN", "true"),
|
|
new PublicProperty("LOCATION", "true"),
|
|
new PublicProperty("MEDIA", "true"),
|
|
new PublicProperty("MIDI_SYSEX", "true"),
|
|
new PublicProperty("MINIMIZE_ON_CLOSE", "ENABLED"),
|
|
new PublicProperty("NOTIFICATIONS", "true"),
|
|
new PublicProperty("OPEN_EXTERNAL", "true"),
|
|
new PublicProperty("POD_URL", "https://my.symphony.com"),
|
|
new PublicProperty("POINTER_LOCK", "true"),
|
|
new Property("MSIINSTALLPERUSER", "1"),
|
|
new Property("PROGRAMSFOLDER", System.Environment.ExpandEnvironmentVariables(@"%PROGRAMFILES%"))
|
|
};
|
|
|
|
// Define the custom actions we want to run, and at what point of the installation we want to execute them.
|
|
project.Actions = new WixSharp.Action[]
|
|
{
|
|
// InstallVariant
|
|
//
|
|
// We want to be able to display the POD URL dialog every time SDA starts after a reinstall, regardless of
|
|
// whether it is a new version or the same version, but we don't want to display it if no reinstallation
|
|
// have been done. To detect this, we always write a new GUID to the fill InstallVariant.info on every
|
|
// installation.
|
|
new ElevatedManagedAction(CustomActions.InstallVariant, Return.check, When.After, Step.InstallFiles, Condition.NOT_BeingRemoved )
|
|
{
|
|
// INSTALLDIR is a built-in property, and we need it to know which path to write the InstallVariant to
|
|
UsesProperties = "INSTALLDIR"
|
|
},
|
|
|
|
// UpdateConfig
|
|
//
|
|
// After installation, the Symphony.config file needs to be updated with values from the install properties,
|
|
// either their default values as specified above, or with the overridden value if an override was specified
|
|
// on the command line when the installer was started.
|
|
new ElevatedManagedAction(CustomActions.UpdateConfig, Return.check, When.After, Step.InstallFiles, Condition.NOT_BeingRemoved )
|
|
{
|
|
// The UpdateConfig action needs the built-in property INSTALLDIR as well as most of the custom properties
|
|
UsesProperties = "INSTALLDIR,POD_URL,MINIMIZE_ON_CLOSE,ALWAYS_ON_TOP,AUTO_START,BRING_TO_FRONT,MEDIA,LOCATION,NOTIFICATIONS,MIDI_SYSEX,POINTER_LOCK,FULL_SCREEN,OPEN_EXTERNAL,CUSTOM_TITLE_BAR,DEV_TOOLS_ENABLED,AUTO_LAUNCH_PATH"
|
|
},
|
|
|
|
// CleanRegistry
|
|
//
|
|
// We have some registry keys which are added by the SDA application when it is first launched. This custom
|
|
// action will clean up those keys on uninstall. The name/location of keys have changed between different
|
|
// versions of SDA, so we clean up all known variations, and ignore any missing ones.
|
|
new ElevatedManagedAction(CustomActions.CleanRegistry, Return.ignore, When.After, Step.RemoveFiles, Condition.BeingUninstalled ),
|
|
|
|
// Start Symphony after installation is complete
|
|
new InstalledFileAction(new Id("symphony_exe"), "", Return.asyncNoWait, When.After, Step.InstallFinalize, Condition.NOT_BeingRemoved)
|
|
};
|
|
|
|
// Use our own Symphony branded bitmap for installation dialogs
|
|
project.BannerImage = "Banner.jpg";
|
|
project.BackgroundImage = "Tabloid.jpg";
|
|
|
|
// Define our own installation flow, using a mix of custom dialogs (defined in their own files) and built-in dialogs
|
|
project.ManagedUI = new ManagedUI();
|
|
project.ManagedUI.InstallDialogs.Add<Symphony.WelcomeDialog>()
|
|
.Add<Symphony.CloseDialog>()
|
|
.Add<Symphony.ProgressDialog>()
|
|
.Add<Symphony.ExitDialog>();
|
|
project.ManagedUI.ModifyDialogs.Add<Symphony.MaintenanceDialog>()
|
|
.Add<Symphony.CloseDialog>()
|
|
.Add<Symphony.ProgressDialog>()
|
|
.Add<Symphony.ExitDialog>();
|
|
|
|
project.Load += project_Load;
|
|
|
|
project.ControlPanelInfo.NoRepair = true;
|
|
project.ControlPanelInfo.NoModify = true;
|
|
project.ControlPanelInfo.ProductIcon = @"..\..\..\images\icon.ico";
|
|
|
|
project.Platform = Platform.x64;
|
|
|
|
// Generate an MSI from all settings done above
|
|
Compiler.BuildMsi(project);
|
|
}
|
|
|
|
static void KillSymphonyProcesses()
|
|
{
|
|
System.Diagnostics.Process.GetProcessesByName("Symphony").ForEach(p => {
|
|
if( System.IO.Path.GetFileName(p.MainModule.FileName) =="Symphony.exe")
|
|
{
|
|
if( !p.HasExited )
|
|
{
|
|
p.Kill();
|
|
p.WaitForExit();
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
static void project_Load(SetupEventArgs e)
|
|
{
|
|
try
|
|
{
|
|
// Try to close all running symphony instances before installing
|
|
KillSymphonyProcesses();
|
|
|
|
// When force terminating Symphony, sometimes this is detected as the renderer process crashing.
|
|
// As this message would confuse users, we wait a short time and then kill all Symphony instances
|
|
// again, which would ensure that the dialog about renderer crash will not be shown.
|
|
|
|
KillSymphonyProcesses(); // Try again immediately, for fast systems which shows dialog quickly
|
|
System.Threading.Thread.Sleep(500);
|
|
KillSymphonyProcesses(); // Try yet again after half a second for slow systems which takes time to show dialog
|
|
|
|
// Try to close all running symphony instances again
|
|
System.Diagnostics.Process.GetProcessesByName("Symphony").ForEach(p => {
|
|
if( System.IO.Path.GetFileName(p.MainModule.FileName) =="Symphony.exe")
|
|
{
|
|
if( !p.HasExited )
|
|
{
|
|
p.Kill();
|
|
p.WaitForExit();
|
|
}
|
|
}
|
|
});
|
|
}
|
|
catch (System.ComponentModel.Win32Exception ex)
|
|
{
|
|
// We always seem to get this specific exception triggered, but the application still close down correctly.
|
|
// The exception description is "Only part of a ReadProcessMemory or WriteProcessMemory request was completed".
|
|
// We ignore that specific exception, so as not to put false error outputs into the log.
|
|
if (ex.NativeErrorCode != 299) {
|
|
e.Session.Log("Error trying to close all Symphony instances: " + ex.ToString() );
|
|
}
|
|
}
|
|
catch (System.Exception ex)
|
|
{
|
|
e.Session.Log("Error trying to close all Symphony instances: " + ex.ToString() );
|
|
}
|
|
}
|
|
}
|
|
|
|
public class CustomActions
|
|
{
|
|
// InstallVariant custom action
|
|
[CustomAction]
|
|
public static ActionResult InstallVariant(Session session)
|
|
{
|
|
try
|
|
{
|
|
// Create the InstallVariant.info file
|
|
var installDir = session.Property("INSTALLDIR");
|
|
var filename = System.IO.Path.Combine(installDir, @"config\InstallVariant.info");
|
|
var installVariantFile = new System.IO.StreamWriter(filename);
|
|
|
|
// Generate new GUID for each time we install, and write it as a text string to the file
|
|
var guid = System.Guid.NewGuid();
|
|
installVariantFile.Write(guid.ToString());
|
|
|
|
installVariantFile.Close();
|
|
}
|
|
catch (System.Exception e)
|
|
{
|
|
session.Log("Error executing InstallVariant: " + e.ToString() );
|
|
return ActionResult.Failure;
|
|
}
|
|
return ActionResult.Success;
|
|
}
|
|
|
|
// UpdateConfig custom action
|
|
[CustomAction]
|
|
public static ActionResult UpdateConfig(Session session)
|
|
{
|
|
try
|
|
{
|
|
// Read the Symphony.config file
|
|
var installDir = session.Property("INSTALLDIR");
|
|
var filename = System.IO.Path.Combine(installDir, @"config\Symphony.config");
|
|
string data = System.IO.File.ReadAllText(filename);
|
|
|
|
// Replace all the relevant settings with values from the properties
|
|
data = ReplaceProperty(data, "url", session.Property("POD_URL"));
|
|
data = ReplaceProperty(data, "minimizeOnClose", session.Property("MINIMIZE_ON_CLOSE"));
|
|
data = ReplaceProperty(data, "alwaysOnTop", session.Property("ALWAYS_ON_TOP"));
|
|
data = ReplaceProperty(data, "launchOnStartup", session.Property("AUTO_START"));
|
|
data = ReplaceProperty(data, "bringToFront", session.Property("BRING_TO_FRONT"));
|
|
data = ReplaceProperty(data, "media", session.Property("MEDIA"));
|
|
data = ReplaceProperty(data, "geolocation", session.Property("LOCATION"));
|
|
data = ReplaceProperty(data, "notifications", session.Property("NOTIFICATIONS"));
|
|
data = ReplaceProperty(data, "midiSysex", session.Property("MIDI_SYSEX"));
|
|
data = ReplaceProperty(data, "pointerLock", session.Property("POINTER_LOCK"));
|
|
data = ReplaceProperty(data, "fullscreen", session.Property("FULL_SCREEN"));
|
|
data = ReplaceProperty(data, "openExternal", session.Property("OPEN_EXTERNAL"));
|
|
data = ReplaceProperty(data, "isCustomTitleBar", session.Property("CUSTOM_TITLE_BAR"));
|
|
data = ReplaceProperty(data, "devToolsEnabled", session.Property("DEV_TOOLS_ENABLED"));
|
|
data = ReplaceProperty(data, "autoLaunchPath", session.Property("AUTO_LAUNCH_PATH"));
|
|
|
|
// Write the contents back to the file
|
|
System.IO.File.WriteAllText(filename, data);
|
|
}
|
|
catch (System.Exception e)
|
|
{
|
|
session.Log("Error executing UpdateConfig: " + e.ToString() );
|
|
return ActionResult.Failure;
|
|
}
|
|
return ActionResult.Success;
|
|
}
|
|
|
|
// Helper function called by UpdadeConfig action, for each config file value that needs to be
|
|
// replaced by a value taken from the property. `data` is the entire contents of the config file.
|
|
// `name` is the name of the setting in the config file (for example "url" or "minimizeOnClose".
|
|
// `value` is the value to insert for the setting, and needs to be grabbed from the propery
|
|
// collection before calling the function. The function returns the full config file content with
|
|
// the requested replacement performed.
|
|
static string ReplaceProperty( string data, string name, string value )
|
|
{
|
|
// Using regular expressions to replace the existing value in the config file with the
|
|
// one from the property. This is the same as the regex we used to have in the old
|
|
// Advanced Installer, which looked like this: "url"\s*:\s*".*" => "url": "[POD_URL]"
|
|
return System.Text.RegularExpressions.Regex.Replace(data, @"""" + name + @"""\s*:\s*"".*""",
|
|
@"""" + name + @""":""" + value.Trim() + @"""");
|
|
}
|
|
|
|
// CleanRegistry custom action
|
|
[CustomAction]
|
|
public static ActionResult CleanRegistry(Session session)
|
|
{
|
|
try
|
|
{
|
|
// Remove registry keys added for auto-launch
|
|
|
|
using( var key = Registry.CurrentUser.OpenSubKey(@"Software\Microsoft\Windows\CurrentVersion\Run", true) )
|
|
{
|
|
if (key != null)
|
|
{
|
|
key.DeleteValue("Symphony", false);
|
|
key.DeleteValue("com.symphony.electron-desktop", false);
|
|
key.DeleteValue("electron.app.Symphony", false);
|
|
}
|
|
}
|
|
using( var key = Registry.Users.OpenSubKey(@"\.DEFAULT\Software\Microsoft\Windows\CurrentVersion\Run", true) )
|
|
{
|
|
if (key != null)
|
|
{
|
|
key.DeleteValue("Symphony", false);
|
|
}
|
|
}
|
|
|
|
|
|
// Remove registry keys added by protocol handlers
|
|
|
|
using( var key = Registry.CurrentUser.OpenSubKey(@"Software\Classes", true) )
|
|
{
|
|
if (key != null)
|
|
{
|
|
key.DeleteSubKeyTree("symphony", false);
|
|
}
|
|
}
|
|
using( var key = Registry.LocalMachine.OpenSubKey(@"Software\Classes", true) )
|
|
{
|
|
if (key != null)
|
|
{
|
|
key.DeleteSubKeyTree("symphony", false);
|
|
}
|
|
}
|
|
using( var key = Registry.ClassesRoot.OpenSubKey(@"\", true) )
|
|
{
|
|
if (key != null)
|
|
{
|
|
key.DeleteSubKeyTree("symphony", false);
|
|
}
|
|
}
|
|
}
|
|
catch (System.Exception e)
|
|
{
|
|
session.Log("Error executing CleanRegistry: " + e.ToString() );
|
|
return ActionResult.Success;
|
|
}
|
|
return ActionResult.Success;
|
|
}
|
|
}
|