Files
UnrealEngineUWP/Engine/Source/Programs/HTML5/HTML5LaunchHelper/Program.cs
Peter Sauerbrei 46a3007613 Copying //UE4/Dev-Mobile to //UE4/Dev-Main (Source: //UE4/Dev-Mobile @ 3496193)
#lockdown Nick.Penwarden
#rb none

=====================================
 MAJOR FEATURES + CHANGES
=====================================

Change 3385029 on 2017/04/07 by Chris.Babcock

	Remove unneeded BILLING permission for Android (it is added by enabling IAP)
	#jira UE-43583
	#ue4
	#android

Change 3388541 on 2017/04/11 by Will.Fissler

	Removed "MacNoEditor" and "WindowsNoEditor" as target platforms in the StrategyTV.uproject.

Change 3390026 on 2017/04/12 by Allan.Bentham

	Allow vertex texture reads on ES3.1 feature level
	#jira UE-43774

Change 3408788 on 2017/04/25 by Dmitriy.Dyomin

	Fixed: -iterativedeploy UAT option

Change 3418253 on 2017/05/02 by Allan.Bentham

	Enable ICF linker option in android tool chain.
	#jira UEMOB-167

Change 3426789 on 2017/05/05 by Jonathan.Fitzpatrick

	#jira UE-43518

	Fixed a missing cast to the proper game mode

Change 3427859 on 2017/05/08 by Dmitriy.Dyomin

	Avoid creating unnecessary FUniqueObjectGuid in foliage (prevents package dirty on actor deletion)

