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 @@
-
+
➕ @T["Add Attribute"]
@@ -280,7 +280,7 @@
row.innerHTML = `
- ✖
+ ✖
`;
attributesContainer.appendChild(row);
});
@@ -407,7 +407,7 @@
@T["Attributes"]
-
+
➕ @T["Add Attribute"]
@@ -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"]
- @T["Inventory asset"]
- @T["Create user"]
+ @T["Inventory asset"]
+ @T["Create user"]
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 @@
- Dummy button
- Dummy button
- Dummy button
+ Dummy button
+ Dummy button
+ Dummy button
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 @@
- @T["Create location"]
+ @T["Create location"]
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 @@
- @T["Create user"]
+ @T["Create user"]
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,
+ });
+}