mirror of
				https://github.com/gnh1201/welsonjs.git
				synced 2025-10-26 18:41:18 +00:00 
			
		
		
		
	Added ConfigureAwait(false) to async HTTP call and moved Compile() invocation to immediately after deserialization. This improves async context handling and ensures blob config is compiled before assignment.
		
			
				
	
	
		
			514 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			514 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| // ResourceServer.cs
 | |
| // SPDX-License-Identifier: GPL-3.0-or-later
 | |
| // SPDX-FileCopyrightText: 2025 Catswords OSS and WelsonJS Contributors
 | |
| // https://github.com/gnh1201/welsonjs
 | |
| // 
 | |
| using System;
 | |
| using System.Collections.Generic;
 | |
| using System.IO;
 | |
| using System.Linq;
 | |
| using System.Net;
 | |
| using System.Net.Http;
 | |
| using System.Reflection;
 | |
| using System.Security.Cryptography;
 | |
| using System.Text;
 | |
| using System.Text.RegularExpressions;
 | |
| using System.Threading;
 | |
| using System.Threading.Tasks;
 | |
| using System.Windows.Forms;
 | |
| using System.Xml.Serialization;
 | |
| 
 | |
| 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 BlobConfig _blobConfig;
 | |
|         private readonly ICompatibleLogger _logger;
 | |
| 
 | |
|         private static readonly HttpClient _httpClient = new HttpClient();
 | |
|         private static readonly string _defaultMimeType = "application/octet-stream";
 | |
| 
 | |
|         static ResourceServer()
 | |
|         {
 | |
|             // Set timeout
 | |
|             int timeout = int.TryParse(Program.GetAppConfig("HttpClientTimeout"), out timeout) ? timeout : 90;
 | |
|             _httpClient.Timeout = TimeSpan.FromSeconds(timeout);
 | |
|         }
 | |
| 
 | |
|         public ResourceServer(string prefix, string resourceName, ICompatibleLogger logger = null)
 | |
|         {
 | |
|             _logger = logger ?? new TraceLogger();
 | |
|             _prefix = prefix;
 | |
|             _listener = new HttpListener();
 | |
|             _resourceName = resourceName;
 | |
| 
 | |
|             // Fetch a blob config from Internet
 | |
|             FetchBlobConfig().ConfigureAwait(false);
 | |
| 
 | |
|             // Add resource tools
 | |
|             _tools.Add(new ResourceTools.Completion(this, _httpClient));
 | |
|             _tools.Add(new ResourceTools.Settings(this, _httpClient));
 | |
|             _tools.Add(new ResourceTools.ChromiumDevTools(this, _httpClient));
 | |
|             _tools.Add(new ResourceTools.DnsQuery(this, _httpClient));
 | |
|             _tools.Add(new ResourceTools.CitiQuery(this, _httpClient));
 | |
|             _tools.Add(new ResourceTools.Tfa(this, _httpClient));
 | |
|             _tools.Add(new ResourceTools.Whois(this, _httpClient));
 | |
| 
 | |
|             // Register the prefix
 | |
|             _listener.Prefixes.Add(prefix);
 | |
|         }
 | |
| 
 | |
|         public string GetPrefix()
 | |
|         {
 | |
|             return _prefix;
 | |
|         }
 | |
| 
 | |
|         public void Start(bool IsOpenWebBrowser = true)
 | |
