using System; using System.Collections.Generic; using System.Diagnostics; using System.IO; 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 _tools = new List(); 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 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 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, "Not Found", "application/xml", 404); } public void ServeResource(HttpListenerContext context, byte[] data, string mimeType = "text/html", int statusCode = 200) { string xmlHeader = ""; if (data == null) { data = Encoding.UTF8.GetBytes(xmlHeader + "\r\nCould not find the resource."); 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 = ""; if (data == null) { data = xmlHeader + "\r\nCould not find the resource."; mimeType = "application/xml"; statusCode = 404; } else if (mimeType == "application/xml" && !data.StartsWith("