SymphonyElectron/installer/win/WixSharpInstaller/Symphony.cs
2020-09-03 08:38:18 +02:00

251 lines
14 KiB
C#

//css_dir ..\..\..\;
//css_ref System.Core.dll;
//css_ref Wix_bin\SDK\Microsoft.Deployment.WindowsInstaller.dll;
//css_ref WixSharp.UI.dll;
using WixSharp;
using Microsoft.Deployment.WindowsInstaller;
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(@"..\..\..\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")
);
// 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);
// 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("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("FULL_SCREEN_CB", "true"),
new PublicProperty("LOCATION", "true"),
new PublicProperty("MEDIA", "true"),
new PublicProperty("MIDI_SYSEX", "true"),
new PublicProperty("MIDI_SYSEX_CB", "true"),
new PublicProperty("MINIMIZE_ON_CLOSE", "ENABLED"),
new PublicProperty("NOTIFICATIONS", "true"),
new PublicProperty("OPEN_EXTERNAL", "true"),
new PublicProperty("OPEN_EXTERNAL_CB", "true"),
new PublicProperty("POD_URL", "https://my.symphony.com"),
new PublicProperty("POINTER_LOCK", "true"),
new PublicProperty("POINTER_LOCK_CB", "true")
};
// 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_Installed )
{
// 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_Installed )
{
// 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"
}
};
// Generate an MSI from all settings done above
Compiler.BuildMsi(project);
}
}
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() + @"""");
}
}