using Mapster; using Microsoft.AspNetCore.Authentication.JwtBearer; using Microsoft.AspNetCore.Builder; using Microsoft.AspNetCore.Hosting; using Microsoft.AspNetCore.Identity; using Microsoft.AspNetCore.Identity.UI.Services; using Microsoft.AspNetCore.Mvc; using Microsoft.AspNetCore.Mvc.Infrastructure; using Microsoft.AspNetCore.Mvc.Routing; using Microsoft.EntityFrameworkCore; using Microsoft.Extensions.Configuration; using Microsoft.Extensions.DependencyInjection; using Microsoft.Extensions.Logging; using Microsoft.IdentityModel.Tokens; using Newtonsoft.Json; using Roadie.Api.Hubs; using Roadie.Api.ModelBinding; using Roadie.Api.Services; using Roadie.Library.Caching; using Roadie.Library.Configuration; using Roadie.Library.Data; using Roadie.Library.Encoding; using Roadie.Library.Identity; using Roadie.Library.Imaging; using Roadie.Library.Utility; using System; using System.Diagnostics; namespace Roadie.Api { public class Startup { private readonly IConfiguration _configuration; private readonly ILoggerFactory _loggerFactory; private ILogger Logger { get; } public Startup(IConfiguration configuration, ILoggerFactory loggerFactory) { this._configuration = configuration; this._loggerFactory = loggerFactory; this.Logger = this._loggerFactory.CreateLogger(); TypeAdapterConfig .NewConfig() .Map(i => i.ArtistId, src => src.Artist == null ? null : (Guid?)src.Artist.RoadieId) .Map(i => i.ReleaseId, src => src.Release == null ? null : (Guid?)src.Release.RoadieId) .Compile(); TypeAdapterConfig.GlobalSettings.Default.PreserveReference(true); } // This method gets called by the runtime. Use this method to configure the HTTP request pipeline. public void Configure(IApplicationBuilder app, ILoggerFactory loggerFactory, IHostingEnvironment env) { if (env.IsDevelopment()) { app.UseDeveloperExceptionPage(); } app.UseCors("CORSPolicy"); app.UseAuthentication(); //app.UseSwagger(); //app.UseSwaggerUI(c => //{ // c.SwaggerEndpoint("/swagger/swagger.json", "Roadie API"); // c.RoutePrefix = string.Empty; //}); app.UseStaticFiles(); app.UseSignalR(routes => { routes.MapHub("/playActivityHub"); routes.MapHub("/scanActivityHub"); }); app.UseMvc(); } // This method gets called by the runtime. Use this method to add services to the container. public void ConfigureServices(IServiceCollection services) { services.AddSingleton(); services.AddSingleton(); services.AddSingleton(); var cacheManager = new DictionaryCacheManager(this._loggerFactory.CreateLogger(), new CachePolicy(TimeSpan.FromHours(4))); services.AddSingleton(cacheManager); services.AddDbContextPool( options => options.UseMySql(this._configuration.GetConnectionString("RoadieDatabaseConnection"), mySqlOptions => { mySqlOptions.ServerVersion(new Version(5, 5), Pomelo.EntityFrameworkCore.MySql.Infrastructure.ServerType.MariaDb); } )); services.AddDbContextPool( options => options.UseMySql(this._configuration.GetConnectionString("RoadieDatabaseConnection"), mySqlOptions => { mySqlOptions.ServerVersion(new Version(5, 5), Pomelo.EntityFrameworkCore.MySql.Infrastructure.ServerType.MariaDb); } )); services.AddIdentity() .AddRoles() .AddEntityFrameworkStores() .AddDefaultTokenProviders(); services.AddAuthorization(options => { options.AddPolicy("Admin", policy => policy.RequireRole("Admin")); options.AddPolicy("Editor", policy => policy.RequireRole("Admin", "Editor")); }); services.Configure(this._configuration); var corsOrigins = (this._configuration["CORSOrigins"] ?? "http://localhost:8080").Split('|'); this.Logger.LogDebug("Setting Up CORS Policy [{0}]", string.Join(", ", corsOrigins)); services.AddCors(options => options.AddPolicy("CORSPolicy", builder => { builder .WithOrigins(corsOrigins) .AllowAnyHeader() .AllowAnyMethod() .AllowCredentials(); })); services.AddSingleton(ctx => { var settings = new RoadieSettings(); var configuration = ctx.GetService(); configuration.GetSection("RoadieSettings").Bind(settings); var hostingEnvironment = ctx.GetService(); settings.ContentPath = hostingEnvironment.WebRootPath; settings.ConnectionString = this._configuration.GetConnectionString("RoadieDatabaseConnection"); var integrationKeys = this._configuration.GetSection("IntegrationKeys") .Get(); if(integrationKeys == null) { Console.WriteLine("Unable to find IntegrationKeys, Integrations will not have proper API keys setup."); } else if (integrationKeys != null) { settings.Integrations.ApiKeys = new System.Collections.Generic.List { new ApiKey { ApiName = "LastFMApiKey", Key = integrationKeys.LastFMApiKey, KeySecret = integrationKeys.LastFMSecret }, new ApiKey { ApiName = "DiscogsConsumerKey", Key = integrationKeys.DiscogsConsumerKey, KeySecret = integrationKeys.DiscogsConsumerSecret }, new ApiKey { ApiName = "BingImageSearch", Key = integrationKeys.BingImageSearch } }; } return settings; }); services.AddSingleton(); services.AddSingleton(); services.AddScoped(); services.AddScoped(); services.AddScoped(); services.AddScoped(); services.AddScoped(); services.AddScoped(); services.AddScoped(); services.AddScoped(); services.AddScoped(); services.AddScoped(); services.AddScoped(); services.AddScoped(); services.AddScoped(); services.AddScoped(); services.AddScoped(); services.AddScoped(); var securityKey = new SymmetricSecurityKey(System.Text.Encoding.Default.GetBytes(this._configuration["Tokens:PrivateKey"])); services.AddAuthentication(options => { options.DefaultAuthenticateScheme = JwtBearerDefaults.AuthenticationScheme; options.DefaultChallengeScheme = JwtBearerDefaults.AuthenticationScheme; }).AddJwtBearer(config => { config.RequireHttpsMetadata = false; config.SaveToken = true; config.TokenValidationParameters = new TokenValidationParameters() { IssuerSigningKey = securityKey, ValidateAudience = true, ValidAudience = this._configuration["Tokens:Audience"], ValidateIssuer = true, ValidIssuer = this._configuration["Tokens:Issuer"], ValidateLifetime = true, ValidateIssuerSigningKey = true }; }); //services.AddSwaggerGen(c => //{ // c.SwaggerDoc("v1", new Info // { // Title = "Roadie API", // Version = "v1" // }); //}); services.AddSignalR(); services.AddMvc(options => { options.RespectBrowserAcceptHeader = true; // false by default options.ModelBinderProviders.Insert(0, new SubsonicRequestBinderProvider()); }) .AddJsonOptions(options => { options.SerializerSettings.NullValueHandling = NullValueHandling.Ignore; options.SerializerSettings.ReferenceLoopHandling = ReferenceLoopHandling.Ignore; }) .AddXmlSerializerFormatters() .SetCompatibilityVersion(CompatibilityVersion.Latest); services.Configure(options => { options.Password.RequireDigit = true; options.Password.RequireLowercase = true; options.Password.RequireNonAlphanumeric = true; options.Password.RequireUppercase = true; options.Password.RequiredLength = 6; options.Password.RequiredUniqueChars = 1; }); services.AddHttpContextAccessor(); services.AddScoped(factory => { var actionContext = factory.GetService() .ActionContext; return new HttpContext(factory.GetService(), new UrlHelper(actionContext)); }); } private class IntegrationKey { public string BingImageSearch { get; set; } public string DiscogsConsumerKey { get; set; } public string DiscogsConsumerSecret { get; set; } public string LastFMApiKey { get; set; } public string LastFMSecret { get; set; } } } }