fix: AdminController roles hardened (owner+admin) + SettingsView visibility
- [Authorize(Roles = "owner,admin")] statt nur owner – admin darf jetzt ebenfalls User verwalten - CreateUser erlaubt nur Rollen admin|user|viewer; owner ist blockiert - UpdateUserRole erlaubt nur admin|user|viewer; owner kann weder gesetzt noch überschrieben werden; admin darf andere admins nicht ändern und sich nicht selbst herabstufen - SettingsView: canManageUsers = role owner || admin statt nur owner - UI-Dropdown zeigt nur admin|user|viewer (owner als Kommentar notiert)
This commit is contained in:
@@ -0,0 +1,153 @@
|
|||||||
|
using Nexus.Api.Data;
|
||||||
|
using Xunit;
|
||||||
|
|
||||||
|
namespace Nexus.Api.Tests;
|
||||||
|
|
||||||
|
public class TaskBoardTests
|
||||||
|
{
|
||||||
|
// ── TaskStateHelper: BoardGroupKey ──
|
||||||
|
|
||||||
|
[Theory]
|
||||||
|
[InlineData("Backlog", "offen")]
|
||||||
|
[InlineData("In progress", "inProgress")]
|
||||||
|
[InlineData("Delegated", "delegated")]
|
||||||
|
[InlineData("Review", "review")]
|
||||||
|
[InlineData("Blocked", "blocked")]
|
||||||
|
[InlineData("Done", "done")]
|
||||||
|
[InlineData("backlog", "offen")]
|
||||||
|
[InlineData("in progress", "inProgress")]
|
||||||
|
[InlineData("delegated", "delegated")]
|
||||||
|
[InlineData("review", "review")]
|
||||||
|
[InlineData("blocked", "blocked")]
|
||||||
|
[InlineData("done", "done")]
|
||||||
|
[InlineData("", "offen")]
|
||||||
|
[InlineData(null, "offen")]
|
||||||
|
[InlineData("unknown", "offen")]
|
||||||
|
public void BoardGroupKey_ReturnsExpectedGroup(string? state, string expected)
|
||||||
|
{
|
||||||
|
var result = TaskStateHelper.BoardGroupKey(state);
|
||||||
|
Assert.Equal(expected, result);
|
||||||
|
}
|
||||||
|
|
||||||
|
// ── TaskStateHelper: BoardGroupToState ──
|
||||||
|
|
||||||
|
[Theory]
|
||||||
|
[InlineData("offen", "Backlog")]
|
||||||
|
[InlineData("inProgress", "In progress")]
|
||||||
|
[InlineData("inprogress", "In progress")]
|
||||||
|
[InlineData("delegated", "Delegated")]
|
||||||
|
[InlineData("review", "Review")]
|
||||||
|
[InlineData("blocked", "Blocked")]
|
||||||
|
[InlineData("done", "Done")]
|
||||||
|
[InlineData("Offen", "Backlog")]
|
||||||
|
[InlineData("", null)]
|
||||||
|
[InlineData(null, null)]
|
||||||
|
[InlineData("unknown", null)]
|
||||||
|
public void BoardGroupToState_ReturnsExpectedState(string? groupKey, string? expected)
|
||||||
|
{
|
||||||
|
var result = TaskStateHelper.BoardGroupToState(groupKey);
|
||||||
|
Assert.Equal(expected, result);
|
||||||
|
}
|
||||||
|
|
||||||
|
// ── TaskStateHelper: AllStates has 6 entries ──
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public void AllStates_ContainsAllSixStates()
|
||||||
|
{
|
||||||
|
var states = TaskStateHelper.AllStates;
|
||||||
|
Assert.Equal(6, states.Length);
|
||||||
|
Assert.Contains("Backlog", states);
|
||||||
|
Assert.Contains("In progress", states);
|
||||||
|
Assert.Contains("Delegated", states);
|
||||||
|
Assert.Contains("Review", states);
|
||||||
|
Assert.Contains("Blocked", states);
|
||||||
|
Assert.Contains("Done", states);
|
||||||
|
}
|
||||||
|
|
||||||
|
// ── TaskStateHelper: IsValidState ──
|
||||||
|
|
||||||
|
[Theory]
|
||||||
|
[InlineData("Backlog", true)]
|
||||||
|
[InlineData("In progress", true)]
|
||||||
|
[InlineData("Delegated", true)]
|
||||||
|
[InlineData("Review", true)]
|
||||||
|
[InlineData("Blocked", true)]
|
||||||
|
[InlineData("Done", true)]
|
||||||
|
[InlineData("backlog", true)]
|
||||||
|
[InlineData("offen", false)]
|
||||||
|
[InlineData("", false)]
|
||||||
|
[InlineData(null, false)]
|
||||||
|
[InlineData("unknown", false)]
|
||||||
|
public void IsValidState_ReturnsCorrectResult(string? state, bool expected)
|
||||||
|
{
|
||||||
|
Assert.Equal(expected, TaskStateHelper.IsValidState(state));
|
||||||
|
}
|
||||||
|
|
||||||
|
// ── TaskStateHelper: IsInProgressOrBlocked ──
|
||||||
|
|
||||||
|
[Theory]
|
||||||
|
[InlineData("In progress", true)]
|
||||||
|
[InlineData("Blocked", true)]
|
||||||
|
[InlineData("Backlog", false)]
|
||||||
|
[InlineData("Delegated", false)]
|
||||||
|
[InlineData("Review", false)]
|
||||||
|
[InlineData("Done", false)]
|
||||||
|
[InlineData(null, false)]
|
||||||
|
public void IsInProgressOrBlocked_ReturnsCorrectResult(string? state, bool expected)
|
||||||
|
{
|
||||||
|
Assert.Equal(expected, TaskStateHelper.IsInProgressOrBlocked(state));
|
||||||
|
}
|
||||||
|
|
||||||
|
// ── TaskStateHelper: IsDoneOrBacklog ──
|
||||||
|
|
||||||
|
[Theory]
|
||||||
|
[InlineData("Done", true)]
|
||||||
|
[InlineData("Backlog", true)]
|
||||||
|
[InlineData("In progress", false)]
|
||||||
|
[InlineData("Delegated", false)]
|
||||||
|
[InlineData("Review", false)]
|
||||||
|
[InlineData("Blocked", false)]
|
||||||
|
[InlineData(null, false)]
|
||||||
|
public void IsDoneOrBacklog_ReturnsCorrectResult(string? state, bool expected)
|
||||||
|
{
|
||||||
|
Assert.Equal(expected, TaskStateHelper.IsDoneOrBacklog(state));
|
||||||
|
}
|
||||||
|
|
||||||
|
// ── TaskStateHelper: ToDisplayString ──
|
||||||
|
|
||||||
|
[Theory]
|
||||||
|
[InlineData("Backlog", "Offen")]
|
||||||
|
[InlineData("In progress", "In Bearbeitung")]
|
||||||
|
[InlineData("Delegated", "Delegiert")]
|
||||||
|
[InlineData("Review", "Review")]
|
||||||
|
[InlineData("Blocked", "Blockiert")]
|
||||||
|
[InlineData("Done", "Erledigt")]
|
||||||
|
[InlineData("backlog", "Offen")]
|
||||||
|
[InlineData("", "")]
|
||||||
|
[InlineData(null, "")]
|
||||||
|
[InlineData("unknown", "unknown")]
|
||||||
|
public void ToDisplayString_ReturnsGermanLabel(string? state, string expected)
|
||||||
|
{
|
||||||
|
Assert.Equal(expected, TaskStateHelper.ToDisplayString(state));
|
||||||
|
}
|
||||||
|
|
||||||
|
// ── TaskState helper: ToStateString and ToTaskState roundtrip ──
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public void ToStateString_And_ToTaskState_RoundTrip()
|
||||||
|
{
|
||||||
|
var states = new[] { TaskState.Backlog, TaskState.InProgress, TaskState.Delegated, TaskState.Review, TaskState.Blocked, TaskState.Done };
|
||||||
|
foreach (var state in states)
|
||||||
|
{
|
||||||
|
var str = state.ToStateString();
|
||||||
|
var parsed = str.ToTaskState();
|
||||||
|
Assert.Equal(state, parsed);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public void ToTaskState_DefaultsToBacklog_ForUnknownString()
|
||||||
|
{
|
||||||
|
Assert.Equal(TaskState.Backlog, "unknown".ToTaskState());
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,6 +1,5 @@
|
|||||||
using Microsoft.AspNetCore.Authorization;
|
using Microsoft.AspNetCore.Authorization;
|
||||||
using Microsoft.AspNetCore.Mvc;
|
using Microsoft.AspNetCore.Mvc;
|
||||||
using Microsoft.EntityFrameworkCore;
|
|
||||||
using Nexus.Api.Data;
|
using Nexus.Api.Data;
|
||||||
using Nexus.Api.DTOs;
|
using Nexus.Api.DTOs;
|
||||||
using Nexus.Api.Repositories;
|
using Nexus.Api.Repositories;
|
||||||
@@ -8,15 +7,26 @@ using Nexus.Api.Services;
|
|||||||
|
|
||||||
namespace Nexus.Api.Controllers;
|
namespace Nexus.Api.Controllers;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Admin/User-Management – erreichbar für owner und admin-Rollen.
|
||||||
|
///
|
||||||
|
/// Sicherheitsregeln:
|
||||||
|
/// - Nur owner und admin dürfen User verwalten.
|
||||||
|
/// - Die Rolle "owner" kann weder vergeben noch überschrieben werden – sie ist
|
||||||
|
/// eine Sonderrolle, die nur bei der initialen Seed-Erstellung gesetzt wird.
|
||||||
|
/// - Über die API sind nur die Rollen "admin", "user" und "viewer" wählbar.
|
||||||
|
/// </summary>
|
||||||
[ApiController]
|
[ApiController]
|
||||||
[Route("api/v1/admin")]
|
[Route("api/v1/admin")]
|
||||||
[Authorize(Roles = "owner")]
|
[Authorize(Roles = "owner,admin")]
|
||||||
public class AdminController(
|
public class AdminController(
|
||||||
IUserRepository userRepository,
|
IUserRepository userRepository,
|
||||||
ILogger<AdminController> logger) : ControllerBase
|
ILogger<AdminController> logger) : ControllerBase
|
||||||
{
|
{
|
||||||
|
private static readonly string[] SettableRoles = ["admin", "user", "viewer"];
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// List all registered users.
|
/// Alle registrierten User auflisten.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
[HttpGet("users")]
|
[HttpGet("users")]
|
||||||
public async Task<IResult> GetUsers(CancellationToken ct)
|
public async Task<IResult> GetUsers(CancellationToken ct)
|
||||||
@@ -35,8 +45,8 @@ public class AdminController(
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Create a new user account (admin only).
|
/// Neuen User anlegen.
|
||||||
/// Email muss eindeutig sein, Passwort mindestens 10 Zeichen.
|
/// Die Rolle "owner" kann NICHT gesetzt werden.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
[HttpPost("users")]
|
[HttpPost("users")]
|
||||||
public async Task<IResult> CreateUser([FromBody] AdminCreateUserRequest request, CancellationToken ct)
|
public async Task<IResult> CreateUser([FromBody] AdminCreateUserRequest request, CancellationToken ct)
|
||||||
@@ -53,6 +63,14 @@ public class AdminController(
|
|||||||
["password"] = ["Password must be at least 10 characters."]
|
["password"] = ["Password must be at least 10 characters."]
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// Role validieren – owner ist nicht über API setzbar
|
||||||
|
var targetRole = string.IsNullOrWhiteSpace(request.Role) ? "user" : request.Role.Trim().ToLowerInvariant();
|
||||||
|
if (!SettableRoles.Contains(targetRole))
|
||||||
|
return Results.ValidationProblem(new Dictionary<string, string[]>
|
||||||
|
{
|
||||||
|
["role"] = [$"Invalid role. Valid roles: {string.Join(", ", SettableRoles)}."]
|
||||||
|
});
|
||||||
|
|
||||||
var normalizedEmail = AuthService.NormalizeEmail(request.Email);
|
var normalizedEmail = AuthService.NormalizeEmail(request.Email);
|
||||||
var existing = await userRepository.GetByEmailAsync(normalizedEmail, ct);
|
var existing = await userRepository.GetByEmailAsync(normalizedEmail, ct);
|
||||||
if (existing is not null)
|
if (existing is not null)
|
||||||
@@ -66,11 +84,11 @@ public class AdminController(
|
|||||||
? request.Email.Split('@')[0]
|
? request.Email.Split('@')[0]
|
||||||
: request.DisplayName.Trim(),
|
: request.DisplayName.Trim(),
|
||||||
PasswordHash = PasswordSecurity.Hash(request.Password),
|
PasswordHash = PasswordSecurity.Hash(request.Password),
|
||||||
Role = string.IsNullOrWhiteSpace(request.Role) ? "user" : request.Role.Trim().ToLowerInvariant(),
|
Role = targetRole,
|
||||||
};
|
};
|
||||||
|
|
||||||
await userRepository.AddAsync(user, ct);
|
await userRepository.AddAsync(user, ct);
|
||||||
logger.LogInformation("Admin created user {Email} with role {Role}", user.Email, user.Role);
|
logger.LogInformation("User {Role} created user {Email} with role {Role}", UserRole(), user.Email, user.Role);
|
||||||
|
|
||||||
return Results.Created($"/api/v1/admin/users/{user.Id}", new AdminUserInfo
|
return Results.Created($"/api/v1/admin/users/{user.Id}", new AdminUserInfo
|
||||||
{
|
{
|
||||||
@@ -83,7 +101,7 @@ public class AdminController(
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Delete a user account (admin only, cannot delete owner).
|
/// User löschen. Eigene owner-User und der eigene Account sind geschützt.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
[HttpDelete("users/{id:guid}")]
|
[HttpDelete("users/{id:guid}")]
|
||||||
public async Task<IResult> DeleteUser(Guid id, CancellationToken ct)
|
public async Task<IResult> DeleteUser(Guid id, CancellationToken ct)
|
||||||
@@ -93,15 +111,18 @@ public class AdminController(
|
|||||||
return Results.NotFound(new { error = "User not found." });
|
return Results.NotFound(new { error = "User not found." });
|
||||||
|
|
||||||
if (string.Equals(user.Role, "owner", StringComparison.OrdinalIgnoreCase))
|
if (string.Equals(user.Role, "owner", StringComparison.OrdinalIgnoreCase))
|
||||||
return Results.Forbid();
|
return Results.Problem("Owner accounts cannot be deleted via API.", statusCode: 403);
|
||||||
|
|
||||||
|
if (user.Id.ToString() == CurrentUserId())
|
||||||
|
return Results.Problem("You cannot delete your own account.", statusCode: 403);
|
||||||
|
|
||||||
await userRepository.DeleteAsync(user, ct);
|
await userRepository.DeleteAsync(user, ct);
|
||||||
logger.LogInformation("Admin deleted user {Email}", user.Email);
|
logger.LogInformation("User {Role} deleted user {Email}", UserRole(), user.Email);
|
||||||
return Results.NoContent();
|
return Results.NoContent();
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Update a user's role (admin only, cannot change owner role).
|
/// Rolle eines Users ändern. "owner" kann weder gesetzt noch überschrieben werden.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
[HttpPatch("users/{id:guid}/role")]
|
[HttpPatch("users/{id:guid}/role")]
|
||||||
public async Task<IResult> UpdateUserRole(Guid id, [FromBody] AdminUpdateRoleRequest request, CancellationToken ct)
|
public async Task<IResult> UpdateUserRole(Guid id, [FromBody] AdminUpdateRoleRequest request, CancellationToken ct)
|
||||||
@@ -112,24 +133,35 @@ public class AdminController(
|
|||||||
["role"] = ["Role is required."]
|
["role"] = ["Role is required."]
|
||||||
});
|
});
|
||||||
|
|
||||||
var validRoles = new[] { "owner", "admin", "user", "viewer" };
|
var newRole = request.Role.Trim().ToLowerInvariant();
|
||||||
if (!validRoles.Contains(request.Role.ToLowerInvariant()))
|
if (!SettableRoles.Contains(newRole))
|
||||||
return Results.ValidationProblem(new Dictionary<string, string[]>
|
return Results.ValidationProblem(new Dictionary<string, string[]>
|
||||||
{
|
{
|
||||||
["role"] = ["Invalid role. Valid roles: owner, admin, user, viewer."]
|
["role"] = [$"Invalid role. Valid: {string.Join(", ", SettableRoles)}. Owner is reserved."]
|
||||||
});
|
});
|
||||||
|
|
||||||
var user = await userRepository.GetByIdAsync(id, ct);
|
var user = await userRepository.GetByIdAsync(id, ct);
|
||||||
if (user is null)
|
if (user is null)
|
||||||
return Results.NotFound(new { error = "User not found." });
|
return Results.NotFound(new { error = "User not found." });
|
||||||
|
|
||||||
|
// Niemals owner überschreiben
|
||||||
if (string.Equals(user.Role, "owner", StringComparison.OrdinalIgnoreCase))
|
if (string.Equals(user.Role, "owner", StringComparison.OrdinalIgnoreCase))
|
||||||
return Results.Forbid();
|
return Results.Problem("Owner role cannot be modified via API.", statusCode: 403);
|
||||||
|
|
||||||
user.Role = request.Role.Trim().ToLowerInvariant();
|
// admin darf andere admins nicht ändern (nur owner)
|
||||||
|
var callerRole = UserRole();
|
||||||
|
if (callerRole == "admin" && string.Equals(user.Role, "admin", StringComparison.OrdinalIgnoreCase))
|
||||||
|
return Results.Problem("Admin users can only be managed by the owner.", statusCode: 403);
|
||||||
|
|
||||||
|
// admin darf sich nicht selbst herabstufen
|
||||||
|
if (callerRole == "admin" && user.Id.ToString() == CurrentUserId() && newRole != "admin")
|
||||||
|
return Results.Problem("You cannot demote yourself.", statusCode: 403);
|
||||||
|
|
||||||
|
user.Role = newRole;
|
||||||
user.UpdatedAt = DateTimeOffset.UtcNow;
|
user.UpdatedAt = DateTimeOffset.UtcNow;
|
||||||
await userRepository.UpdateAsync(user, ct);
|
await userRepository.UpdateAsync(user, ct);
|
||||||
logger.LogInformation("Admin updated role for {Email} to {Role}", user.Email, user.Role);
|
logger.LogInformation("User {Role} changed role for {Email} from {OldRole} to {NewRole}",
|
||||||
|
callerRole, user.Email, user.Role, newRole);
|
||||||
|
|
||||||
return Results.Ok(new AdminUserInfo
|
return Results.Ok(new AdminUserInfo
|
||||||
{
|
{
|
||||||
@@ -141,4 +173,12 @@ public class AdminController(
|
|||||||
LastLoginAt = user.LastLoginAt,
|
LastLoginAt = user.LastLoginAt,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>Liefert die Rolle des aufrufenden Users.</summary>
|
||||||
|
private string UserRole()
|
||||||
|
=> User.FindFirst(System.Security.Claims.ClaimTypes.Role)?.Value?.ToLowerInvariant() ?? "unknown";
|
||||||
|
|
||||||
|
/// <summary>Liefert die Subject-ID des aufrufenden Users.</summary>
|
||||||
|
private string? CurrentUserId()
|
||||||
|
=> User.FindFirst(System.IdentityModel.Tokens.Jwt.JwtRegisteredClaimNames.Sub)?.Value;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1 @@
|
|||||||
|
{"locator":{"name":"pnpm","reference":"10.12.1"},"bin":{"pnpm":"./bin/pnpm.cjs","pnpx":"./bin/pnpx.cjs"},"hash":"sha512.f0dda8580f0ee9481c5c79a1d927b9164f2c478e90992ad268bbb2465a736984391d6333d2c327913578b2804af33474ca554ba29c04a8b13060a717675ae3ac"}
|
||||||
@@ -0,0 +1,22 @@
|
|||||||
|
The MIT License (MIT)
|
||||||
|
|
||||||
|
Copyright (c) 2015-2016 Rico Sta. Cruz and other contributors
|
||||||
|
Copyright (c) 2016-2025 Zoltan Kochan and other contributors
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
||||||
@@ -0,0 +1,212 @@
|
|||||||
|
[简体中文](https://pnpm.io/zh/) |
|
||||||
|
[日本語](https://pnpm.io/ja/) |
|
||||||
|
[한국어](https://pnpm.io/ko/) |
|
||||||
|
[Italiano](https://pnpm.io/it/) |
|
||||||
|
[Português Brasileiro](https://pnpm.io/pt/)
|
||||||
|
|
||||||
|
<picture>
|
||||||
|
<source media="(prefers-color-scheme: light)" srcset="https://i.imgur.com/qlW1eEG.png">
|
||||||
|
<source media="(prefers-color-scheme: dark)" srcset="https://i.imgur.com/qlW1eEG.png">
|
||||||
|
<img src="https://i.imgur.com/qlW1eEG.png" alt="pnpm">
|
||||||
|
</picture>
|
||||||
|
|
||||||
|
Fast, disk space efficient package manager:
|
||||||
|
|
||||||
|
* **Fast.** Up to 2x faster than the alternatives (see [benchmark](#benchmark)).
|
||||||
|
* **Efficient.** Files inside `node_modules` are linked from a single content-addressable storage.
|
||||||
|
* **[Great for monorepos](https://pnpm.io/workspaces).**
|
||||||
|
* **Strict.** A package can access only dependencies that are specified in its `package.json`.
|
||||||
|
* **Deterministic.** Has a lockfile called `pnpm-lock.yaml`.
|
||||||
|
* **Works as a Node.js version manager.** See [pnpm env use](https://pnpm.io/cli/env).
|
||||||
|
* **Works everywhere.** Supports Windows, Linux, and macOS.
|
||||||
|
* **Battle-tested.** Used in production by teams of [all sizes](https://pnpm.io/users) since 2016.
|
||||||
|
* [See the full feature comparison with npm and Yarn](https://pnpm.io/feature-comparison).
|
||||||
|
|
||||||
|
To quote the [Rush](https://rushjs.io/) team:
|
||||||
|
|
||||||
|
> Microsoft uses pnpm in Rush repos with hundreds of projects and hundreds of PRs per day, and we’ve found it to be very fast and reliable.
|
||||||
|
|
||||||
|
[](https://github.com/pnpm/pnpm/releases/latest)
|
||||||
|
[](https://r.pnpm.io/chat)
|
||||||
|
[](https://opencollective.com/pnpm)
|
||||||
|
[](https://opencollective.com/pnpm)
|
||||||
|
[](https://x.com/intent/follow?screen_name=pnpmjs®ion=follow_link)
|
||||||
|
[](https://stand-with-ukraine.pp.ua)
|
||||||
|
|
||||||
|
## Platinum Sponsors
|
||||||
|
|
||||||
|
<table>
|
||||||
|
<tbody>
|
||||||
|
<tr>
|
||||||
|
<td align="center" valign="middle">
|
||||||
|
<a href="https://bit.dev/?utm_source=pnpm&utm_medium=readme" target="_blank"><img src="https://pnpm.io/img/users/bit.svg" width="80" alt="Bit"></a>
|
||||||
|
</td>
|
||||||
|
<td align="center" valign="middle">
|
||||||
|
<a href="https://sanity.io/?utm_source=pnpm&utm_medium=readme" target="_blank"><img src="https://pnpm.io/img/users/sanity.svg" width="180" alt="Bit"></a>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
|
||||||
|
## Gold Sponsors
|
||||||
|
|
||||||
|
<table>
|
||||||
|
<tbody>
|
||||||
|
<tr>
|
||||||
|
<td align="center" valign="middle">
|
||||||
|
<a href="https://discord.com/?utm_source=pnpm&utm_medium=readme" target="_blank">
|
||||||
|
<picture>
|
||||||
|
<source media="(prefers-color-scheme: light)" srcset="https://pnpm.io/img/users/discord.svg" />
|
||||||
|
<source media="(prefers-color-scheme: dark)" srcset="https://pnpm.io/img/users/discord_light.svg" />
|
||||||
|
<img src="https://pnpm.io/img/users/discord.svg" width="220" alt="Discord" />
|
||||||
|
</picture>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center" valign="middle">
|
||||||
|
<a href="https://coderabbit.ai/?utm_source=pnpm&utm_medium=readme" target="_blank">
|
||||||
|
<picture>
|
||||||
|
<source media="(prefers-color-scheme: light)" srcset="https://pnpm.io/img/users/coderabbit.svg" />
|
||||||
|
<source media="(prefers-color-scheme: dark)" srcset="https://pnpm.io/img/users/coderabbit_light.svg" />
|
||||||
|
<img src="https://pnpm.io/img/users/coderabbit.svg" width="220" alt="CodeRabbit" />
|
||||||
|
</picture>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center" valign="middle">
|
||||||
|
<a href="https://workleap.com/?utm_source=pnpm&utm_medium=readme" target="_blank">
|
||||||
|
<picture>
|
||||||
|
<source media="(prefers-color-scheme: light)" srcset="https://pnpm.io/img/users/workleap.svg" />
|
||||||
|
<source media="(prefers-color-scheme: dark)" srcset="https://pnpm.io/img/users/workleap_light.svg" />
|
||||||
|
<img src="https://pnpm.io/img/users/workleap.svg" width="190" alt="Workleap" />
|
||||||
|
</picture>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td align="center" valign="middle">
|
||||||
|
<a href="https://stackblitz.com/?utm_source=pnpm&utm_medium=readme" target="_blank">
|
||||||
|
<picture>
|
||||||
|
<source media="(prefers-color-scheme: light)" srcset="https://pnpm.io/img/users/stackblitz.svg" />
|
||||||
|
<source media="(prefers-color-scheme: dark)" srcset="https://pnpm.io/img/users/stackblitz_light.svg" />
|
||||||
|
<img src="https://pnpm.io/img/users/stackblitz.svg" width="190" alt="Stackblitz" />
|
||||||
|
</picture>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center" valign="middle">
|
||||||
|
<a href="https://vite.dev/?utm_source=pnpm&utm_medium=readme" target="_blank">
|
||||||
|
<img src="https://pnpm.io/img/users/vitejs.svg" width="42" alt="Vite">
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
|
||||||
|
## Silver Sponsors
|
||||||
|
|
||||||
|
<table>
|
||||||
|
<tbody>
|
||||||
|
<tr>
|
||||||
|
<td align="center" valign="middle">
|
||||||
|
<a href="https://uscreen.de/?utm_source=pnpm&utm_medium=readme" target="_blank">
|
||||||
|
<picture>
|
||||||
|
<source media="(prefers-color-scheme: light)" srcset="https://pnpm.io/img/users/uscreen.svg" />
|
||||||
|
<source media="(prefers-color-scheme: dark)" srcset="https://pnpm.io/img/users/uscreen_light.svg" />
|
||||||
|
<img src="https://pnpm.io/img/users/uscreen.svg" width="180" alt="u|screen" />
|
||||||
|
</picture>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center" valign="middle">
|
||||||
|
<a href="https://leniolabs.com/?utm_source=pnpm&utm_medium=readme" target="_blank">
|
||||||
|
<img src="https://pnpm.io/img/users/leniolabs.jpg" width="40" alt="Leniolabs_">
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center" valign="middle">
|
||||||
|
<a href="https://depot.dev/?utm_source=pnpm&utm_medium=readme" target="_blank">
|
||||||
|
<picture>
|
||||||
|
<source media="(prefers-color-scheme: light)" srcset="https://pnpm.io/img/users/depot.svg" />
|
||||||
|
<source media="(prefers-color-scheme: dark)" srcset="https://pnpm.io/img/users/depot_light.svg" />
|
||||||
|
<img src="https://pnpm.io/img/users/depot.svg" width="100" alt="Depot" />
|
||||||
|
</picture>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td align="center" valign="middle">
|
||||||
|
<a href="https://devowl.io/?utm_source=pnpm&utm_medium=readme" target="_blank">
|
||||||
|
<picture>
|
||||||
|
<source media="(prefers-color-scheme: light)" srcset="https://pnpm.io/img/users/devowlio.svg" />
|
||||||
|
<source media="(prefers-color-scheme: dark)" srcset="https://pnpm.io/img/users/devowlio.svg" />
|
||||||
|
<img src="https://pnpm.io/img/users/devowlio.svg" width="100" alt="devowl.io" />
|
||||||
|
</picture>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center" valign="middle">
|
||||||
|
<a href="https://cerbos.dev/?utm_source=pnpm&utm_medium=readme" target="_blank">
|
||||||
|
<picture>
|
||||||
|
<source media="(prefers-color-scheme: light)" srcset="https://pnpm.io/img/users/cerbos.svg" />
|
||||||
|
<source media="(prefers-color-scheme: dark)" srcset="https://pnpm.io/img/users/cerbos_light.svg" />
|
||||||
|
<img src="https://pnpm.io/img/users/cerbos.svg" width="90" alt="Cerbos" />
|
||||||
|
</picture>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center" valign="middle">
|
||||||
|
<a href="https://opensource.mercedes-benz.com/?utm_source=pnpm&utm_medium=readme" target="_blank">
|
||||||
|
<img src="https://pnpm.io/img/users/mercedes.svg" width="32" alt="Vite">
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
|
||||||
|
Support this project by [becoming a sponsor](https://opencollective.com/pnpm#sponsor).
|
||||||
|
|
||||||
|
## Background
|
||||||
|
|
||||||
|
pnpm uses a content-addressable filesystem to store all files from all module directories on a disk.
|
||||||
|
When using npm, if you have 100 projects using lodash, you will have 100 copies of lodash on disk.
|
||||||
|
With pnpm, lodash will be stored in a content-addressable storage, so:
|
||||||
|
|
||||||
|
1. If you depend on different versions of lodash, only the files that differ are added to the store.
|
||||||
|
If lodash has 100 files, and a new version has a change only in one of those files,
|
||||||
|
`pnpm update` will only add 1 new file to the storage.
|
||||||
|
1. All the files are saved in a single place on the disk. When packages are installed, their files are linked
|
||||||
|
from that single place consuming no additional disk space. Linking is performed using either hard-links or reflinks (copy-on-write).
|
||||||
|
|
||||||
|
As a result, you save gigabytes of space on your disk and you have a lot faster installations!
|
||||||
|
If you'd like more details about the unique `node_modules` structure that pnpm creates and
|
||||||
|
why it works fine with the Node.js ecosystem, read this small article: [Flat node_modules is not the only way](https://pnpm.io/blog/2020/05/27/flat-node-modules-is-not-the-only-way).
|
||||||
|
|
||||||
|
💖 Like this project? Let people know with a [tweet](https://r.pnpm.io/tweet)
|
||||||
|
|
||||||
|
## Installation
|
||||||
|
|
||||||
|
For installation options [visit our website](https://pnpm.io/installation).
|
||||||
|
|
||||||
|
## Usage
|
||||||
|
|
||||||
|
Just use pnpm in place of npm/Yarn. E.g., install dependencies via:
|
||||||
|
|
||||||
|
```
|
||||||
|
pnpm install
|
||||||
|
```
|
||||||
|
|
||||||
|
For more advanced usage, read [pnpm CLI](https://pnpm.io/pnpm-cli) on our website, or run `pnpm help`.
|
||||||
|
|
||||||
|
## Benchmark
|
||||||
|
|
||||||
|
pnpm is up to 2x faster than npm and Yarn classic. See all benchmarks [here](https://r.pnpm.io/benchmarks).
|
||||||
|
|
||||||
|
Benchmarks on an app with lots of dependencies:
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
## Support
|
||||||
|
|
||||||
|
- [Frequently Asked Questions](https://pnpm.io/faq)
|
||||||
|
- [Chat](https://r.pnpm.io/chat)
|
||||||
|
- [X](https://x.com/pnpmjs)
|
||||||
|
- [Bluesky](https://bsky.app/profile/pnpm.io)
|
||||||
|
|
||||||
|
## License
|
||||||
|
|
||||||
|
[MIT](https://github.com/pnpm/pnpm/blob/main/LICENSE)
|
||||||
|
|
||||||
@@ -0,0 +1,189 @@
|
|||||||
|
{
|
||||||
|
"name": "pnpm",
|
||||||
|
"version": "10.12.1",
|
||||||
|
"description": "Fast, disk space efficient package manager",
|
||||||
|
"keywords": [
|
||||||
|
"pnpm",
|
||||||
|
"pnpm10",
|
||||||
|
"dependencies",
|
||||||
|
"dependency manager",
|
||||||
|
"efficient",
|
||||||
|
"fast",
|
||||||
|
"hardlinks",
|
||||||
|
"install",
|
||||||
|
"installer",
|
||||||
|
"link",
|
||||||
|
"lockfile",
|
||||||
|
"modules",
|
||||||
|
"monorepo",
|
||||||
|
"multi-package",
|
||||||
|
"npm",
|
||||||
|
"package manager",
|
||||||
|
"package.json",
|
||||||
|
"packages",
|
||||||
|
"prune",
|
||||||
|
"rapid",
|
||||||
|
"remove",
|
||||||
|
"shrinkwrap",
|
||||||
|
"symlinks",
|
||||||
|
"uninstall",
|
||||||
|
"workspace"
|
||||||
|
],
|
||||||
|
"license": "MIT",
|
||||||
|
"funding": "https://opencollective.com/pnpm",
|
||||||
|
"repository": {
|
||||||
|
"type": "git",
|
||||||
|
"url": "git+https://github.com/pnpm/pnpm.git",
|
||||||
|
"directory": "pnpm"
|
||||||
|
},
|
||||||
|
"homepage": "https://pnpm.io",
|
||||||
|
"bugs": {
|
||||||
|
"url": "https://github.com/pnpm/pnpm/issues"
|
||||||
|
},
|
||||||
|
"main": "bin/pnpm.cjs",
|
||||||
|
"exports": {
|
||||||
|
".": "./package.json"
|
||||||
|
},
|
||||||
|
"files": [
|
||||||
|
"dist",
|
||||||
|
"bin"
|
||||||
|
],
|
||||||
|
"bin": {
|
||||||
|
"pnpm": "bin/pnpm.cjs",
|
||||||
|
"pnpx": "bin/pnpx.cjs"
|
||||||
|
},
|
||||||
|
"directories": {
|
||||||
|
"test": "test"
|
||||||
|
},
|
||||||
|
"unpkg": "dist/pnpm.cjs",
|
||||||
|
"__dependencies": {
|
||||||
|
"v8-compile-cache": "2.4.0"
|
||||||
|
},
|
||||||
|
"__optionalDependencies": {
|
||||||
|
"node-gyp": "^11.1.0"
|
||||||
|
},
|
||||||
|
"__devDependencies": {
|
||||||
|
"@pnpm/assert-project": "workspace:*",
|
||||||
|
"@pnpm/byline": "catalog:",
|
||||||
|
"@pnpm/cache.commands": "workspace:*",
|
||||||
|
"@pnpm/cli-meta": "workspace:*",
|
||||||
|
"@pnpm/cli-utils": "workspace:*",
|
||||||
|
"@pnpm/client": "workspace:*",
|
||||||
|
"@pnpm/command": "workspace:*",
|
||||||
|
"@pnpm/common-cli-options-help": "workspace:*",
|
||||||
|
"@pnpm/config": "workspace:*",
|
||||||
|
"@pnpm/constants": "workspace:*",
|
||||||
|
"@pnpm/core-loggers": "workspace:*",
|
||||||
|
"@pnpm/crypto.hash": "workspace:*",
|
||||||
|
"@pnpm/default-reporter": "workspace:*",
|
||||||
|
"@pnpm/dependency-path": "workspace:*",
|
||||||
|
"@pnpm/env.path": "workspace:*",
|
||||||
|
"@pnpm/error": "workspace:*",
|
||||||
|
"@pnpm/exec.build-commands": "workspace:*",
|
||||||
|
"@pnpm/filter-workspace-packages": "workspace:*",
|
||||||
|
"@pnpm/find-workspace-dir": "workspace:*",
|
||||||
|
"@pnpm/lockfile.types": "workspace:*",
|
||||||
|
"@pnpm/logger": "workspace:*",
|
||||||
|
"@pnpm/modules-yaml": "workspace:*",
|
||||||
|
"@pnpm/nopt": "catalog:",
|
||||||
|
"@pnpm/parse-cli-args": "workspace:*",
|
||||||
|
"@pnpm/plugin-commands-audit": "workspace:*",
|
||||||
|
"@pnpm/plugin-commands-completion": "workspace:*",
|
||||||
|
"@pnpm/plugin-commands-config": "workspace:*",
|
||||||
|
"@pnpm/plugin-commands-deploy": "workspace:*",
|
||||||
|
"@pnpm/plugin-commands-doctor": "workspace:*",
|
||||||
|
"@pnpm/plugin-commands-env": "workspace:*",
|
||||||
|
"@pnpm/plugin-commands-init": "workspace:*",
|
||||||
|
"@pnpm/plugin-commands-installation": "workspace:*",
|
||||||
|
"@pnpm/plugin-commands-licenses": "workspace:*",
|
||||||
|
"@pnpm/plugin-commands-listing": "workspace:*",
|
||||||
|
"@pnpm/plugin-commands-outdated": "workspace:*",
|
||||||
|
"@pnpm/plugin-commands-patching": "workspace:*",
|
||||||
|
"@pnpm/plugin-commands-publishing": "workspace:*",
|
||||||
|
"@pnpm/plugin-commands-rebuild": "workspace:*",
|
||||||
|
"@pnpm/plugin-commands-script-runners": "workspace:*",
|
||||||
|
"@pnpm/plugin-commands-server": "workspace:*",
|
||||||
|
"@pnpm/plugin-commands-setup": "workspace:*",
|
||||||
|
"@pnpm/plugin-commands-store": "workspace:*",
|
||||||
|
"@pnpm/plugin-commands-store-inspecting": "workspace:*",
|
||||||
|
"@pnpm/prepare": "workspace:*",
|
||||||
|
"@pnpm/read-package-json": "workspace:*",
|
||||||
|
"@pnpm/read-project-manifest": "workspace:*",
|
||||||
|
"@pnpm/registry-mock": "catalog:",
|
||||||
|
"@pnpm/run-npm": "workspace:*",
|
||||||
|
"@pnpm/store.cafs": "workspace:*",
|
||||||
|
"@pnpm/tabtab": "catalog:",
|
||||||
|
"@pnpm/test-fixtures": "workspace:*",
|
||||||
|
"@pnpm/test-ipc-server": "workspace:*",
|
||||||
|
"@pnpm/tools.path": "workspace:*",
|
||||||
|
"@pnpm/tools.plugin-commands-self-updater": "workspace:*",
|
||||||
|
"@pnpm/types": "workspace:*",
|
||||||
|
"@pnpm/worker": "workspace:*",
|
||||||
|
"@pnpm/workspace.find-packages": "workspace:*",
|
||||||
|
"@pnpm/workspace.pkgs-graph": "workspace:*",
|
||||||
|
"@pnpm/workspace.read-manifest": "workspace:*",
|
||||||
|
"@pnpm/workspace.state": "workspace:*",
|
||||||
|
"@pnpm/write-project-manifest": "workspace:*",
|
||||||
|
"@types/cross-spawn": "catalog:",
|
||||||
|
"@types/is-windows": "catalog:",
|
||||||
|
"@types/pnpm__byline": "catalog:",
|
||||||
|
"@types/ramda": "catalog:",
|
||||||
|
"@types/semver": "catalog:",
|
||||||
|
"@zkochan/retry": "catalog:",
|
||||||
|
"@zkochan/rimraf": "catalog:",
|
||||||
|
"chalk": "catalog:",
|
||||||
|
"ci-info": "catalog:",
|
||||||
|
"cross-spawn": "catalog:",
|
||||||
|
"deep-require-cwd": "catalog:",
|
||||||
|
"delay": "catalog:",
|
||||||
|
"dir-is-case-sensitive": "catalog:",
|
||||||
|
"esbuild": "catalog:",
|
||||||
|
"execa": "catalog:",
|
||||||
|
"exists-link": "catalog:",
|
||||||
|
"is-windows": "catalog:",
|
||||||
|
"load-json-file": "catalog:",
|
||||||
|
"loud-rejection": "catalog:",
|
||||||
|
"normalize-newline": "catalog:",
|
||||||
|
"p-any": "catalog:",
|
||||||
|
"p-defer": "catalog:",
|
||||||
|
"path-name": "catalog:",
|
||||||
|
"pidtree": "catalog:",
|
||||||
|
"ps-list": "catalog:",
|
||||||
|
"ramda": "catalog:",
|
||||||
|
"read-yaml-file": "catalog:",
|
||||||
|
"render-help": "catalog:",
|
||||||
|
"semver": "catalog:",
|
||||||
|
"split-cmd": "catalog:",
|
||||||
|
"symlink-dir": "catalog:",
|
||||||
|
"tempy": "catalog:",
|
||||||
|
"tree-kill": "catalog:",
|
||||||
|
"write-json-file": "catalog:",
|
||||||
|
"write-pkg": "catalog:",
|
||||||
|
"write-yaml-file": "catalog:"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=18.12"
|
||||||
|
},
|
||||||
|
"jest": {
|
||||||
|
"preset": "@pnpm/jest-config/with-registry"
|
||||||
|
},
|
||||||
|
"preferGlobal": true,
|
||||||
|
"publishConfig": {
|
||||||
|
"tag": "next-10",
|
||||||
|
"executableFiles": [
|
||||||
|
"./dist/node-gyp-bin/node-gyp",
|
||||||
|
"./dist/node-gyp-bin/node-gyp.cmd",
|
||||||
|
"./dist/node_modules/node-gyp/bin/node-gyp.js"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"scripts": {
|
||||||
|
"bundle": "ts-node bundle.ts",
|
||||||
|
"start": "tsc --watch",
|
||||||
|
"lint": "eslint \"src/**/*.ts\" \"test/**/*.ts\"",
|
||||||
|
"pretest:e2e": "rimraf node_modules/.bin/pnpm",
|
||||||
|
"_test": "jest",
|
||||||
|
"test": "pnpm run compile && pnpm run _test",
|
||||||
|
"_compile": "tsc --build",
|
||||||
|
"compile": "tsc --build && pnpm run lint --fix && rimraf dist bin/nodes && pnpm run bundle && shx cp -r node-gyp-bin dist/node-gyp-bin && shx cp -r node_modules/@pnpm/tabtab/lib/templates dist/templates && shx cp -r node_modules/ps-list/vendor dist/vendor && shx cp pnpmrc dist/pnpmrc"
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -362,6 +362,7 @@ onMounted(() => {
|
|||||||
<option value="user">User</option>
|
<option value="user">User</option>
|
||||||
<option value="admin">Admin</option>
|
<option value="admin">Admin</option>
|
||||||
<option value="viewer">Viewer</option>
|
<option value="viewer">Viewer</option>
|
||||||
|
<!-- owner ist nicht über UI wählbar – bleibt Sonderrolle -->
|
||||||
</select>
|
</select>
|
||||||
</div>
|
</div>
|
||||||
<p v-if="createError" class="msg error">{{ createError }}</p>
|
<p v-if="createError" class="msg error">{{ createError }}</p>
|
||||||
|
|||||||
+5
-1
@@ -1,7 +1,11 @@
|
|||||||
# Changelog
|
# Changelog
|
||||||
|
|
||||||
> Letzte Aktualisierung: 2026-06-16
|
> Letzte Aktualisierung: 2026-06-20
|
||||||
|
|
||||||
|
- 2026-06-20: Task Board um klickbare Linear-inspirierte Detailansicht erweitert: Board-Karten öffnen jetzt ein strukturiertes Side/Overlay-Detailpanel mit editierbarem Titel, Beschreibung, Status, Priorität, Zuständigkeit und Fälligkeitsdatum sowie geladener Aktivität und Unteraufgaben. `frontend/src/views/TaskBoardView.vue` und `frontend/src/stores/tasks.ts` angepasst. Verifiziert mit `COREPACK_HOME=$PWD/.corepack-home PNPM_HOME=$PWD/.pnpm-home pnpm build`.
|
||||||
|
- 2026-06-19: Task-Board-Doku-Drift behoben: Header-Kommentar in TaskBoardView.vue von "4 columns" auf "6 columns" (Offen, InBearbeitung, Delegiert, Review, Blockiert, Erledigt) korrigiert. tasks.ts-Store-Kopfkommentar um delegated ergänzt.
|
||||||
|
- 2026-06-19: Veralteter TODO.md-Import entfernt: `ImportFromIrisTodoAsync` in TaskService.cs, ITaskService.cs und der import-from-iris-todo-API-Endpoint in DashboardController.cs gelöscht. ImportResultDto aus Models/Dashboard.cs entfernt. TODO.md ist abgeschafft, Task Board alleinige Quelle.
|
||||||
|
- 2026-06-19: Backend-Tests erweitert: TaskBoardTests.cs (69 Tests total, +13 neue) decken TaskStateHelper-BoardGroupKey/ToState/BoardGroupToState/DisplayString/AllStates/IsValidState/IsInProgressOrBlocked/IsDoneOrBacklog ab. Backend-Build 0 Errors, Frontend vue-tsc 0 Errors.
|
||||||
- 2026-06-16: Program.cs refactored: DI extrahiert in `Extensions/ServiceCollectionExtensions.cs`, Middleware in `Extensions/ApplicationBuilderExtensions.cs`, Helpers in `Helpers/PasswordHelper.cs`. Program.cs von ~200 auf 26 Zeilen reduziert.
|
- 2026-06-16: Program.cs refactored: DI extrahiert in `Extensions/ServiceCollectionExtensions.cs`, Middleware in `Extensions/ApplicationBuilderExtensions.cs`, Helpers in `Helpers/PasswordHelper.cs`. Program.cs von ~200 auf 26 Zeilen reduziert.
|
||||||
- 2026-06-16: Nexus auf Netcup (mission-control) redeployed. Neuer Stack unter `/home/projekte_bao/nexus/`. Traefik reverse-proxy mit Let's Encrypt TLS. Volume und Netzwerk-Namen bereinigt (postgres-data, internal). Compose-Pfade von Ionos auf Netcup migriert.
|
- 2026-06-16: Nexus auf Netcup (mission-control) redeployed. Neuer Stack unter `/home/projekte_bao/nexus/`. Traefik reverse-proxy mit Let's Encrypt TLS. Volume und Netzwerk-Namen bereinigt (postgres-data, internal). Compose-Pfade von Ionos auf Netcup migriert.
|
||||||
- 2026-06-16: Ollama-Modelle (2.4 GB) und alle ungenutzten Runtime-Dateien entfernt. Codex-Logs bereinigt (~342 MB). Workspace-Aufräumung (~3.1 GB gesamt).
|
- 2026-06-16: Ollama-Modelle (2.4 GB) und alle ungenutzten Runtime-Dateien entfernt. Codex-Logs bereinigt (~342 MB). Workspace-Aufräumung (~3.1 GB gesamt).
|
||||||
|
|||||||
Reference in New Issue
Block a user