mirror of
https://github.com/chylex/Discord-History-Tracker.git
synced 2025-10-24 02:23:39 +02:00
Compare commits
2 Commits
v47.2
...
84732af1e1
Author | SHA1 | Date | |
---|---|---|---|
84732af1e1
|
|||
c2ae54cdcb
|
@@ -51,6 +51,7 @@
|
|||||||
|
|
||||||
<Style Selector="Expander">
|
<Style Selector="Expander">
|
||||||
<Setter Property="MinHeight" Value="40" />
|
<Setter Property="MinHeight" Value="40" />
|
||||||
|
<Setter Property="Padding" Value="12" />
|
||||||
<Setter Property="HorizontalAlignment" Value="Stretch" />
|
<Setter Property="HorizontalAlignment" Value="Stretch" />
|
||||||
</Style>
|
</Style>
|
||||||
|
|
||||||
|
@@ -11,25 +11,41 @@
|
|||||||
<pages:TrackingPageModel />
|
<pages:TrackingPageModel />
|
||||||
</Design.DataContext>
|
</Design.DataContext>
|
||||||
|
|
||||||
<StackPanel Spacing="10">
|
<UserControl.Styles>
|
||||||
<TextBlock TextWrapping="Wrap">
|
<Style Selector="TextBlock">
|
||||||
<TextBlock.Text>
|
<Setter Property="TextWrapping" Value="Wrap" />
|
||||||
<MultiBinding StringFormat="To start tracking messages, copy the tracking script and paste it into the console of either the Discord app, or your browser. The console is usually opened by pressing {0}.">
|
</Style>
|
||||||
<Binding Path="OpenDevToolsShortcutText" />
|
<Style Selector="WrapPanel > Button">
|
||||||
</MultiBinding>
|
<Setter Property="Margin" Value="0 0 10 10" />
|
||||||
</TextBlock.Text>
|
</Style>
|
||||||
</TextBlock>
|
</UserControl.Styles>
|
||||||
<StackPanel DockPanel.Dock="Left" Orientation="Horizontal" Spacing="10">
|
|
||||||
<Button x:Name="CopyTrackingScript" Click="CopyTrackingScriptButton_OnClick" IsEnabled="{Binding IsCopyTrackingScriptButtonEnabled}">Copy Tracking Script</Button>
|
<StackPanel Spacing="25">
|
||||||
</StackPanel>
|
<Expander Header="Method 1: Manual" IsExpanded="True">
|
||||||
<TextBlock TextWrapping="Wrap" Margin="0 5 0 0">
|
<StackPanel Orientation="Vertical" Spacing="10">
|
||||||
<TextBlock.Text>
|
<TextBlock>
|
||||||
<MultiBinding StringFormat="By default, the Discord app does not allow opening the console. The button below will change a hidden setting in the Discord app that controls whether the {0} shortcut is enabled.">
|
<TextBlock.Text>
|
||||||
<Binding Path="OpenDevToolsShortcutText" />
|
<MultiBinding StringFormat="Use {0} to open the Dev Tools in your browser or the Discord app. Copy the tracking script, and paste it into the console.">
|
||||||
</MultiBinding>
|
<Binding Path="OpenDevToolsShortcutText" />
|
||||||
</TextBlock.Text>
|
</MultiBinding>
|
||||||
</TextBlock>
|
</TextBlock.Text>
|
||||||
<Button DockPanel.Dock="Right" Command="{Binding OnClickToggleAppDevTools}" Content="{Binding ToggleAppDevToolsButtonText}" IsEnabled="{Binding IsToggleAppDevToolsButtonEnabled}" />
|
</TextBlock>
|
||||||
|
<Button x:Name="CopyTrackingScript" Click="CopyTrackingScriptButton_OnClick">Copy Tracking Script</Button>
|
||||||
|
<TextBlock Margin="0 5 0 0">
|
||||||
|
By default, the Discord app blocks the shortcut for opening Dev Tools. The button below will change a hidden setting in the Discord app that controls it.
|
||||||
|
</TextBlock>
|
||||||
|
<Button Command="{Binding OnClickToggleAppDevTools}" Content="{Binding ToggleAppDevToolsButtonText}" IsEnabled="{Binding IsToggleAppDevToolsButtonEnabled}" />
|
||||||
|
</StackPanel>
|
||||||
|
</Expander>
|
||||||
|
<Expander Header="Method 2: Userscript" IsExpanded="True" Padding="12 12 12 2.5">
|
||||||
|
<StackPanel Orientation="Vertical" Spacing="10">
|
||||||
|
<TextBlock>If your browser has a userscript manager, you can install a userscript that adds a DHT icon next to the Help icon. Clicking the DHT icon opens a prompt, where you paste the Connection Code.</TextBlock>
|
||||||
|
<WrapPanel>
|
||||||
|
<Button Command="{Binding OnClickInstallOrUpdateUserscript}">Install or Update Userscript</Button>
|
||||||
|
<Button x:Name="CopyConnectionCode" Click="CopyConnectionScriptButton_OnClick">Copy Connection Code</Button>
|
||||||
|
</WrapPanel>
|
||||||
|
</StackPanel>
|
||||||
|
</Expander>
|
||||||
</StackPanel>
|
</StackPanel>
|
||||||
|
|
||||||
</UserControl>
|
</UserControl>
|
||||||
|
@@ -1,4 +1,5 @@
|
|||||||
using System;
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
using System.Diagnostics.CodeAnalysis;
|
using System.Diagnostics.CodeAnalysis;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using Avalonia.Controls;
|
using Avalonia.Controls;
|
||||||
@@ -8,24 +9,36 @@ namespace DHT.Desktop.Main.Pages;
|
|||||||
|
|
||||||
[SuppressMessage("ReSharper", "MemberCanBeInternal")]
|
[SuppressMessage("ReSharper", "MemberCanBeInternal")]
|
||||||
public sealed partial class TrackingPage : UserControl {
|
public sealed partial class TrackingPage : UserControl {
|
||||||
private bool isCopyingScript;
|
private readonly HashSet<Button> copyingButtons = new (ReferenceEqualityComparer.Instance);
|
||||||
|
|
||||||
public TrackingPage() {
|
public TrackingPage() {
|
||||||
InitializeComponent();
|
InitializeComponent();
|
||||||
}
|
}
|
||||||
|
|
||||||
public async void CopyTrackingScriptButton_OnClick(object? sender, RoutedEventArgs e) {
|
public async void CopyTrackingScriptButton_OnClick(object? sender, RoutedEventArgs e) {
|
||||||
|
await HandleCopyButton(CopyTrackingScript, "Script Copied!", static model => model.OnClickCopyTrackingScript());
|
||||||
|
}
|
||||||
|
|
||||||
|
public async void CopyConnectionScriptButton_OnClick(object? sender, RoutedEventArgs e) {
|
||||||
|
await HandleCopyButton(CopyConnectionCode, "Code Copied!", static model => model.OnClickCopyConnectionCode());
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task HandleCopyButton(Button button, string copiedText, Func<TrackingPageModel, Task<bool>> onClick) {
|
||||||
if (DataContext is TrackingPageModel model) {
|
if (DataContext is TrackingPageModel model) {
|
||||||
object? originalText = CopyTrackingScript.Content;
|
object? originalText = button.Content;
|
||||||
CopyTrackingScript.MinWidth = CopyTrackingScript.Bounds.Width;
|
button.MinWidth = button.Bounds.Width;
|
||||||
|
|
||||||
if (await model.OnClickCopyTrackingScript() && !isCopyingScript) {
|
if (await onClick(model) && copyingButtons.Add(button)) {
|
||||||
isCopyingScript = true;
|
button.IsEnabled = false;
|
||||||
CopyTrackingScript.Content = "Script Copied!";
|
button.Content = copiedText;
|
||||||
|
|
||||||
await Task.Delay(TimeSpan.FromSeconds(2));
|
try {
|
||||||
CopyTrackingScript.Content = originalText;
|
await Task.Delay(TimeSpan.FromSeconds(2));
|
||||||
isCopyingScript = false;
|
} finally {
|
||||||
|
copyingButtons.Remove(button);
|
||||||
|
button.IsEnabled = true;
|
||||||
|
button.Content = originalText;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -1,20 +1,22 @@
|
|||||||
using System;
|
using System;
|
||||||
|
using System.Text.RegularExpressions;
|
||||||
using System.Threading;
|
using System.Threading;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using System.Web;
|
using System.Web;
|
||||||
using Avalonia.Controls;
|
using Avalonia.Controls;
|
||||||
using Avalonia.Input.Platform;
|
using Avalonia.Input.Platform;
|
||||||
|
using DHT.Desktop.Common;
|
||||||
using DHT.Desktop.Dialogs.Message;
|
using DHT.Desktop.Dialogs.Message;
|
||||||
using DHT.Desktop.Discord;
|
using DHT.Desktop.Discord;
|
||||||
using DHT.Desktop.Server;
|
using DHT.Desktop.Server;
|
||||||
|
using DHT.Utils.Logging;
|
||||||
using PropertyChanged.SourceGenerator;
|
using PropertyChanged.SourceGenerator;
|
||||||
using static DHT.Desktop.Program;
|
using static DHT.Desktop.Program;
|
||||||
|
|
||||||
namespace DHT.Desktop.Main.Pages;
|
namespace DHT.Desktop.Main.Pages;
|
||||||
|
|
||||||
sealed partial class TrackingPageModel {
|
sealed partial class TrackingPageModel {
|
||||||
[Notify(Setter.Private)]
|
private static readonly Log Log = Log.ForType<TrackingPageModel>();
|
||||||
private bool isCopyTrackingScriptButtonEnabled = true;
|
|
||||||
|
|
||||||
[Notify(Setter.Private)]
|
[Notify(Setter.Private)]
|
||||||
private bool? areDevToolsEnabled = null;
|
private bool? areDevToolsEnabled = null;
|
||||||
@@ -51,32 +53,9 @@ sealed partial class TrackingPageModel {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public async Task<bool> OnClickCopyTrackingScript() {
|
public async Task<bool> OnClickCopyTrackingScript() {
|
||||||
IsCopyTrackingScriptButtonEnabled = false;
|
string url = ServerConfiguration.HttpHost + $"/get-tracking-script?token={HttpUtility.UrlEncode(ServerConfiguration.Token)}";
|
||||||
|
|
||||||
try {
|
|
||||||
return await CopyTrackingScript();
|
|
||||||
} finally {
|
|
||||||
IsCopyTrackingScriptButtonEnabled = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private async Task<bool> CopyTrackingScript() {
|
|
||||||
string url = $"http://127.0.0.1:{ServerConfiguration.Port}/get-tracking-script?token={HttpUtility.UrlEncode(ServerConfiguration.Token)}";
|
|
||||||
string script = (await Resources.ReadTextAsync("tracker-loader.js")).Trim().Replace("{url}", url);
|
string script = (await Resources.ReadTextAsync("tracker-loader.js")).Trim().Replace("{url}", url);
|
||||||
|
return await TryCopy(script, "Copy Tracking Script");
|
||||||
IClipboard? clipboard = window.Clipboard;
|
|
||||||
if (clipboard == null) {
|
|
||||||
await Dialog.ShowOk(window, "Copy Tracking Script", "Clipboard is not available on this system.");
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
await clipboard.SetTextAsync(script);
|
|
||||||
return true;
|
|
||||||
} catch {
|
|
||||||
await Dialog.ShowOk(window, "Copy Tracking Script", "An error occurred while copying to clipboard.");
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private async Task InitializeDevToolsToggle() {
|
private async Task InitializeDevToolsToggle() {
|
||||||
@@ -132,4 +111,44 @@ sealed partial class TrackingPageModel {
|
|||||||
throw new ArgumentOutOfRangeException();
|
throw new ArgumentOutOfRangeException();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public async Task OnClickInstallOrUpdateUserscript() {
|
||||||
|
try {
|
||||||
|
SystemUtils.OpenUrl(ServerConfiguration.HttpHost + "/get-userscript/dht.user.js");
|
||||||
|
} catch (Exception e) {
|
||||||
|
await Dialog.ShowOk(window, "Install or Update Userscript", "Could not open the browser: " + e.Message);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[GeneratedRegex("^[a-zA-Z0-9]{1,100}$")]
|
||||||
|
private static partial Regex ConnectionCodeTokenRegex();
|
||||||
|
|
||||||
|
public async Task<bool> OnClickCopyConnectionCode() {
|
||||||
|
const string Title = "Copy Connection Code";
|
||||||
|
|
||||||
|
if (ConnectionCodeTokenRegex().IsMatch(ServerConfiguration.Token)) {
|
||||||
|
return await TryCopy(ServerConfiguration.Port + ":" + ServerConfiguration.Token, Title);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
await Dialog.ShowOk(window, Title, "The internal server token cannot be used to create a connection code. Check the 'Advanced' tab and ensure the configured token is 1-100 characters long, and only contains plain letters and numbers.");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task<bool> TryCopy(string script, string errorDialogTitle) {
|
||||||
|
IClipboard? clipboard = window.Clipboard;
|
||||||
|
if (clipboard == null) {
|
||||||
|
await Dialog.ShowOk(window, errorDialogTitle, "Clipboard is not available on this system.");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
await clipboard.SetTextAsync(script);
|
||||||
|
return true;
|
||||||
|
} catch (Exception e) {
|
||||||
|
Log.Error(e);
|
||||||
|
await Dialog.ShowOk(window, errorDialogTitle, "An error occurred while copying to clipboard.");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@@ -51,10 +51,9 @@ sealed partial class ViewerPageModel : IDisposable {
|
|||||||
|
|
||||||
public async void OnClickOpenViewer() {
|
public async void OnClickOpenViewer() {
|
||||||
try {
|
try {
|
||||||
string serverUrl = "http://127.0.0.1:" + ServerConfiguration.Port;
|
|
||||||
string serverToken = ServerConfiguration.Token;
|
string serverToken = ServerConfiguration.Token;
|
||||||
string sessionId = state.ViewerSessions.Register(new ViewerSession(FilterModel.CreateFilter())).ToString();
|
string sessionId = state.ViewerSessions.Register(new ViewerSession(FilterModel.CreateFilter())).ToString();
|
||||||
SystemUtils.OpenUrl(serverUrl + "/viewer/?token=" + HttpUtility.UrlEncode(serverToken) + "&session=" + HttpUtility.UrlEncode(sessionId));
|
SystemUtils.OpenUrl(ServerConfiguration.HttpHost + "/viewer/?token=" + HttpUtility.UrlEncode(serverToken) + "&session=" + HttpUtility.UrlEncode(sessionId));
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
await Dialog.ShowOk(window, "Open Viewer", "Could not open viewer: " + e.Message);
|
await Dialog.ShowOk(window, "Open Viewer", "Could not open viewer: " + e.Message);
|
||||||
}
|
}
|
||||||
|
@@ -5,4 +5,6 @@ namespace DHT.Desktop.Server;
|
|||||||
static class ServerConfiguration {
|
static class ServerConfiguration {
|
||||||
public static ushort Port { get; set; } = ServerUtils.FindAvailablePort(min: 50000, max: 60000);
|
public static ushort Port { get; set; } = ServerUtils.FindAvailablePort(min: 50000, max: 60000);
|
||||||
public static string Token { get; set; } = ServerUtils.GenerateRandomToken(20);
|
public static string Token { get; set; } = ServerUtils.GenerateRandomToken(20);
|
||||||
|
|
||||||
|
public static string HttpHost => "http://127.0.0.1:" + Port;
|
||||||
}
|
}
|
||||||
|
242
app/Resources/Tracker/loader/dht.user.js
Normal file
242
app/Resources/Tracker/loader/dht.user.js
Normal file
@@ -0,0 +1,242 @@
|
|||||||
|
// ==UserScript==
|
||||||
|
// @name Discord History Tracker
|
||||||
|
// @license MIT
|
||||||
|
// @namespace https://chylex.com
|
||||||
|
// @homepageURL https://dht.chylex.com/
|
||||||
|
// @supportURL https://github.com/chylex/Discord-History-Tracker/issues
|
||||||
|
// @include https://discord.com/*
|
||||||
|
// @run-at document-idle
|
||||||
|
// @grant none
|
||||||
|
// @noframes
|
||||||
|
// ==/UserScript==
|
||||||
|
|
||||||
|
function startMutationObserver(callback) {
|
||||||
|
let debounceTimeout = null;
|
||||||
|
|
||||||
|
new MutationObserver((_, observer) => {
|
||||||
|
window.clearTimeout(debounceTimeout);
|
||||||
|
debounceTimeout = window.setTimeout(() => callback(observer), 20);
|
||||||
|
}).observe(document.body, { childList: true, subtree: true });
|
||||||
|
}
|
||||||
|
|
||||||
|
startMutationObserver(observer => {
|
||||||
|
const triggerButton = installConnectDialogButton();
|
||||||
|
if (triggerButton === null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
observer.disconnect();
|
||||||
|
|
||||||
|
startMutationObserver(() => {
|
||||||
|
if (!triggerButton.isConnected) {
|
||||||
|
getHelpIcon()?.parentElement.parentElement.insertAdjacentElement("afterend", triggerButton);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
function installConnectDialogButton() {
|
||||||
|
const helpIcon = getHelpIcon();
|
||||||
|
if (!helpIcon) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const helpIconWrapper = helpIcon.parentElement;
|
||||||
|
const helpIconLink = helpIconWrapper.parentElement;
|
||||||
|
|
||||||
|
helpIconLink.insertAdjacentHTML("afterend", `
|
||||||
|
<div id="dht-connector-show-dialog" role="button" class="${helpIconWrapper.getAttribute("class")}">
|
||||||
|
<svg viewBox="0 0 24 20" fill="currentColor" class="${helpIcon.getAttribute("class")}" style="width: auto;">
|
||||||
|
<path d="M7.446,9.924c-0,0.846 -0.105,1.593 -0.315,2.234c-0.21,0.644 -0.497,1.181 -0.86,1.615c-0.365,0.431 -0.795,0.761 -1.292,0.982c-0.494,0.22 -1.029,0.332 -1.605,0.332l-3.374,-0l0,-10.174l3.021,0c0.645,0 1.239,0.1 1.781,0.296c0.542,0.198 1.011,0.501 1.4,0.911c0.391,0.408 0.695,0.928 0.914,1.56c0.22,0.629 0.33,1.378 0.33,2.244Zm-1.765,-0c0,-0.592 -0.067,-1.1 -0.198,-1.524c-0.134,-0.422 -0.318,-0.77 -0.554,-1.042c-0.236,-0.272 -0.518,-0.473 -0.848,-0.604c-0.332,-0.128 -0.697,-0.195 -1.096,-0.195l-1.237,-0l0,6.882l1.483,0c0.351,0 0.676,-0.076 0.977,-0.224c0.298,-0.15 0.556,-0.372 0.776,-0.668c0.22,-0.295 0.389,-0.663 0.513,-1.101c0.122,-0.439 0.184,-0.947 0.184,-1.524Z"/>
|
||||||
|
<path d="M14.147,15.087l-0,-4.362l-3.637,-0l-0,4.362l-1.748,-0l-0,-10.174l1.748,0l-0,4.052l3.637,-0l-0,-4.052l1.75,0l0,10.174l-1.75,-0Z"/>
|
||||||
|
<path d="M21.297,6.559l-0,8.528l-1.748,-0l-0,-8.528l-2.699,-0l0,-1.646l7.15,0l0,1.646l-2.703,-0Z"/>
|
||||||
|
</svg>
|
||||||
|
</div>
|
||||||
|
`);
|
||||||
|
|
||||||
|
const button = document.getElementById("dht-connector-show-dialog");
|
||||||
|
button.addEventListener("click", showConnectDialog);
|
||||||
|
return button;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getHelpIcon() {
|
||||||
|
return document.querySelector("div[class*='bar_'] a[href$='://support.discord.com'] > div[class*='clickable'] > svg");
|
||||||
|
}
|
||||||
|
|
||||||
|
function showConnectDialog() {
|
||||||
|
const dialogElement = document.createElement("dialog");
|
||||||
|
dialogElement.id = "dht-connector-dialog";
|
||||||
|
dialogElement.innerHTML = `
|
||||||
|
<form id="dht-connector-dialog-form">
|
||||||
|
<label for="dht-connector-dialog-input-code">Connection Code</label>
|
||||||
|
<input id="dht-connector-dialog-input-code" type="text">
|
||||||
|
<p id="dht-connector-dialog-input-error"></p>
|
||||||
|
|
||||||
|
<div id="dht-connector-dialog-buttons">
|
||||||
|
<button type="submit" id="dht-connector-dialog-button-connect">Connect</button>
|
||||||
|
<button type="button" id="dht-connector-dialog-button-close">Cancel</button>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
<style>
|
||||||
|
#dht-connector-dialog {
|
||||||
|
width: 300px;
|
||||||
|
padding: 18px;
|
||||||
|
font-size: 16px;
|
||||||
|
border: none;
|
||||||
|
border-radius: 8px;
|
||||||
|
background-color: #313338;
|
||||||
|
}
|
||||||
|
|
||||||
|
#dht-connector-dialog::backdrop {
|
||||||
|
background-color: rgba(0, 0, 0, 0.6);
|
||||||
|
}
|
||||||
|
|
||||||
|
#dht-connector-dialog label {
|
||||||
|
display: block;
|
||||||
|
margin: 0 0 9px;
|
||||||
|
font-weight: 500;
|
||||||
|
letter-spacing: .02em;
|
||||||
|
color: #fbfbfb;
|
||||||
|
}
|
||||||
|
|
||||||
|
#dht-connector-dialog-input-error {
|
||||||
|
margin: 10px 1px;
|
||||||
|
color: #f23f43;
|
||||||
|
font-size: 15px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#dht-connector-dialog input[type="text"] {
|
||||||
|
box-sizing: border-box;
|
||||||
|
width: 100%;
|
||||||
|
padding: 12px 10px;
|
||||||
|
border: 1px solid rgba(151, 151, 159, 0.2);
|
||||||
|
border-radius: 8px;
|
||||||
|
color: #efeff0;
|
||||||
|
background-color: #1d1d21;
|
||||||
|
}
|
||||||
|
|
||||||
|
#dht-connector-dialog-buttons {
|
||||||
|
margin-top: 15px;
|
||||||
|
display: flex;
|
||||||
|
justify-content: end;
|
||||||
|
gap: 12px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#dht-connector-dialog button {
|
||||||
|
padding: 8px 16px;
|
||||||
|
border: none;
|
||||||
|
border-radius: 8px;
|
||||||
|
font-size: 14px;
|
||||||
|
font-weight: 500;
|
||||||
|
color: #fff;
|
||||||
|
background-color: #5865f2;
|
||||||
|
transition: background-color 200ms ease, color 200ms ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
#dht-connector-dialog button:hover {
|
||||||
|
background-color: #4752c4;
|
||||||
|
}
|
||||||
|
|
||||||
|
#dht-connector-dialog button[disabled] {
|
||||||
|
cursor: not-allowed;
|
||||||
|
opacity: 0.5;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
`;
|
||||||
|
|
||||||
|
document.body.insertAdjacentElement("beforeend", dialogElement);
|
||||||
|
|
||||||
|
const formElement = document.getElementById("dht-connector-dialog-form");
|
||||||
|
const codeErrorElement = document.getElementById("dht-connector-dialog-input-error");
|
||||||
|
const codeInputElement = document.getElementById("dht-connector-dialog-input-code");
|
||||||
|
const connectButtonElement = document.getElementById("dht-connector-dialog-button-connect");
|
||||||
|
const closeButtonElement = document.getElementById("dht-connector-dialog-button-close");
|
||||||
|
|
||||||
|
dialogElement.addEventListener("close", function() {
|
||||||
|
dialogElement.remove();
|
||||||
|
});
|
||||||
|
|
||||||
|
closeButtonElement.addEventListener("click", function() {
|
||||||
|
dialogElement.close();
|
||||||
|
});
|
||||||
|
|
||||||
|
codeInputElement.addEventListener("paste", function() {
|
||||||
|
setTimeout(async function() {
|
||||||
|
const code = parseConnectionCode(codeInputElement.value);
|
||||||
|
if (code !== null) {
|
||||||
|
await onSubmit(code);
|
||||||
|
}
|
||||||
|
}, 0);
|
||||||
|
});
|
||||||
|
|
||||||
|
formElement.addEventListener("submit", async function(e) {
|
||||||
|
e.preventDefault();
|
||||||
|
|
||||||
|
const code = parseConnectionCode(codeInputElement.value);
|
||||||
|
if (code === null) {
|
||||||
|
codeErrorElement.innerText = "Code is not valid.";
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
await onSubmit(code);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
let isSubmitting = false;
|
||||||
|
|
||||||
|
async function onSubmit(code) {
|
||||||
|
if (isSubmitting) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
isSubmitting = true;
|
||||||
|
connectButtonElement.setAttribute("disabled", "");
|
||||||
|
try {
|
||||||
|
await loadTrackingScript(code.port, code.token);
|
||||||
|
dialogElement.close();
|
||||||
|
} catch (e) {
|
||||||
|
console.error("Could not load tracking script", e);
|
||||||
|
codeErrorElement.innerText = "Could not load tracking script";
|
||||||
|
} finally {
|
||||||
|
isSubmitting = false;
|
||||||
|
connectButtonElement.removeAttribute("disabled");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
dialogElement.showModal();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {string} code
|
||||||
|
* @return {?{port: number, token: string}}
|
||||||
|
*/
|
||||||
|
function parseConnectionCode(code) {
|
||||||
|
if (code.length > 1000) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const match = code.match(/^(\d{1,5}):([a-zA-Z0-9]{1,100})$/);
|
||||||
|
if (!match) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const port = Number(match[1]);
|
||||||
|
if (port < 0 || port > 65536) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return { port, token: match[2] };
|
||||||
|
}
|
||||||
|
|
||||||
|
async function loadTrackingScript(port, token) {
|
||||||
|
const response = await fetch("http://127.0.0.1:" + port + "/get-tracking-script?token=" + encodeURIComponent(token));
|
||||||
|
|
||||||
|
if (!response.ok) {
|
||||||
|
throw "Invalid response";
|
||||||
|
}
|
||||||
|
|
||||||
|
if (response.headers.get("X-DHT") !== "1") {
|
||||||
|
throw response.status + " " + response.statusText;
|
||||||
|
}
|
||||||
|
|
||||||
|
// noinspection DynamicallyGeneratedCodeJS
|
||||||
|
eval(await response.text());
|
||||||
|
}
|
@@ -3,18 +3,17 @@ using System.Net;
|
|||||||
using System.Text.Json;
|
using System.Text.Json;
|
||||||
using System.Threading;
|
using System.Threading;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using DHT.Server.Database;
|
|
||||||
using DHT.Utils.Http;
|
using DHT.Utils.Http;
|
||||||
using DHT.Utils.Logging;
|
using DHT.Utils.Logging;
|
||||||
using Microsoft.AspNetCore.Http;
|
using Microsoft.AspNetCore.Http;
|
||||||
|
using Microsoft.AspNetCore.StaticFiles;
|
||||||
using Microsoft.Extensions.Primitives;
|
using Microsoft.Extensions.Primitives;
|
||||||
|
|
||||||
namespace DHT.Server.Endpoints;
|
namespace DHT.Server.Endpoints;
|
||||||
|
|
||||||
abstract class BaseEndpoint(IDatabaseFile db) {
|
abstract class BaseEndpoint {
|
||||||
private static readonly Log Log = Log.ForType<BaseEndpoint>();
|
private static readonly Log Log = Log.ForType<BaseEndpoint>();
|
||||||
|
private static readonly FileExtensionContentTypeProvider ContentTypeProvider = new ();
|
||||||
protected IDatabaseFile Db { get; } = db;
|
|
||||||
|
|
||||||
public async Task Handle(HttpContext ctx) {
|
public async Task Handle(HttpContext ctx) {
|
||||||
HttpResponse response = ctx.Response;
|
HttpResponse response = ctx.Response;
|
||||||
@@ -49,6 +48,16 @@ abstract class BaseEndpoint(IDatabaseFile db) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected static async Task WriteFileIfFound(HttpResponse response, string relativeFilePath, byte[]? bytes, CancellationToken cancellationToken) {
|
||||||
|
if (bytes == null) {
|
||||||
|
throw new HttpException(HttpStatusCode.NotFound, "File not found: " + relativeFilePath);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
string? contentType = ContentTypeProvider.TryGetContentType(relativeFilePath, out string? type) ? type : null;
|
||||||
|
await response.WriteFileAsync(contentType, bytes, cancellationToken);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
protected static Guid GetSessionId(HttpRequest request) {
|
protected static Guid GetSessionId(HttpRequest request) {
|
||||||
if (request.Query.TryGetValue("session", out StringValues sessionIdValue) && sessionIdValue.Count == 1 && Guid.TryParse(sessionIdValue[0], out Guid sessionId)) {
|
if (request.Query.TryGetValue("session", out StringValues sessionIdValue) && sessionIdValue.Count == 1 && Guid.TryParse(sessionIdValue[0], out Guid sessionId)) {
|
||||||
return sessionId;
|
return sessionId;
|
||||||
|
@@ -10,12 +10,12 @@ using Microsoft.AspNetCore.Http;
|
|||||||
|
|
||||||
namespace DHT.Server.Endpoints;
|
namespace DHT.Server.Endpoints;
|
||||||
|
|
||||||
sealed class GetDownloadedFileEndpoint(IDatabaseFile db) : BaseEndpoint(db) {
|
sealed class GetDownloadedFileEndpoint(IDatabaseFile db) : BaseEndpoint {
|
||||||
protected override async Task Respond(HttpRequest request, HttpResponse response, CancellationToken cancellationToken) {
|
protected override async Task Respond(HttpRequest request, HttpResponse response, CancellationToken cancellationToken) {
|
||||||
string url = WebUtility.UrlDecode((string) request.RouteValues["url"]!);
|
string url = WebUtility.UrlDecode((string) request.RouteValues["url"]!);
|
||||||
string normalizedUrl = DiscordCdn.NormalizeUrl(url);
|
string normalizedUrl = DiscordCdn.NormalizeUrl(url);
|
||||||
|
|
||||||
if (!await Db.Downloads.GetSuccessfulDownloadWithData(normalizedUrl, WriteDataTo(response), cancellationToken)) {
|
if (!await db.Downloads.GetSuccessfulDownloadWithData(normalizedUrl, WriteDataTo(response), cancellationToken)) {
|
||||||
response.Redirect(url, permanent: false);
|
response.Redirect(url, permanent: false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -2,7 +2,6 @@ using System.Net.Mime;
|
|||||||
using System.Threading;
|
using System.Threading;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using System.Web;
|
using System.Web;
|
||||||
using DHT.Server.Database;
|
|
||||||
using DHT.Server.Service;
|
using DHT.Server.Service;
|
||||||
using DHT.Utils.Http;
|
using DHT.Utils.Http;
|
||||||
using DHT.Utils.Resources;
|
using DHT.Utils.Resources;
|
||||||
@@ -10,7 +9,7 @@ using Microsoft.AspNetCore.Http;
|
|||||||
|
|
||||||
namespace DHT.Server.Endpoints;
|
namespace DHT.Server.Endpoints;
|
||||||
|
|
||||||
sealed class GetTrackingScriptEndpoint(IDatabaseFile db, ServerParameters parameters, ResourceLoader resources) : BaseEndpoint(db) {
|
sealed class GetTrackingScriptEndpoint(ServerParameters parameters, ResourceLoader resources) : BaseEndpoint {
|
||||||
protected override async Task Respond(HttpRequest request, HttpResponse response, CancellationToken cancellationToken) {
|
protected override async Task Respond(HttpRequest request, HttpResponse response, CancellationToken cancellationToken) {
|
||||||
string bootstrap = await resources.ReadTextAsync("Tracker/bootstrap.js");
|
string bootstrap = await resources.ReadTextAsync("Tracker/bootstrap.js");
|
||||||
string script = bootstrap.Replace("= 0; /*[PORT]*/", "= " + parameters.Port + ";")
|
string script = bootstrap.Replace("= 0; /*[PORT]*/", "= " + parameters.Port + ";")
|
||||||
|
18
app/Server/Endpoints/GetUserscriptEndpoint.cs
Normal file
18
app/Server/Endpoints/GetUserscriptEndpoint.cs
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
using System.Threading;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using DHT.Server.Service.Middlewares;
|
||||||
|
using DHT.Utils.Resources;
|
||||||
|
using Microsoft.AspNetCore.Http;
|
||||||
|
|
||||||
|
namespace DHT.Server.Endpoints;
|
||||||
|
|
||||||
|
[ServerAuthorizationMiddleware.NoAuthorization]
|
||||||
|
sealed class GetUserscriptEndpoint(ResourceLoader resources) : BaseEndpoint {
|
||||||
|
protected override async Task Respond(HttpRequest request, HttpResponse response, CancellationToken cancellationToken) {
|
||||||
|
const string FileName = "dht.user.js";
|
||||||
|
const string ResourcePath = "Tracker/loader/" + FileName;
|
||||||
|
|
||||||
|
byte[]? resourceBytes = await resources.ReadBytesAsyncIfExists(ResourcePath);
|
||||||
|
await WriteFileIfFound(response, FileName, resourceBytes, cancellationToken);
|
||||||
|
}
|
||||||
|
}
|
@@ -8,12 +8,12 @@ using Microsoft.AspNetCore.Http;
|
|||||||
|
|
||||||
namespace DHT.Server.Endpoints;
|
namespace DHT.Server.Endpoints;
|
||||||
|
|
||||||
sealed class GetViewerMessagesEndpoint(IDatabaseFile db, ViewerSessions viewerSessions) : BaseEndpoint(db) {
|
sealed class GetViewerMessagesEndpoint(IDatabaseFile db, ViewerSessions viewerSessions) : BaseEndpoint {
|
||||||
protected override Task Respond(HttpRequest request, HttpResponse response, CancellationToken cancellationToken) {
|
protected override Task Respond(HttpRequest request, HttpResponse response, CancellationToken cancellationToken) {
|
||||||
Guid sessionId = GetSessionId(request);
|
Guid sessionId = GetSessionId(request);
|
||||||
ViewerSession session = viewerSessions.Get(sessionId);
|
ViewerSession session = viewerSessions.Get(sessionId);
|
||||||
|
|
||||||
response.ContentType = "application/x-ndjson";
|
response.ContentType = "application/x-ndjson";
|
||||||
return ViewerJsonExport.GetMessages(response.Body, Db, session.MessageFilter, cancellationToken);
|
return ViewerJsonExport.GetMessages(response.Body, db, session.MessageFilter, cancellationToken);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -9,12 +9,12 @@ using Microsoft.AspNetCore.Http;
|
|||||||
|
|
||||||
namespace DHT.Server.Endpoints;
|
namespace DHT.Server.Endpoints;
|
||||||
|
|
||||||
sealed class GetViewerMetadataEndpoint(IDatabaseFile db, ViewerSessions viewerSessions) : BaseEndpoint(db) {
|
sealed class GetViewerMetadataEndpoint(IDatabaseFile db, ViewerSessions viewerSessions) : BaseEndpoint {
|
||||||
protected override Task Respond(HttpRequest request, HttpResponse response, CancellationToken cancellationToken) {
|
protected override Task Respond(HttpRequest request, HttpResponse response, CancellationToken cancellationToken) {
|
||||||
Guid sessionId = GetSessionId(request);
|
Guid sessionId = GetSessionId(request);
|
||||||
ViewerSession session = viewerSessions.Get(sessionId);
|
ViewerSession session = viewerSessions.Get(sessionId);
|
||||||
|
|
||||||
response.ContentType = MediaTypeNames.Application.Json;
|
response.ContentType = MediaTypeNames.Application.Json;
|
||||||
return ViewerJsonExport.GetMetadata(response.Body, Db, session.MessageFilter, cancellationToken);
|
return ViewerJsonExport.GetMetadata(response.Body, db, session.MessageFilter, cancellationToken);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -9,14 +9,14 @@ using Microsoft.AspNetCore.Http;
|
|||||||
|
|
||||||
namespace DHT.Server.Endpoints;
|
namespace DHT.Server.Endpoints;
|
||||||
|
|
||||||
sealed class TrackChannelEndpoint(IDatabaseFile db) : BaseEndpoint(db) {
|
sealed class TrackChannelEndpoint(IDatabaseFile db) : BaseEndpoint {
|
||||||
protected override async Task Respond(HttpRequest request, HttpResponse response, CancellationToken cancellationToken) {
|
protected override async Task Respond(HttpRequest request, HttpResponse response, CancellationToken cancellationToken) {
|
||||||
JsonElement root = await ReadJson(request);
|
JsonElement root = await ReadJson(request);
|
||||||
Data.Server server = ReadServer(root.RequireObject("server"), "server");
|
Data.Server server = ReadServer(root.RequireObject("server"), "server");
|
||||||
Channel channel = ReadChannel(root.RequireObject("channel"), "channel", server.Id);
|
Channel channel = ReadChannel(root.RequireObject("channel"), "channel", server.Id);
|
||||||
|
|
||||||
await Db.Servers.Add([server]);
|
await db.Servers.Add([server]);
|
||||||
await Db.Channels.Add([channel]);
|
await db.Channels.Add([channel]);
|
||||||
}
|
}
|
||||||
|
|
||||||
private static Data.Server ReadServer(JsonElement json, string path) {
|
private static Data.Server ReadServer(JsonElement json, string path) {
|
||||||
|
@@ -16,7 +16,7 @@ using Microsoft.AspNetCore.Http;
|
|||||||
|
|
||||||
namespace DHT.Server.Endpoints;
|
namespace DHT.Server.Endpoints;
|
||||||
|
|
||||||
sealed class TrackMessagesEndpoint(IDatabaseFile db) : BaseEndpoint(db) {
|
sealed class TrackMessagesEndpoint(IDatabaseFile db) : BaseEndpoint {
|
||||||
private const string HasNewMessages = "1";
|
private const string HasNewMessages = "1";
|
||||||
private const string NoNewMessages = "0";
|
private const string NoNewMessages = "0";
|
||||||
|
|
||||||
@@ -38,9 +38,9 @@ sealed class TrackMessagesEndpoint(IDatabaseFile db) : BaseEndpoint(db) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
var addedMessageFilter = new MessageFilter { MessageIds = addedMessageIds };
|
var addedMessageFilter = new MessageFilter { MessageIds = addedMessageIds };
|
||||||
bool anyNewMessages = await Db.Messages.Count(addedMessageFilter, CancellationToken.None) < addedMessageIds.Count;
|
bool anyNewMessages = await db.Messages.Count(addedMessageFilter, CancellationToken.None) < addedMessageIds.Count;
|
||||||
|
|
||||||
await Db.Messages.Add(messages);
|
await db.Messages.Add(messages);
|
||||||
|
|
||||||
await response.WriteTextAsync(anyNewMessages ? HasNewMessages : NoNewMessages, cancellationToken);
|
await response.WriteTextAsync(anyNewMessages ? HasNewMessages : NoNewMessages, cancellationToken);
|
||||||
}
|
}
|
||||||
|
@@ -9,7 +9,7 @@ using Microsoft.AspNetCore.Http;
|
|||||||
|
|
||||||
namespace DHT.Server.Endpoints;
|
namespace DHT.Server.Endpoints;
|
||||||
|
|
||||||
sealed class TrackUsersEndpoint(IDatabaseFile db) : BaseEndpoint(db) {
|
sealed class TrackUsersEndpoint(IDatabaseFile db) : BaseEndpoint {
|
||||||
protected override async Task Respond(HttpRequest request, HttpResponse response, CancellationToken cancellationToken) {
|
protected override async Task Respond(HttpRequest request, HttpResponse response, CancellationToken cancellationToken) {
|
||||||
JsonElement root = await ReadJson(request);
|
JsonElement root = await ReadJson(request);
|
||||||
|
|
||||||
@@ -24,7 +24,7 @@ sealed class TrackUsersEndpoint(IDatabaseFile db) : BaseEndpoint(db) {
|
|||||||
users[i++] = ReadUser(user, "user");
|
users[i++] = ReadUser(user, "user");
|
||||||
}
|
}
|
||||||
|
|
||||||
await Db.Users.Add(users);
|
await db.Users.Add(users);
|
||||||
}
|
}
|
||||||
|
|
||||||
private static User ReadUser(JsonElement json, string path) {
|
private static User ReadUser(JsonElement json, string path) {
|
||||||
|
@@ -1,18 +1,14 @@
|
|||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Net;
|
|
||||||
using System.Threading;
|
using System.Threading;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using DHT.Server.Database;
|
using DHT.Server.Service.Middlewares;
|
||||||
using DHT.Utils.Http;
|
|
||||||
using DHT.Utils.Resources;
|
using DHT.Utils.Resources;
|
||||||
using Microsoft.AspNetCore.Http;
|
using Microsoft.AspNetCore.Http;
|
||||||
using Microsoft.AspNetCore.StaticFiles;
|
|
||||||
|
|
||||||
namespace DHT.Server.Endpoints;
|
namespace DHT.Server.Endpoints;
|
||||||
|
|
||||||
sealed class ViewerEndpoint(IDatabaseFile db, ResourceLoader resources) : BaseEndpoint(db) {
|
[ServerAuthorizationMiddleware.NoAuthorization]
|
||||||
private static readonly FileExtensionContentTypeProvider ContentTypeProvider = new ();
|
sealed class ViewerEndpoint(ResourceLoader resources) : BaseEndpoint {
|
||||||
|
|
||||||
private readonly Dictionary<string, byte[]?> cache = new ();
|
private readonly Dictionary<string, byte[]?> cache = new ();
|
||||||
private readonly SemaphoreSlim cacheSemaphore = new (1);
|
private readonly SemaphoreSlim cacheSemaphore = new (1);
|
||||||
|
|
||||||
@@ -31,12 +27,6 @@ sealed class ViewerEndpoint(IDatabaseFile db, ResourceLoader resources) : BaseEn
|
|||||||
cacheSemaphore.Release();
|
cacheSemaphore.Release();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (resourceBytes == null) {
|
await WriteFileIfFound(response, path, resourceBytes, cancellationToken);
|
||||||
throw new HttpException(HttpStatusCode.NotFound, "File not found: " + path);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
string? contentType = ContentTypeProvider.TryGetContentType(path, out string? type) ? type : null;
|
|
||||||
await response.WriteFileAsync(contentType, resourceBytes, cancellationToken);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -29,6 +29,11 @@
|
|||||||
<Link>Resources/Tracker/%(RecursiveDir)%(Filename)%(Extension)</Link>
|
<Link>Resources/Tracker/%(RecursiveDir)%(Filename)%(Extension)</Link>
|
||||||
<Visible>false</Visible>
|
<Visible>false</Visible>
|
||||||
</EmbeddedResource>
|
</EmbeddedResource>
|
||||||
|
<EmbeddedResource Include="../Resources/Tracker/loader/**">
|
||||||
|
<LogicalName>Tracker\loader\%(RecursiveDir)%(Filename)%(Extension)</LogicalName>
|
||||||
|
<Link>Resources/Tracker/loader/%(RecursiveDir)%(Filename)%(Extension)</Link>
|
||||||
|
<Visible>false</Visible>
|
||||||
|
</EmbeddedResource>
|
||||||
<EmbeddedResource Include="../Resources/Tracker/scripts/**">
|
<EmbeddedResource Include="../Resources/Tracker/scripts/**">
|
||||||
<LogicalName>Tracker\scripts\%(RecursiveDir)%(Filename)%(Extension)</LogicalName>
|
<LogicalName>Tracker\scripts\%(RecursiveDir)%(Filename)%(Extension)</LogicalName>
|
||||||
<Link>Resources/Tracker/scripts/%(RecursiveDir)%(Filename)%(Extension)</Link>
|
<Link>Resources/Tracker/scripts/%(RecursiveDir)%(Filename)%(Extension)</Link>
|
||||||
|
@@ -1,4 +1,6 @@
|
|||||||
|
using System;
|
||||||
using System.Net;
|
using System.Net;
|
||||||
|
using System.Reflection;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using DHT.Utils.Logging;
|
using DHT.Utils.Logging;
|
||||||
using Microsoft.AspNetCore.Http;
|
using Microsoft.AspNetCore.Http;
|
||||||
@@ -6,25 +8,11 @@ using Microsoft.Extensions.Primitives;
|
|||||||
|
|
||||||
namespace DHT.Server.Service.Middlewares;
|
namespace DHT.Server.Service.Middlewares;
|
||||||
|
|
||||||
sealed class ServerAuthorizationMiddleware {
|
sealed class ServerAuthorizationMiddleware(RequestDelegate next, ServerParameters serverParameters) {
|
||||||
private static readonly Log Log = Log.ForType<ServerAuthorizationMiddleware>();
|
private static readonly Log Log = Log.ForType<ServerAuthorizationMiddleware>();
|
||||||
|
|
||||||
private readonly RequestDelegate next;
|
|
||||||
private readonly ServerParameters serverParameters;
|
|
||||||
|
|
||||||
public ServerAuthorizationMiddleware(RequestDelegate next, ServerParameters serverParameters) {
|
|
||||||
this.next = next;
|
|
||||||
this.serverParameters = serverParameters;
|
|
||||||
}
|
|
||||||
|
|
||||||
public async Task InvokeAsync(HttpContext context) {
|
public async Task InvokeAsync(HttpContext context) {
|
||||||
HttpRequest request = context.Request;
|
if (SkipAuthorization(context) || CheckToken(context.Request)) {
|
||||||
|
|
||||||
bool success = HttpMethods.IsGet(request.Method)
|
|
||||||
? CheckToken(request.Query["token"])
|
|
||||||
: CheckToken(request.Headers["X-DHT-Token"]);
|
|
||||||
|
|
||||||
if (success) {
|
|
||||||
await next(context);
|
await next(context);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
@@ -32,6 +20,16 @@ sealed class ServerAuthorizationMiddleware {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static bool SkipAuthorization(HttpContext context) {
|
||||||
|
return context.GetEndpoint()?.RequestDelegate?.Target?.GetType().GetCustomAttribute<NoAuthorization>() != null;
|
||||||
|
}
|
||||||
|
|
||||||
|
private bool CheckToken(HttpRequest request) {
|
||||||
|
return HttpMethods.IsGet(request.Method)
|
||||||
|
? CheckToken(request.Query["token"])
|
||||||
|
: CheckToken(request.Headers["X-DHT-Token"]);
|
||||||
|
}
|
||||||
|
|
||||||
private bool CheckToken(StringValues token) {
|
private bool CheckToken(StringValues token) {
|
||||||
if (token.Count == 1 && token[0] == serverParameters.Token) {
|
if (token.Count == 1 && token[0] == serverParameters.Token) {
|
||||||
return true;
|
return true;
|
||||||
@@ -41,4 +39,7 @@ sealed class ServerAuthorizationMiddleware {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[AttributeUsage(AttributeTargets.Class)]
|
||||||
|
public sealed class NoAuthorization : Attribute;
|
||||||
}
|
}
|
||||||
|
@@ -38,25 +38,19 @@ sealed class Startup {
|
|||||||
public void Configure(IApplicationBuilder app, IHostApplicationLifetime lifetime, IDatabaseFile db, ServerParameters parameters, ResourceLoader resources, ViewerSessions viewerSessions) {
|
public void Configure(IApplicationBuilder app, IHostApplicationLifetime lifetime, IDatabaseFile db, ServerParameters parameters, ResourceLoader resources, ViewerSessions viewerSessions) {
|
||||||
app.UseMiddleware<ServerLoggingMiddleware>();
|
app.UseMiddleware<ServerLoggingMiddleware>();
|
||||||
app.UseCors();
|
app.UseCors();
|
||||||
|
|
||||||
app.Map("/viewer", node => {
|
|
||||||
node.UseRouting();
|
|
||||||
node.UseEndpoints(endpoints => {
|
|
||||||
endpoints.MapGet("/{**path}", new ViewerEndpoint(db, resources).Handle);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
app.UseMiddleware<ServerAuthorizationMiddleware>();
|
|
||||||
|
|
||||||
app.UseRouting();
|
app.UseRouting();
|
||||||
|
app.UseMiddleware<ServerAuthorizationMiddleware>();
|
||||||
app.UseEndpoints(endpoints => {
|
app.UseEndpoints(endpoints => {
|
||||||
endpoints.MapGet("/get-tracking-script", new GetTrackingScriptEndpoint(db, parameters, resources).Handle);
|
|
||||||
endpoints.MapGet("/get-viewer-metadata", new GetViewerMetadataEndpoint(db, viewerSessions).Handle);
|
|
||||||
endpoints.MapGet("/get-viewer-messages", new GetViewerMessagesEndpoint(db, viewerSessions).Handle);
|
|
||||||
endpoints.MapGet("/get-downloaded-file/{url}", new GetDownloadedFileEndpoint(db).Handle);
|
endpoints.MapGet("/get-downloaded-file/{url}", new GetDownloadedFileEndpoint(db).Handle);
|
||||||
|
endpoints.MapGet("/get-tracking-script", new GetTrackingScriptEndpoint(parameters, resources).Handle);
|
||||||
|
endpoints.MapGet("/get-userscript/{**ignored}", new GetUserscriptEndpoint(resources).Handle);
|
||||||
|
endpoints.MapGet("/get-viewer-messages", new GetViewerMessagesEndpoint(db, viewerSessions).Handle);
|
||||||
|
endpoints.MapGet("/get-viewer-metadata", new GetViewerMetadataEndpoint(db, viewerSessions).Handle);
|
||||||
|
endpoints.MapGet("/viewer/{**path}", new ViewerEndpoint(resources).Handle);
|
||||||
|
|
||||||
endpoints.MapPost("/track-channel", new TrackChannelEndpoint(db).Handle);
|
endpoints.MapPost("/track-channel", new TrackChannelEndpoint(db).Handle);
|
||||||
endpoints.MapPost("/track-users", new TrackUsersEndpoint(db).Handle);
|
|
||||||
endpoints.MapPost("/track-messages", new TrackMessagesEndpoint(db).Handle);
|
endpoints.MapPost("/track-messages", new TrackMessagesEndpoint(db).Handle);
|
||||||
|
endpoints.MapPost("/track-users", new TrackUsersEndpoint(db).Handle);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user