Change 3428842 on 2017/05/08 by Chris.Babcock

	Fix environment variable leakage in ant.bat patch (already in 4.16, didn't make the integration/merge)
	Fix Intermediate/Android/APK/src cleanup (already in 4.16, didn't make the integration/merge)
	#ue4
	#android

Change 3432096 on 2017/05/09 by Dmitriy.Dyomin

	Android LaunchOn improvements

Change 3433937 on 2017/05/10 by Chris.Babcock

	Enable XGE on non-build machine
	#ue4
	#android

Change 3434556 on 2017/05/11 by Dmitriy.Dyomin

	Added mobile separate translucency
	#jira UEMOB-146

Change 3436664 on 2017/05/12 by Dmitriy.Dyomin

	Fixed: missing translucent objects on mobile, fallout from separate translucency

Change 3437328 on 2017/05/12 by Allan.Bentham

	Add android versions of PRAGMA_DISABLE_OPTIMIZATION_ACTUAL and PRAGMA_ENABLE_OPTIMIZATION_ACTUAL

Change 3446874 on 2017/05/18 by Chris.Babcock

	Change FGenericPlatformMemoryConstants and FGenericPlatformMemoryStats to use uint64 instead of SIZE_T to handle >4GB Android devices running in ARMv7 mode
	#jira
	#ue4
	#android

Change 3448354 on 2017/05/19 by Dmitriy.Dyomin

	Added: Support sRGB texture sampling on Android ES 3.1 and Vulkan
	#jira UEMOB-190

Change 3451129 on 2017/05/21 by Dmitriy.Dyomin

	Added project option to limit gpu skinning to 2 bone per vertex (Rendering Settings -> Optimizations -> Limit GPU skinning to 2 bones influence)
	#jira UEMOB-154

Change 3451131 on 2017/05/21 by Dmitriy.Dyomin

	Fixed: NavMesh streaming - stable tile addressing

Change 3451141 on 2017/05/21 by Dmitriy.Dyomin

	Avoid drawing quads for clears on mobile

Change 3453549 on 2017/05/23 by Dmitriy.Dyomin

	Fixed wrong memreport for STAT_TextureMemoryCube, STAT_PrecomputedLightVolumeMemory, STAT_ReflectionCaptureMemory

Change 3458488 on 2017/05/25 by Dmitriy.Dyomin

	Added RenderDoc integration for Android

Change 3458589 on 2017/05/25 by Dmitriy.Dyomin

	Fixed foliage occlusion culling after world origin was rebased

Change 3462146 on 2017/05/26 by Nick.Shin

	HTML5 - merge from Release-4.16 to Dev-Mobile

	#jira none

	#rnx

Change 3462166 on 2017/05/26 by Nick.Shin

	HTML5 - fix viewport after returning from fullscreen

	PR: 113b9ea104

	#jira UE-44419  HTML5 - View does not redraw properly after returning from Fullscreen

	#rn fix viewport after returning from fullscreen

Change 3464093 on 2017/05/28 by Jack.Porter

	Fix for GenerateProjectFiles warnings

	#codereview: Nick.Shin

Change 3465335 on 2017/05/30 by Nick.Shin

	HTML5LaunchHelper.exe - current working directory "/" check

	#jira UE-45302  HTML5LaunchHelper.exe hosts the files in the current working directory on Linux

	#rnx

Change 3465499 on 2017/05/30 by Nick.Shin

	HTML5 - TaskGraph crash fix & compiler fix when STATS disabled

	#jira UE-44811  Projects crash when launching onto Firefox 64-bit

	#rnx

Change 3468295 on 2017/05/31 by Chris.Babcock

	Allow mediaplayer audio to be disable on Android
	#jira UE-45570
	#ue4
	#android

Change 3469099 on 2017/06/01 by Dmitriy.Dyomin

	Fixing mobile separate translucency after merge

Change 3470541 on 2017/06/01 by Chris.Babcock

	Fix Android.NewKeyboard behavior
	#jira UE-45612
	#ue4
	#android

Change 3470576 on 2017/06/01 by Chris.Babcock

	Blacklist DefaultBloomKernel on mobile platforms (unneeded and takes 32MB)
	#jira UE-45548
	#ue4
	#android

Change 3471583 on 2017/06/02 by Allan.Bentham

	#jira UEMOB-361
	Add experimental mobile PIE with device preview launch option.

Change 3471708 on 2017/06/02 by Allan.Bentham

	Fixes for no unity no pch CIS build.
	Add missing #includes

Change 3474619 on 2017/06/05 by Chris.Babcock

	Add support for optional Gradle build system
	#jira UEMOB-229
	#ue4
	#android

Change 3477357 on 2017/06/07 by Dmitriy.Dyomin

	Added GLES for RenderDoc capture on Android

	#contributed by Jimmy Lee (https://github.com/Oculus-VR/UnrealEngine/pull/7)

Change 3477953 on 2017/06/07 by Nick.Shin

	HTML5 memory/executable size pass

	these fixes contains:
	+ build shipping asmjs compressed files (remove serving non-compressed data file)
	+ phsyx updated emscripten toolchain cmake config override (i.e. removed EPIC_BUILD_FLAGS match)
	+ retired "/Script/BuildSettings.BuildSettings" config code
	+ added better verbose feedback to print optimization levels during packaging

	#jira UEMOB-382  HTML5 memory/executable size pass

	#rn a lot of stability fixes

Change 3479142 on 2017/06/07 by Chris.Babcock

	Update Clang version checks and handle 3.9
	#jira UE-45812
	#ue4
	#android

Change 3479416 on 2017/06/08 by Dmitriy.Dyomin

	Fixed UBT crash introduced in CL# 3477357

Change 3479425 on 2017/06/08 by Dmitriy.Dyomin

	Fixed: CustomDepth sampling outside of PP materials on Mobile
	#jira UE-44700

Change 3479600 on 2017/06/08 by Dmitriy.Dyomin

	Do "-skipdeploy" when packaging

Change 3481938 on 2017/06/09 by Dmitriy.Dyomin

	Fixed: LG G6, Samsung Galaxy S8 Letter box issue
	#jira UE-45164

Change 3482725 on 2017/06/09 by Chris.Babcock

	Fix out of bounds access to iChild
	#jira none

Change 3482735 on 2017/06/09 by Chris.Babcock

	Support for NDK14b and start of NDK15 support (Clang 5.0)
	#jira UEMOB-240
	#ue4
	#android

Change 3484209 on 2017/06/11 by Dmitriy.Dyomin

	fixed warning introduced in CL# 3481938

Change 3484256 on 2017/06/11 by Dmitriy.Dyomin

	Fixed: HighresShot with 'Use Customdepth as mask' in Feature level ES2 (Android preview rendering level) leads to Engine crash
	#jira UE-43655

	also requires content changes in CL# 3484255

Change 3484295 on 2017/06/12 by Dmitriy.Dyomin

	Fixed: Deferred Decals move with the camera in HTML5
	#jira UE-45606

Change 3484748 on 2017/06/12 by Chris.Babcock

	Add detection of Houdini (running on Intel Android CPU emulating ARM)
	#jira UE-45934
	#ue4
	#android

Change 3484766 on 2017/06/12 by Chris.Babcock

	Add missing log message for UsingHoudini
	#jira UE-45934
	#ue4
	#android

Change 3485762 on 2017/06/12 by Chris.Babcock

	Check in Gradle TPS
	#jira none
	#ue4
	#android

Change 3486596 on 2017/06/13 by Jack.Porter

	Fixed merge error

Change 3487559 on 2017/06/13 by Peter.Sauerbrei

	disable bEnableREmoteNotifications in binary builds

	#jira UE-44156

Change 3487875 on 2017/06/13 by Peter.Sauerbrei

	make it so we don't crash if the device isn't paired

	#jira UE-38247

Change 3487949 on 2017/06/13 by Peter.Sauerbrei

	fix for casing of DotNET in several locations from PR#3112 (portaloffreedom and Madh93)

	#jira UE-40396

Change 3488155 on 2017/06/13 by Peter.Sauerbrei

	fix for intermediate being lower cased when we want mixed case, keeps commandline txt file lower
	cased (original PR#2939 from kosz78, modified from that change)

	#jira UE-38737

Change 3488428 on 2017/06/13 by Chris.Babcock

	Fix permissions on gradlew on Mac and Linux
	#jira UE-46002
	#ue4
	#android

Change 3488735 on 2017/06/13 by Dmitriy.Dyomin

	Removed MDG note about crash on none-mali devices

Change 3488961 on 2017/06/14 by Dmitriy.Dyomin

	Fixed: scene capture component was applying only default ShowFlags in game

Change 3489162 on 2017/06/14 by Jack.Porter

	Removed checkbox "Deferred Rendering with Metal" on iOS.  This feature is no longer supported and will be replaced by a Metal 2-based renderer.

	#jira UE-41766

Change 3489192 on 2017/06/14 by Peter.Sauerbrei

	hide 32-bit and OpenGL options for IOS

	#jira none

Change 3489207 on 2017/06/14 by Peter.Sauerbrei

	make the MetalMRT setting hidden instead of removed

Change 3489593 on 2017/06/14 by Jack.Porter

	Removed the Android_All cook flavor as it's deprecated in favor of Android_Multi
	#jira UE-45469

Change 3491385 on 2017/06/15 by Dmitriy.Dyomin

	Fixed: SM_FireFX Particle not rendering for various Android texture compressions
	#jira UE-46083

Change 3491402 on 2017/06/15 by Dmitriy.Dyomin

	Fixed: Static + CSM shadows cause a crash
	#jira UE-46091

Change 3493586 on 2017/06/15 by Chris.Babcock

	Remove extra > in AndroidManifest
	#jira UE-46134
	#ue4
	#android

Change 3496193 on 2017/06/16 by Chris.Babcock

	Fix DeviceProfileManager setting in WEX
	#jira UE-46176
	#ue4
	#android

[CL 3496903 by Peter Sauerbrei in Main branch]
2017-06-16 20:17:59 -04:00

592 lines
16 KiB
C#

// Copyright 1998-2017 Epic Games, Inc. All Rights Reserved.
using System;
using System.Net;
using System.Net.Sockets;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Diagnostics;
using System.IO;
using System.Reflection;
using System.Web;
namespace HTML5LaunchHelper
{
class HttpServer
{
#region extension to MIME type list
// some basic mime types, not really important but for completeness sake.
private static IDictionary<string, string> MimeTypeMapping = new Dictionary<string, string>(StringComparer.InvariantCultureIgnoreCase)
{
{".bin", "application/octet-stream"},
{".css", "text/css"},
{".dll", "application/octet-stream"},
{".dmg", "application/octet-stream"},
{".ear", "application/java-archive"},
{".eot", "application/octet-stream"},
{".exe", "application/octet-stream"},
{".flv", "video/x-flv"},
{".gif", "image/gif"},
{".hqx", "application/mac-binhex40"},
{".htc", "text/x-component"},
{".htm", "text/html"},
{".html", "text/html"},
{".ico", "image/x-icon"},
{".img", "application/octet-stream"},
{".iso", "application/octet-stream"},
{".jar", "application/java-archive"},
{".jardiff", "application/x-java-archive-diff"},
{".jng", "image/x-jng"},
{".jnlp", "application/x-java-jnlp-file"},
{".jpeg", "image/jpeg"},
{".jpg", "image/jpeg"},
{".js", "application/x-javascript"},
{".mml", "text/mathml"},
{".mng", "video/x-mng"},
{".mov", "video/quicktime"},
{".mp3", "audio/mpeg"},
{".mpeg", "video/mpeg"},
{".mpg", "video/mpeg"},
{".msi", "application/octet-stream"},
{".msm", "application/octet-stream"},
{".msp", "application/octet-stream"},
{".pdb", "application/x-pilot"},
{".pdf", "application/pdf"},
{".pem", "application/x-x509-ca-cert"},
{".pl", "application/x-perl"},
{".pm", "application/x-perl"},
{".png", "image/png"},
{".prc", "application/x-pilot"},
{".ra", "audio/x-realaudio"},
{".rar", "application/x-rar-compressed"},
{".rpm", "application/x-redhat-package-manager"},
{".rss", "text/xml"},
{".run", "application/x-makeself"},
{".sea", "application/x-sea"},
{".shtml", "text/html"},
{".sit", "application/x-stuffit"},
{".swf", "application/x-shockwave-flash"},
{".tcl", "application/x-tcl"},
{".tk", "application/x-tcl"},
{".txt", "text/plain"},
{".war", "application/java-archive"},
{".wbmp", "image/vnd.wap.wbmp"},
{".wmv", "video/x-ms-wmv"},
{".xml", "text/xml"},
{".xpi", "application/x-xpinstall"},
{".zip", "application/zip"},
};
#endregion
private HttpListener WebServer = new HttpListener();
private string Root;
public HttpServer(int Port, string ServerRoot, bool UseAllPrefixes)
{
Root = ServerRoot;
WebServer.Prefixes.Add(string.Format("http://localhost:{0}/", Port.ToString()));
if (UseAllPrefixes)
{
WebServer.Prefixes.Add (string.Format ("http://127.0.0.1:{0}/", Port.ToString ()));
WebServer.Prefixes.Add (string.Format ("http://{0}:{1}/", Environment.MachineName, Port.ToString ()));
IPHostEntry host = Dns.GetHostEntry (Dns.GetHostName ());
foreach (IPAddress ip in host.AddressList) {
if (ip.AddressFamily == AddressFamily.InterNetwork) {
WebServer.Prefixes.Add (string.Format ("http://{0}:{1}/", ip.ToString (), Port.ToString ()));
}
}
}
}
public void Run()
{
System.Console.WriteLine("Starting Server at " + WebServer.Prefixes.First().ToString());
WebServer.Start();
Task.Factory.StartNew(()
=>
{
while( WebServer.IsListening)
{
// Handle requests in threaded mode.
Task.Factory.StartNew((Ctx)
=>
{
var Context = Ctx as HttpListenerContext;
try
{
RequestHandler(Context);
}
catch { }
finally
{
Context.Response.Close();
}
}, WebServer.GetContext());
}
}
);
}
private void RequestHandler(HttpListenerContext Context)
{
if (Directory.Exists(Root + Context.Request.Url.LocalPath))
{
// Process the list of files found in the directory.
string[] fileEntries = Directory.GetFileSystemEntries(Root + Context.Request.Url.LocalPath);
string Response = "<html>\n" +
"<body>\n" +
"<h2>Unreal WebServer</h2>\n" +
"<h3>Directory listing for " + Context.Request.Url.LocalPath + "</h3>\n" +
"<hr>\n";
Response += "<table>\n";
Response += "<tr>\n" +
"\t<th>Filename</th>\n"+
"\t<th>TimeStamp</th>\n"+
"</tr>\n";
foreach (string fileName in fileEntries)
{
string Slash = Directory.Exists(fileName) ? "/" : "" ;
string Url = fileName.Replace(Root, WebServer.Prefixes.First().ToString());
Response += "<tr>\n\t<td><a href=\"" + Url + "\">" + Path.GetFileName(fileName) + Slash + "</a></td>\n\t<td> " + File.GetLastAccessTime(fileName).ToString() + "</td>\n</tr>\n";
}
Response += "</table></html>";
byte[] buf = Encoding.UTF8.GetBytes(Response);
Context.Response.AddHeader("Access-Control-Allow-Origin", "*");
Context.Response.ContentLength64 = buf.Length;
Context.Response.OutputStream.Write(buf, 0, buf.Length);
Context.Response.Close();
}
else if (File.Exists(Root + Context.Request.Url.LocalPath))
{
string RequestedFile = Root + Context.Request.Url.LocalPath;
string RequestedFileCompressed = Root + Context.Request.Url.LocalPath + "gz";
if (File.Exists(RequestedFileCompressed))
{
RequestedFile = RequestedFileCompressed;
}
System.Console.WriteLine("Serving " + RequestedFile);
using (Stream source = File.OpenRead(RequestedFile))
{
string Extention = Path.GetExtension(RequestedFile);
string MimeType = "text/html";
if (MimeTypeMapping.ContainsKey(Extention))
{
MimeType = MimeTypeMapping[Extention];
}
// This is the crux of serving pre-compressed files.
if (Extention.EndsWith("gz"))
{
Context.Response.AddHeader("Content-Encoding", "gzip");
}
byte[] buffer = new byte[source.Length];
source.Read(buffer, 0, buffer.Length);
Context.Response.AddHeader("Access-Control-Allow-Origin", "*");
Context.Response.ContentType = MimeType;
Context.Response.ContentLength64 = buffer.Length;
Context.Response.OutputStream.Write(buffer, 0, buffer.Length);
Context.Response.Close();
}
}
else
{
string RequestedFile = Root + Context.Request.Url.LocalPath;
System.Console.WriteLine("Not Serving " + RequestedFile);
string Response = "<html>\n" +
"<body>\n" +
"<h2>404 Not Found</h2>\n" +
"</html>\n";
byte[] buf = Encoding.UTF8.GetBytes(Response);
Context.Response.ContentLength64 = buf.Length;
Context.Response.StatusCode = (int)HttpStatusCode.NotFound;
Context.Response.ContentType = "text/html";
Context.Response.OutputStream.Write(buf, 0, buf.Length);
Context.Response.Close();
}
}
public void Stop()
{
WebServer.Stop();
WebServer.Close();
}
}
class ArgumentName : Attribute
{
public string Name;
public ArgumentName(string _Name)
{
Name = _Name;
}
}
class DefaultArgument: Attribute
{
public string Value;
public DefaultArgument(string _Value)
{
Value = _Value;
}
}
// Various command line options supported.
class Arguments
{
// if this is set - This browser is spawned and the web server blocks till the browser quits.
// if this not set - Just the server starts up and waits for key to quit.
[ArgumentName("-Browser="), DefaultArgument("")]
public string Browser
{
get;
set;
}
[ArgumentName("-ServerRoot="), DefaultArgument("./")]
public string ServerRoot
{
get;
set;
}
[ArgumentName("-ServerPort="), DefaultArgument("8000")]
public string ServerPort
{
get;
set;
}
[ArgumentName("-BrowserCommandLine="), DefaultArgument("")]
public string BrowserCommandLine
{
get;
set;
}
[ArgumentName("-UseAllPrefixes="), DefaultArgument("FALSE")]
public string UseAllPrefixes
{
get;
set;
}
public Arguments()
{}
public bool Parse(string[] args)
{
PropertyInfo[] Infos = typeof(Arguments).GetProperties();
foreach( var Info in Infos )
{
object[] Attributes = Info.GetCustomAttributes(false);
string Name = null;
string DefaultValue = null;
foreach( var Att in Attributes)
{
if (Att.GetType() == typeof(ArgumentName))
{
Name = (Att as ArgumentName).Name;
}
if (Att.GetType() == typeof(DefaultArgument))
{
DefaultValue = (Att as DefaultArgument).Value;
}
}
bool found = false;
foreach(var arg in args)
{
if (arg.StartsWith(Name))
{
string val = arg.Replace(Name, "");
Info.SetValue(this, val);
found = true;
break;
}
}
if (!found && DefaultValue != null)
{
Info.SetValue(this, DefaultValue);
}
else if ( !found && DefaultValue == null)
{
return false;
}
}
return true;
}
public void ShowParsedValues()
{
PropertyInfo[] Infos = typeof(Arguments).GetProperties();
foreach (var Info in Infos)
{
object[] Attributes = Info.GetCustomAttributes(false);
string Name = null;
foreach (var Att in Attributes)
{
if (Att.GetType() == typeof(ArgumentName))
{
Name = (Att as ArgumentName).Name;
}
}
if (Info.GetValue(this) != null)
{
System.Console.WriteLine("Name: " + Name + " " + Info.GetValue(this).ToString());
}
}
}
public void ShowAllOptions()
{
PropertyInfo[] Infos = typeof(Arguments).GetProperties();
foreach (var Info in Infos)
{
object[] Attributes = Info.GetCustomAttributes(false);
string Name = null;
string DefaultValue = null;
foreach (var Att in Attributes)
{
if (Att.GetType() == typeof(ArgumentName))
{
Name = (Att as ArgumentName).Name;
}
if (Att.GetType() == typeof(DefaultArgument))
{
DefaultValue = (Att as DefaultArgument).Value;
}
}
System.Console.WriteLine("Option: {0}, Default Value {1}", Name, DefaultValue == null ? " None, this option is required " : DefaultValue);
}
}
}
class Program
{
static private List<Process> ProcessesToKill = new List<Process>();
static private List<Process> ProcessesToWatch = new List<Process>();
static bool IsRunningOnMac()
{
PlatformID Platform = Environment.OSVersion.Platform;
switch (Platform)
{
case PlatformID.Unix:
return System.IO.File.Exists("/System/Library/CoreServices/SystemVersion.plist");
case PlatformID.MacOSX:
return true;
}
return false;
}
static Process SpawnBrowserProcess(string bpath, string args)
{
var bIsSafari = bpath.Contains("Safari");
var Result = new Process();
if (IsRunningOnMac())
{
string BrowserArgs = bIsSafari ? "" : args;
Result.StartInfo.FileName = "/usr/bin/open";
Result.StartInfo.UseShellExecute = false;
Result.StartInfo.RedirectStandardOutput = true;
Result.StartInfo.RedirectStandardInput = true;
Result.StartInfo.Arguments = String.Format("-nW \"{0}\" --args {1}", bpath, BrowserArgs);
Result.EnableRaisingEvents = true;
}
else
{
Result.StartInfo.FileName = bpath;
Result.StartInfo.UseShellExecute = false;
Result.StartInfo.RedirectStandardOutput = true;
Result.StartInfo.RedirectStandardInput = true;
Result.StartInfo.Arguments = args;
Result.EnableRaisingEvents = true;
}
Result.Start();
if (bIsSafari)
{
// Give Safari time to open...
System.Threading.Thread.Sleep(1500);
var Proc = new Process();
Proc.StartInfo.FileName = "/usr/bin/osascript";
Proc.StartInfo.UseShellExecute = false;
Proc.StartInfo.RedirectStandardOutput = true;
Proc.StartInfo.RedirectStandardInput = true;
Proc.StartInfo.Arguments = String.Format("-e 'tell application \"Safari\" to open location \"{0}\"'", args);
Proc.EnableRaisingEvents = true;
Proc.Start();
Proc.WaitForExit();
}
System.Console.WriteLine("Spawning Browser Process {0} with args {1}\n", bpath, args);
return Result;
}
static void SpawnBrowserAndBlock(Arguments Args)
{
// Browsers can be multiprocess programs (Chrome, basically)
bool bMultiprocessBrowser = Args.Browser.Contains("chrome");
// so we need to catch spawning of other child processes. The trick is
// they aren't really child-processes at all. There appears to be no real binding between the two,
// So we kind of fudged it a bit here.
var PrevProcesses = Process.GetProcesses();
var FirstProcess = SpawnBrowserProcess(Args.Browser, Args.BrowserCommandLine);
ProcessesToWatch.Add(FirstProcess);
var ProcName = FirstProcess.ProcessName;
// We should now have a list of processes to watch to exit.
// Loop over the calling WaitForExit() until the list is empty.
while (ProcessesToWatch.Count() > 0)
{
for (var i=0; i<ProcessesToWatch.Count(); ++i)
{
if (ProcessesToWatch[i].HasExited)
{
ProcessesToWatch.RemoveAt(i);
}
}
if (bMultiprocessBrowser && FirstProcess != null && FirstProcess.HasExited)
{
var CurrentProcesses = Process.GetProcesses();
foreach (var item in CurrentProcesses)
{
var bWasAlive = false;
foreach (var pitem in PrevProcesses)
{
if (pitem.Id == item.Id)
{
bWasAlive = true;
}
}
if (!bWasAlive)
{
try
{
if (!item.HasExited && item.ProcessName.StartsWith(ProcName))
{
var PID = item.Id;
System.Console.WriteLine("Found Process {0} with PID {1} which started at {2}. Waiting on that process to end.", item.ProcessName, item.Id, item.StartTime.ToString());
ProcessesToWatch.Add(item);
}
}
catch { }
}
}
FirstProcess = null;
}
// It is considered an error if any service processes have died.
foreach (var proc in ProcessesToKill)
{
if (proc.HasExited)
{
System.Console.WriteLine("A spawned thread has died. Do you have a python server instance running?");
HardShutdown();
}
}
System.Threading.Thread.Sleep(250);
}
System.Console.WriteLine("All processes being watched have exited.\n");
// All processes we cared about have finished, So it is time to clean up the services we spawned for them.
foreach (var proc in ProcessesToKill)
{
if (!proc.HasExited)
{
System.Console.WriteLine("Killing spawned process {0}.\n", proc.Id);
proc.Kill();
proc.WaitForExit();
}
}
}
static void HardShutdown()
{
foreach (var proc in ProcessesToKill)
{
if (!proc.HasExited)
{
proc.Kill();
proc.WaitForExit();
}
}
foreach (var proc in ProcessesToWatch)
{
if (!proc.HasExited)
{
proc.Kill();
proc.WaitForExit();
}
}
}
static int Main(string[] args)
{
var Args = new Arguments();
if (Args.Parse(args))
{
string cwd = Directory.GetCurrentDirectory();
if ( Args.ServerRoot.Equals("./") && cwd.Equals("/") ) // UE-45302
{
string path = System.IO.Path.GetDirectoryName( System.Reflection.Assembly.GetExecutingAssembly().GetName().CodeBase ).Replace("file:","");
Directory.SetCurrentDirectory(path);
}
Args.ShowParsedValues();
}
else
{
System.Console.WriteLine("Incorrect Command line Options.. Exiting");
Args.ShowAllOptions();
return 0;
}
var Server = new HttpServer(Convert.ToInt32(Args.ServerPort),Args.ServerRoot, Args.UseAllPrefixes == "FALSE" ? false : true );
Server.Run();
if ( Args.Browser != "" )
{
if ((!File.Exists(Args.Browser) && !IsRunningOnMac()) || (!Directory.Exists(Args.Browser) && IsRunningOnMac()))
{
System.Console.WriteLine("Browser Not found, Please check -Browser= option");
return 0;
}
SpawnBrowserAndBlock(Args);
}
else
{
System.Console.WriteLine("Press Any key Quit Server");
System.Console.ReadKey();
}
Server.Stop();
return 0;
}
}
}