mirror of
https://github.com/claunia/marechai.git
synced 2025-12-16 11:04:25 +00:00
314 lines
16 KiB
C#
314 lines
16 KiB
C#
using System;
|
|
using System.Linq;
|
|
using System.Text.Json;
|
|
using System.Text.Json.Serialization;
|
|
using Aaru.CommonTypes.Interop;
|
|
using Marechai.Database;
|
|
using Marechai.Database.Models;
|
|
using Marechai.Helpers;
|
|
using Marechai.Server.Helpers;
|
|
using Marechai.Server.Services;
|
|
using Markdig;
|
|
using Microsoft.AspNetCore.Authentication.JwtBearer;
|
|
using Microsoft.AspNetCore.Builder;
|
|
using Microsoft.AspNetCore.Routing;
|
|
using Microsoft.EntityFrameworkCore;
|
|
using Microsoft.Extensions.Configuration;
|
|
using Microsoft.Extensions.DependencyInjection;
|
|
using Microsoft.Extensions.Hosting;
|
|
using Microsoft.IdentityModel.Tokens;
|
|
using Version = Marechai.Server.Interop.Version;
|
|
|
|
namespace Marechai.Server;
|
|
|
|
file class Program
|
|
{
|
|
private static IDbCore _database;
|
|
|
|
public static void Main(string[] args)
|
|
{
|
|
Console.Clear();
|
|
|
|
Console.Write("\e[32m . ,,\n" +
|
|
"\e[32m ;,. '0d.\n" +
|
|
"\e[32m oc oWd \e[31m" +
|
|
@"________/\\\\\\\\\___/\\\\\\\\\\\_________/\\\\\\\\\___/\\\\____________/\\\\_" +
|
|
"\n\e[0m" +
|
|
"\e[32m ;X. 'WN' \e[31m" +
|
|
@" _____/\\\////////___\/////\\\///_______/\\\////////___\/\\\\\\________/\\\\\\_" +
|
|
"\n\e[0m" +
|
|
"\e[32m oMo cMM: \e[31m" +
|
|
@" ___/\\\/________________\/\\\________/\\\/____________\/\\\//\\\____/\\\//\\\_" +
|
|
"\n\e[0m" +
|
|
"\e[32m ;MM. .MMM; \e[31m" +
|
|
@" __/\\\__________________\/\\\_______/\\\______________\/\\\\///\\\/\\\/_\/\\\_" +
|
|
"\n\e[0m" +
|
|
"\e[32m NMM WMMW \e[31m" +
|
|
@" _\/\\\__________________\/\\\______\/\\\______________\/\\\__\///\\\/___\/\\\_" +
|
|
"\n\e[0m" +
|
|
"\e[32m 'MMM MMMM; \e[31m" +
|
|
@" _\//\\\_________________\/\\\______\//\\\_____________\/\\\____\///_____\/\\\_" +
|
|
"\n\e[0m" +
|
|
"\e[32m ,MMM: dMMMM: \e[31m" +
|
|
@" __\///\\\_______________\/\\\_______\///\\\___________\/\\\_____________\/\\\_" +
|
|
"\n\e[0m" +
|
|
"\e[32m .MMMW. :MMMMM. \e[31m" +
|
|
@" ____\////\\\\\\\\\___/\\\\\\\\\\\_____\////\\\\\\\\\__\/\\\_____________\/\\\_" +
|
|
"\n\e[0m" +
|
|
"\e[32m XMMMW: .:xKNMMMMMMN0d, lMMMMMd \e[31m" +
|
|
@" _______\/////////___\///////////_________\/////////___\///______________\///__" +
|
|
"\n\e[0m" +
|
|
"\e[32m :MMMMMK; cWMNkl:;;;:lxKMXc .0MMMMMO \e[37;1m MARECHAI\e[0m\n" +
|
|
"\e[32m ..KMMMMMMNo,. ,OMMMMMMW:,. \e[37;1m Master repository of computing history artifacts information\e[0m\n" +
|
|
"\e[32m .;d0NMMMMMMMMMMMMMMW0d:' .;lOWMMMMMMMMMMMMMXkl. \e[37;1m Version \e[0m\e[33m{0}\e[37;1m-\e[0m\e[31m{1}\e[0m\n" +
|
|
"\e[32m :KMMMMMMMMMMMMMMMMMMMMMMMMc WMMMMMMMMMMMMMMMMMMMMMMWk'\e[0m\n" +
|
|
"\e[32m ;NMMMMWX0kkkkO0XMMMMMMMMMMM0' dNMMMMMMMMMMW0xl:;,;:oOWMMX; \e[37;1m Running under \e[35;1m{2}\e[37;1m, \e[35m{3}-bit\e[37;1m in \e[35m{4}-bit\e[37;1m mode.\e[0m\n" +
|
|
"\e[32m xMMWk:. .c0MMMMMW' OMMMMMM0c'.. .oNMO \e[37;1m Using \e[33;1m{5}\e[37;1m version \e[31;1m{6}\e[0m\n" +
|
|
"\e[32m OMNc .MNc oWMMk 'WMMNl. .MMK ;KX.\e[0m\n" +
|
|
"\e[32m xMO WMN ; ., , ': ,MMx lK\e[0m\n" +
|
|
"\e[32m ,Md cMMl .XMMMWWMMMO XMW. :\e[0m\n" +
|
|
"\e[32m Ok xMMl XMMMMMMMMc 0MW,\e[0m\n" +
|
|
"\e[32m 0 oMM0' lMMMMMMMM. :NMN'\e[0m\n" +
|
|
"\e[32m . .0MMKl ;MMMMMMMM oNMWd\e[0m\n" +
|
|
"\e[32m .dNW cMMMMMMMM, XKl\e[0m\n" +
|
|
"\e[32m 0MMMMMMMMK\e[0m\n" +
|
|
"\e[32m ;MMMMMMMMMMO \e[37;1m Proudly presented to you by:\e[0m\n" +
|
|
"\e[32m 'WMMMMKxMMMMM0 \e[34;1m Natalia Portillo\e[0m\n" +
|
|
"\e[32m oMMMMNc :WMMMMN:\e[0m\n" +
|
|
"\e[32m .dWMMM0; dWMMMMXl. \e[37;1m Thanks to all contributors, collaborators, translators, donators and friends.\e[0m\n" +
|
|
"\e[32m .......,cd0WMMNk: c0MMMMMWKkolc:clodc'\e[0m\n" +
|
|
"\e[32m .';loddol:'. ':oxkkOkkxoc,.\e[0m\n" +
|
|
"\e[0m\n",
|
|
Version.GetVersion(),
|
|
#if DEBUG
|
|
"DEBUG"
|
|
#else
|
|
"RELEASE"
|
|
#endif
|
|
,
|
|
DetectOS.GetPlatformName(DetectOS.GetRealPlatformID()),
|
|
Environment.Is64BitOperatingSystem ? 64 : 32,
|
|
Environment.Is64BitProcess ? 64 : 32,
|
|
DetectOS.IsMono ? "Mono" : ".NET Core",
|
|
DetectOS.IsMono ? Version.GetMonoVersion() : Version.GetNetCoreVersion());
|
|
|
|
Console.WriteLine("\e[31;1mUpdating MySQL database without Entity Framework if it exists...\e[0m");
|
|
_database = new Mysql();
|
|
|
|
IConfigurationBuilder configurationBuilder = new ConfigurationBuilder().AddJsonFile("appsettings.json");
|
|
IConfigurationRoot configuration = configurationBuilder.Build();
|
|
string connectionString = configuration.GetConnectionString("DefaultConnection");
|
|
|
|
if(connectionString is null)
|
|
Console.WriteLine("\e[31;1mCould not find a correct connection string...\e[0m");
|
|
else
|
|
{
|
|
string server = null, user = null, database = null, password = null;
|
|
ushort port = 0;
|
|
string[] pieces = connectionString.Split(';');
|
|
|
|
foreach(string piece in pieces)
|
|
{
|
|
if(piece.StartsWith("server=", StringComparison.Ordinal))
|
|
server = piece[7..];
|
|
else if(piece.StartsWith("user=", StringComparison.Ordinal))
|
|
user = piece[5..];
|
|
else if(piece.StartsWith("password=", StringComparison.Ordinal))
|
|
password = piece[9..];
|
|
else if(piece.StartsWith("database=", StringComparison.Ordinal))
|
|
database = piece[9..];
|
|
else if(piece.StartsWith("port=", StringComparison.Ordinal))
|
|
{
|
|
string portString = piece[5..];
|
|
|
|
ushort.TryParse(portString, out port);
|
|
}
|
|
}
|
|
|
|
if(server is null || user is null || database is null || password is null || port == 0)
|
|
Console.WriteLine("\e[31;1mCould not find a correct connection string...\e[0m");
|
|
|
|
else
|
|
{
|
|
bool res = _database.OpenDb(server, user, database, password, port);
|
|
|
|
if(res)
|
|
{
|
|
Console.WriteLine("\e[31;1mClosing database...\e[0m");
|
|
_database.CloseDb();
|
|
}
|
|
}
|
|
}
|
|
|
|
DateTime start = DateTime.Now;
|
|
Console.WriteLine("\e[31;1mRendering new country flags...\e[0m");
|
|
SvgRender.RenderCountries();
|
|
DateTime end = DateTime.Now;
|
|
|
|
Console.WriteLine("\e[31;1mTook \e[32;1m{0} seconds\e[31;1m...\e[0m", (end - start).TotalSeconds);
|
|
|
|
WebApplicationBuilder builder = WebApplication.CreateBuilder(args);
|
|
|
|
// Add services to the container.
|
|
builder.Services.AddControllers()
|
|
.AddJsonOptions(options =>
|
|
{
|
|
options.JsonSerializerOptions.Converters.Add(new IsoDateTimeConverter());
|
|
options.JsonSerializerOptions.PropertyNamingPolicy = JsonNamingPolicy.CamelCase;
|
|
options.JsonSerializerOptions.WriteIndented = true;
|
|
options.JsonSerializerOptions.ReferenceHandler = ReferenceHandler.IgnoreCycles;
|
|
});
|
|
|
|
// Learn more about configuring OpenAPI at https://aka.ms/aspnet/openapi
|
|
builder.Services.AddOpenApi();
|
|
|
|
// Add custom route constraints
|
|
builder.Services.Configure<RouteOptions>(options =>
|
|
{
|
|
options.ConstraintMap["ulong"] = typeof(UlongRouteConstraint);
|
|
options.ConstraintMap["char"] = typeof(CharRouteConstraint);
|
|
});
|
|
|
|
builder.Services.AddAuthentication(options =>
|
|
{
|
|
options.DefaultAuthenticateScheme = JwtBearerDefaults.AuthenticationScheme;
|
|
options.DefaultChallengeScheme = JwtBearerDefaults.AuthenticationScheme;
|
|
options.DefaultScheme = JwtBearerDefaults.AuthenticationScheme;
|
|
})
|
|
.AddJwtBearer(options =>
|
|
{
|
|
options.SaveToken = true;
|
|
|
|
options.TokenValidationParameters = new TokenValidationParameters
|
|
{
|
|
ClockSkew = TimeSpan.Zero,
|
|
ValidateIssuer = true,
|
|
ValidateAudience = true,
|
|
ValidateLifetime = false,
|
|
ValidateIssuerSigningKey = true,
|
|
ValidIssuer = "apiWithAuthBackend",
|
|
ValidAudience = "apiWithAuthBackend",
|
|
IssuerSigningKey =
|
|
new
|
|
SymmetricSecurityKey("!SomethingSecret!!SomethingSecret!!SomethingSecret!!SomethingSecret!"u8
|
|
.ToArray())
|
|
};
|
|
});
|
|
|
|
builder.Services.AddDbContextFactory<MarechaiContext>(options => options.UseLazyLoadingProxies()
|
|
.UseMySql(builder.Configuration
|
|
.GetConnectionString("DefaultConnection"),
|
|
new
|
|
MariaDbServerVersion(new System.
|
|
Version(10, 5, 0)),
|
|
b => b.UseMicrosoftJson()));
|
|
|
|
builder.Services.AddDefaultIdentity<ApplicationUser>(options => options.SignIn.RequireConfirmedAccount = true)
|
|
.AddRoles<ApplicationRole>()
|
|
.AddEntityFrameworkStores<MarechaiContext>();
|
|
|
|
builder.Services.AddScoped<TokenService, TokenService>();
|
|
|
|
// Read allowed CORS origins from configuration
|
|
string[] allowedOrigins = builder.Configuration.GetSection("CORS:AllowedOrigins").Get<string[]>();
|
|
|
|
builder.Services.AddCors(options =>
|
|
{
|
|
options.AddPolicy("AllowFrontend",
|
|
policy =>
|
|
{
|
|
switch(allowedOrigins)
|
|
{
|
|
case ["*"]:
|
|
policy.AllowAnyOrigin().AllowAnyHeader().AllowAnyMethod();
|
|
|
|
break;
|
|
case { Length: > 0 }:
|
|
policy.WithOrigins(allowedOrigins).AllowAnyHeader().AllowAnyMethod();
|
|
|
|
break;
|
|
}
|
|
});
|
|
});
|
|
|
|
WebApplication app = builder.Build();
|
|
|
|
// Configure the HTTP request pipeline.
|
|
if(app.Environment.IsDevelopment()) app.MapOpenApi();
|
|
|
|
app.UseHttpsRedirection();
|
|
|
|
// Use CORS before authentication/authorization
|
|
app.UseCors("AllowFrontend");
|
|
|
|
app.UseAuthentication();
|
|
app.UseAuthorization();
|
|
|
|
app.MapControllers();
|
|
|
|
using(IServiceScope scope = app.Services.CreateScope())
|
|
{
|
|
IServiceProvider services = scope.ServiceProvider;
|
|
|
|
try
|
|
{
|
|
start = DateTime.Now;
|
|
Console.WriteLine("\e[31;1mUpdating database with Entity Framework...\e[0m");
|
|
MarechaiContext context = services.GetRequiredService<MarechaiContext>();
|
|
context.Database.Migrate();
|
|
end = DateTime.Now;
|
|
|
|
Console.WriteLine("\e[31;1mTook \e[32;1m{0} seconds\e[31;1m...\e[0m", (end - start).TotalSeconds);
|
|
|
|
start = DateTime.Now;
|
|
Console.WriteLine("\e[31;1mImporting company logos...\e[0m");
|
|
SvgRender.ImportCompanyLogos(context);
|
|
end = DateTime.Now;
|
|
|
|
Console.WriteLine("\e[31;1mTook \e[32;1m{0} seconds\e[31;1m...\e[0m", (end - start).TotalSeconds);
|
|
|
|
start = DateTime.Now;
|
|
Console.WriteLine("\e[31;1mRendering markdown in company descriptions...\e[0m");
|
|
MarkdownPipeline pipeline = new MarkdownPipelineBuilder().UseAdvancedExtensions().Build();
|
|
|
|
foreach(CompanyDescription companyDescription in
|
|
context.CompanyDescriptions.Where(cd => cd.Html == null))
|
|
{
|
|
companyDescription.Html = Markdown.ToHtml(companyDescription.Text, pipeline);
|
|
context.Update(companyDescription);
|
|
}
|
|
|
|
context.SaveChanges();
|
|
|
|
end = DateTime.Now;
|
|
|
|
Console.WriteLine("\e[31;1mTook \e[32;1m{0} seconds\e[31;1m...\e[0m", (end - start).TotalSeconds);
|
|
|
|
start = DateTime.Now;
|
|
Console.WriteLine("\e[31;1mEnsuring photo folders exist...\e[0m");
|
|
Photos.EnsureCreated("wwwroot", false, "machines");
|
|
end = DateTime.Now;
|
|
|
|
start = DateTime.Now;
|
|
Console.WriteLine("\e[31;1mEnsuring scan folders exist...\e[0m");
|
|
Photos.EnsureCreated("wwwroot", true, "books");
|
|
Photos.EnsureCreated("wwwroot", true, "documents");
|
|
Photos.EnsureCreated("wwwroot", true, "magazines");
|
|
end = DateTime.Now;
|
|
|
|
Console.WriteLine("\e[31;1mTook \e[32;1m{0} seconds\e[31;1m...\e[0m", (end - start).TotalSeconds);
|
|
}
|
|
catch(Exception ex)
|
|
{
|
|
Console.WriteLine("\e[31;1mCould not open database...\e[0m");
|
|
#if DEBUG
|
|
Console.WriteLine("\e[31;1mException: {0}\e[0m", ex.Message);
|
|
#endif
|
|
return;
|
|
}
|
|
}
|
|
|
|
Console.WriteLine("\e[31;1mStarting API server...\e[0m");
|
|
app.Run();
|
|
}
|
|
} |