diff --git a/docs/Testresults/20251010_QA_K6_16VUs30s.txt b/docs/Testresults/20251010_QA_K6_16VUs30s.txt new file mode 100644 index 0000000..d7c6f3e --- /dev/null +++ b/docs/Testresults/20251010_QA_K6_16VUs30s.txt @@ -0,0 +1,23 @@ + █ TOTAL RESULTS + + checks_total.......: 2016 65.329201/s + checks_succeeded...: 100.00% 2016 out of 2016 + checks_failed......: 0.00% 0 out of 2016 + + ✓ authorized request succeeded + + HTTP + http_req_duration..............: avg=13.38ms min=4.54ms med=11.01ms max=993.51ms p(90)=18.27ms p(95)=21.24ms + { expected_response:true }...: avg=13.38ms min=4.54ms med=11.01ms max=993.51ms p(90)=18.27ms p(95)=21.24ms + http_req_failed................: 0.00% 0 out of 4044 + http_reqs......................: 4044 131.047267/s + + EXECUTION + iteration_duration.............: avg=27.16ms min=14.2ms med=20.91ms max=1s p(90)=34ms p(95)=38.13ms + iterations.....................: 2016 65.329201/s + vus............................: 16 min=16 max=16 + vus_max........................: 16 min=16 max=16 + + NETWORK + data_received..................: 73 MB 2.4 MB/s + data_sent......................: 2.0 MB 66 kB/s \ No newline at end of file diff --git a/docs/Testresults/20251010_QA_Lighthouse.html b/docs/Testresults/20251010_QA_Lighthouse.html new file mode 100644 index 0000000..4836c91 --- /dev/null +++ b/docs/Testresults/20251010_QA_Lighthouse.html @@ -0,0 +1,65 @@ + + + + + + + + Lighthouse Report + + + + + +
+ + + + + + diff --git a/src/Program.cs b/src/Program.cs index 0dad9d7..9c35a95 100644 --- a/src/Program.cs +++ b/src/Program.cs @@ -48,8 +48,8 @@ builder.Services.AddAuthentication(CookieAuthenticationDefaults.AuthenticationSc builder.Services.AddResponseCompression(options => { options.EnableForHttps = true; - options.MimeTypes = new[] - { + options.MimeTypes = + [ "text/plain", "text/css", "application/javascript", @@ -58,7 +58,7 @@ builder.Services.AddResponseCompression(options => "text/xml", "application/json", "image/svg+xml" - }; + ]; }); var app = builder.Build(); @@ -89,7 +89,22 @@ if (app.Environment.IsDevelopment()) app.UseSwaggerUI(); } -app.UseStaticFiles(); +app.UseStaticFiles(new StaticFileOptions +{ + OnPrepareResponse = ctx => + { + string requestPath = ctx.Context.Request.Path.ToString(); + if (requestPath.EndsWith(".css") || requestPath.EndsWith(".js")) + { + ctx.Context.Response.GetTypedHeaders().CacheControl = + new Microsoft.Net.Http.Headers.CacheControlHeaderValue() + { + Public = true, + MaxAge = TimeSpan.FromDays(365) + }; + } + } +}); app.UseRouting(); app.UseAuthorization(); app.UseResponseCompression(); diff --git a/src/Views/Home/Assets.cshtml b/src/Views/Home/Assets.cshtml index 354d0cd..605b34c 100644 --- a/src/Views/Home/Assets.cshtml +++ b/src/Views/Home/Assets.cshtml @@ -12,7 +12,7 @@
-
@@ -231,7 +231,7 @@
- @@ -280,7 +280,7 @@ row.innerHTML = ` - + `; attributesContainer.appendChild(row); }); @@ -407,7 +407,7 @@
@T["Attributes"]
- @@ -457,7 +457,7 @@ document.addEventListener('DOMContentLoaded', () => { row.innerHTML = ` - + `; updateAttributesContainer.appendChild(row); }); @@ -504,7 +504,7 @@ document.addEventListener('DOMContentLoaded', () => { row.innerHTML = ` - + `; updateAttributesContainer.appendChild(row); } diff --git a/src/Views/Home/Index.cshtml b/src/Views/Home/Index.cshtml index f99bc02..512f298 100644 --- a/src/Views/Home/Index.cshtml +++ b/src/Views/Home/Index.cshtml @@ -10,7 +10,7 @@

@T["Overview"]

- - + +
diff --git a/src/Views/Home/Inventory.cshtml b/src/Views/Home/Inventory.cshtml index 8b97126..ca1195d 100644 --- a/src/Views/Home/Inventory.cshtml +++ b/src/Views/Home/Inventory.cshtml @@ -12,9 +12,9 @@
- - - + + +
diff --git a/src/Views/Home/Locations.cshtml b/src/Views/Home/Locations.cshtml index 2f598df..e36a0b6 100644 --- a/src/Views/Home/Locations.cshtml +++ b/src/Views/Home/Locations.cshtml @@ -12,7 +12,7 @@
- +
diff --git a/src/Views/Home/Users.cshtml b/src/Views/Home/Users.cshtml index f57572e..c526b36 100644 --- a/src/Views/Home/Users.cshtml +++ b/src/Views/Home/Users.cshtml @@ -13,7 +13,7 @@
- +
diff --git a/src/Views/Shared/_Layout.cshtml b/src/Views/Shared/_Layout.cshtml index 857088e..6ca4965 100644 --- a/src/Views/Shared/_Layout.cshtml +++ b/src/Views/Shared/_Layout.cshtml @@ -4,15 +4,17 @@ + @ViewData["Title"] - Berufsschule_HAM - @* *@ + @* *@ - - + @* *@ + + @@ -71,10 +73,10 @@ © 2025 - Berufsschule_HAM - @* *@ + @* *@ - @* *@ + @* *@ diff --git a/tests/k6/loadtest.js b/tests/k6/loadtest.js new file mode 100644 index 0000000..c027a80 --- /dev/null +++ b/tests/k6/loadtest.js @@ -0,0 +1,37 @@ +import http from 'k6/http'; +import { check } from 'k6'; + +export let options = { + vus: 16, // number of virtual users + duration: '30s', // test duration +}; +export default function () { + let environment = "http://localhost:5275"; + let username = "admin"; + let password = "admin"; + // Step 1: login, disable redirect following + let loginRes = http.post( + environment + '/Home/Login', + { Username: username, Password: password }, + { redirects: 0 } + ); + + // Step 2: extract the auth cookie manually + const setCookieHeader = loginRes.headers['Set-Cookie']; + + // Use a regex to extract the cookie value + const match = setCookieHeader.match(/\.AspNetCore\.Cookies=([^;]+)/); + const authCookie = match ? match[1] : null; + + if (!authCookie) { + throw new Error('Login failed: no auth cookie found'); + } + + // Step 3: use it on a protected page + const headers = { Cookie: `.AspNetCore.Cookies=${authCookie}` }; + const res = http.get(environment + '/Home/Assets', { headers }); + + check(res, { + 'authorized request succeeded': (r) => r.status === 200, + }); +}