using System; using System.Collections.Generic; using System.IO; using System.Linq; using System.Net; using System.Net.Http; using System.Reflection; using System.Text; using System.Threading; using System.Threading.Tasks; using System.Windows.Forms; using System.Xml; using System.Xml.Linq; 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 ExecutablesCollector _executablesCollector; public ResourceServer(string prefix, string resourceName) { _prefix = prefix; _listener = new HttpListener(); _listener.Prefixes.Add(prefix); _resourceName = resourceName; _executablesCollector = new ExecutablesCollector(); } 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 { ProcessRequest(await _listener.GetContextAsync()); } catch (Exception ex) { if (token.IsCancellationRequested || !_isRunning) break; MessageBox.Show($"Error: {ex.Message}"); } } } private void ProcessRequest(HttpListenerContext context) { string path = context.Request.Url.AbsolutePath.TrimStart('/'); // Serve the favicon.ico file if ("favicon.ico".Equals(path, StringComparison.OrdinalIgnoreCase)) { ServeResource(context, GetResource("favicon"), "image/x-icon"); return; } // Serve the code completion (word suggestion) const string completionPrefix = "completion/"; if (path.StartsWith(completionPrefix, StringComparison.OrdinalIgnoreCase)) { ServeCompletion(context, path.Substring(completionPrefix.Length)); return; } // Serve the DevTools Protocol const string devtoolsPrefix = "devtools/"; if (path.StartsWith(devtoolsPrefix, StringComparison.OrdinalIgnoreCase)) { ServeDevTools(context, path.Substring(devtoolsPrefix.Length - 1)); return; } // Serve a resource ServeResource(context, GetResource(_resourceName), "text/html"); } private void ServeCompletion(HttpListenerContext context, string word) { try { List executables = _executablesCollector.GetExecutables(); CompletionItem[] completionItems = executables .Where(exec => exec.IndexOf(word, 0, StringComparison.OrdinalIgnoreCase) > -1) .Take(100) // Limit the number of results .Select(exec => new CompletionItem { Label = Path.GetFileName(exec), Kind = "Text", Documentation = $"An executable file: {exec}", InsertText = exec }) .ToArray(); XElement response = new XElement("suggestions", completionItems.Select(item => new XElement("item", new XElement("label", item.Label), new XElement("kind", item.Kind), new XElement("documentation", item.Documentation), new XElement("insertText", item.InsertText) )) ); ServeResource(context, response.ToString(), "application/xml"); } catch (Exception ex) { ServeResource(context, $"Failed to process completion request. {ex.Message}", "application/xml", 500); } } private void ServeDevTools(HttpListenerContext context, string endpoint) { try { using (HttpClient client = new HttpClient()) { string url = "http://localhost:9222" + endpoint; string data = client.GetStringAsync(url).GetAwaiter().GetResult(); ServeResource(context, data, "application/json"); } } catch (Exception ex) { ServeResource(context, $"Failed to process DevTools request. {ex.Message}", "application/xml", 500); } } private 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); } } private 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("