Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
SeleniumHQ
GitHub Repository: SeleniumHQ/Selenium
Path: blob/trunk/dotnet/test/testing.webserver/AppServer.cs
10193 views
// <copyright file="AppServer.cs" company="Selenium Committers">
// Licensed to the Software Freedom Conservancy (SFC) under one
// or more contributor license agreements.  See the NOTICE file
// distributed with this work for additional information
// regarding copyright ownership.  The SFC licenses this file
// to you under the Apache License, Version 2.0 (the
// "License"); you may not use this file except in compliance
// with the License.  You may obtain a copy of the License at
//
//   http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing,
// software distributed under the License is distributed on an
// "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
// KIND, either express or implied.  See the License for the
// specific language governing permissions and limitations
// under the License.
// </copyright>

using System;
using System.Collections.Concurrent;
using System.IO;
using System.Linq;
using System.Net;
using System.Security.Cryptography;
using System.Security.Cryptography.X509Certificates;
using System.Threading.Tasks;
using Microsoft.AspNetCore.Builder;
using Microsoft.AspNetCore.Hosting;
using Microsoft.AspNetCore.Http;
using Microsoft.AspNetCore.Routing;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.FileProviders;
using OpenQA.Selenium.Testing.WebServer.Handlers;

namespace OpenQA.Selenium.Testing.WebServer;

public class AppServer : IAsyncDisposable
{
    private WebApplication? _app;
    private readonly string _webContentRoot = FindWebContentRoot();
    private readonly ConcurrentDictionary<string, string> _pages = new();

    public async Task<(string HttpUrl, string HttpsUrl)> StartAsync()
    {
        var builder = WebApplication.CreateSlimBuilder();

        var certificate = GenerateSelfSignedCertificate();

        builder.WebHost.ConfigureKestrel(options =>
        {
            options.Listen(IPAddress.Loopback, 0);
            options.Listen(IPAddress.Loopback, 0, listenOptions =>
            {
                listenOptions.UseHttps(certificate);
            });
        });
        builder.Services.AddDirectoryBrowser();

        _app = builder.Build();

        MapEndpoints(_app);
        MapEndpoints(_app.MapGroup("/common"));

        if (Directory.Exists(_webContentRoot))
        {
            var fileProvider = new PhysicalFileProvider(_webContentRoot);

            _app.UseStaticFiles(new StaticFileOptions
            {
                FileProvider = fileProvider,
                ServeUnknownFileTypes = true
            });

            _app.UseStaticFiles(new StaticFileOptions
            {
                FileProvider = fileProvider,
                RequestPath = "/common",
                ServeUnknownFileTypes = true
            });

            _app.UseDirectoryBrowser(new DirectoryBrowserOptions
            {
                FileProvider = fileProvider
            });

            _app.UseDirectoryBrowser(new DirectoryBrowserOptions
            {
                FileProvider = fileProvider,
                RequestPath = "/common"
            });
        }

        await _app.StartAsync();

        int httpPort = new Uri(_app.Urls.First(u => u.StartsWith("http://"))).Port;
        int httpsPort = new Uri(_app.Urls.First(u => u.StartsWith("https://"))).Port;

        return ($"http://localhost:{httpPort}", $"https://localhost:{httpsPort}");
    }

    public async Task StopAsync()
    {
        if (_app is not null)
        {
            await _app.StopAsync();
            await _app.DisposeAsync();
            _app = null;
        }
    }

    public async ValueTask DisposeAsync()
    {
        await StopAsync();
    }

    private void MapEndpoints(IEndpointRouteBuilder endpoints)
    {
        endpoints.MapGet("/basicAuth", BasicAuthHandler.Handle);
        endpoints.MapGet("/echo", (Delegate)EchoHandler.Handle);
        endpoints.MapGet("/cookie", CookieHandler.Handle);
        endpoints.MapGet("/encoding", EncodingHandler.Handle);
        endpoints.MapGet("/sleep", (Delegate)SleepHandler.Handle);
        endpoints.MapGet("/redirect", RedirectHandler.Handle);
        endpoints.MapGet("/page/{pageNumber}", PageHandler.Handle);
        endpoints.MapGet("/utf8/{*path}", (HttpContext context, string path) => Utf8Handler.Handle(context, path, _webContentRoot));
        endpoints.MapPost("/createPage", (Delegate)((HttpContext context) => CreatePageHandler.Handle(context, _pages)));
        endpoints.MapPost("/upload", (Delegate)UploadHandler.Handle);

        endpoints.MapGet("/.well-known/web-identity", (HttpContext context) => FedCmHandler.HandleWebIdentity(context));
        endpoints.MapGet("/fedcm/config.json", (HttpContext context) => FedCmHandler.HandleConfig(context));
        endpoints.MapPost("/fedcm/id_assertion.json", (HttpContext context) => FedCmHandler.HandleIdAssertion(context));

        endpoints.MapGet("/temp/{fileName}", (string fileName) => CreatePageHandler.ServePage(fileName, _pages));
    }

    private static X509Certificate2 GenerateSelfSignedCertificate()
    {
        using var ecdsa = ECDsa.Create();
        var request = new CertificateRequest("CN=localhost", ecdsa, HashAlgorithmName.SHA256);
        return request.CreateSelfSigned(DateTimeOffset.UtcNow, DateTimeOffset.UtcNow.AddYears(1));
    }

    private static string FindWebContentRoot()
    {
        var info = new DirectoryInfo(AppContext.BaseDirectory);
        while (info is not null && info != info.Root)
        {
            string webPath = Path.Combine(info.FullName, "common", "src", "web");
            if (Directory.Exists(webPath))
            {
                return webPath;
            }
            info = info.Parent;
        }

        return string.Empty;
    }
}