/* Technitium DNS Server Copyright (C) 2023 Shreyas Zare (shreyas@technitium.com) This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see . */ using DnsServerCore; using Microsoft.Extensions.Hosting; using System; using System.Reflection; using System.Threading; using System.Threading.Tasks; using TechnitiumLibrary.Net.Firewall; namespace DnsServerWindowsService { public class DnsServiceWorker : BackgroundService { readonly DnsWebService _service; public DnsServiceWorker() { string configFolder = null; string[] args = Environment.GetCommandLineArgs(); if (args.Length == 2) configFolder = args[1]; _service = new DnsWebService(configFolder, new Uri("https://go.technitium.com/?id=43"), new Uri("https://go.technitium.com/?id=44")); } public override async Task StartAsync(CancellationToken cancellationToken) { CheckFirewallEntries(); await _service.StartAsync(); } public override async Task StopAsync(CancellationToken cancellationToken) { await _service.StopAsync(); } public override void Dispose() { if (_service != null) _service.Dispose(); } protected override Task ExecuteAsync(CancellationToken stoppingToken) { return Task.CompletedTask; } private void CheckFirewallEntries() { string appPath = Assembly.GetEntryAssembly().Location; if (appPath.EndsWith(".dll", StringComparison.OrdinalIgnoreCase)) appPath = appPath.Substring(0, appPath.Length - 4) + ".exe"; if (!WindowsFirewallEntryExists(appPath)) AddWindowsFirewallEntry(appPath); } private bool WindowsFirewallEntryExists(string appPath) { try { return WindowsFirewall.RuleExistsVista("", appPath) == RuleStatus.Allowed; } catch { return false; } } private bool AddWindowsFirewallEntry(string appPath) { try { RuleStatus status = WindowsFirewall.RuleExistsVista("", appPath); switch (status) { case RuleStatus.Blocked: case RuleStatus.Disabled: WindowsFirewall.RemoveRuleVista("", appPath); break; case RuleStatus.Allowed: return true; } WindowsFirewall.AddRuleVista("Technitium DNS Server", "Allows incoming connection request to the DNS server.", FirewallAction.Allow, appPath, Protocol.ANY, null, null, null, null, InterfaceTypeFlags.All, true, Direction.Inbound, true); //add web console rule try { WindowsFirewall.RemoveRuleVista("Technitium DNS Server Web Console", ""); } catch { } try { WindowsFirewall.AddRuleVista("Technitium DNS Server Web Console", "Allows access to the DNS server web console.", FirewallAction.Allow, null, Protocol.TCP, _service.WebServiceHttpPort + ", " + _service.WebServiceTlsPort, null, null, null, InterfaceTypeFlags.All, true, Direction.Inbound, true); } catch { } return true; } catch { return false; } } } }