using System ;
using System.Collections.Generic ;
using System.Configuration ;
using System.Diagnostics ;
using System.IO ;
using System.Net ;
using System.Threading ;
using System.Threading.Tasks ;
using System.Windows ;
using System.Linq ;
using Ionic.Zip ;
using MediaBrowser.Installer.Code ;
using Microsoft.Win32 ;
using ServiceStack.Text ;
namespace MediaBrowser.Installer
{
/// <summary>
/// Interaction logic for MainWindow.xaml
/// </summary>
public partial class MainWindow : Window
{
protected PackageVersionClass PackageClass = PackageVersionClass . Release ;
protected Version RequestedVersion = new Version ( 4 , 0 , 0 , 0 ) ;
protected Version ActualVersion ;
protected string PackageName = "MBServer" ;
protected string RootSuffix = "-Server" ;
protected string TargetExe = "MediaBrowser.ServerApplication.exe" ;
protected string TargetArgs = "" ;
protected string FriendlyName = "Media Browser Server" ;
protected string Archive = null ;
protected bool InstallPismo = true ;
protected string RootPath = Path . Combine ( Environment . GetFolderPath ( Environment . SpecialFolder . CommonApplicationData ) , "MediaBrowser-Server" ) ;
protected string EndInstallPath = Path . Combine ( Environment . GetFolderPath ( Environment . SpecialFolder . CommonApplicationData ) , "MediaBrowser-Server" ) ;
protected bool IsUpdate = false ;
protected bool SystemClosing = false ;
protected string TempLocation = Path . Combine ( Path . GetTempPath ( ) , "MediaBrowser" ) ;
protected WebClient MainClient = new WebClient ( ) ;
public MainWindow ( )
{
try
{
GetArgs ( ) ;
InitializeComponent ( ) ;
DoInstall ( Archive ) ;
}
catch ( Exception e )
{
MessageBox . Show ( "Error: " + e . Message + " \n\n" + e . StackTrace ) ;
}
}
private void btnCancel_Click ( object sender , RoutedEventArgs e )
{
this . Close ( ) ;
}
protected override void OnClosing ( System . ComponentModel . CancelEventArgs e )
{
if ( ! SystemClosing & & MessageBox . Show ( "Cancel Installation - Are you sure?" , "Cancel" , MessageBoxButton . YesNo ) = = MessageBoxResult . No )
{
e . Cancel = true ;
}
if ( MainClient . IsBusy )
{
MainClient . CancelAsync ( ) ;
while ( MainClient . IsBusy )
{
// wait to finish
}
}
MainClient . Dispose ( ) ;
ClearTempLocation ( TempLocation ) ;
base . OnClosing ( e ) ;
}
protected void SystemClose ( string message = null )
{
if ( message ! = null )
{
MessageBox . Show ( message , "Error" ) ;
}
SystemClosing = true ;
this . Close ( ) ;
}
protected void GetArgs ( )
{
//cmd line args should be name/value pairs like: product=server archive="c:\.." caller=34552
var cmdArgs = Environment . GetCommandLineArgs ( ) ;
var args = new Dictionary < string , string > ( StringComparer . OrdinalIgnoreCase ) ;
foreach ( var pair in cmdArgs )
{
var nameValue = pair . Split ( '=' ) ;
if ( nameValue . Length = = 2 )
{
args [ nameValue [ 0 ] ] = nameValue [ 1 ] ;
}
}
Archive = args . GetValueOrDefault ( "archive" , null ) ;
if ( args . GetValueOrDefault ( "pismo" , "true" ) = = "false" ) InstallPismo = false ;
var product = args . GetValueOrDefault ( "product" , null ) ? ? ConfigurationManager . AppSettings [ "product" ] ? ? "server" ;
PackageClass = ( PackageVersionClass ) Enum . Parse ( typeof ( PackageVersionClass ) , args . GetValueOrDefault ( "class" , null ) ? ? ConfigurationManager . AppSettings [ "class" ] ? ? "Release" ) ;
RequestedVersion = new Version ( args . GetValueOrDefault ( "version" , "4.0" ) ) ;
var callerId = args . GetValueOrDefault ( "caller" , null ) ;
if ( callerId ! = null )
{
// Wait for our caller to exit
try
{
var process = Process . GetProcessById ( Convert . ToInt32 ( callerId ) ) ;
process . WaitForExit ( ) ;
}
catch ( ArgumentException )
{
// wasn't running
}
IsUpdate = true ;
}
//MessageBox.Show(string.Format("Called with args: product: {0} archive: {1} caller: {2}", product, Archive, callerId));
switch ( product . ToLower ( ) )
{
case "mbt" :
PackageName = "MBTheater" ;
RootSuffix = "-Theater" ;
TargetExe = "MediaBrowser.UI.exe" ;
FriendlyName = "Media Browser Theater" ;
RootPath = EndInstallPath = Path . Combine ( Environment . GetFolderPath ( Environment . SpecialFolder . CommonApplicationData ) , "MediaBrowser" + RootSuffix ) ;
EndInstallPath = Path . Combine ( RootPath , "system" ) ;
break ;
case "mbc" :
PackageName = "MBClassic" ;
RootSuffix = "-WMC" ;
TargetExe = "ehshell.exe" ;
TargetArgs = @"/nostartupanimation /entrypoint:{CE32C570-4BEC-4aeb-AD1D-CF47B91DE0B2}\{FC9ABCCC-36CB-47ac-8BAB-03E8EF5F6F22}" ;
FriendlyName = "Media Browser Classic" ;
RootPath = Path . Combine ( Environment . GetFolderPath ( Environment . SpecialFolder . CommonApplicationData ) , "MediaBrowser" + RootSuffix ) ;
EndInstallPath = Path . Combine ( Environment . GetFolderPath ( Environment . SpecialFolder . Windows ) , "ehome" ) ;
break ;
default :
PackageName = "MBServer" ;
RootSuffix = "-Server" ;
TargetExe = "MediaBrowser.ServerApplication.exe" ;
FriendlyName = "Media Browser Server" ;
RootPath = Path . Combine ( Environment . GetFolderPath ( Environment . SpecialFolder . CommonApplicationData ) , "MediaBrowser" + RootSuffix ) ;
EndInstallPath = Path . Combine ( RootPath , "system" ) ;
break ;
}
}
/// <summary>
/// Execute the install process
/// </summary>
/// <returns></returns>
protected async Task DoInstall ( string archive )
{
lblStatus . Text = string . Format ( "Installing {0}..." , FriendlyName ) ;
// Determine Package version
var version = archive = = null ? await GetPackageVersion ( ) : null ;
ActualVersion = version ! = null ? version . version : new Version ( 3 , 0 ) ;
// Now try and shut down the server if that is what we are installing and it is running
var procs = Process . GetProcessesByName ( "mediabrowser.serverapplication" ) ;
var server = procs . Length > 0 ? procs [ 0 ] : null ;
if ( ! IsUpdate & & PackageName = = "MBServer" & & server ! = null )
{
lblStatus . Text = "Shutting Down Media Browser Server..." ;
using ( var client = new WebClient ( ) )
{
try
{
client . UploadString ( "http://localhost:8096/mediabrowser/System/Shutdown" , "" ) ;
try
{
server . WaitForExit ( 30000 ) ; //don't hang indefinitely
}
catch ( ArgumentException )
{
// already gone
}
}
catch ( WebException e )
{
if ( e . Status = = WebExceptionStatus . Timeout | | e . Message . StartsWith ( "Unable to connect" , StringComparison . OrdinalIgnoreCase ) ) return ; // just wasn't running
MessageBox . Show ( "Error shutting down server. Please be sure it is not running before hitting OK.\n\n" + e . Status + "\n\n" + e . Message ) ;
}
}
}
else
{
if ( ! IsUpdate & & PackageName = = "MBTheater" )
{
// Uninstalling MBT - shut it down if it is running
var processes = Process . GetProcessesByName ( "mediabrowser.ui" ) ;
if ( processes . Length > 0 )
{
lblStatus . Text = "Shutting Down Media Browser Theater..." ;
try
{
processes [ 0 ] . Kill ( ) ;
}
catch ( Exception ex )
{
MessageBox . Show ( "Unable to shutdown Media Browser Theater. Please ensure it is not running before hitting OK.\n\n" + ex . Message , "Error" ) ;
}
}
}
}
// Download if we don't already have it
if ( archive = = null )
{
lblStatus . Text = string . Format ( "Downloading {0} (version {1})..." , FriendlyName , version . versionStr ) ;
try
{
archive = await DownloadPackage ( version ) ;
}
catch ( Exception e )
{
SystemClose ( "Error Downloading Package - " + e . GetType ( ) . FullName + "\n\n" + e . Message ) ;
return ;
}
}
if ( archive = = null ) return ; //we canceled or had an error that was already reported
if ( Path . GetExtension ( archive ) = = ".msi" )
{
// Create directory for our installer log
if ( ! Directory . Exists ( RootPath ) ) Directory . CreateDirectory ( RootPath ) ;
var logPath = Path . Combine ( RootPath , "Logs" ) ;
if ( ! Directory . Exists ( logPath ) ) Directory . CreateDirectory ( logPath ) ;
// Run in silent mode and wait for it to finish
// First uninstall any previous version
lblStatus . Text = "Uninstalling any previous version..." ;
var logfile = Path . Combine ( RootPath , "logs" , "UnInstall.log" ) ;
var uninstaller = Process . Start ( "msiexec" , "/x \"" + archive + "\" /quiet /le \"" + logfile + "\"" ) ;
if ( uninstaller ! = null ) uninstaller . WaitForExit ( ) ;
// And now installer
lblStatus . Text = "Installing " + FriendlyName ;
logfile = Path . Combine ( RootPath , "logs" , "Install.log" ) ;
var installer = Process . Start ( archive , "/quiet /le \"" + logfile + "\"" ) ;
installer . WaitForExit ( ) ; // let this throw if there is a problem
}
else
{
// Extract
lblStatus . Text = "Extracting Package..." ;
try
{
ExtractPackage ( archive ) ;
// We're done with it so delete it (this is necessary for update operations)
TryDelete ( archive ) ;
}
catch ( Exception e )
{
SystemClose ( "Error Extracting - " + e . GetType ( ) . FullName + "\n\n" + e . Message ) ;
// Delete archive even if failed so we don't try again with this one
TryDelete ( archive ) ;
return ;
}
// Create shortcut
lblStatus . Text = "Creating Shortcuts..." ;
var fullPath = Path . Combine ( RootPath , "System" , TargetExe ) ;
try
{
CreateShortcuts ( fullPath ) ;
}
catch ( Exception e )
{
SystemClose ( "Error Creating Shortcut - " + e . GetType ( ) . FullName + "\n\n" + e . Message ) ;
return ;
}
// Install Pismo
if ( InstallPismo )
{
lblStatus . Text = "Installing ISO Support..." ;
try
{
PismoInstall ( ) ;
}
catch ( Exception e )
{
SystemClose ( "Error Installing ISO support - " + e . GetType ( ) . FullName + "\n\n" + e . Message ) ;
}
}
// Now delete the pismo install files
Directory . Delete ( Path . Combine ( RootPath , "Pismo" ) , true ) ;
}
// And run
lblStatus . Text = string . Format ( "Starting {0}..." , FriendlyName ) ;
try
{
Process . Start ( Path . Combine ( EndInstallPath , TargetExe ) , TargetArgs ) ;
}
catch ( Exception e )
{
SystemClose ( "Error Executing - " + Path . Combine ( EndInstallPath , TargetExe ) + " " + TargetArgs + "\n\n" + e . GetType ( ) . FullName + "\n\n" + e . Message ) ;
return ;
}
SystemClose ( ) ;
}
private bool TryDelete ( string file )
{
try
{
File . Delete ( file ) ;
}
catch ( FileNotFoundException )
{
}
catch ( Exception e )
{
return false ;
}
return true ;
}
private void PismoInstall ( )
{
// Kick off the Pismo installer and wait for it to end
var pismo = new Process ( ) ;
pismo . StartInfo . WindowStyle = ProcessWindowStyle . Hidden ;
pismo . StartInfo . FileName = Path . Combine ( RootPath , "Pismo" , "pfminst.exe" ) ;
pismo . StartInfo . Arguments = "install" ;
pismo . Start ( ) ;
pismo . WaitForExit ( ) ;
}
protected async Task < PackageVersionInfo > GetPackageVersion ( )
{
try
{
// get the package information for the server
var json = await MainClient . DownloadStringTaskAsync ( "http://www.mb3admin.com/admin/service/package/retrieveAll?name=" + PackageName ) ;
var packages = JsonSerializer . DeserializeFromString < List < PackageInfo > > ( json ) ;
var version = packages [ 0 ] . versions . Where ( v = > v . classification < = PackageClass ) . OrderByDescending ( v = > v . version ) . FirstOrDefault ( v = > v . version < = RequestedVersion ) ;
if ( version = = null )
{
SystemClose ( "Could not locate download package. Aborting." ) ;
return null ;
}
return version ;
}
catch ( Exception e )
{
SystemClose ( e . GetType ( ) . FullName + "\n\n" + e . Message ) ;
}
return null ;
}
/// <summary>
/// Download our specified package to an archive in a temp location
/// </summary>
/// <returns>The fully qualified name of the downloaded package</returns>
protected async Task < string > DownloadPackage ( PackageVersionInfo version )
{
var success = false ;
var retryCount = 0 ;
var archiveFile = Path . Combine ( PrepareTempLocation ( ) , version . targetFilename ) ;
try
{
while ( ! success & & retryCount < 3 )
{
// setup download progress and download the package
MainClient . DownloadProgressChanged + = DownloadProgressChanged ;
try
{
await MainClient . DownloadFileTaskAsync ( version . sourceUrl , archiveFile ) ;
success = true ;
}
catch ( WebException e )
{
if ( e . Status = = WebExceptionStatus . RequestCanceled )
{
return null ;
}
if ( retryCount < 3 & & ( e . Status = = WebExceptionStatus . Timeout | | e . Status = = WebExceptionStatus . ConnectFailure | | e . Status = = WebExceptionStatus . ProtocolError ) )
{
Thread . Sleep ( 500 ) ; //wait just a sec
PrepareTempLocation ( ) ; //clear this out
retryCount + + ;
}
else
{
throw ;
}
}
}
return archiveFile ;
}
catch ( Exception e )
{
SystemClose ( e . GetType ( ) . FullName + "\n\n" + e . Message ) ;
}
return "" ;
}
void DownloadProgressChanged ( object sender , DownloadProgressChangedEventArgs e )
{
rectProgress . Width = ( this . Width * e . ProgressPercentage ) / 100f ;
}
/// <summary>
/// Extract the provided archive to our program root
/// It is assumed the archive is a zip file relative to that root (with all necessary sub-folders)
/// </summary>
/// <param name="archive"></param>
protected void ExtractPackage ( string archive )
{
// Delete old content of system
var systemDir = Path . Combine ( RootPath , "System" ) ;
var backupDir = Path . Combine ( RootPath , "System.old" ) ;
if ( Directory . Exists ( systemDir ) )
{
try
{
if ( Directory . Exists ( backupDir ) ) Directory . Delete ( backupDir , true ) ;
}
catch ( Exception e )
{
throw new ApplicationException ( "Could not delete previous backup directory.\n\n" + e . Message ) ;
}
try
{
Directory . Move ( systemDir , backupDir ) ;
}
catch ( Exception e )
{
throw new ApplicationException ( "Could not move system directory to backup.\n\n" + e . Message ) ;
}
}
// And extract
var retryCount = 0 ;
var success = false ;
while ( ! success & & retryCount < 3 )
{
try
{
using ( var fileStream = File . OpenRead ( archive ) )
{
using ( var zipFile = ZipFile . Read ( fileStream ) )
{
zipFile . ExtractAll ( RootPath , ExtractExistingFileAction . OverwriteSilently ) ;
success = true ;
}
}
}
catch ( Exception e )
{
if ( retryCount < 3 )
{
Thread . Sleep ( 250 ) ;
retryCount + + ;
}
else
{
//Rollback
RollBack ( systemDir , backupDir ) ;
TryDelete ( archive ) ; // so we don't try again if its an update
throw new ApplicationException ( string . Format ( "Could not extract {0} to {1} after {2} attempts.\n\n{3}" , archive , RootPath , retryCount , e . Message ) ) ;
}
}
}
}
protected void RollBack ( string systemDir , string backupDir )
{
if ( Directory . Exists ( backupDir ) )
{
if ( Directory . Exists ( systemDir ) ) Directory . Delete ( systemDir ) ;
Directory . Move ( backupDir , systemDir ) ;
}
}
/// <summary>
/// Create a shortcut in the current user's start menu
/// Only do current user to avoid need for admin elevation
/// </summary>
/// <param name="targetExe"></param>
protected void CreateShortcuts ( string targetExe )
{
// get path to all users start menu
var startMenu = Path . Combine ( Environment . GetFolderPath ( Environment . SpecialFolder . StartMenu ) , "Media Browser 3" ) ;
if ( ! Directory . Exists ( startMenu ) ) Directory . CreateDirectory ( startMenu ) ;
var product = new ShellShortcut ( Path . Combine ( startMenu , FriendlyName + ".lnk" ) ) { Path = targetExe , Description = "Run " + FriendlyName } ;
product . Save ( ) ;
if ( PackageName = = "MBServer" )
{
var path = Path . Combine ( startMenu , "MB Dashboard.lnk" ) ;
var dashboard = new ShellShortcut ( path )
{ Path = @"http://localhost:8096/mediabrowser/dashboard/dashboard.html" , Description = "Open the Media Browser Server Dashboard (configuration)" } ;
dashboard . Save ( ) ;
}
CreateUninstaller ( Path . Combine ( Path . GetDirectoryName ( targetExe ) ? ? "" , "MediaBrowser.Uninstaller.exe" ) + " " + ( PackageName = = "MBServer" ? "server" : "mbt" ) , targetExe ) ;
}
/// <summary>
/// Create uninstall entry in add/remove
/// </summary>
/// <param name="uninstallPath"></param>
/// <param name="targetExe"></param>
private void CreateUninstaller ( string uninstallPath , string targetExe )
{
var parent = Registry . CurrentUser . OpenSubKey ( @"SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall" , true ) ;
{
if ( parent = = null )
{
var rootParent = Registry . CurrentUser . OpenSubKey ( @"SOFTWARE\Microsoft\Windows\CurrentVersion" , true ) ;
{
if ( rootParent ! = null )
{
parent = rootParent . CreateSubKey ( "Uninstall" ) ;
if ( parent = = null )
{
MessageBox . Show ( "Unable to create Uninstall registry key. Program is still installed sucessfully." ) ;
return ;
}
}
}
}
try
{
RegistryKey key = null ;
try
{
const string guidText = "{4E76DB4E-1BB9-4A7B-860C-7940779CF7A0}" ;
key = parent . OpenSubKey ( guidText , true ) ? ?
parent . CreateSubKey ( guidText ) ;
if ( key = = null )
{
MessageBox . Show ( String . Format ( "Unable to create uninstaller entry'{0}\\{1}'. Program is still installed successfully." , @"SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall" , guidText ) ) ;
return ;
}
key . SetValue ( "DisplayName" , FriendlyName ) ;
key . SetValue ( "ApplicationVersion" , ActualVersion ) ;
key . SetValue ( "Publisher" , "Media Browser Team" ) ;
key . SetValue ( "DisplayIcon" , targetExe ) ;
key . SetValue ( "DisplayVersion" , ActualVersion . ToString ( 2 ) ) ;
key . SetValue ( "URLInfoAbout" , "http://www.mediabrowser3.com" ) ;
key . SetValue ( "Contact" , "http://community.mediabrowser.tv" ) ;
key . SetValue ( "InstallDate" , DateTime . Now . ToString ( "yyyyMMdd" ) ) ;
key . SetValue ( "UninstallString" , uninstallPath ) ;
}
finally
{
if ( key ! = null )
{
key . Close ( ) ;
}
}
}
catch ( Exception ex )
{
MessageBox . Show ( "An error occurred writing uninstall information to the registry." ) ;
}
}
}
/// <summary>
/// Prepare a temporary location to download to
/// </summary>
/// <returns>The path to the temporary location</returns>
protected string PrepareTempLocation ( )
{
ClearTempLocation ( TempLocation ) ;
Directory . CreateDirectory ( TempLocation ) ;
return TempLocation ;
}
/// <summary>
/// Clear out (delete recursively) the supplied temp location
/// </summary>
/// <param name="location"></param>
protected void ClearTempLocation ( string location )
{
if ( Directory . Exists ( location ) )
{
Directory . Delete ( location , true ) ;
}
}
}
}