|         {
 | |
|             if (_isRunning) return;
 | |
| 
 | |
|             _isRunning = true;
 | |
|             _cts = new CancellationTokenSource();
 | |
|             _listener.Start();
 | |
| 
 | |
|             // Open the web browser
 | |
|             if (IsOpenWebBrowser)
 | |
|             {
 | |
|                 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;
 | |
|                     _logger?.Error($"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, string prefix = null)
 | |
|         {
 | |
|             byte[] data;
 | |
|             string mimeType;
 | |
| 
 | |
|             if (!String.IsNullOrEmpty(prefix))
 | |
|             {
 | |
|                 string url = $"{prefix}{path}";
 | |
| 
 | |
|                 try
 | |
|                 {
 | |
|                     HttpRequestMessage request = new HttpRequestMessage(HttpMethod.Get, url);
 | |
|                     var ua = context?.Request?.UserAgent;
 | |
|                     if (!string.IsNullOrEmpty(ua))
 | |
|                     {
 | |
|                         request.Headers.UserAgent.ParseAdd(ua);
 | |
|                     }
 | |
|                     HttpResponseMessage response = await _httpClient.SendAsync(request);
 | |
| 
 | |
|                     if (!response.IsSuccessStatusCode)
 | |
|                     {
 | |
|                         _logger?.Error($"Failed to serve blob. URL: {url}, 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)
 | |
|                 {
 | |
|                     _logger?.Error($"Failed to serve blob. URL: {url}, Exception: {ex.Message}");
 | |
|                     return false;
 | |
|                 }
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 // use the cached data
 | |
|                 if (TryGetCachedBlob(path, out mimeType, true))
 | |
|                 {
 | |
|                     if (TryGetCachedBlob(path, out data))
 | |
|                     {
 | |
|                         if (String.IsNullOrEmpty(mimeType))
 | |
|                         {
 | |
|                             mimeType = _defaultMimeType;
 | |
|                         }
 | |
| 
 | |
|                         ServeResource(context, data, mimeType);
 | |
|                         return true;
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 // use a blob source
 | |
|                 if (await TryServeFromBlob(context, path))
 | |
|                 {
 | |
|                     return true;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         private async Task<bool> TryServeFromBlob(HttpListenerContext context, string path)
 | |
|         {
 | |
|             if (_blobConfig != null)
 | |
|             {
 | |
|                 foreach (var route in _blobConfig.Routes)
 | |
|                 {
 | |
|                     foreach (var (regex, index) in route.RegexConditions.Select((r, i) => (r, i)))
 | |
|                     {
 | |
|                         if (!regex.Compiled.IsMatch(path)) continue;
 | |
| 
 | |
|                         var match = (index < route.Matches.Count) ? route.Matches[index] : route.Matches.First();
 | |
|                         var _path = route.StripPrefix ? path.Substring(match.Length) : path;
 | |
| 
 | |
|                         foreach (var prefixUrl in route.PrefixUrls)
 | |
|                         {
 | |
|                             if (await ServeBlob(context, _path, prefixUrl))
 | |
|                                 return true;
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             // fallback
 | |
|             string prefix = Program.GetAppConfig("BlobStoragePrefix");
 | |
|             if (await ServeBlob(context, path, prefix))
 | |
|                 return true;
 | |
| 
 | |
|             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)
 | |
|             {
 | |
|                 _logger?.Error($"Cache Read 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)
 | |
|             {
 | |
|                 _logger?.Error($"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();
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         private async Task FetchBlobConfig()
 | |
|         {
 | |
|             try
 | |
|             {
 | |
|                 string url = Program.GetAppConfig("BlobConfigUrl");
 | |
|                 if (string.IsNullOrWhiteSpace(url))
 | |
|                 {
 | |
|                     _logger?.Warn("BlobConfigUrl is not configured.");
 | |
|                     return;
 | |
|                 }
 | |
| 
 | |
|                 using (var response = await _httpClient.GetStreamAsync(url).ConfigureAwait(false))
 | |
|                 using (var reader = new StreamReader(response))
 | |
|                 {
 | |
|                     var serializer = new XmlSerializer(typeof(BlobConfig));
 | |
|                     var cfg = (BlobConfig)serializer.Deserialize(reader);
 | |
|                     cfg?.Compile();
 | |
|                     _blobConfig = cfg;
 | |
|                 }
 | |
|             }
 | |
|             catch (Exception ex)
 | |
|             {
 | |
|                 _logger?.Error($"Failed to fetch a blob config. Exception: {ex}");
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     [XmlRoot("blobConfig")]
 | |
|     public class BlobConfig
 | |
|     {
 | |
|         [XmlArray("routes")]
 | |
|         [XmlArrayItem("route")]
 | |
|         public List<BlobRoute> Routes { get; set; } = new List<BlobRoute>();
 | |
| 
 | |
|         public void Compile()
 | |
|         {
 | |
|             foreach (var route in Routes)
 | |
|             {
 | |
|                 if (route.Matches == null) continue;
 | |
| 
 | |
|                 route.RegexConditions = new List<RegexCondition>();
 | |
|                 foreach (var match in route.Matches)
 | |
|                 {
 | |
|                     route.RegexConditions.Add(new RegexCondition
 | |
|                     {
 | |
|                         Pattern = match,
 | |
|                         Compiled = new Regex(
 | |
|                             match.StartsWith("^") ? match : "^" + Regex.Escape(match),
 | |
|                             RegexOptions.Compiled)
 | |
|                     });
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     public class BlobRoute
 | |
|     {
 | |
|         [XmlArray("matches")]
 | |
|         [XmlArrayItem("match")]
 | |
|         public List<string> Matches { get; set; }
 | |
| 
 | |
|         [XmlArray("prefixUrls")]
 | |
|         [XmlArrayItem("url")]
 | |
|         public List<string> PrefixUrls { get; set; }
 | |
| 
 | |
|         [XmlAttribute("stripPrefix")]
 | |
|         public bool StripPrefix { get; set; }
 | |
| 
 | |
|         [XmlIgnore]
 | |
|         public List<RegexCondition> RegexConditions { get; set; }
 | |
|     }
 | |
| 
 | |
|     public class RegexCondition
 | |
|     {
 | |
|         [XmlIgnore]
 | |
|         public string Pattern { get; set; }
 | |
| 
 | |
|         [XmlIgnore]
 | |
|         public Regex Compiled { get; set; }
 | |
|     }
 | |
| } |