using System; using System.Collections.Generic; using System.Diagnostics; using System.IO; using System.IO.Compression; using System.Net; using System.Net.Http; using System.Reflection; using System.Security.Cryptography; using System.Text; using System.Threading; using System.Threading.Tasks; using System.Windows.Forms; namespace WelsonJS.Launcher { public class ResourceServer { private readonly HttpListener _listener; private CancellationTokenSource _cts; private Task _serverTask; private bool _isRunning; private string _prefix; private string _resourceName; private List<IResourceTool> _tools = new List<IResourceTool>(); private const int _blobTimeout = 5000; private readonly HttpClient _blobClient = new HttpClient(); private readonly string _defaultMimeType = "application/octet-stream"; public ResourceServer(string prefix, string resourceName) { _prefix = prefix; _listener = new HttpListener(); _listener.Prefixes.Add(prefix); _resourceName = resourceName; _blobClient.Timeout = TimeSpan.FromMilliseconds(_blobTimeout); // Add resource tools _tools.Add(new ResourceTools.Completion(this)); _tools.Add(new ResourceTools.Config(this)); _tools.Add(new ResourceTools.DevTools(this)); _tools.Add(new ResourceTools.DnsQuery(this)); _tools.Add(new ResourceTools.Tfa(this)); _tools.Add(new ResourceTools.Whois(this)); } public string GetPrefix() { return _prefix; } public void Start() { if (_isRunning) return; _isRunning = true; _cts = new CancellationTokenSource(); _listener.Start(); // Open the web browser Program.OpenWebBrowser(_prefix); // Run a task with cancellation token _serverTask = Task.Run(() => ListenLoop(_cts.Token)); } public void Stop() { _isRunning = false; _cts.Cancel(); _listener.Stop(); MessageBox.Show("Server stopped."); } public bool IsRunning() { return _isRunning; } private async Task ListenLoop(CancellationToken token) { while (!token.IsCancellationRequested && _isRunning) { try { await ProcessRequest(await _listener.GetContextAsync()); } catch (Exception ex) { if (token.IsCancellationRequested || !_isRunning) break; MessageBox.Show($"Error: {ex.Message}"); } } } private async Task ProcessRequest(HttpListenerContext context) { string path = context.Request.Url.AbsolutePath.TrimStart('/'); // Serve from a resource name if (String.IsNullOrEmpty(path)) { ServeResource(context, GetResource(_resourceName), "text/html"); return; } // Serve the favicon.ico file if ("favicon.ico".Equals(path, StringComparison.OrdinalIgnoreCase)) { ServeResource(context, GetResource("favicon"), "image/x-icon"); return; } // Serve from a resource tool foreach (var tool in _tools) { if (tool.CanHandle(path)) { await tool.HandleAsync(context, path); return; } } // Serve from the blob server if (await ServeBlob(context, path)) return; // Fallback to serve from a resource name ServeResource(context, GetResource(_resourceName), "text/html"); } private async Task<bool> ServeBlob(HttpListenerContext context, string path) { byte[] data; string mimeType; // Try serve data from the cached blob if (TryGetCachedBlob(path, out mimeType, true)) { if (TryGetCachedBlob(path, out data)) { if (String.IsNullOrEmpty(mimeType)) { mimeType = _defaultMimeType; } ServeResource(context, data, mimeType); return true; } } // If not cached yet try { string blobServerPrefix = Program.GetAppConfig("BlobServerPrefix"); string url = $"{blobServerPrefix}{path}"; HttpRequestMessage request = new HttpRequestMessage(HttpMethod.Get, url); request.Headers.UserAgent.ParseAdd(context.Request.UserAgent); HttpResponseMessage response = await _blobClient.SendAsync(request); if (!response.IsSuccessStatusCode) { Trace.TraceError($"Failed to serve blob. Status: {response.StatusCode}"); return false; } data = await response.Content.ReadAsByteArrayAsync(); mimeType = response.Content.Headers.ContentType?.MediaType ?? _defaultMimeType; ServeResource(context, data, mimeType); _ = TrySaveCachedBlob(path, data, mimeType); return true; } catch (Exception ex) { Trace.TraceError($"Failed to serve blob. Exception: {ex.Message}"); return false; } } private string GetCachedBlobPath(string path) { // Get a hash from the path string hashedPath; using (MD5 md5 = MD5.Create()) { byte[] bHashedPath = md5.ComputeHash(Encoding.UTF8.GetBytes(path)); hashedPath = BitConverter.ToString(bHashedPath).Replace("-", "").ToLowerInvariant(); } // Get a sub-directory paths from the hashed path string[] subDirectoryPaths = new string[] { hashedPath.Substring(0, 2), hashedPath.Substring(2, 2), hashedPath.Substring(4, 2) }; // Return the cache path return Path.Combine(Program.GetAppDataPath(), "BlobCache", String.Join("\\", subDirectoryPaths), hashedPath); } private bool TryGetCachedBlob(string path, out byte[] data, bool isMetadata = false) { string cachePath = GetCachedBlobPath(path); if (isMetadata) { cachePath = $"{cachePath}.meta"; } try { if (File.Exists(cachePath)) { data = File.ReadAllBytes(cachePath); return true; } } catch (Exception ex) { Trace.TraceError($"Error: {ex.Message}"); } data = null; return false; } private bool TryGetCachedBlob(string path, out string data, bool isMetadata = false) { byte[] bData; if (TryGetCachedBlob(path, out bData, isMetadata)) { data = Encoding.UTF8.GetString(bData); return true; } data = null; return false; } private async Task<bool> TrySaveCachedBlob(string path, byte[] data, string mimeType) { await Task.Delay(0); try { string cachePath = GetCachedBlobPath(path); string cacheDirectory = Path.GetDirectoryName(cachePath); // Is exists the cached blob directory if (!Directory.Exists(cacheDirectory)) { Directory.CreateDirectory(cacheDirectory); } // Save the cache File.WriteAllBytes(cachePath, data); // Save the cache meta File.WriteAllBytes($"{cachePath}.meta", Encoding.UTF8.GetBytes(mimeType)); return true; } catch (Exception ex) { Trace.TraceError($"Error: {ex.Message}"); return false; } } public void ServeResource(HttpListenerContext context) { ServeResource(context, "<error>Not Found</error>", "application/xml", 404); } public void ServeResource(HttpListenerContext context, byte[] data, string mimeType = "text/html", int statusCode = 200) { string xmlHeader = "<?xml version=\"1.0\" encoding=\"UTF-8\"?>"; if (data == null) { data = Encoding.UTF8.GetBytes(xmlHeader + "\r\n<error>Could not find the resource.</error>"); mimeType = "application/xml"; statusCode = 404; } context.Response.StatusCode = statusCode; context.Response.ContentType = mimeType; context.Response.ContentLength64 = data.Length; using (Stream outputStream = context.Response.OutputStream) { outputStream.Write(data, 0, data.Length); } } public void ServeResource(HttpListenerContext context, string data, string mimeType = "text/html", int statusCode = 200) { string xmlHeader = "<?xml version=\"1.0\" encoding=\"UTF-8\"?>"; if (data == null) { data = xmlHeader + "\r\n<error>Could not find the resource.</error>"; mimeType = "application/xml"; statusCode = 404; } else if (mimeType == "application/xml" && !data.StartsWith("<?xml")) { data = xmlHeader + "\r\n" + data; } ServeResource(context, Encoding.UTF8.GetBytes(data), mimeType, statusCode); } private byte[] GetResource(string resourceName) { // Try to fetch embedded resource. byte[] data = GetEmbeddedResource(typeof(Program).Namespace + "." + resourceName); if (data != null) return data; // Fallback: Try to fetch resource from ResourceManager. return GetResourceFromManager(resourceName); } private byte[] GetEmbeddedResource(string fullResourceName) { Assembly assembly = Assembly.GetExecutingAssembly(); using (Stream stream = assembly.GetManifestResourceStream(fullResourceName)) { if (stream != null) { using (MemoryStream memoryStream = new MemoryStream()) { stream.CopyTo(memoryStream); return memoryStream.ToArray(); } } } return null; } private byte[] GetResourceFromManager(string resourceName) { object resourceObject = Properties.Resources.ResourceManager.GetObject(resourceName); switch (resourceObject) { case byte[] resourceBytes: return resourceBytes; case System.Drawing.Icon icon: return ConvertIconToBytes(icon); default: return null; } } private byte[] ConvertIconToBytes(System.Drawing.Icon icon) { using (MemoryStream memoryStream = new MemoryStream()) { icon.Save(memoryStream); return memoryStream.ToArray(); } } } }