mirror of
https://github.com/wabbajack-tools/wabbajack.git
synced 2024-08-30 18:42:17 +00:00
WJ accepts Github PAT as author keys
This commit is contained in:
parent
f21b4569a2
commit
b65a014ff7
@ -1,12 +1,15 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
|
using System.Net.Http;
|
||||||
|
using System.Net.Http.Json;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using Microsoft.Extensions.Logging;
|
using Microsoft.Extensions.Logging;
|
||||||
using Octokit;
|
using Octokit;
|
||||||
using Wabbajack.DTOs;
|
using Wabbajack.DTOs;
|
||||||
using Wabbajack.DTOs.GitHub;
|
using Wabbajack.DTOs.GitHub;
|
||||||
using Wabbajack.DTOs.JsonConverters;
|
using Wabbajack.DTOs.JsonConverters;
|
||||||
|
using Wabbajack.Networking.GitHub.DTOs;
|
||||||
|
|
||||||
namespace Wabbajack.Networking.GitHub;
|
namespace Wabbajack.Networking.GitHub;
|
||||||
|
|
||||||
@ -16,9 +19,11 @@ public class Client
|
|||||||
private readonly DTOSerializer _dtos;
|
private readonly DTOSerializer _dtos;
|
||||||
private readonly ILogger<Client> _logger;
|
private readonly ILogger<Client> _logger;
|
||||||
private readonly GithubAuthTokenProvider _token;
|
private readonly GithubAuthTokenProvider _token;
|
||||||
|
private readonly HttpClient _httpClient;
|
||||||
|
|
||||||
public Client(ILogger<Client> logger, DTOSerializer dtos, GitHubClient client)
|
public Client(ILogger<Client> logger, DTOSerializer dtos, GitHubClient client, HttpClient httpClient)
|
||||||
{
|
{
|
||||||
|
_httpClient = httpClient;
|
||||||
_logger = logger;
|
_logger = logger;
|
||||||
_client = client;
|
_client = client;
|
||||||
_dtos = dtos;
|
_dtos = dtos;
|
||||||
@ -71,6 +76,16 @@ public class Client
|
|||||||
return (result.Sha, result.Content);
|
return (result.Sha, result.Content);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public async Task<UserInfo?> GetUserInfoFromPAT(string pat)
|
||||||
|
{
|
||||||
|
var msg = new HttpRequestMessage(HttpMethod.Get, "https://api.github.com/user");
|
||||||
|
msg.Headers.Add("User-Agent", "wabbajack");
|
||||||
|
msg.Headers.Add("Authorization", "Token " + pat);
|
||||||
|
var result = await _httpClient.SendAsync(msg);
|
||||||
|
if (!result.IsSuccessStatusCode) return null;
|
||||||
|
return await result.Content.ReadFromJsonAsync<UserInfo>();
|
||||||
|
}
|
||||||
|
|
||||||
public async Task PutData(string owner, string repo, string path, string message, string content, string oldSha)
|
public async Task PutData(string owner, string repo, string path, string message, string content, string oldSha)
|
||||||
{
|
{
|
||||||
await _client.Repository.Content.UpdateFile(owner, repo, path, new UpdateFileRequest(message, content, oldSha));
|
await _client.Repository.Content.UpdateFile(owner, repo, path, new UpdateFileRequest(message, content, oldSha));
|
||||||
|
104
Wabbajack.Networking.GitHub/DTOs/UserInfo.cs
Normal file
104
Wabbajack.Networking.GitHub/DTOs/UserInfo.cs
Normal file
@ -0,0 +1,104 @@
|
|||||||
|
using System;
|
||||||
|
using System.Text.Json.Serialization;
|
||||||
|
|
||||||
|
namespace Wabbajack.Networking.GitHub.DTOs;
|
||||||
|
|
||||||
|
|
||||||
|
public class UserInfo
|
||||||
|
{
|
||||||
|
[JsonPropertyName("login")]
|
||||||
|
public string Login { get; set; } = "";
|
||||||
|
|
||||||
|
[JsonPropertyName("id")]
|
||||||
|
public int Id { get; set; }
|
||||||
|
|
||||||
|
[JsonPropertyName("node_id")]
|
||||||
|
public string NodeId { get; set; }
|
||||||
|
|
||||||
|
[JsonPropertyName("avatar_url")]
|
||||||
|
public string AvatarUrl { get; set; }
|
||||||
|
|
||||||
|
[JsonPropertyName("gravatar_id")]
|
||||||
|
public string GravatarId { get; set; }
|
||||||
|
|
||||||
|
[JsonPropertyName("url")]
|
||||||
|
public string Url { get; set; }
|
||||||
|
|
||||||
|
[JsonPropertyName("html_url")]
|
||||||
|
public string HtmlUrl { get; set; }
|
||||||
|
|
||||||
|
[JsonPropertyName("followers_url")]
|
||||||
|
public string FollowersUrl { get; set; }
|
||||||
|
|
||||||
|
[JsonPropertyName("following_url")]
|
||||||
|
public string FollowingUrl { get; set; }
|
||||||
|
|
||||||
|
[JsonPropertyName("gists_url")]
|
||||||
|
public string GistsUrl { get; set; }
|
||||||
|
|
||||||
|
[JsonPropertyName("starred_url")]
|
||||||
|
public string StarredUrl { get; set; }
|
||||||
|
|
||||||
|
[JsonPropertyName("subscriptions_url")]
|
||||||
|
public string SubscriptionsUrl { get; set; }
|
||||||
|
|
||||||
|
[JsonPropertyName("organizations_url")]
|
||||||
|
public string OrganizationsUrl { get; set; }
|
||||||
|
|
||||||
|
[JsonPropertyName("repos_url")]
|
||||||
|
public string ReposUrl { get; set; }
|
||||||
|
|
||||||
|
[JsonPropertyName("events_url")]
|
||||||
|
public string EventsUrl { get; set; }
|
||||||
|
|
||||||
|
[JsonPropertyName("received_events_url")]
|
||||||
|
public string ReceivedEventsUrl { get; set; }
|
||||||
|
|
||||||
|
[JsonPropertyName("type")]
|
||||||
|
public string Type { get; set; }
|
||||||
|
|
||||||
|
[JsonPropertyName("site_admin")]
|
||||||
|
public bool SiteAdmin { get; set; }
|
||||||
|
|
||||||
|
[JsonPropertyName("name")]
|
||||||
|
public string Name { get; set; }
|
||||||
|
|
||||||
|
[JsonPropertyName("company")]
|
||||||
|
public object Company { get; set; }
|
||||||
|
|
||||||
|
[JsonPropertyName("blog")]
|
||||||
|
public string Blog { get; set; }
|
||||||
|
|
||||||
|
[JsonPropertyName("location")]
|
||||||
|
public string Location { get; set; }
|
||||||
|
|
||||||
|
[JsonPropertyName("email")]
|
||||||
|
public string Email { get; set; }
|
||||||
|
|
||||||
|
[JsonPropertyName("hireable")]
|
||||||
|
public object Hireable { get; set; }
|
||||||
|
|
||||||
|
[JsonPropertyName("bio")]
|
||||||
|
public object Bio { get; set; }
|
||||||
|
|
||||||
|
[JsonPropertyName("twitter_username")]
|
||||||
|
public object TwitterUsername { get; set; }
|
||||||
|
|
||||||
|
[JsonPropertyName("public_repos")]
|
||||||
|
public int PublicRepos { get; set; }
|
||||||
|
|
||||||
|
[JsonPropertyName("public_gists")]
|
||||||
|
public int PublicGists { get; set; }
|
||||||
|
|
||||||
|
[JsonPropertyName("followers")]
|
||||||
|
public int Followers { get; set; }
|
||||||
|
|
||||||
|
[JsonPropertyName("following")]
|
||||||
|
public int Following { get; set; }
|
||||||
|
|
||||||
|
[JsonPropertyName("created_at")]
|
||||||
|
public DateTime CreatedAt { get; set; }
|
||||||
|
|
||||||
|
[JsonPropertyName("updated_at")]
|
||||||
|
public DateTime UpdatedAt { get; set; }
|
||||||
|
}
|
@ -1,16 +1,13 @@
|
|||||||
using System;
|
using System.Security.Claims;
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Linq;
|
|
||||||
using System.Security.Claims;
|
|
||||||
using System.Text.Encodings.Web;
|
using System.Text.Encodings.Web;
|
||||||
using System.Threading.Tasks;
|
|
||||||
using Microsoft.AspNetCore.Authentication;
|
using Microsoft.AspNetCore.Authentication;
|
||||||
using Microsoft.AspNetCore.Http;
|
using Microsoft.AspNetCore.Http;
|
||||||
|
using Microsoft.Extensions.Caching.Memory;
|
||||||
using Microsoft.Extensions.Logging;
|
using Microsoft.Extensions.Logging;
|
||||||
using Microsoft.Extensions.Options;
|
using Microsoft.Extensions.Options;
|
||||||
using Wabbajack.DTOs.JsonConverters;
|
using Wabbajack.DTOs.JsonConverters;
|
||||||
using Wabbajack.Paths;
|
using Wabbajack.Networking.GitHub;
|
||||||
using Wabbajack.Paths.IO;
|
using Wabbajack.Networking.GitHub.DTOs;
|
||||||
using Wabbajack.Server.DataModels;
|
using Wabbajack.Server.DataModels;
|
||||||
using Wabbajack.Server.DTOs;
|
using Wabbajack.Server.DTOs;
|
||||||
|
|
||||||
@ -33,10 +30,13 @@ public class ApiKeyAuthenticationHandler : AuthenticationHandler<ApiKeyAuthentic
|
|||||||
private readonly Task<HashSet<string>> _tarKeys;
|
private readonly Task<HashSet<string>> _tarKeys;
|
||||||
private readonly Metrics _metricsStore;
|
private readonly Metrics _metricsStore;
|
||||||
private readonly TarLog _tarLog;
|
private readonly TarLog _tarLog;
|
||||||
|
private readonly Client _githubClient;
|
||||||
|
private readonly MemoryCache _githubCache;
|
||||||
|
|
||||||
public ApiKeyAuthenticationHandler(
|
public ApiKeyAuthenticationHandler(
|
||||||
IOptionsMonitor<ApiKeyAuthenticationOptions> options,
|
IOptionsMonitor<ApiKeyAuthenticationOptions> options,
|
||||||
AuthorKeys authorKeys,
|
AuthorKeys authorKeys,
|
||||||
|
Client githubClient,
|
||||||
ILoggerFactory logger,
|
ILoggerFactory logger,
|
||||||
UrlEncoder encoder,
|
UrlEncoder encoder,
|
||||||
ISystemClock clock,
|
ISystemClock clock,
|
||||||
@ -51,6 +51,8 @@ public class ApiKeyAuthenticationHandler : AuthenticationHandler<ApiKeyAuthentic
|
|||||||
_dtos = dtos;
|
_dtos = dtos;
|
||||||
_authorKeys = authorKeys;
|
_authorKeys = authorKeys;
|
||||||
_settings = settings;
|
_settings = settings;
|
||||||
|
_githubClient = githubClient;
|
||||||
|
_githubCache = new MemoryCache(new MemoryCacheOptions());
|
||||||
}
|
}
|
||||||
|
|
||||||
protected override async Task<AuthenticateResult> HandleAuthenticateAsync()
|
protected override async Task<AuthenticateResult> HandleAuthenticateAsync()
|
||||||
@ -58,7 +60,6 @@ public class ApiKeyAuthenticationHandler : AuthenticationHandler<ApiKeyAuthentic
|
|||||||
var metricsKey = Request.Headers[_settings.MetricsKeyHeader].FirstOrDefault();
|
var metricsKey = Request.Headers[_settings.MetricsKeyHeader].FirstOrDefault();
|
||||||
// Never needed this, disabled for now
|
// Never needed this, disabled for now
|
||||||
//await LogRequest(metricsKey);
|
//await LogRequest(metricsKey);
|
||||||
var ip = Request.HttpContext.Connection.RemoteIpAddress?.ToString() ?? "";
|
|
||||||
if (metricsKey != default)
|
if (metricsKey != default)
|
||||||
{
|
{
|
||||||
if (await _tarLog.Contains(metricsKey))
|
if (await _tarLog.Contains(metricsKey))
|
||||||
@ -69,7 +70,7 @@ public class ApiKeyAuthenticationHandler : AuthenticationHandler<ApiKeyAuthentic
|
|||||||
Action = "tarlog",
|
Action = "tarlog",
|
||||||
MetricsKey = metricsKey,
|
MetricsKey = metricsKey,
|
||||||
UserAgent = Request.Headers.UserAgent,
|
UserAgent = Request.Headers.UserAgent,
|
||||||
Ip = ip
|
Ip = Request.HttpContext.Connection.RemoteIpAddress?.ToString() ?? ""
|
||||||
});
|
});
|
||||||
await Task.Delay(TimeSpan.FromSeconds(20));
|
await Task.Delay(TimeSpan.FromSeconds(20));
|
||||||
throw new Exception("Error, lipsum timeout of the cross distant cloud.");
|
throw new Exception("Error, lipsum timeout of the cross distant cloud.");
|
||||||
@ -88,8 +89,15 @@ public class ApiKeyAuthenticationHandler : AuthenticationHandler<ApiKeyAuthentic
|
|||||||
{
|
{
|
||||||
var owner = await _authorKeys.AuthorForKey(authorKey);
|
var owner = await _authorKeys.AuthorForKey(authorKey);
|
||||||
if (owner == null)
|
if (owner == null)
|
||||||
|
{
|
||||||
|
var ghUser = await GetGithubUserInfo(authorKey);
|
||||||
|
|
||||||
|
if (ghUser == null)
|
||||||
return AuthenticateResult.Fail("Invalid author key");
|
return AuthenticateResult.Fail("Invalid author key");
|
||||||
|
|
||||||
|
owner = "github/" + ghUser.Login;
|
||||||
|
}
|
||||||
|
|
||||||
var claims = new List<Claim> {new(ClaimTypes.Name, owner)};
|
var claims = new List<Claim> {new(ClaimTypes.Name, owner)};
|
||||||
|
|
||||||
claims.Add(new Claim(ClaimTypes.Role, "Author"));
|
claims.Add(new Claim(ClaimTypes.Role, "Author"));
|
||||||
@ -120,6 +128,18 @@ public class ApiKeyAuthenticationHandler : AuthenticationHandler<ApiKeyAuthentic
|
|||||||
return AuthenticateResult.NoResult();
|
return AuthenticateResult.NoResult();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected async Task<UserInfo?> GetGithubUserInfo(string authToken)
|
||||||
|
{
|
||||||
|
if (_githubCache.TryGetValue<UserInfo>(authToken, out var value)) return value;
|
||||||
|
|
||||||
|
var info = await _githubClient.GetUserInfoFromPAT(authToken);
|
||||||
|
if (info != null)
|
||||||
|
_githubCache.Set(authToken, info,
|
||||||
|
new MemoryCacheEntryOptions().SetSlidingExpiration(TimeSpan.FromHours(6)));
|
||||||
|
|
||||||
|
return info;
|
||||||
|
}
|
||||||
|
|
||||||
protected override async Task HandleChallengeAsync(AuthenticationProperties properties)
|
protected override async Task HandleChallengeAsync(AuthenticationProperties properties)
|
||||||
{
|
{
|
||||||
Response.StatusCode = 401;
|
Response.StatusCode = 401;
|
||||||
|
@ -79,7 +79,7 @@ public class MetricsController : ControllerBase
|
|||||||
Subject = value,
|
Subject = value,
|
||||||
MetricsKey = metricsKey,
|
MetricsKey = metricsKey,
|
||||||
UserAgent = Request.Headers.UserAgent.FirstOrDefault() ?? "<unknown>",
|
UserAgent = Request.Headers.UserAgent.FirstOrDefault() ?? "<unknown>",
|
||||||
Ip = Request.HttpContext.Connection.RemoteIpAddress?.ToString() ?? ""
|
Ip = Request.Headers["cf-connecting-ip"].FirstOrDefault() ?? (Request.HttpContext.Connection.RemoteIpAddress?.ToString() ?? "")
|
||||||
});
|
});
|
||||||
return new Result {Timestamp = date};
|
return new Result {Timestamp = date};
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user