// Copyright (c) .NET Foundation. All rights reserved. // Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information. using System; using System.IO; using System.Linq; using System.Threading.Tasks; using System.Threading; using System.Text; using System.Net.WebSockets; using Microsoft.AspNetCore.Builder; using Microsoft.AspNetCore.Hosting; using Microsoft.AspNetCore.Http; using Microsoft.AspNetCore.Http.Features; using Microsoft.Net.Http.Headers; using Microsoft.Extensions.DependencyInjection; using Microsoft.Extensions.Primitives; namespace ANCMStressTestApp { public class Startup { // This method gets called by the runtime. Use this method to add services to the container. // For more information on how to configure your application, visit https://go.microsoft.com/fwlink/?LinkID=398940 public void ConfigureServices(IServiceCollection services) { } public void Configure(IApplicationBuilder app) { app.Map("/HelloWorld", HelloWorld); app.Map("/ConnectionClose", ConnectionClose); app.Map("/EchoPostData", EchoPostData); app.Map("/LargeResponseBody", LargeResponseBody); app.Map("/ResponseHeaders", ResponseHeaders); app.Map("/EnvironmentVariables", EnvironmentVariables); app.Map("/RequestInformation", RequestInformation); app.Map("/WebSocket", WebSocket); app.Run(async context => { await context.Response.WriteAsync("Default Page"); }); } private void HelloWorld(IApplicationBuilder app) { app.Run(async context => { await context.Response.WriteAsync("Hello World"); }); } private void ConnectionClose(IApplicationBuilder app) { app.Run(async context => { context.Response.Headers[HeaderNames.Connection] = "close"; await context.Response.WriteAsync("Connnection Close"); await context.Response.Body.FlushAsync(); }); } private void EchoPostData(IApplicationBuilder app) { app.Run(async context => { string responseBody = string.Empty; if (string.Equals(context.Request.Method, "POST", StringComparison.OrdinalIgnoreCase)) { using (StreamReader reader = new StreamReader(context.Request.Body, Encoding.UTF8)) { responseBody = await reader.ReadToEndAsync(); } } else { responseBody = "NoAction"; } await context.Response.WriteAsync(responseBody); }); } private void LargeResponseBody(IApplicationBuilder app) { app.Run(async context => { if (int.TryParse(context.Request.Query["length"], out var length)) { await context.Response.WriteAsync(new string('a', length)); } }); } private void ResponseHeaders(IApplicationBuilder app) { app.Run(async context => { context.Response.Headers["UnknownHeader"] = "test123=foo"; context.Response.ContentType = "text/plain"; context.Response.Headers["MultiHeader"] = new StringValues(new string[] { "1", "2" }); await context.Response.WriteAsync("Request Complete"); }); } private void EnvironmentVariables(IApplicationBuilder app) { app.Run(async context => { context.Response.ContentType = "text/plain"; await context.Response.WriteAsync("Environment Variables:" + Environment.NewLine); var vars = Environment.GetEnvironmentVariables(); foreach (var key in vars.Keys.Cast().OrderBy(key => key, StringComparer.OrdinalIgnoreCase)) { var value = vars[key]; await context.Response.WriteAsync(key + ": " + value + Environment.NewLine); } await context.Response.WriteAsync(Environment.NewLine); }); } private void RequestInformation(IApplicationBuilder app) { app.Run(async context => { context.Response.ContentType = "text/plain"; await context.Response.WriteAsync("Address:" + Environment.NewLine); await context.Response.WriteAsync("Scheme: " + context.Request.Scheme + Environment.NewLine); await context.Response.WriteAsync("Host: " + context.Request.Headers["Host"] + Environment.NewLine); await context.Response.WriteAsync("PathBase: " + context.Request.PathBase.Value + Environment.NewLine); await context.Response.WriteAsync("Path: " + context.Request.Path.Value + Environment.NewLine); await context.Response.WriteAsync("Query: " + context.Request.QueryString.Value + Environment.NewLine); await context.Response.WriteAsync(Environment.NewLine); await context.Response.WriteAsync("Connection:" + Environment.NewLine); await context.Response.WriteAsync("RemoteIp: " + context.Connection.RemoteIpAddress + Environment.NewLine); await context.Response.WriteAsync("RemotePort: " + context.Connection.RemotePort + Environment.NewLine); await context.Response.WriteAsync("LocalIp: " + context.Connection.LocalIpAddress + Environment.NewLine); await context.Response.WriteAsync("LocalPort: " + context.Connection.LocalPort + Environment.NewLine); await context.Response.WriteAsync(Environment.NewLine); await context.Response.WriteAsync("Headers:" + Environment.NewLine); foreach (var header in context.Request.Headers) { await context.Response.WriteAsync(header.Key + ": " + header.Value + Environment.NewLine); } await context.Response.WriteAsync(Environment.NewLine); }); } private void WebSocket(IApplicationBuilder app) { app.Run(async context => { var upgradeFeature = context.Features.Get(); // Generate WebSocket response headers string key = context.Request.Headers[Constants.Headers.SecWebSocketKey].ToString(); var responseHeaders = HandshakeHelpers.GenerateResponseHeaders(key); foreach (var headerPair in responseHeaders) { context.Response.Headers[headerPair.Key] = headerPair.Value; } // Upgrade the connection Stream opaqueTransport = await upgradeFeature.UpgradeAsync(); // Get the WebSocket object var ws = WebSocketProtocol.CreateFromStream(opaqueTransport, isServer: true, subProtocol: null, keepAliveInterval: TimeSpan.FromMinutes(2)); var appLifetime = app.ApplicationServices.GetRequiredService(); await Echo(ws, appLifetime.ApplicationStopping); }); } private async Task Echo(WebSocket webSocket, CancellationToken token) { try { var buffer = new byte[1024 * 4]; var result = await webSocket.ReceiveAsync(new ArraySegment(buffer), token); bool closeFromServer = false; string closeFromServerCmd = "CloseFromServer"; int closeFromServerLength = closeFromServerCmd.Length; while (!result.CloseStatus.HasValue && !token.IsCancellationRequested && !closeFromServer) { if (result.Count == closeFromServerLength && Encoding.ASCII.GetString(buffer).Substring(0, result.Count) == closeFromServerCmd) { // The client sent "CloseFromServer" text message to request the server to close (a test scenario). closeFromServer = true; } else { await webSocket.SendAsync(new ArraySegment(buffer, 0, result.Count), result.MessageType, result.EndOfMessage, token); result = await webSocket.ReceiveAsync(new ArraySegment(buffer), token); } } if (result.CloseStatus.HasValue) { // Client-initiated close handshake await webSocket.CloseAsync(result.CloseStatus.Value, result.CloseStatusDescription, CancellationToken.None); } else { // Server-initiated close handshake due to either of the two conditions: // (1) The applicaton host is performing a graceful shutdown. // (2) The client sent "CloseFromServer" text message to request the server to close (a test scenario). await webSocket.CloseOutputAsync(WebSocketCloseStatus.NormalClosure, closeFromServerCmd, CancellationToken.None); // The server has sent the Close frame. // Stop sending but keep receiving until we get the Close frame from the client. while (!result.CloseStatus.HasValue) { result = await webSocket.ReceiveAsync(new ArraySegment(buffer), CancellationToken.None); } } } catch (Exception e) { Console.WriteLine("{0} Exception caught!", e); } } } }