Skip to content

Conversation

@elibosley
Copy link
Member

@elibosley elibosley commented Jan 16, 2025

Summary by CodeRabbit

Release Notes

  • Permission Management

    • Removed legacy role-based permission system.
    • Introduced new granular permission model for API keys and users.
    • Added support for resource-specific permissions.
  • API Key Management

    • Enhanced API key handling with more flexible permission configuration.
    • Improved synchronization of API key permissions.
    • Updated GraphQL schema to support new permission structure.
  • Authentication

    • Deprecated previous role assignment methods.
    • Introduced new MeResolver for retrieving user information.
    • Simplified permission checking mechanisms.
  • GraphQL Schema

    • Updated user and API key types to include permissions.
    • Added new Permission type for more precise access control.
  • Testing

    • Updated test suites to reflect new permission and authentication models.
    • Improved test coverage for new functionality.
    • Added new tests for MeResolver.
  • Chores

    • Removed unused files and methods related to legacy permission management.
    • Updated configuration files to streamline testing processes.
    • Removed deprecated API key files from the system.

@coderabbitai
Copy link
Contributor

coderabbitai bot commented Jan 16, 2025

Walkthrough

The pull request introduces significant changes to the permissions and role management system in the Unraid API. This includes the removal of existing role-based access control (RBAC) files, such as permissions.json, and the deletion of related tests and functionalities. New structures for managing permissions in API keys are introduced, along with updates to GraphQL schemas to accommodate these changes. Overall, the modifications streamline and modernize how permissions are handled within the application.

Changes

File Change Summary
api/dev/data/permissions.json Deleted role definitions for admin, user, and guest
api/package.json Added container:stop script
api/src/__test__/core/permissions.test.ts Deleted test file for setupPermissions function
api/src/core/index.ts Removed exports for permission-related modules
api/src/core/modules/index.ts Removed exports for permission-related modules
api/src/core/modules/get-me.ts Deleted getMe function for retrieving user info
api/src/core/modules/get-permissions.ts Deleted getPermissions function for user permissions
api/src/core/permission-manager.ts Removed PermissionManager class and related functionality
api/src/core/permissions.ts Deleted role-based access control definitions and setup function
api/src/core/utils/permissions/check-permission.ts Updated checkPermission function to disable permission checks
api/src/core/utils/permissions/ensure-permission.ts Deprecated ensurePermission function with modified parameters
api/src/core/utils/permissions/get-permissions.ts Deleted getPermissions function for role permissions
api/src/graphql/schema/types/auth/auth.graphql Added new Permission type and updated ApiKey and ApiKeyWithSecret types
api/src/graphql/schema/types/users/me.graphql Updated permissions field in Me type to use Permission objects
api/src/graphql/schema/types/users/user.graphql Added permissions field to UserAccount and User types
api/src/unraid-api/auth/api-key.service.ts Refactored API key management with permission synchronization
api/src/unraid-api/auth/auth.service.ts Introduced permission management functionality for API keys
api/src/unraid-api/graph/resolvers/me/me.resolver.ts Added new MeResolver with user information and permissions
api/src/unraid-api/graph/resolvers/resolvers.module.ts Added MeResolver to the ResolversModule
api/src/unraid-api/graph/sandbox-plugin.ts Updated renderSandboxPage function to return landing page
api/src/unraid-api/auth/api-key.service.spec.ts Updated tests to reflect new API key permissions structure
api/src/unraid-api/auth/auth.service.spec.ts Updated tests to include permissions in mockApiKeyWithSecret
api/src/index.ts Removed imports and method call related to local API key creation
api/vite.config.ts Updated test setup configuration to include reflect-metadata
api/src/mothership/utils/create-local-connect-api-key.ts Deleted function for creating local API key

Possibly related PRs

  • feat: generate key one time #992: The changes in this PR involve modifications to API key roles, which are relevant to the role definitions and permissions removed in the main PR, specifically regarding the "connect" role.
  • feat(api): graphql sandbox on unraid servers #1047: This PR introduces a GraphQL sandbox that may utilize the permissions structure defined in the main PR, particularly in how roles and permissions are managed within the GraphQL context.

Suggested reviewers

  • elibosley

Poem

🐰 Permissions dance, a rabbit's delight,
Old roles fade, new freedoms take flight
API keys sing their flexible song
GraphQL whispers where they belong
A code ballet of access reborn! 🔑✨

✨ Finishing Touches
  • 📝 Generate Docstrings (Beta)

🪧 Tips

Chat

There are 3 ways to chat with CodeRabbit:

  • Review comments: Directly reply to a review comment made by CodeRabbit. Example:
    • I pushed a fix in commit <commit_id>, please review it.
    • Generate unit testing code for this file.
    • Open a follow-up GitHub issue for this discussion.
  • Files and specific lines of code (under the "Files changed" tab): Tag @coderabbitai in a new review comment at the desired location with your query. Examples:
    • @coderabbitai generate unit testing code for this file.
    • @coderabbitai modularize this function.
  • PR comments: Tag @coderabbitai in a new PR comment to ask questions about the PR branch. For the best results, please provide a very specific query, as very limited context is provided in this mode. Examples:
    • @coderabbitai gather interesting stats about this repository and render them as a table. Additionally, render a pie chart showing the language distribution in the codebase.
    • @coderabbitai read src/utils.ts and generate unit testing code.
    • @coderabbitai read the files in the src/scheduler package and generate a class diagram using mermaid and a README in the markdown format.
    • @coderabbitai help me debug CodeRabbit configuration file.

Note: Be mindful of the bot's finite context window. It's strongly recommended to break down tasks such as reading entire modules into smaller chunks. For a focused discussion, use review comments to chat about specific files and their changes, instead of using the PR comments.

CodeRabbit Commands (Invoked using PR comments)

  • @coderabbitai pause to pause the reviews on a PR.
  • @coderabbitai resume to resume the paused reviews.
  • @coderabbitai review to trigger an incremental review. This is useful when automatic reviews are disabled for the repository.
  • @coderabbitai full review to do a full review from scratch and review all the files again.
  • @coderabbitai summary to regenerate the summary of the PR.
  • @coderabbitai generate docstrings to generate docstrings for this PR. (Beta)
  • @coderabbitai resolve resolve all the CodeRabbit review comments.
  • @coderabbitai configuration to show the current CodeRabbit configuration for the repository.
  • @coderabbitai help to get help.

Other keywords and placeholders

  • Add @coderabbitai ignore anywhere in the PR description to prevent this PR from being reviewed.
  • Add @coderabbitai summary to generate the high-level summary at a specific location in the PR description.
  • Add @coderabbitai anywhere in the PR title to generate the title automatically.

CodeRabbit Configuration File (.coderabbit.yaml)

  • You can programmatically configure CodeRabbit by adding a .coderabbit.yaml file to the root of your repository.
  • Please see the configuration documentation for more information.
  • If your editor has YAML language server enabled, you can add the path at the top of this file to enable auto-completion and validation: # yaml-language-server: $schema=https://coderabbit.ai/integrations/schema.v2.json

Documentation and Community

  • Visit our Documentation for detailed information on how to use CodeRabbit.
  • Join our Discord Community to get help, request features, and share feedback.
  • Follow us on X/Twitter for updates and announcements.

Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 14

🧹 Nitpick comments (10)
api/src/unraid-api/auth/api-key.service.ts (3)

20-20: Consider using a Map for efficient API key lookups.

Changing memoryApiKeys from a Map to an Array may decrease the performance of lookup operations, especially as the number of API keys grows. Using a Map allows for constant-time complexity when retrieving keys by their ID or other unique fields.

Apply this diff to revert back to a Map:

-protected memoryApiKeys: Array<ApiKeyWithSecret> = [];
+protected memoryApiKeys: Map<string, ApiKeyWithSecret> = new Map();

Update the methods that access memoryApiKeys accordingly to work with the Map.


98-107: Improve error handling in loadAllFromDisk.

Currently, if an API key file has invalid JSON or does not conform to the schema, an error is logged, and the process continues. Consider aggregating these errors and providing a summary, or alerting administrators about corrupted or invalid files for further investigation.


127-133: Simplify the return statement in findById.

The method can be simplified by returning the parsed key directly if it exists, which improves readability.

Apply this diff to streamline the method:

findById(id: string): ApiKey | null {
    const key = this.findByField('id', id);

-   if (key) {
-       return ApiKeySchema().parse(key);
-   }
-   return null;
+   return key ? ApiKeySchema().parse(key) : null;
}
api/src/core/utils/permissions/ensure-permission.ts (1)

6-6: Provide guidance on migrating to Casbin authorization.

The deprecation notice mentions "Use casbin auth in nest instead," but without additional context or references. Consider providing a link to documentation or guidelines on how to migrate to Casbin for authorization within NestJS.

api/src/unraid-api/auth/casbin/casbin.service.ts (1)

1-1: Remove unused OnModuleInit import.

The OnModuleInit interface is imported but not implemented by the CasbinService class.

-import { Injectable, InternalServerErrorException, Logger, OnModuleInit } from '@nestjs/common';
+import { Injectable, InternalServerErrorException, Logger } from '@nestjs/common';
api/src/unraid-api/graph/sandbox-plugin.ts (1)

67-68: Remove unnecessary empty line.

There's an extra empty line that doesn't add to code readability.

-    
    return serverListener.renderLandingPage();
api/src/unraid-api/auth/auth.service.ts (2)

32-32: Remove debugging console.log statement.

Production code should not contain console.log statements.

-            console.log('Permissions', apiKeyEntity)

220-220: Consider documenting the empty permissions array.

Add a comment explaining why the session user starts with empty permissions despite having the ADMIN role.

             roles: [Role.ADMIN],
-            permissions: [],
+            // Session users inherit permissions from their roles
+            permissions: [],
api/src/graphql/schema/types/auth/auth.graphql (1)

1-4: Consider using an enum for actions instead of string array.

The actions field using [String!] allows any string value, which could lead to inconsistency and typos. Consider defining an enum type for allowed actions to ensure type safety.

enum PermissionAction {
  CREATE
  READ
  UPDATE
  DELETE
  # Add other actions as needed
}

type Permission {
  resource: Resource!
  actions: [PermissionAction!]
}
api/src/graphql/schema/types/users/user.graphql (1)

6-6: Add documentation for the permissions field.

Consider adding descriptive documentation for the permissions field in both the interface and type definitions to improve schema clarity.

interface UserAccount {
+  """
+  List of explicit permissions granted to the account
+  """
  permissions: [Permission!]
}

type User implements UserAccount {
+  """
+  List of explicit permissions granted to the user
+  """
  permissions: [Permission!]
}

Also applies to: 65-65

📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro (Legacy)

📥 Commits

Reviewing files that changed from the base of the PR and between 7ee2acb and 8f22648.

⛔ Files ignored due to path filters (2)
  • api/src/graphql/generated/api/operations.ts is excluded by !**/generated/**
  • api/src/graphql/generated/api/types.ts is excluded by !**/generated/**
📒 Files selected for processing (26)
  • api/dev/Unraid.net/myservers.cfg (1 hunks)
  • api/dev/data/permissions.json (0 hunks)
  • api/package.json (1 hunks)
  • api/src/__test__/core/permissions.test.ts (0 hunks)
  • api/src/core/index.ts (0 hunks)
  • api/src/core/modules/get-me.ts (0 hunks)
  • api/src/core/modules/get-permissions.ts (0 hunks)
  • api/src/core/modules/index.ts (0 hunks)
  • api/src/core/permission-manager.ts (0 hunks)
  • api/src/core/permissions.ts (0 hunks)
  • api/src/core/utils/permissions/check-permission.ts (1 hunks)
  • api/src/core/utils/permissions/ensure-permission.ts (1 hunks)
  • api/src/core/utils/permissions/get-permissions.ts (0 hunks)
  • api/src/graphql/schema/types/auth/auth.graphql (2 hunks)
  • api/src/graphql/schema/types/users/me.graphql (1 hunks)
  • api/src/graphql/schema/types/users/user.graphql (2 hunks)
  • api/src/unraid-api/auth/api-key.service.ts (6 hunks)
  • api/src/unraid-api/auth/auth.service.ts (5 hunks)
  • api/src/unraid-api/auth/casbin/casbin.service.ts (2 hunks)
  • api/src/unraid-api/auth/user.decorator.ts (1 hunks)
  • api/src/unraid-api/graph/resolvers/auth/auth.resolver.spec.ts (0 hunks)
  • api/src/unraid-api/graph/resolvers/auth/auth.resolver.ts (0 hunks)
  • api/src/unraid-api/graph/resolvers/me/me.resolver.spec.ts (1 hunks)
  • api/src/unraid-api/graph/resolvers/me/me.resolver.ts (1 hunks)
  • api/src/unraid-api/graph/resolvers/resolvers.module.ts (2 hunks)
  • api/src/unraid-api/graph/sandbox-plugin.ts (1 hunks)
💤 Files with no reviewable changes (11)
  • api/src/core/index.ts
  • api/src/test/core/permissions.test.ts
  • api/src/core/modules/get-me.ts
  • api/src/unraid-api/graph/resolvers/auth/auth.resolver.ts
  • api/src/core/utils/permissions/get-permissions.ts
  • api/src/core/modules/get-permissions.ts
  • api/src/core/permission-manager.ts
  • api/src/core/modules/index.ts
  • api/src/unraid-api/graph/resolvers/auth/auth.resolver.spec.ts
  • api/src/core/permissions.ts
  • api/dev/data/permissions.json
🧰 Additional context used
🪛 Gitleaks (8.21.2)
api/dev/Unraid.net/myservers.cfg

11-11: Detected a Generic API Key, potentially exposing access to various services and sensitive operations.

(generic-api-key)


12-12: Detected a Generic API Key, potentially exposing access to various services and sensitive operations.

(generic-api-key)

⏰ Context from checks skipped due to timeout of 90000ms (3)
  • GitHub Check: Build Web App
  • GitHub Check: Build and Test API
  • GitHub Check: Cloudflare Pages
🔇 Additional comments (7)
api/src/unraid-api/auth/api-key.service.ts (1)

89-89: Initialize permissions appropriately when creating a new API key.

The permissions array is initialized to an empty array, but there might be default permissions that should be assigned based on the roles or other criteria. Ensure that the permissions are set correctly for the new API key.

Do you want me to assist in defining default permissions for new API keys based on roles?

api/src/unraid-api/graph/resolvers/me/me.resolver.ts (1)

13-18: Verify permission configuration aligns with business requirements.

The current permission setup allows ANY possession level for reading ME resource. Consider if this aligns with your security requirements.

✅ Verification successful

Permission configuration is appropriate for the ME resource

The ANY possession level is correct for this endpoint because:

  • This is a "current user" endpoint that only returns the authenticated user's own data
  • The implementation uses @graphqluser() decorator which ensures the data is automatically scoped to the requesting user
  • The Casbin policy confirms this is an intentional design choice
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Search for other ME resource permission configurations
rg -l "resource: Resource.ME" | xargs rg "possession:"

Length of output: 94


Script:

#!/bin/bash
# Search for ME resource usage and implementation
echo "=== ME Resource References ==="
rg "Resource.ME" -B 2 -A 2

echo -e "\n=== ME Resolver Implementation ==="
ast-grep --pattern 'class MeResolver {
  $$$
}'

Length of output: 2516

api/src/unraid-api/auth/casbin/casbin.service.ts (1)

8-8: LGTM! Good refactor moving enforcer to class property.

The refactor to use a class property for the enforcer instance improves state management and reusability.

Also applies to: 20-23

api/src/unraid-api/graph/resolvers/resolvers.module.ts (1)

22-22: LGTM! Clean integration of MeResolver.

The MeResolver is properly imported and added to the providers array, following the module's existing pattern.

Also applies to: 44-44

api/src/unraid-api/auth/auth.service.ts (1)

34-34: Verify permission synchronization order.

The code synchronizes roles before permissions. Ensure this order is intentional and doesn't create race conditions or permission gaps.

Also applies to: 46-46

✅ Verification successful

The permission synchronization order is correct and beneficial.

The current order (roles then permissions) is intentional and provides these advantages:

  • Ensures role-granted permissions are established first
  • Prevents race conditions through sequential await operations
  • Maintains permission consistency as custom permissions are added last
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Check for any role-dependent permission logic in the codebase
ast-grep --pattern 'if ($user.roles.$$$) {
  $$$
  addPermission($$$)
  $$$
}'

Length of output: 80


Script:

#!/bin/bash
# Find syncApiKeyPermissions implementation
ast-grep --pattern 'syncApiKeyPermissions($$$) {
  $$$
}'

# Look for role-related sync methods and their usage
rg "sync.*Role|role.*Sync" -A 5

# Get more context around the current usage
rg "syncApiKeyPermissions" -A 5 -B 5

Length of output: 10100

api/src/graphql/schema/types/users/me.graphql (1)

16-16: LGTM! Consistent with UserAccount interface.

The nullable permissions field aligns with the interface definition and maintains type consistency.

api/package.json (1)

28-28: LGTM! Consistent with existing container management scripts.

The new script follows the established pattern and provides a useful container management command.

Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 1

♻️ Duplicate comments (1)
api/src/unraid-api/graph/resolvers/me/me.resolver.spec.ts (1)

35-58: 🛠️ Refactor suggestion

Expand test coverage with additional scenarios.

While the basic happy path is covered, several critical test scenarios are missing:

  • Permission decorator validation
  • Error handling scenarios
  • Required field validation
  • Edge cases with missing or invalid user data

This was previously mentioned in a past review. Please add the suggested test cases to ensure robust coverage.

🧹 Nitpick comments (5)
api/src/unraid-api/auth/api-key.service.ts (2)

94-94: Consider initializing default permissions based on roles.

Currently, new API keys are created with an empty permissions array. Consider deriving default permissions from the assigned roles to ensure proper access control from creation.


103-134: Consider enhancing error logging for better debugging.

While the error handling is robust, consider adding more detailed logging:

  • Log the total number of files processed
  • Log successful API key loads
  • Include file paths in error messages
api/src/unraid-api/auth/auth.service.spec.ts (1)

35-40: Consider adding more test cases for permissions.

The mock data only includes a single permission. Consider adding test cases for:

  • Multiple permissions
  • Different resources and actions
  • Edge cases like empty permissions array
api/src/unraid-api/auth/api-key.service.spec.ts (1)

56-61: Enhance test coverage for permissions-related functionality.

Consider adding test cases for:

  • Validation of permissions structure
  • Error handling for invalid permissions
  • Persistence of permissions during API key updates
api/src/unraid-api/graph/resolvers/me/me.resolver.spec.ts (1)

10-59: Improve test suite structure and organization.

Consider restructuring the test suite to group related test cases:

  1. Add a describe block for successful scenarios
  2. Add a describe block for error scenarios
  3. Add a describe block for permission validation

Example structure:

describe('MeResolver', () => {
  // Setup code...

  describe('Successful scenarios', () => {
    it('should return user information');
    it('should handle minimal user data');
  });

  describe('Error scenarios', () => {
    it('should handle missing user data');
    it('should handle invalid permissions');
  });

  describe('Permission validation', () => {
    it('should deny access without proper permissions');
    it('should allow access with proper permissions');
  });
});
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro (Legacy)

📥 Commits

Reviewing files that changed from the base of the PR and between 8f22648 and 5f7ae01.

📒 Files selected for processing (4)
  • api/src/unraid-api/auth/api-key.service.spec.ts (8 hunks)
  • api/src/unraid-api/auth/api-key.service.ts (5 hunks)
  • api/src/unraid-api/auth/auth.service.spec.ts (2 hunks)
  • api/src/unraid-api/graph/resolvers/me/me.resolver.spec.ts (1 hunks)
⏰ Context from checks skipped due to timeout of 90000ms (3)
  • GitHub Check: Build Web App
  • GitHub Check: Build and Test API
  • GitHub Check: Cloudflare Pages
🔇 Additional comments (6)
api/src/unraid-api/auth/api-key.service.ts (6)

17-27: Review initialization order to prevent race conditions.

The setupWatch is called in the constructor before memoryApiKeys is populated, which could lead to inconsistent state if file changes occur before onModuleInit completes.


29-36: LGTM! Well-structured initialization with proper error handling.

The implementation correctly loads API keys from disk and includes appropriate error handling and logging.


42-47: Review the file watcher implementation for potential issues.

The implementation has two potential issues:

  1. The event handler parameters for the change event should be (path) => { ... } rather than (event, path) => { ... }.
  2. Clearing memoryApiKeys before reloading may result in temporary empty state.

137-151: LGTM! Well-structured implementation with proper error handling.

The method effectively uses findByField for consistency and includes appropriate error handling and schema validation.


Line range hint 158-170: Validate the field parameter to prevent invalid lookups.

There's no validation for the field parameter, which could lead to undefined behavior if an incorrect field name is provided.


221-224: LGTM! Correct file path handling.

The implementation now correctly uses join for file path construction and includes proper error handling.

Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 1

📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro (Legacy)

📥 Commits

Reviewing files that changed from the base of the PR and between 8309253 and e48fe00.

⛔ Files ignored due to path filters (1)
  • api/src/graphql/generated/api/operations.ts is excluded by !**/generated/**
📒 Files selected for processing (2)
  • api/src/graphql/schema/types/auth/auth.graphql (2 hunks)
  • api/src/unraid-api/graph/resolvers/me/me.resolver.ts (1 hunks)
🚧 Files skipped from review as they are similar to previous changes (1)
  • api/src/graphql/schema/types/auth/auth.graphql
⏰ Context from checks skipped due to timeout of 90000ms (3)
  • GitHub Check: Build Web App
  • GitHub Check: Build and Test API
  • GitHub Check: Cloudflare Pages
🔇 Additional comments (2)
api/src/unraid-api/graph/resolvers/me/me.resolver.ts (2)

1-7: LGTM! Well-organized imports.

The imports are clean, necessary, and follow NestJS best practices with proper use of path aliases.


12-26: Add input validation and error handling.

The implementation still lacks validation for user properties.

Consider implementing field-level permissions.

The current implementation uses a single permission check for all fields. Consider implementing field-level permissions for sensitive data like roles and permissions.

Example implementation:

@ResolveField()
@UsePermissions({
  action: AuthActionVerb.READ,
  resource: Resource.ROLES,
  possession: AuthPossession.ANY,
})
public async roles() {
  // Implementation
}

Simplify the return statement using object shorthand notation.

     public async me(@GraphqlUser() user: UserAccount): Promise<Me> {
+        const { description, permissions, id, name, roles } = user;
         return {
-            description: user.description,
-            permissions: user.permissions,
-            id: user.id,
-            name: user.name,
-            roles: user.roles,
+            description,
+            permissions,
+            id,
+            name,
+            roles,
         };
     }

@github-actions
Copy link
Contributor

This plugin has been deployed to Cloudflare R2 and is available for testing.
Download it at this URL: https://preview.dl.unraid.net/unraid-api/pr/1068/dynamix.unraid.net.staging.plg

Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 0

♻️ Duplicate comments (2)
api/src/unraid-api/auth/api-key.service.ts (2)

28-29: ⚠️ Potential issue

Move setupWatch initialization to onModuleInit.

The file watcher is being set up before memoryApiKeys is populated, which could lead to race conditions. The watcher might receive events before the initial data is loaded.

Apply this diff to fix the initialization order:

constructor() {
    this.basePath = getters.paths()['auth-keys'];
    ensureDirSync(this.basePath);
-   this.setupWatch();
}

async onModuleInit() {
    try {
        this.memoryApiKeys = await this.loadAllFromDisk();
        await this.createLocalApiKeyForConnectIfNecessary();
+       this.setupWatch();
    } catch (error) {
        this.logger.error('Failed to initialize API keys:', error);
        throw error;
    }
}

46-50: ⚠️ Potential issue

Fix file watcher implementation issues.

The current implementation has two issues:

  1. Incorrect event handler parameters for chokidar's 'change' event
  2. Race condition when updating memoryApiKeys

Apply this diff to fix both issues:

-watch(this.basePath, { ignoreInitial: false }).on('change', async (event, path) => {
-    this.logger.debug(`API key storage event: ${event} on ${path}`);
-    this.memoryApiKeys = [];
-    this.memoryApiKeys = await this.loadAllFromDisk();
+watch(this.basePath, { ignoreInitial: false }).on('change', async (path) => {
+    this.logger.debug(`API key changed: ${path}`);
+    const updatedApiKeys = await this.loadAllFromDisk();
+    this.memoryApiKeys = updatedApiKeys;
});
🧹 Nitpick comments (4)
api/src/unraid-api/auth/api-key.service.ts (4)

97-97: Add type assertion for permissions array initialization.

Consider adding a type assertion to ensure type safety of the permissions array.

-apiKey.permissions = [];
+apiKey.permissions = [] as Array<{ resource: string; actions: string[] }>;

106-142: Enhance logging for Connect API key creation process.

While the implementation is solid, adding more detailed logging would help with debugging and monitoring.

 if (remote.apikey && (!remote.localApiKey || !(await this.findByKey(remote.localApiKey)))) {
+    this.logger.debug('Checking for existing Connect API key...');
     const hasExistingKey = this.findByField('name', 'Connect');

     if (hasExistingKey) {
+        this.logger.debug('Connect API key already exists');
         return;
     }
+    this.logger.debug('Creating new Connect API key...');

157-174: Enhance error messages with actionable information.

The error messages could be more specific about what action users or administrators should take.

-throw new Error('Authentication system error: Corrupted key file');
+throw new Error('Authentication system error: Corrupted key file. Please check the file format and restore from backup if necessary.');

-throw new Error('Invalid API key structure');
+throw new Error('Invalid API key structure. Please verify the key file matches the expected schema.');

-throw new Error('Failed to list API keys');
+throw new Error('Failed to list API keys. Please check directory permissions and ensure the auth-keys directory exists.');

178-191: Add input validation for ID parameter.

Consider adding validation for the ID parameter to ensure it's a valid UUID.

 findById(id: string): ApiKey | null {
+    if (!id || !/^[0-9a-f]{8}-[0-9a-f]{4}-4[0-9a-f]{3}-[89ab][0-9a-f]{3}-[0-9a-f]{12}$/i.test(id)) {
+        throw new GraphQLError('Invalid API key ID format');
+    }
     try {
         const key = this.findByField('id', id);
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro (Legacy)

📥 Commits

Reviewing files that changed from the base of the PR and between df548f3 and 9c4b340.

📒 Files selected for processing (3)
  • api/src/index.ts (1 hunks)
  • api/src/mothership/utils/create-local-connect-api-key.ts (0 hunks)
  • api/src/unraid-api/auth/api-key.service.ts (4 hunks)
💤 Files with no reviewable changes (1)
  • api/src/mothership/utils/create-local-connect-api-key.ts
🚧 Files skipped from review as they are similar to previous changes (1)
  • api/src/index.ts
⏰ Context from checks skipped due to timeout of 90000ms (3)
  • GitHub Check: Build Web App
  • GitHub Check: Build and Test API
  • GitHub Check: Cloudflare Pages
🔇 Additional comments (2)
api/src/unraid-api/auth/api-key.service.ts (2)

31-39: Well-structured initialization with proper error handling!

The implementation includes comprehensive error handling and logging, ensuring robust initialization of the API key service.


262-265: Correctly implemented file path handling!

The implementation now properly uses join for file path construction and includes proper formatting for the JSON output.

@github-actions
Copy link
Contributor

This plugin has been deployed to Cloudflare R2 and is available for testing.
Download it at this URL: https://preview.dl.unraid.net/unraid-api/pr/1068/dynamix.unraid.net.staging.plg

@github-actions
Copy link
Contributor

This plugin has been deployed to Cloudflare R2 and is available for testing.
Download it at this URL: https://preview.dl.unraid.net/unraid-api/pr/1068/dynamix.unraid.net.staging.plg

Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 0

♻️ Duplicate comments (1)
api/src/unraid-api/graph/resolvers/me/me.resolver.ts (1)

10-12: 🛠️ Refactor suggestion

Remove unnecessary constructor and enhance Resolver type information.

The empty constructor serves no purpose and should be removed. Also, the @Resolver() decorator should specify the type it resolves.

-@Resolver()
+@Resolver('Me')
 export class MeResolver {
-    constructor() {}
🧰 Tools
🪛 Biome (1.9.4)

[error] 12-12: This constructor is unnecessary.

Unsafe fix: Remove the unnecessary constructor.

(lint/complexity/noUselessConstructor)

🧹 Nitpick comments (1)
api/src/unraid-api/graph/resolvers/me/me.resolver.ts (1)

14-14: Add return type to Query decorator.

The @Query() decorator should specify its return type for better type safety and documentation.

-    @Query()
+    @Query(() => Me)
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro (Legacy)

📥 Commits

Reviewing files that changed from the base of the PR and between f1a012c and 5205515.

📒 Files selected for processing (2)
  • api/src/index.ts (0 hunks)
  • api/src/unraid-api/graph/resolvers/me/me.resolver.ts (1 hunks)
💤 Files with no reviewable changes (1)
  • api/src/index.ts
🧰 Additional context used
🪛 Biome (1.9.4)
api/src/unraid-api/graph/resolvers/me/me.resolver.ts

[error] 12-12: This constructor is unnecessary.

Unsafe fix: Remove the unnecessary constructor.

(lint/complexity/noUselessConstructor)

⏰ Context from checks skipped due to timeout of 90000ms (1)
  • GitHub Check: Cloudflare Pages
🔇 Additional comments (1)
api/src/unraid-api/graph/resolvers/me/me.resolver.ts (1)

20-34: LGTM! Input validation and error handling look good.

The implementation includes proper input validation, error handling, and default values for optional fields.

Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 4

🧹 Nitpick comments (6)
api/src/utils.ts (1)

247-261: LGTM! Consider enhancing type validation.

The type predicate implementation is correct and follows TypeScript best practices. However, consider strengthening the validation for more robust type safety:

 export function isUserAccount(user: any): user is UserAccount {
     return (
         user &&
-        typeof user.id === 'string' &&
-        typeof user.name === 'string' &&
-        Array.isArray(user.permissions) &&
-        Array.isArray(user.roles)
+        typeof user.id === 'string' && user.id.length > 0 &&
+        typeof user.name === 'string' && user.name.length > 0 &&
+        Array.isArray(user.permissions) && user.permissions.every(p => typeof p === 'string') &&
+        Array.isArray(user.roles) && user.roles.every(r => typeof r === 'string')
     );
 }

This enhancement:

  • Ensures non-empty strings for id and name
  • Validates that array elements are strings
api/src/unraid-api/graph/resolvers/me/me.resolver.spec.ts (1)

37-39: Expand the basic test coverage.

While checking if the resolver is defined is a good start, consider adding more basic validation tests:

  1. Verify resolver instance type
  2. Verify AuthZService injection

Example tests to add:

it('should be instance of MeResolver', () => {
  expect(resolver).toBeInstanceOf(MeResolver);
});

it('should have AuthZService injected', () => {
  expect(authzService).toBeDefined();
  expect(authzService.checkPermission).toBeDefined();
});
api/src/unraid-api/auth/auth.service.ts (1)

32-32: Remove debug console.log statement.

Production code should use the logger service instead of console.log for consistency and proper log level management.

-            console.log('Permissions', apiKeyEntity);
+            this.logger.debug('API Key permissions:', apiKeyEntity);
api/src/unraid-api/auth/api-key.service.ts (3)

22-22: Consider using Map for better lookup performance.

Changing memoryApiKeys from Map to Array might impact performance for key lookups. Using Array.find() has O(n) complexity, while Map.get() has O(1).

-    protected memoryApiKeys: Array<ApiKeyWithSecret> = [];
+    protected memoryApiKeys: Map<string, ApiKeyWithSecret> = new Map();

35-37: Enhance error logging for initialization failures.

The current error logging could be more detailed to help with troubleshooting initialization issues.

-            this.logger.error('Failed to initialize API keys:', error);
+            this.logger.error('Failed to initialize API keys:', {
+                error: error instanceof Error ? error.message : String(error),
+                stack: error instanceof Error ? error.stack : undefined,
+                phase: 'initialization'
+            });

148-166: Optimize file loading for large directories.

Consider using Promise.all for parallel file reading to improve performance with many API keys.

-            for (const file of files) {
-                if (file.endsWith('.json')) {
-                    try {
-                        const content = await readFile(join(this.basePath, file), 'utf8');
-                        const apiKey = ApiKeyWithSecretSchema().parse(JSON.parse(content));
-                        apiKeys.push(apiKey);
-                    } catch (error) {
+            const readPromises = files
+                .filter(file => file.endsWith('.json'))
+                .map(async file => {
+                    try {
+                        const content = await readFile(join(this.basePath, file), 'utf8');
+                        return ApiKeyWithSecretSchema().parse(JSON.parse(content));
+                    } catch (error) {
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro (Legacy)

📥 Commits

Reviewing files that changed from the base of the PR and between 5205515 and 58ce565.

📒 Files selected for processing (5)
  • api/src/unraid-api/auth/api-key.service.ts (4 hunks)
  • api/src/unraid-api/auth/auth.service.ts (5 hunks)
  • api/src/unraid-api/auth/user.decorator.ts (1 hunks)
  • api/src/unraid-api/graph/resolvers/me/me.resolver.spec.ts (1 hunks)
  • api/src/utils.ts (1 hunks)
🚧 Files skipped from review as they are similar to previous changes (1)
  • api/src/unraid-api/auth/user.decorator.ts
🧰 Additional context used
🪛 Biome (1.9.4)
api/src/unraid-api/auth/api-key.service.ts

[error] 200-200: Do not access Object.prototype method 'hasOwnProperty' from target object.

It's recommended using Object.hasOwn() instead of using Object.hasOwnProperty().
See MDN web docs for more details.

(lint/suspicious/noPrototypeBuiltins)

⏰ Context from checks skipped due to timeout of 90000ms (3)
  • GitHub Check: Build Web App
  • GitHub Check: Build and Test API
  • GitHub Check: Cloudflare Pages
🔇 Additional comments (5)
api/src/unraid-api/auth/auth.service.ts (2)

100-114: Implementation looks good!

The syncApiKeyPermissions method follows best practices with proper permission cleanup, parallel execution, and error handling.


226-226: ⚠️ Potential issue

Verify session user permissions.

The session user has the ADMIN role but returns an empty permissions array. This seems inconsistent with the method's comment that states "Session receives administrator permissions". This could lead to unexpected authorization issues.

Consider either:

  1. Syncing permissions based on the ADMIN role, or
  2. Updating the method's documentation to clarify the permission model
-            permissions: [],
+            // TODO: Sync these permissions with ADMIN role capabilities
+            permissions: [
+                {
+                    resource: '*',
+                    actions: ['*']
+                }
+            ],
api/src/unraid-api/auth/api-key.service.ts (3)

48-49: Avoid clearing memoryApiKeys before reloading to prevent race conditions.

Clearing memoryApiKeys before reloading may result in temporary empty state.


266-269: LGTM! Proper file path handling implemented.

The implementation correctly uses path.join and proper JSON formatting.


200-201: 🛠️ Refactor suggestion

Use Object.hasOwn instead of hasOwnProperty.

Replace Object.prototype.hasOwnProperty with Object.hasOwn for better safety and maintainability.

-        if (!ApiKeyWithSecretSchema().shape.hasOwnProperty(field)) {
+        if (!Object.hasOwn(ApiKeyWithSecretSchema().shape, field)) {

Likely invalid or redundant comment.

🧰 Tools
🪛 Biome (1.9.4)

[error] 200-200: Do not access Object.prototype method 'hasOwnProperty' from target object.

It's recommended using Object.hasOwn() instead of using Object.hasOwnProperty().
See MDN web docs for more details.

(lint/suspicious/noPrototypeBuiltins)

Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 2

♻️ Duplicate comments (3)
api/src/unraid-api/auth/api-key.service.ts (3)

26-28: ⚠️ Potential issue

Set secure file permissions for API key storage.

The API key storage directory needs proper file permissions to prevent unauthorized access.

Apply this diff to set secure file permissions:

 constructor() {
     this.basePath = getters.paths()['auth-keys'];
     ensureDirSync(this.basePath);
+    chmodSync(this.basePath, 0o700);
 }

Ensure you've imported the necessary function:

import { chmodSync } from 'fs';

45-51: ⚠️ Potential issue

Fix race condition in setupWatch and add cleanup.

The current implementation has two issues:

  1. Race condition when clearing and reloading memoryApiKeys
  2. No cleanup of the watcher when the service is destroyed

Apply this diff to fix these issues:

+private watcher: ReturnType<typeof watch> | null = null;
+
 private setupWatch() {
-    watch(this.basePath, { ignoreInitial: false }).on('change', async (path) => {
+    this.watcher = watch(this.basePath, { ignoreInitial: false });
+    this.watcher.on('change', async (path) => {
         this.logger.debug(`API key changed: ${path}`);
-        this.memoryApiKeys = [];
-        this.memoryApiKeys = await this.loadAllFromDisk();
+        const updatedKeys = await this.loadAllFromDisk();
+        this.memoryApiKeys = updatedKeys;
     });
 }
+
+async onModuleDestroy() {
+    if (this.watcher) {
+        await this.watcher.close();
+        this.watcher = null;
+    }
+}

Don't forget to implement OnModuleDestroy:

-export class ApiKeyService implements OnModuleInit {
+export class ApiKeyService implements OnModuleInit, OnModuleDestroy {

106-142: ⚠️ Potential issue

Implement atomic operations for Connect API key creation.

The current implementation has potential race conditions and non-atomic operations when creating and updating the Connect API key.

Apply this diff to implement atomic operations:

 private async createLocalApiKeyForConnectIfNecessary() {
     if (getters.config().status !== FileLoadStatus.LOADED) {
         this.logger.error('Config file not loaded, cannot create local API key');
         return;
     }

     const { remote } = getters.config();
-    if (remote.apikey && (!remote.localApiKey || !(await this.findByKey(remote.localApiKey)))) {
-        const hasExistingKey = this.findByField('name', 'Connect');
-
-        if (hasExistingKey) {
-            return;
-        }
+    if (!remote.apikey || (remote.localApiKey && await this.findByKey(remote.localApiKey))) {
+        return;
+    }
+
+    const existingKey = this.findByField('name', 'Connect');
+    if (existingKey) {
+        return;
+    }

     try {
+        // Start atomic operation
         const localApiKey = await this.create(
             'Connect',
             'API key for Connect user',
             [Role.CONNECT],
             true
         );

         if (!localApiKey?.key) {
             throw new Error('Failed to create local API key - no key returned');
         }

-        if (localApiKey?.key) {
-            store.dispatch(
-                updateUserConfig({
-                    remote: {
-                        localApiKey: localApiKey.key,
-                    },
-                })
-            );
-        } else {
-            this.logger.error('Failed to create local API key - no key returned');
-            throw new Error('Failed to create local API key');
-        }
+        await store.dispatch(
+            updateUserConfig({
+                remote: {
+                    localApiKey: localApiKey.key,
+                },
+            })
+        );
+    } catch (error) {
+        // Rollback on failure
+        if (localApiKey?.id) {
+            await this.deleteApiKey(localApiKey.id);
+        }
+        throw error;
     }
 }
🧹 Nitpick comments (5)
api/src/unraid-api/graph/resolvers/me/me.resolver.spec.ts (4)

21-28: Enhance type safety and verification in AuthZService mock.

The current mock implementation could be improved for better type safety and verification:

 checkPermission: vi.fn().mockImplementation((action, resource, possession) => {
-    // Return false for specific test scenarios
-    if (action === 'write' && resource === Resource.ME) {
-        return Promise.resolve(false);
-    }
-    return Promise.resolve(true);
+    // Verify possession parameter
+    if (!['own', 'any'].includes(possession)) {
+        throw new Error(`Invalid possession: ${possession}`);
+    }
+    
+    // Type-safe resource check
+    if (!Object.values(Resource).includes(resource)) {
+        throw new Error(`Invalid resource: ${resource}`);
+    }
+    
+    // Type-safe action check
+    const validActions = ['read', 'write', 'delete'];
+    if (!validActions.includes(action)) {
+        throw new Error(`Invalid action: ${action}`);
+    }
+    
+    return Promise.resolve(action !== 'write' || resource !== Resource.ME);
 }),

37-39: Enhance basic resolver verification.

While checking if the resolver is defined is good, consider adding more basic verification:

 it('should be defined', () => {
     expect(resolver).toBeDefined();
+    expect(resolver).toBeInstanceOf(MeResolver);
+    expect(resolver.me).toBeInstanceOf(Function);
+    expect(authzService).toBeDefined();
+    expect(authzService.checkPermission).toBeInstanceOf(Function);
 });

57-64: Remove redundant assertions.

The block-level equality check is redundant with the individual property checks that follow:

-        expect(result).toEqual({
-            id: mockUser.id,
-            name: mockUser.name,
-            description: mockUser.description,
-            roles: mockUser.roles,
-            permissions: mockUser.permissions,
-        } as Me);
-        expect(result).toBeDefined();
+        expect(result).toBeDefined();

69-73: Avoid non-null assertion operator and improve type safety.

Replace non-null assertions with proper type checking:

-        expect(result.permissions!).toHaveLength(1);
-        expect(result.permissions![0]).toEqual({
+        expect(result.permissions).toBeDefined();
+        expect(result.permissions).toHaveLength(1);
+        
+        const [permission] = result.permissions ?? [];
+        expect(permission).toBeDefined();
+        expect(permission).toEqual({
             resource: Resource.ME,
             actions: expect.arrayContaining(['read']),
         });
api/src/unraid-api/auth/api-key.service.ts (1)

144-176: Improve file handling in loadAllFromDisk.

The current implementation could be optimized to handle file resources better and provide more detailed error messages.

Apply this diff to improve file handling:

 async loadAllFromDisk(): Promise<ApiKeyWithSecret[]> {
+    const apiKeys: ApiKeyWithSecret[] = [];
     try {
         const files = await readdir(this.basePath);
-        const apiKeys: ApiKeyWithSecret[] = [];
 
         for (const file of files) {
             if (file.endsWith('.json')) {
+                const filePath = join(this.basePath, file);
                 try {
-                    const content = await readFile(join(this.basePath, file), 'utf8');
+                    const content = await readFile(filePath, 'utf8');
                     const apiKey = ApiKeyWithSecretSchema().parse(JSON.parse(content));
 
                     apiKeys.push(apiKey);
                 } catch (error) {
                     if (error instanceof SyntaxError) {
-                        throw new Error('Authentication system error: Corrupted key file');
+                        throw new Error(`Authentication system error: Corrupted key file ${file}`);
                     }
                     if (error instanceof ZodError) {
                         this.logger.error(`Invalid API key structure in file ${file}`, error.errors);
-                        throw new Error('Invalid API key structure');
+                        throw new Error(`Invalid API key structure in file ${file}`);
                     }
-                    this.logger.warn(`Error reading API key file ${file}: ${error}`);
+                    const errorMessage = error instanceof Error ? error.message : String(error);
+                    this.logger.warn(`Error reading API key file ${file}: ${errorMessage}`);
                 }
             }
         }
-        return apiKeys;
     } catch (error) {
         if (error instanceof Error) {
             throw error;
         }
         this.logger.error(`Failed to read API key directory: ${error}`);
         throw new Error('Failed to list API keys');
     }
+    return apiKeys;
 }
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro (Legacy)

📥 Commits

Reviewing files that changed from the base of the PR and between 58ce565 and 9311000.

📒 Files selected for processing (2)
  • api/src/unraid-api/auth/api-key.service.ts (4 hunks)
  • api/src/unraid-api/graph/resolvers/me/me.resolver.spec.ts (1 hunks)
⏰ Context from checks skipped due to timeout of 90000ms (3)
  • GitHub Check: Build and Test API
  • GitHub Check: Build Web App
  • GitHub Check: Cloudflare Pages
🔇 Additional comments (1)
api/src/unraid-api/auth/api-key.service.ts (1)

276-279: ⚠️ Potential issue

Set secure file permissions when saving API keys.

API key files should have restricted permissions to prevent unauthorized access.

Apply this diff to set secure file permissions:

 await writeFile(
     join(this.basePath, `${validatedApiKey.id}.json`),
     JSON.stringify(sortedApiKey, null, 2)
 );
+await chmodSync(join(this.basePath, `${validatedApiKey.id}.json`), 0o600);

Likely invalid or redundant comment.

@github-actions
Copy link
Contributor

This plugin has been deployed to Cloudflare R2 and is available for testing.
Download it at this URL: https://preview.dl.unraid.net/unraid-api/pr/1068/dynamix.unraid.net.staging.plg

@github-actions github-actions bot mentioned this pull request Feb 20, 2025
elibosley pushed a commit that referenced this pull request Feb 20, 2025
:robot: I have created a release *beep* *boop*
---


<details><summary>4.1.2</summary>

## 4.1.2 (2025-02-20)


### Features

* add api key creation logic
([81382bc](https://github.com/unraid/api/commit/81382bcf1d26364ad9c5445530f648209101cf91))
* add category.json
([c9e87e2](https://github.com/unraid/api/commit/c9e87e2e5b47a8801b7865ed586c803d0b470915))
* add command to package.json scripts
([0dfb07f](https://github.com/unraid/api/commit/0dfb07f9eb519e60441f4123423f65acfdffca3b))
* add csrf support to api & web components
([#999](https://github.com/unraid/api/issues/999))
([19241ed](https://github.com/unraid/api/commit/19241ed55f5112f878b9890d8695badf7eb1c3eb))
* add description flag, remove console log, and update readme
([c416c30](https://github.com/unraid/api/commit/c416c30951de4ed6b8d7a8c014403772db1c2015))
* add developer docs
([#1128](https://github.com/unraid/api/issues/1128))
([bb2e340](https://github.com/unraid/api/commit/bb2e340b68268d5121db650b27e8b2580c7966bb))
* add line about recommendation for sso command
([44727a8](https://github.com/unraid/api/commit/44727a8d1a7c16c566678da43119b17a6303e375))
* add log rotation
([f5c7ad9](https://github.com/unraid/api/commit/f5c7ad9221f80e4630e69f78d57f08f4c7252719))
* add logging around fixture downloads
([a1ce27b](https://github.com/unraid/api/commit/a1ce27b17c970657f52635600f0d13116523f928))
* add logrotate cron again
([4f85f66](https://github.com/unraid/api/commit/4f85f6687f920dae50277e726e2db2c3d946e867))
* add patch for auth-request.php
([ec6ec56](https://github.com/unraid/api/commit/ec6ec562f43aac9947de2e9c269181303f42b2db))
* add user with cli
([37458cd](https://github.com/unraid/api/commit/37458cd7408a1ad8aedca66a55ff13ac19ee30db))
* address log level feedback
([49774aa](https://github.com/unraid/api/commit/49774aae459797f04ef2866ca064050aa476ae91))
* allow csrf passing through querystring
([dba38c0](https://github.com/unraid/api/commit/dba38c0d149a77e4104c718c53d426330a17f2fa))
* allow deletion and creation of files with patches
([32c9524](https://github.com/unraid/api/commit/32c952402c25e8340b1c628b4d0fdc4816b28ade))
* always ensureDirectory for keys exists
([c6e9f80](https://github.com/unraid/api/commit/c6e9f804c58e44b46bce9f0da2260888544354cd))
* **api:** graphql sandbox on unraid servers
([#1047](https://github.com/unraid/api/issues/1047))
([ec504f3](https://github.com/unraid/api/commit/ec504f39297c92b64d9d3cc2f8f482cc1f3a2e44))
* **api:** omit tz from sys time date format by default
([b2acde3](https://github.com/unraid/api/commit/b2acde3351d7afe18a2902e90b672537aadabffd))
* **api:** rm 2fa & t2fa from myservers config type
([#996](https://github.com/unraid/api/issues/996))
([89e791a](https://github.com/unraid/api/commit/89e791ad2e6f0395bee05e3f8bdcb2c8d72305dd))
* async disk mapping
([bbb27e6](https://github.com/unraid/api/commit/bbb27e686897e4f9a0c926553d75aa046d7a8323))
* async hypervisor and FIXED vm listing
([e79f4dd](https://github.com/unraid/api/commit/e79f4ddbc7061c249efb8214a311bb629628f669))
* auto-docusaurus-prs
([#1127](https://github.com/unraid/api/issues/1127))
([1147e76](https://github.com/unraid/api/commit/1147e762ae2fed6dea198fa38d6bcc514a1e66fb))
* begin building plugin with node instead of bash
([#1120](https://github.com/unraid/api/issues/1120))
([253b65a](https://github.com/unraid/api/commit/253b65a85ab9c5f53d53ef265b41aa132678f278))
* better patch application
([a3e7daa](https://github.com/unraid/api/commit/a3e7daa6a6565ac81004ffd13da35d8b95b429cf))
* better pm2 calls, log lines
([338ce30](https://github.com/unraid/api/commit/338ce3061310dfc42ad5f65edacbe5272de4afc7))
* cleanup config entries
([943e73f](https://github.com/unraid/api/commit/943e73fa696b6ecec3227be914ab4962c4fee79d))
* cleanup disclaimer and command to add users
([6be3af8](https://github.com/unraid/api/commit/6be3af8d7569d9c413dd9349df52e3fa4cb4f631))
* cli Commands
([f8e5367](https://github.com/unraid/api/commit/f8e5367f3eb47daa5bcbd7711ae5835369502a1d))
* CLI options for adding and deleting users
([16bf6d4](https://github.com/unraid/api/commit/16bf6d4c27ae8fa8d6d05ec4b28ce49a12673278))
* coderabbit suggestion
([11ac36c](https://github.com/unraid/api/commit/11ac36c3616a90853d91467526fd39ecba17db88))
* configure PM2 on startup
([2b908f1](https://github.com/unraid/api/commit/2b908f100b9eefaccf2264d5ff9945667568acf0))
* convert to pnpm monorepo
([#1137](https://github.com/unraid/api/issues/1137))
([8d89f8b](https://github.com/unraid/api/commit/8d89f8b20d6f3983d4e85b33827a857aa862db37))
* create key cli command logic and add to index command list
([9b2a62d](https://github.com/unraid/api/commit/9b2a62d642b0942e3787e4ddd582a66e40321ab2))
* csv validation
([84aae15](https://github.com/unraid/api/commit/84aae15a73014592c226fa3701e34e57c7b60b46))
* default value for option
([6513fc4](https://github.com/unraid/api/commit/6513fc49de61c836e1aabf32a874d7da7da18adb))
* disable casbin logging
([2518e7c](https://github.com/unraid/api/commit/2518e7c506f0d3aa9f44031d61dce95d9db0a4cf))
* docstrings
([b836ba7](https://github.com/unraid/api/commit/b836ba72516c554ee8973d69aaaa4ed35b465fa7))
* dont pass entire server state for privacy
([54e3f17](https://github.com/unraid/api/commit/54e3f17bd9e541f50970c696bbe8b602ec38a748))
* download fixtures from the web
([1258c2b](https://github.com/unraid/api/commit/1258c2bc1813f0fa3cd52b4932302ad12b4edd01))
* enable sandbox in dev mode
([4536d70](https://github.com/unraid/api/commit/4536d7092d77c68f5a996fd63bf74ce6e64f5efe))
* enable sandbox with developer command
([c354d48](https://github.com/unraid/api/commit/c354d482283295547afeb99c5e110b0181197c44))
* enable token sign in with comma separated subs in myservers.config
([ebed5bd](https://github.com/unraid/api/commit/ebed5bddea1445d9aaaee60d54758dc74b77271e))
* exit cli after running command
([04bf528](https://github.com/unraid/api/commit/04bf528616fcbdf916916734a12d5fd32db9a06d))
* extensive file checking
([ab881c8](https://github.com/unraid/api/commit/ab881c8aed8dd4aa9fd71c32b50d3514d1496fa5))
* fallback to local
([a2579c2](https://github.com/unraid/api/commit/a2579c2a7f80f54b4cc61533aec9ecc41a7e7f54))
* faster failure logic
([b439434](https://github.com/unraid/api/commit/b439434f1574e174fcf23f3a5f5b8df8e092eb1e))
* fix docusaurus build + update snapshot
([23b27bd](https://github.com/unraid/api/commit/23b27bd63ea99f4137538eab40501daa67d7e3f5))
* force linting on build
([43e6639](https://github.com/unraid/api/commit/43e663998a55e83c142067cb64ae7a331395fe68))
* generate key one time
([afe53c3](https://github.com/unraid/api/commit/afe53c30ea9987e6d8728faa2cb7291f8a126ecb))
* glob for files
([3fe281f](https://github.com/unraid/api/commit/3fe281f1ae28e3cbc089b5244a6ae2863b20adcb))
* hypervisor async imports
([32686ca](https://github.com/unraid/api/commit/32686ca4f0c25c43c6a9f7162bb8179b39e58f7e))
* ignore generated code
([68265a2](https://github.com/unraid/api/commit/68265a26efa588b60001310b9a11b398f04ae88f))
* improve packing
([9ef02d5](https://github.com/unraid/api/commit/9ef02d53666b70d41fdd186364808deac715e1ff))
* initial patcher implementation using the diff tool
([c87acbb](https://github.com/unraid/api/commit/c87acbb146c2e4e30997c964cd8be325dee68cea))
* initial setup of permissions on keys
([#1068](https://github.com/unraid/api/issues/1068))
([cf0fa85](https://github.com/unraid/api/commit/cf0fa850954ea2f018e338a132149f872b966df4))
* initial version of modification service
([b80469d](https://github.com/unraid/api/commit/b80469d38e519a7ba0e6eae636cda2a821e2d465))
* inject after form
([a4b276f](https://github.com/unraid/api/commit/a4b276f7874580bbf9827025730777715c9983da))
* kill timeout extended
([22d4026](https://github.com/unraid/api/commit/22d40264a02672a818053b5280d63a03ff7336b9))
* log size and only tar files
([731f2f8](https://github.com/unraid/api/commit/731f2f8e77a77b544a7f526c78aabfacca71eee4))
* logrotate test
([4504c39](https://github.com/unraid/api/commit/4504c39a2bbcf51385578b69a9fdc7b81a950e98))
* manually install libvirt in build process to ensure it is included in
the final build
([e695481](https://github.com/unraid/api/commit/e695481363f0d5d7add9d0e0d50d1e113b3024f6))
* more pm2 fixes
([8257bdf](https://github.com/unraid/api/commit/8257bdff3624211ee645349abdec303bf271538e))
* move fixtures into __test__ folder
([22a901d](https://github.com/unraid/api/commit/22a901de9b0c274d3f75ed4b4618cd6cd90324ba))
* myservers_fb keepalive location
([e07e7f3](https://github.com/unraid/api/commit/e07e7f335c8ea4a73966ada90c26b7c82dbb025e))
* only write config when a specific config update action occurs
([ec29778](https://github.com/unraid/api/commit/ec29778e37a50f43eb164991bcf2a6ff9c266033))
* properly read log level from environment
([b5151e9](https://github.com/unraid/api/commit/b5151e9ba76a6814e24e8da34e8a3c1bf1cc2144))
* public index
([f0641ea](https://github.com/unraid/api/commit/f0641ea7ca0919884dc3b8642c2e6694398e3246))
* remove sso if disabled on Unraid-API start
([3bc407c](https://github.com/unraid/api/commit/3bc407c54e8e7aeadebd9ac223d71f21ef97fca1))
* remove sso user command
([bbd809b](https://github.com/unraid/api/commit/bbd809b83826e81eef38a06e66f3393e4f83e81e))
* remove sso user options
([e34041f](https://github.com/unraid/api/commit/e34041f86ef7ab6cf5e2fdf7efb86458d190edc1))
* remove unused config sections
([f0b9c4f](https://github.com/unraid/api/commit/f0b9c4f44ab0ee8f75bf96fde2413988ef4f6a8c))
* remove unused fields
([d2d0f7c](https://github.com/unraid/api/commit/d2d0f7cd9acb53ea2372245d7ef669c7ca24ee8a))
* remove unused vars
([0507713](https://github.com/unraid/api/commit/0507713972e344ad47bd077554d5888269669e9c))
* rename api key resource back to api_key
([ee9666b](https://github.com/unraid/api/commit/ee9666b317d7feb5c15d53e2a6b902c7771c8c7a))
* rename modification file
([70a93f2](https://github.com/unraid/api/commit/70a93f2cc63e0e62242be6fe1a717515a6fbec85))
* reorder index
([858553f](https://github.com/unraid/api/commit/858553f0debb6424ae0614640b82a050c33f175a))
* restart the API when an SSO user is added
([a6b0c90](https://github.com/unraid/api/commit/a6b0c906a423df048401750943f02dfdc9bc2619))
* restoring sso error
([234bf7d](https://github.com/unraid/api/commit/234bf7dfa4b0be88b6cc13996d8f29ec819da26e))
* revert local api key value
([ff40e7a](https://github.com/unraid/api/commit/ff40e7ae392052d3d9e1b084c5f4851e8ebd529e))
* rollback if patch exists before applying
([c2f4e8d](https://github.com/unraid/api/commit/c2f4e8d4e5c758601bd20ba491fd077b434ba45e))
* secondary changes
([d75331a](https://github.com/unraid/api/commit/d75331a67e3566875ce8642fce80195e79932a4c))
* service tests for modifier service
([08c1502](https://github.com/unraid/api/commit/08c150259f2b4630d973803f4edff69c8bf0ec3a))
* session issues
([5981693](https://github.com/unraid/api/commit/5981693abd605337f9174ba4c85fd1bfc243edeb))
* shared call to createPatch
([eb3e263](https://github.com/unraid/api/commit/eb3e263fb32a748bfa06ec6d119ee51d242707cf))
* simplify docs
([d428030](https://github.com/unraid/api/commit/d428030b806f55b62421559d434fc723786b03ad))
* style improvements
([b0f395e](https://github.com/unraid/api/commit/b0f395ef76f11047eaa13091df277df0459e9d8f))
* swap to async exit hook
([4302f31](https://github.com/unraid/api/commit/4302f316820a109c76408092994727b2dc030a15))
* switch to nest-commander
([1ab2ab5](https://github.com/unraid/api/commit/1ab2ab5b58a1f49cd6b05aaa84bfeced49d68e8e))
* try catch restart
([89abee6](https://github.com/unraid/api/commit/89abee680bdbdaa9946ddb991f0e6b5ada9ccdf7))
* **ui:** webgui-compatible web component library
([#1075](https://github.com/unraid/api/issues/1075))
([1c7b2e0](https://github.com/unraid/api/commit/1c7b2e091b0975438860a8e1fc3db5fd8d3fcf93))
* unnecessary comment
([0c52256](https://github.com/unraid/api/commit/0c5225612875b96319b28ef447db69ecab15cfda))
* unraid single sign on with account app
([5183104](https://github.com/unraid/api/commit/5183104b322a328eea3e4b2f6d86fd9d4b1c76e3))
* update packageManager field for pnpm
([8d5db7a](https://github.com/unraid/api/commit/8d5db7a9bfdf528e2d58b20cc62434ea5929d24f))
* upgrade dependencies
([0a0cac3](https://github.com/unraid/api/commit/0a0cac3da74c2fe20f7100a9ad5d1caafa74b157))
* use execa for start and stop
([46ab014](https://github.com/unraid/api/commit/46ab0144d41b425015487c251c1884744223ba29))
* use zod to parse config
([19cf1be](https://github.com/unraid/api/commit/19cf1be079f2ccb9e0cfa10f2fb97a18f15c5729))
* validate token format in both PHP and CLI
([6ef05a3](https://github.com/unraid/api/commit/6ef05a3d7770f799e7d587c2cef8d29f6058bee1))
* **web:** add delete all notifications button to archive view in
notifications sidebar
([3bda9d6](https://github.com/unraid/api/commit/3bda9d6a4ca01cc5580012b0133e72929d6dab40))
* **web:** enhance notifications indicator in UPC
([#950](https://github.com/unraid/api/issues/950))
([6376848](https://github.com/unraid/api/commit/63768486e4ec64ab32666a26adf96f4db4a53e81))
* **web:** pull date format from display/date and time settings
([b058067](https://github.com/unraid/api/commit/b058067b628ca7866a9ba0a6c4c5e4d5505d98cb))
* **web:** rm api-key validation from connect sign in
([#986](https://github.com/unraid/api/issues/986))
([7b105d1](https://github.com/unraid/api/commit/7b105d18678e88a064f0643d6e857704789e0ee8))
* zod config no longer any
([c32c5f5](https://github.com/unraid/api/commit/c32c5f57127b9469bde8806d78dc364562e73d9f))


### Bug Fixes

* allow concurrent testing with a shared patcher instance
([623846e](https://github.com/unraid/api/commit/623846ef46eb24a32c62516de58e8bc5d0219833))
* **api:** append time to formatted date when a custom date format is
selected
([0ac8ed9](https://github.com/unraid/api/commit/0ac8ed9d9e7e239e471eedf466832aed0270d123))
* **api:** change log output location for diagnostic compatibility
([#1130](https://github.com/unraid/api/issues/1130))
([cba1551](https://github.com/unraid/api/commit/cba155138379d47bc3151c7c27d745ba6a345d83))
* **api:** delay pm2 start until server has booted
([bd3188e](https://github.com/unraid/api/commit/bd3188efea4d3656994ffae32bd53f821c96358d))
* **api:** exclude duplicates from legacy script in archive retrieval
([8644e13](https://github.com/unraid/api/commit/8644e130979ed8740c5a8da0b3984266e2b3684c))
* **api:** improve defaults in PM2 service
([#1116](https://github.com/unraid/api/issues/1116))
([57526de](https://github.com/unraid/api/commit/57526dede69e3a6547d05183e43c5b36dd1cae89))
* **api:** logrotate modification & permissions
([#1145](https://github.com/unraid/api/issues/1145))
([5209df2](https://github.com/unraid/api/commit/5209df2776e1a985e82bedc655fe28acf1fd0bde))
* **api:** make cookie recognition during websocket connection more
([353e012](https://github.com/unraid/api/commit/353e012db8ab5280863f32392c520b4a330c13cc))
* **api:** pm2 start script & limit auto restarts
([#1040](https://github.com/unraid/api/issues/1040))
([ebcd347](https://github.com/unraid/api/commit/ebcd3479e735724626ffc6907c338d5080898bee))
* **api:** retry mothership connection up to 3x before logout
([#1069](https://github.com/unraid/api/issues/1069))
([c27bb1b](https://github.com/unraid/api/commit/c27bb1be4c7a9ab201585586f3bc5e4afa1c7791))
* **api:** sanitize incoming user session id's
([f5e3424](https://github.com/unraid/api/commit/f5e3424b79702e8f959b5519e83370a9e1d2033b))
* **api:** slow init of unraid-api cli
([#1022](https://github.com/unraid/api/issues/1022))
([5dbbae7](https://github.com/unraid/api/commit/5dbbae796792a62234497d056eac019aa084b21c))
* **api:** update deploy-dev script to dist instead of src
([55cce09](https://github.com/unraid/api/commit/55cce09e65521762a6fe388d5b9b88ace1337c26))
* **api:** validate cookie session data
([491f680](https://github.com/unraid/api/commit/491f680607ce7244d9e47a457e44cde711fbe00c))
* apply and rollback error handling
([e22191b](https://github.com/unraid/api/commit/e22191bc77bc09f5c6c4ad57e5073829cf966ba4))
* authorization type error
([#987](https://github.com/unraid/api/issues/987))
([7a4799e](https://github.com/unraid/api/commit/7a4799e9cd4caef6acfc3661d205a377fcf499ab))
* back to default configs
([b5711c9](https://github.com/unraid/api/commit/b5711c91284072991bcf409ac6126cd4b46afc7c))
* backup restore formatting
([15210f6](https://github.com/unraid/api/commit/15210f64b0938ec884a3ef4379d245c661eab9a3))
* basic test fixed
([2f38035](https://github.com/unraid/api/commit/2f38035520ca0fe796c981d08b9136d89ffc5888))
* better js file handling
([ddf160e](https://github.com/unraid/api/commit/ddf160e878a352842e813154b607945ccc7b4081))
* better loader functionality and error handling
([8a57d2d](https://github.com/unraid/api/commit/8a57d2dccbcb9c2effc5df5d8c69ad02713de24a))
* better logging when error
([6e4e3f8](https://github.com/unraid/api/commit/6e4e3f85abf64f8d799e33c33823810e71ef13e2))
* cleaner logs for starting API
([79f26ef](https://github.com/unraid/api/commit/79f26ef251cb42e7f2106d00c6c05e2bf17b8227))
* clearer error messaging
([e373849](https://github.com/unraid/api/commit/e37384966c5b9079bb507052dcaba56232c1c42a))
* code review feedback
([c66079e](https://github.com/unraid/api/commit/c66079e9a8e0ef47e5054118d0581bec708ac604))
* completion script registration
([05c8c9b](https://github.com/unraid/api/commit/05c8c9bf078ece2061ad8ae32497f52b8c9b94dc))
* connect key role
([2dcfc1c](https://github.com/unraid/api/commit/2dcfc1c19a1d085df84f0b1b50c096e3220205dd))
* create api key for connect on startup
([58329bc](https://github.com/unraid/api/commit/58329bc29521ebc26b27ee20013ac3926c5088c2))
* create api key permissions
([cefb644](https://github.com/unraid/api/commit/cefb644bd7fa513f553ca0ca4c49f0fb42a74112))
* create connect key
([6b1ab7b](https://github.com/unraid/api/commit/6b1ab7b74ae1d2938fa9105180a5f66e9604fd41))
* cwd on ecosystem.config.json
([dfd0da4](https://github.com/unraid/api/commit/dfd0da4ca23078f6de2e54d5e5bd6cba06334abc))
* default overwrite false test
([cf59107](https://github.com/unraid/api/commit/cf59107e568d91be684176335db5300bee9be865))
* delete .original files
([a9eb21a](https://github.com/unraid/api/commit/a9eb21aac0f373990aaa3f7a99731612540533cf))
* deprecated version warning
([89d0bd2](https://github.com/unraid/api/commit/89d0bd2e6da35fb1e8d95627d38edb54f82e0c6b))
* **deps:** update all non-major dependencies
([#1158](https://github.com/unraid/api/issues/1158))
([45ebc8b](https://github.com/unraid/api/commit/45ebc8b6e07c53ad3eee28d5cf8ac9cd0d827754))
* **deps:** update apollo graphql packages
([7b1ee99](https://github.com/unraid/api/commit/7b1ee9940cca46e563bb79c7056996315f9decc5))
* **deps:** update dependency @apollo/client to v3.12.6
([bb7800a](https://github.com/unraid/api/commit/bb7800a8c088705fd8310671a9896cbe9b0184e5))
* **deps:** update dependency @apollo/client to v3.12.9
([6607cf2](https://github.com/unraid/api/commit/6607cf20c10a091d466c6a8031eebc17feb3e3fc))
* **deps:** update dependency @graphql-tools/load-files to v7.0.1
([4e5c724](https://github.com/unraid/api/commit/4e5c7242e43cc356f1c69adcfcd25b57896af476))
* **deps:** update dependency @nestjs/schedule to v4.1.2
([faf0de5](https://github.com/unraid/api/commit/faf0de5a19256efb83dc45a484e3cba65596ccd7))
* **deps:** update dependency chokidar to v4.0.3
([d63a93c](https://github.com/unraid/api/commit/d63a93c55004d17b6d17634c55ffbc5670ebbec7))
* **deps:** update dependency dockerode to v4
([#830](https://github.com/unraid/api/issues/830))
([c331ecd](https://github.com/unraid/api/commit/c331ecd50c4910fd6c35e5ad92b3f676d552febc))
* **deps:** update dependency dotenv to v16.4.7
([c66a650](https://github.com/unraid/api/commit/c66a6502b027853046d126a14ddee870ffabd10c))
* **deps:** update dependency execa to v9.5.2
([d487c90](https://github.com/unraid/api/commit/d487c90ccc20162c76f0cdf49a736c1fee4271bd))
* **deps:** update dependency express to v4.21.2
([a070306](https://github.com/unraid/api/commit/a07030684c8777e47eb4a51be0ea680b7f217e74))
* **deps:** update dependency got to v14.4.5
([975a47c](https://github.com/unraid/api/commit/975a47c7d47841c49443f46264feb54abf53698c))
* **deps:** update dependency graphql-ws to v5.16.2
([25d8f08](https://github.com/unraid/api/commit/25d8f085b67c2e53876d837c739214dc874116b8))
* **deps:** update dependency ini to v4.1.3
([4c88cbe](https://github.com/unraid/api/commit/4c88cbee4b2d5f6717241dadac23bfe90ce15193))
* **deps:** update dependency node-window-polyfill to v1.0.4
([8bfa88f](https://github.com/unraid/api/commit/8bfa88f4bc932eb82dd9b33a494811ea15764758))
* **deps:** update dependency openid-client to v6.1.7
([0f50517](https://github.com/unraid/api/commit/0f50517a8544e1eb9b08ad1b3f05f798491b7f23))
* **deps:** update dependency p-retry to v6.2.1
([c6f3241](https://github.com/unraid/api/commit/c6f324155019e066701723a57b642c6e3ba8332d))
* **deps:** update dependency pm2 to v5.4.3
([a754090](https://github.com/unraid/api/commit/a75409026dd4e3d9ed120802012b67b179327448))
* **deps:** update dependency uuid to v11.0.5
([7e3398b](https://github.com/unraid/api/commit/7e3398b2efabf1a5407d6e20c165eb4923b3bab2))
* **deps:** update graphql-tools monorepo
([cd7e2fe](https://github.com/unraid/api/commit/cd7e2feea199276a1d431cf355e54e12e5960d9a))
* **deps:** update graphqlcodegenerator monorepo
([0446c59](https://github.com/unraid/api/commit/0446c5924a6a9dd15b875628ca0f1197cfe521c4))
* **deps:** update graphqlcodegenerator monorepo
([15c789d](https://github.com/unraid/api/commit/15c789dbb34b85bed55c2731fb8ae8260f5f311f))
* **deps:** update nest monorepo to v10.4.15
([07b1ea9](https://github.com/unraid/api/commit/07b1ea9a10634a597909ae1d237cc3b1e7f959b7))
* **deps:** update nest-graphql monorepo to v12.2.2
([91aabd9](https://github.com/unraid/api/commit/91aabd9ffbfb8c2ceb4110217dfc05de8859077d))
* do not process.exit on restart or stop command
([933575f](https://github.com/unraid/api/commit/933575fc2badbb09b3a9d3c66724e37a9ee246f2))
* don't check code for execa
([508a5eb](https://github.com/unraid/api/commit/508a5eb49d9514dca9953317d9fa93314fe63e4c))
* dont remove login file without a backup presetn
([0370e4f](https://github.com/unraid/api/commit/0370e4f7ea3e3df0d2264264324d8e53ffc0c086))
* ensure directory exists before making connect key
([9e27ec9](https://github.com/unraid/api/commit/9e27ec98b68a49bdd6dc4b03de8c0cc3a1470a5e))
* excess spacing in api-key.service
([1deb002](https://github.com/unraid/api/commit/1deb0023287a39d40e52e89c515a28e62352f62c))
* extra log line
([1183063](https://github.com/unraid/api/commit/1183063aa7063afd8222def18f5e1fd6077e8c88))
* extra spacing in config.ts
([f3ee7be](https://github.com/unraid/api/commit/f3ee7be80f2c60266fbb13597a70f0a389fb577f))
* file modification service fixes
([aa5b3f4](https://github.com/unraid/api/commit/aa5b3f4e47ed88df23af00dfcccb7b64786b6231))
* find by key, not ID
([3c3fa1e](https://github.com/unraid/api/commit/3c3fa1e27cfabbe6926c3da8870751397eed1def))
* fix libvirt bindings
([#1167](https://github.com/unraid/api/issues/1167))
([5817e5d](https://github.com/unraid/api/commit/5817e5d4b1697bbdfaa1984ccd650a232233cc15))
* forced restarting on commands
([925866d](https://github.com/unraid/api/commit/925866d389e337fcb8c249ead929e1f65854465b))
* format authrequest mod as other files
([180a81d](https://github.com/unraid/api/commit/180a81dbae8e749eae237fc8cee2950c790eedf0))
* initial feedback about report addressed
([5dee09c](https://github.com/unraid/api/commit/5dee09c77ad375de2eca59f650e5fea2070087b5))
* install as-integrations/fastify
([ff4546d](https://github.com/unraid/api/commit/ff4546d6692d2a4799f2dbeef0d5e5c6bac62561))
* length
([83579f1](https://github.com/unraid/api/commit/83579f1fbd03ffe929d009c20d214b4de62835c6))
* lint
([0f218b8](https://github.com/unraid/api/commit/0f218b8b72e397734823efab8f2141973a3a80ce))
* lint
([82bca54](https://github.com/unraid/api/commit/82bca54c594265ddf23a298691bd7ef6d4b47f32))
* lint
([ceb443d](https://github.com/unraid/api/commit/ceb443da15d177a950c36af61b93a7126cf4ca85))
* lint
([da04e7c](https://github.com/unraid/api/commit/da04e7ce0873d7802a936952d91e9867f0868a6e))
* lint
([7d87f0e](https://github.com/unraid/api/commit/7d87f0eee23dfa0f391fd342d38ed9084f18d8d4))
* logrotate error
([8c64dd2](https://github.com/unraid/api/commit/8c64dd2f2c65aa83ce0e2d501357ee595c976e56))
* mock ensureDirSync
([7e012e6](https://github.com/unraid/api/commit/7e012e6a2eb96ccddf5a1f69d7580b4bdfe7a0a9))
* more generic test
([0f651db](https://github.com/unraid/api/commit/0f651dbf61a1822b492aa80030f0bc231bc6f606))
* only instantiate service one time
([933dc81](https://github.com/unraid/api/commit/933dc81b6c50db5a33f586f7094e1ea524b9a9fa))
* only toast unread notifications, not archived ones
([cc59be6](https://github.com/unraid/api/commit/cc59be6cb3efc71226ee50f9f04e37a2e4b50de6))
* padding and glob function issues
([1d3f2eb](https://github.com/unraid/api/commit/1d3f2eb8213115c3385ac2d29ee8f53560347ba8))
* patch-utils unused
([047808d](https://github.com/unraid/api/commit/047808dce0cd9e9b4b273a9124dbd45ca9446208))
* paths now correct, better download logic
([16db2d9](https://github.com/unraid/api/commit/16db2d908dcb2c65508b367712c51bf9872a95e5))
* properly log error with template string
([3781f1f](https://github.com/unraid/api/commit/3781f1f41c7f0eef604daee0402ed9a2bb27cd46))
* pull token from query not params
([2e827e7](https://github.com/unraid/api/commit/2e827e7cabe4a6a069d4e8779015e5896d8a1d1d))
* remove devDependencies from output package json
([294869b](https://github.com/unraid/api/commit/294869bbea7f8a1863f8aafae6b074330e057679))
* remove isNaN in favor of number.isNaN
([03e3a46](https://github.com/unraid/api/commit/03e3a46092db613281176b88cae284f6448027c6))
* remove memory key generation
([b84db13](https://github.com/unraid/api/commit/b84db1322104c7f26f7b6378f25a2757b3010c6d))
* remove uneeded env variable
([f688a35](https://github.com/unraid/api/commit/f688a350d3d0a1c47be5896e6fbf92eeb8433967))
* remove unused constructor
([e0e2a7b](https://github.com/unraid/api/commit/e0e2a7b41c5e599ed4cf3bf49c7faea3b71f0b70))
* remove usage of Role.UPC
([d1e2f6e](https://github.com/unraid/api/commit/d1e2f6e0b391cb4eca75a0997b41cb99a9953d42))
* report issues + pm2 issues
([28c383e](https://github.com/unraid/api/commit/28c383e1d111d4ac4226d7d966533ba80ca5d9a1))
* reset config to be closer to default
([b7fbb0b](https://github.com/unraid/api/commit/b7fbb0b6af0453f5f6a17087bb7e68c393b9fe3f))
* resource busy when removing all subdirectories
([29936c9](https://github.com/unraid/api/commit/29936c90938fb83bc2f154315ca63a9d7cc98552))
* restart command elegant
([296117b](https://github.com/unraid/api/commit/296117b51aac8a4c15366f2271af858868b6e071))
* revert dockerode upgrade
([#1140](https://github.com/unraid/api/issues/1140))
([a74a379](https://github.com/unraid/api/commit/a74a379a93fd15a315e31191de1bf69c5879f8a6))
* revert myservers.cfg
([d0896f3](https://github.com/unraid/api/commit/d0896f3ef8aebdd9c76d805ed6a35b4a5d5a1b08))
* sandbox defaults in dev mode wrong
([2a24919](https://github.com/unraid/api/commit/2a2491936cf85013be836450ab7ed0cc11207e11))
* sequential test execution for generic-modification
([79ee1f7](https://github.com/unraid/api/commit/79ee1f7552cee47c6f5a8eb5942468292212e2f2))
* shell path to unraid-api
([15d11e4](https://github.com/unraid/api/commit/15d11e477bb2a08d785a7b22bd51900279a55508))
* simplify api setup index
([701b1fb](https://github.com/unraid/api/commit/701b1fbd9096c9675475062eaf32a2cbfb0567b9))
* simplify upcoming features
([8af79b2](https://github.com/unraid/api/commit/8af79b27501b42e1c1f7697756a56a9001000d8f))
* staging build issues
([e6bcb8d](https://github.com/unraid/api/commit/e6bcb8de7daee463f7ac0dbf977e085e108302ba))
* start command simplification
([e1faf3a](https://github.com/unraid/api/commit/e1faf3aa8db5973eb1bb0ea7a4844f820504618d))
* stop command exits
([2dbfdb6](https://github.com/unraid/api/commit/2dbfdb670a773114c0fdc68c7cf9d29fa4e28a9b))
* swap to placeholder key
([d1864d0](https://github.com/unraid/api/commit/d1864d0020ed56ab2368d23b48604b55cff21ae4))
* test issues
([e4b55b1](https://github.com/unraid/api/commit/e4b55b133bb2dc4bf2ccfd6fd2fc244daadbea53))
* test simplification to ensure no redownloads
([e07dad3](https://github.com/unraid/api/commit/e07dad3a6947aa186c4ac03032b5b3813cd046b6))
* tests
([25c1c1a](https://github.com/unraid/api/commit/25c1c1a55a3fb32b76bf5cb7257a4ba44f717a89))
* tests and validate token clears screen
([7f48ddd](https://github.com/unraid/api/commit/7f48dddcd2e2ea1ae3a55ecc54d5ac274535b714))
* type for generic test
([e856535](https://github.com/unraid/api/commit/e85653592a9d6eadcd0be89bf90a96c5d313fda3))
* unit test issues
([c58f7a7](https://github.com/unraid/api/commit/c58f7a7f246902c7d354eb51d1c87c8ea3b636a3))
* unit tests updated
([9548505](https://github.com/unraid/api/commit/954850535bec6b09aaf66b01d3ee749c8a22de5d))
* unneeded await on api-key service
([0325be7](https://github.com/unraid/api/commit/0325be757ee4c04b5c23365ff592f521a492595b))
* unused imports
([a5447aa](https://github.com/unraid/api/commit/a5447aa2f4c99968651fa3750d6bf0e8d68678de))
* update tests
([d0696a9](https://github.com/unraid/api/commit/d0696a93810893ccd6c676df1c639ca279992428))
* use an enum and defaults for sandbox value
([eb56483](https://github.com/unraid/api/commit/eb56483ba2693944d39f4409c91b75ee82a7d30b))
* use batchProcess
([ffbb9d7](https://github.com/unraid/api/commit/ffbb9d7750568bfa849d21e051503d1fcca5355f))
* use cwd when running application
([e016652](https://github.com/unraid/api/commit/e01665264b6f45366cdacf60c0f3553adfbd85d3))
* use placeholder in test API key
([c6b7755](https://github.com/unraid/api/commit/c6b7755214de8bedd5c0f2735473c2a559b1e26f))
* watch all events to load keys
([59ca177](https://github.com/unraid/api/commit/59ca17787e4d36113b0a8c5ef2117acfc491c49c))
* **web:** name of toaster component
([e093242](https://github.com/unraid/api/commit/e093242d20ddd72567396f4a53238250f2199a64))


### Miscellaneous Chores

* release 4.1.2
([dbab290](https://github.com/unraid/api/commit/dbab290b429f9eff8fa903d193de2bd02bb392bd))
</details>

<details><summary>4.1.2</summary>

## 4.1.2 (2025-02-20)


### Features

* add csrf support to api & web components
([#999](https://github.com/unraid/api/issues/999))
([19241ed](https://github.com/unraid/api/commit/19241ed55f5112f878b9890d8695badf7eb1c3eb))
* allow csrf passing through querystring
([dba38c0](https://github.com/unraid/api/commit/dba38c0d149a77e4104c718c53d426330a17f2fa))
* automatic session setup for dev
([36d630e](https://github.com/unraid/api/commit/36d630e89bbf9bc7e3ae64bdf5cf73a8536d44ab))
* back to callbackUrl
([e39b120](https://github.com/unraid/api/commit/e39b1203a315889c5b5232ecfd32c7377ae04800))
* begin fixing dark mode in the webcomponents
([5f7dcdb](https://github.com/unraid/api/commit/5f7dcdb1a7e7bce87b29add7849c94a0353c2c96))
* begin nuking alpha beta gamma
([25acd4b](https://github.com/unraid/api/commit/25acd4b39fff9a0cb573f9e90c52830fef41d737))
* convert to pnpm monorepo
([#1137](https://github.com/unraid/api/issues/1137))
([8d89f8b](https://github.com/unraid/api/commit/8d89f8b20d6f3983d4e85b33827a857aa862db37))
* delete unused imports
([97a3772](https://github.com/unraid/api/commit/97a3772d95aff534d85c410e58391d30494d9237))
* disable button on submit
([2ceb5da](https://github.com/unraid/api/commit/2ceb5da3c70826cc50df476decb6b117025f46c0))
* dont pass entire server state for privacy
([54e3f17](https://github.com/unraid/api/commit/54e3f17bd9e541f50970c696bbe8b602ec38a748))
* eliminate all alpha beta gamma variable usage
([fbdbce9](https://github.com/unraid/api/commit/fbdbce97ec2171ec7057f0f159e73032e984705a))
* error state outside of button
([18c63e0](https://github.com/unraid/api/commit/18c63e0b0c7451c99eacabb504e18f8070ff7dc2))
* hide sign in from the dropdown text
([3e68aaf](https://github.com/unraid/api/commit/3e68aaf8cdc0fb20c6e1b819a8571f419d94a811))
* lots of progress on colors
([dc8b2ee](https://github.com/unraid/api/commit/dc8b2ee01b454d307e779d495dbcf11227760480))
* move ssoenabled to a boolean flag rather than ids
([404a02b](https://github.com/unraid/api/commit/404a02b26bae6554d15e317f613ebc727c8f702f))
* move variable declarations to theme.ts
([3c82ee1](https://github.com/unraid/api/commit/3c82ee1e9acc197c9768a624cdef8c2e23c56d00))
* nuxt config simplification and formatting
([02ffde2](https://github.com/unraid/api/commit/02ffde24d19594949faa97f9d070383b498fdcbe))
* or button on sign in page
([1433e93](https://github.com/unraid/api/commit/1433e938d7ac01af326e2875c582a6aa6d622615))
* remove apiKey from server
([b110a11](https://github.com/unraid/api/commit/b110a118fb153c0af09a74755deb468b3760ba27))
* remove sso if disabled on Unraid-API start
([3bc407c](https://github.com/unraid/api/commit/3bc407c54e8e7aeadebd9ac223d71f21ef97fca1))
* responsive notifications
([d427054](https://github.com/unraid/api/commit/d427054443176563faa3e44249219c1d938e4b07))
* set background color on webcomponents
([b66e684](https://github.com/unraid/api/commit/b66e6847c895f216a5dec42410186b81a31af1a9))
* sidebar notification count
([694f01b](https://github.com/unraid/api/commit/694f01b6c4ab83c4131ae42bc11002d0300497c5))
* sso button token exchange
([f6f2390](https://github.com/unraid/api/commit/f6f2390b0169ceaf90ab88edfab3f2809bfe86b5))
* sso testing page and form disable on submit
([ffc6d8a](https://github.com/unraid/api/commit/ffc6d8a286d7c6ba751894464000f9870784507c))
* state using crypto
([afce130](https://github.com/unraid/api/commit/afce13099f5018d0c39765bfdd181adc8383a105))
* style improvements
([b0f395e](https://github.com/unraid/api/commit/b0f395ef76f11047eaa13091df277df0459e9d8f))
* **ui:** webgui-compatible web component library
([#1075](https://github.com/unraid/api/issues/1075))
([1c7b2e0](https://github.com/unraid/api/commit/1c7b2e091b0975438860a8e1fc3db5fd8d3fcf93))
* unraid single sign on with account app
([5183104](https://github.com/unraid/api/commit/5183104b322a328eea3e4b2f6d86fd9d4b1c76e3))
* unraid ui component library
([#976](https://github.com/unraid/api/issues/976))
([03e2fee](https://github.com/unraid/api/commit/03e2feebc73d620b21e54912e0bbddc1826880e1))
* update based on review feedback
([4383971](https://github.com/unraid/api/commit/43839711e3365e31120e156abac3746c55e8e694))
* update packageManager field for pnpm
([8d5db7a](https://github.com/unraid/api/commit/8d5db7a9bfdf528e2d58b20cc62434ea5929d24f))
* use state passing to validate requests
([4480c14](https://github.com/unraid/api/commit/4480c14c932fd8b42ba44989abdbecb49252e6f3))
* use text-secondary-foreground instead of gray
([463a1f7](https://github.com/unraid/api/commit/463a1f7b611599a19a23d3c75156c0a16da83312))
* viewport watch refactor
([9aefa38](https://github.com/unraid/api/commit/9aefa382ec64f08b1da8a3748ce16f637d562c8c))
* warning on missing fields
([0ef9aec](https://github.com/unraid/api/commit/0ef9aecccdde879e3be44d0b2a0fa4d8befc53b5))
* **web:** activation modal steps, updated copy
([#1079](https://github.com/unraid/api/issues/1079))
([8af9d8c](https://github.com/unraid/api/commit/8af9d8c58895010e3ddc03cc5fa075ac1e264f50))
* **web:** add an 'all' option to notification filter
([7c2a72e](https://github.com/unraid/api/commit/7c2a72e0c9537827c3c96df7b6378c03e2cc2852))
* **web:** add confirmation before archiving or deleting all
notifications
([d16f08c](https://github.com/unraid/api/commit/d16f08c266953ddb84223f90f1275d19c9d3c380))
* **web:** add count labels to notification tabs
([4caea3d](https://github.com/unraid/api/commit/4caea3dfc2c7067062f3ce8d863f9385ad030dbd))
* **web:** add delete all notifications button to archive view in
notifications sidebar
([3bda9d6](https://github.com/unraid/api/commit/3bda9d6a4ca01cc5580012b0133e72929d6dab40))
* **web:** add link to settings in notification sidebar
([f1a4d87](https://github.com/unraid/api/commit/f1a4d873481c212ffde1af7e38327a53a7e41d43))
* **web:** add loading and error states to notification sidebar
([2e9183a](https://github.com/unraid/api/commit/2e9183a479e0ec5f7cfc34bb81ccfd05e4bd2b29))
* **web:** clear notifications indicator after opening sidebar
([68958d1](https://github.com/unraid/api/commit/68958d17b78220c77c3cda4f0f4068b3ce623688))
* **web:** enhance notifications indicator in UPC
([#950](https://github.com/unraid/api/issues/950))
([6376848](https://github.com/unraid/api/commit/63768486e4ec64ab32666a26adf96f4db4a53e81))
* **web:** improve notification count syncing
([#1148](https://github.com/unraid/api/issues/1148))
([af2057c](https://github.com/unraid/api/commit/af2057c643640270e3e152ff8e08c3045e622437))
* **web:** move notification indicator icons to top-right of bell icon
([2fe4303](https://github.com/unraid/api/commit/2fe4303387023d303d7e50fc4d9a41f1eafdcc45))
* **web:** open official release notes via header os version
([54a893f](https://github.com/unraid/api/commit/54a893f396b29251b982ff1f26d376d24b962b93))
* **web:** pull date format from display/date and time settings
([b058067](https://github.com/unraid/api/commit/b058067b628ca7866a9ba0a6c4c5e4d5505d98cb))
* **web:** remove notification indicator pulse
([f320a77](https://github.com/unraid/api/commit/f320a77330c8cc7b92e170b0099d6c7f93b11c0e))
* **web:** support markdown in notification messages
([90cbef7](https://github.com/unraid/api/commit/90cbef774962e9d8ede47df7a4c1ca06f2a6651b))
* **web:** use Markdown helper class to interact with markdown
([f9c2d35](https://github.com/unraid/api/commit/f9c2d353133b01e74fe1bfbc420df3980d944012))


### Bug Fixes

* 12 hour timestamp logic corrected
([03be43b](https://github.com/unraid/api/commit/03be43b4579f1dcf6a666a144f75b3063576748a))
* **api:** make cookie recognition during websocket connection more
([353e012](https://github.com/unraid/api/commit/353e012db8ab5280863f32392c520b4a330c13cc))
* authorization type error
([#987](https://github.com/unraid/api/issues/987))
([7a4799e](https://github.com/unraid/api/commit/7a4799e9cd4caef6acfc3661d205a377fcf499ab))
* check width before changing viewport
([f07381b](https://github.com/unraid/api/commit/f07381b243501ecc6d54063881faad77a99a7655))
* connect breaks default css of header
([#1155](https://github.com/unraid/api/issues/1155))
([4ac9aa3](https://github.com/unraid/api/commit/4ac9aa3e409d0d89f2be61bfbafb8d7b5a5b3b00))
* dark theme as array
([1021d0d](https://github.com/unraid/api/commit/1021d0da0d7a919dedec70656bb52775575aa9e7))
* **deps:** update all non-major dependencies
([#1158](https://github.com/unraid/api/issues/1158))
([45ebc8b](https://github.com/unraid/api/commit/45ebc8b6e07c53ad3eee28d5cf8ac9cd0d827754))
* **deps:** update apollo graphql packages
([7b1ee99](https://github.com/unraid/api/commit/7b1ee9940cca46e563bb79c7056996315f9decc5))
* **deps:** update dependency @apollo/client to v3.12.6
([22ce615](https://github.com/unraid/api/commit/22ce61574f862eac4cdf8c00141bfbf1ac948055))
* **deps:** update dependency @apollo/client to v3.12.6
([bb7800a](https://github.com/unraid/api/commit/bb7800a8c088705fd8310671a9896cbe9b0184e5))
* **deps:** update dependency @apollo/client to v3.12.9
([6607cf2](https://github.com/unraid/api/commit/6607cf20c10a091d466c6a8031eebc17feb3e3fc))
* **deps:** update dependency @floating-ui/dom to v1.6.13
([08798d2](https://github.com/unraid/api/commit/08798d2f77683412807d684d7a8e63f1aadb0c34))
* **deps:** update dependency @floating-ui/dom to v1.6.13
([4d4c218](https://github.com/unraid/api/commit/4d4c218ac78e82a18679ec7b4939523db032b99b))
* **deps:** update dependency @floating-ui/vue to v1.1.6
([b4b7d89](https://github.com/unraid/api/commit/b4b7d898b62f746180b7f5730b5d9b5033dcecc2))
* **deps:** update dependency @floating-ui/vue to v1.1.6
([4c07d38](https://github.com/unraid/api/commit/4c07d389523f277950b8d2d359102f889587e5ce))
* **deps:** update dependency focus-trap to v7.6.4
([41ff232](https://github.com/unraid/api/commit/41ff232a3232dda66e5cdc2d4808a820a90a5d34))
* **deps:** update dependency focus-trap to v7.6.4
([f0e3038](https://github.com/unraid/api/commit/f0e3038ee7426aafb6cef01b85b47893c2238302))
* **deps:** update dependency graphql-ws to v5.16.2
([a189a03](https://github.com/unraid/api/commit/a189a0308a734e66750fe5059f7c59d8c9532bd8))
* **deps:** update dependency graphql-ws to v5.16.2
([25d8f08](https://github.com/unraid/api/commit/25d8f085b67c2e53876d837c739214dc874116b8))
* **deps:** update dependency radix-vue to v1.9.12
([0fd433f](https://github.com/unraid/api/commit/0fd433fe2a6b3f787624cb5a98efeae0f6c31cfd))
* **deps:** update dependency radix-vue to v1.9.13
([249feff](https://github.com/unraid/api/commit/249feff5cfe0bbb60bfa8f943b76b9c16c6c161b))
* **deps:** update graphqlcodegenerator monorepo
([0446c59](https://github.com/unraid/api/commit/0446c5924a6a9dd15b875628ca0f1197cfe521c4))
* dev mode
([fd64e01](https://github.com/unraid/api/commit/fd64e01e0c87db03fc2d4d0f32a0e8205fbe8b84))
* downgrade marked to fix changelog preview issue
([cfb3a45](https://github.com/unraid/api/commit/cfb3a45533d3c1bd31c44094f7ae2912e77a673e))
* edit settings padding issue
([adf349b](https://github.com/unraid/api/commit/adf349b76560b5f1fd4c320da35b3c6f660895fb))
* empty manifest and version alignment
([c4c44d9](https://github.com/unraid/api/commit/c4c44d92caa593795b9cb111a27828ecb8f62dbd))
* further resolve sso sub ids issues
([ef3d0ea](https://github.com/unraid/api/commit/ef3d0ead687d4a6071da290c0df29c12163303e1))
* improve typing and format lookup
([c6097f8](https://github.com/unraid/api/commit/c6097f86e42fcc57209c1344029abe854198edca))
* integration of `unraid-ui` tailwind config in `web`
([#1074](https://github.com/unraid/api/issues/1074))
([f3cd85b](https://github.com/unraid/api/commit/f3cd85bd3f02bdbe4c44136189d1c61935015844))
* lowercase or
([386cbde](https://github.com/unraid/api/commit/386cbdef5c9158290e03c670efb992cf11d5af1b))
* marked single input
([ceacbbe](https://github.com/unraid/api/commit/ceacbbe5d46466627df0fccc5ca8e7c56fa36a37))
* missing server type
([f1b721b](https://github.com/unraid/api/commit/f1b721bd72b875d9ff8c0bca2cc1eee506ba7697))
* more color work
([c48f826](https://github.com/unraid/api/commit/c48f8268def64ef8828dea556360b375b8cb32c7))
* oauth2 api prefix
([ec00add](https://github.com/unraid/api/commit/ec00adde20d4d9eca28f6b18615073305f491a73))
* only toast unread notifications, not archived ones
([cc59be6](https://github.com/unraid/api/commit/cc59be6cb3efc71226ee50f9f04e37a2e4b50de6))
* padding and glob function issues
([1d3f2eb](https://github.com/unraid/api/commit/1d3f2eb8213115c3385ac2d29ee8f53560347ba8))
* pass token to password field
([499b023](https://github.com/unraid/api/commit/499b023d359ed5181450ee9e04cbbf4531a4a680))
* re-add type-check
([60e9d1d](https://github.com/unraid/api/commit/60e9d1d912c983cf04e3e6cf15e221c39938612a))
* recreate package-lock to fix issues
([ad5a537](https://github.com/unraid/api/commit/ad5a53793d25ac9f63bae6df6c2a30d8d2780c67))
* remove console log
([8e75b82](https://github.com/unraid/api/commit/8e75b8254bbda93ded786750226090b769bed5c4))
* remove extra space
([a99ee03](https://github.com/unraid/api/commit/a99ee03fc37059b3a018db289c43fc419a634524))
* remove line from or in button
([1a1bce7](https://github.com/unraid/api/commit/1a1bce7b64b1cf90505f811e11b585ff87476f72))
* remove unused date-fns
([fe94ef5](https://github.com/unraid/api/commit/fe94ef5ba88df56aad87089081dd5fe4518fa414))
* remove unused login entries
([7833b5d](https://github.com/unraid/api/commit/7833b5db386f724318857fc31d825fb3534c84b9))
* render function fixed
([8008ab4](https://github.com/unraid/api/commit/8008ab46fb2f231b68201758a258fd43e2e1672e))
* revert changes to indicator.vue
([84d2a83](https://github.com/unraid/api/commit/84d2a832c0f64e52be05670eb438b21bff2e5163))
* switch to useToggle
([848233f](https://github.com/unraid/api/commit/848233f05465053876ac6f9f6ac4bfad2a48abff))
* theme store now uses singular variables object
([5ca6e40](https://github.com/unraid/api/commit/5ca6e40b2d4942385b12a4325d6b8a551cb3f44b))
* thorw on invalid token body
([f1af763](https://github.com/unraid/api/commit/f1af763eaf0dd8215eed470293d3a7f98784f38a))
* type error on element render
([a2563eb](https://github.com/unraid/api/commit/a2563eb8e710a9ac7259c4260cad9a3454565dae))
* upc header text color
([f989026](https://github.com/unraid/api/commit/f9890260d1c4abe69dac3ac4c05ebab17aab5161))
* use foreground text color for UPC
([87b8165](https://github.com/unraid/api/commit/87b816550d413dc9023c5057efe18b9cb26761e7))
* **web:** broken modals
([aebf339](https://github.com/unraid/api/commit/aebf3392595d45c84a84668f461c632a2d62e7dd))
* **web:** display error message in sidebar when api is offline
([#984](https://github.com/unraid/api/issues/984))
([125c0a1](https://github.com/unraid/api/commit/125c0a140b4e9b5401bacf1addab1820c412917e))
* **web:** edge case where archived notifications don't appear
([0a8c574](https://github.com/unraid/api/commit/0a8c5746fc2b8f8639643f013c1f19f0d7236d41))
* **web:** env var typo
([22cf90b](https://github.com/unraid/api/commit/22cf90b27fadec3024d9d038c53683e8f8a723bc))
* **web:** escaping html-encoded symbols like apostrophes in
translations ([#1002](https://github.com/unraid/api/issues/1002))
([04a3362](https://github.com/unraid/api/commit/04a33621e1d406d75ed0ff9af9f1f945813a1e8d))
* **web:** flash of disconnected api state on page load
([a8c02f4](https://github.com/unraid/api/commit/a8c02f4c49433b440a6f9c70f269bf69076655dc))
* **web:** infinite scroll loop when there's only 1 page of
notifications
([e9f2fc4](https://github.com/unraid/api/commit/e9f2fc424c350d07c756ae7573e90f615bcae25b))
* **web:** infinite trigger at bottom of infinite scroll
([eb691d3](https://github.com/unraid/api/commit/eb691d3514d8dc079987bfa566de4aa86094ef67))
* **web:** notification styles & alignment
([#968](https://github.com/unraid/api/issues/968))
([0d65e12](https://github.com/unraid/api/commit/0d65e12cede3324261fd3b219745b1e7793a33de))
* **web:** refetch notifications for sidebar when new notifications
arrive
([591bf4a](https://github.com/unraid/api/commit/591bf4a643ccc13c151c0a8cafad833d3137043e))
* **web:** remove warn and error console log removal
([#1086](https://github.com/unraid/api/issues/1086))
([9375639](https://github.com/unraid/api/commit/9375639e4a71ecfe8d4b877301c1f9bb22800a72))
* **web:** replaces dompurify with isomorphic-dompurify to enable
server-side usage with same syntax
([27834d9](https://github.com/unraid/api/commit/27834d92375be96a7842e81cc88eb083bcf127b8))
* **web:** reset infinite scroll when notification filters change
([da6de2c](https://github.com/unraid/api/commit/da6de2ccdb710772a199c8cba8952adc247412db))
* **web:** sanitize changelog markup after parsing
([c960292](https://github.com/unraid/api/commit/c96029273283f5970a5029eea1d7f451bbd0071b))
* **web:** stop opening notification sidebar to archive tab
([325e75f](https://github.com/unraid/api/commit/325e75f5d444908a2227fbe2e94be9ab5196ad8e))
* **web:** theme header differences
([#1085](https://github.com/unraid/api/issues/1085))
([1ccdd8d](https://github.com/unraid/api/commit/1ccdd8dc71ee5e1e3aacabd113d1cf213ca7c7ae))
* **web:** track 'notification seen' state across tabs & page loads
([#1121](https://github.com/unraid/api/issues/1121))
([64cf6ec](https://github.com/unraid/api/commit/64cf6ecc6aec25cd8edee5659efb09f288bb9908))
* **web:** update unread total immediately upon archiving
([#982](https://github.com/unraid/api/issues/982))
([ff5fd8e](https://github.com/unraid/api/commit/ff5fd8e5eb8eb4803db1265e31b0c1352af20251))


### Miscellaneous Chores

* release 4.1.2
([dbab290](https://github.com/unraid/api/commit/dbab290b429f9eff8fa903d193de2bd02bb392bd))
</details>

<details><summary>4.1.2</summary>

## 4.1.2 (2025-02-20)


### Features

* add unraid-ui documentation
([#1142](https://github.com/unraid/api/issues/1142))
([c557806](https://github.com/unraid/api/commit/c55780680ae905558b79dfefa91b116aef22b105))
* attempt to resolve performance issues with rm earlier in build …
([#1152](https://github.com/unraid/api/issues/1152))
([2a1aa95](https://github.com/unraid/api/commit/2a1aa95bd62ebfe42b62b8e7105c7a92b00cfca9))
* convert to pnpm monorepo
([#1137](https://github.com/unraid/api/issues/1137))
([8d89f8b](https://github.com/unraid/api/commit/8d89f8b20d6f3983d4e85b33827a857aa862db37))
* style improvements
([b0f395e](https://github.com/unraid/api/commit/b0f395ef76f11047eaa13091df277df0459e9d8f))
* **ui:** webgui-compatible web component library
([#1075](https://github.com/unraid/api/issues/1075))
([1c7b2e0](https://github.com/unraid/api/commit/1c7b2e091b0975438860a8e1fc3db5fd8d3fcf93))
* unraid ui component library
([#976](https://github.com/unraid/api/issues/976))
([03e2fee](https://github.com/unraid/api/commit/03e2feebc73d620b21e54912e0bbddc1826880e1))
* update packageManager field for pnpm
([8d5db7a](https://github.com/unraid/api/commit/8d5db7a9bfdf528e2d58b20cc62434ea5929d24f))


### Bug Fixes

* **deps:** update all non-major dependencies
([#1158](https://github.com/unraid/api/issues/1158))
([45ebc8b](https://github.com/unraid/api/commit/45ebc8b6e07c53ad3eee28d5cf8ac9cd0d827754))
* **deps:** update dependency radix-vue to v1.9.12
([0fd433f](https://github.com/unraid/api/commit/0fd433fe2a6b3f787624cb5a98efeae0f6c31cfd))
* **deps:** update dependency radix-vue to v1.9.13
([249feff](https://github.com/unraid/api/commit/249feff5cfe0bbb60bfa8f943b76b9c16c6c161b))
* empty manifest and version alignment
([c4c44d9](https://github.com/unraid/api/commit/c4c44d92caa593795b9cb111a27828ecb8f62dbd))
* integration of `unraid-ui` tailwind config in `web`
([#1074](https://github.com/unraid/api/issues/1074))
([f3cd85b](https://github.com/unraid/api/commit/f3cd85bd3f02bdbe4c44136189d1c61935015844))
* storybook resolution issue
([#1153](https://github.com/unraid/api/issues/1153))
([52c70b9](https://github.com/unraid/api/commit/52c70b9d85469008894d44788429ba298b082ac7))


### Miscellaneous Chores

* release 4.1.2
([dbab290](https://github.com/unraid/api/commit/dbab290b429f9eff8fa903d193de2bd02bb392bd))
</details>

<details><summary>4.1.2</summary>

## 4.1.2 (2025-02-20)


### Features

* add validation step to ensure that variables are set
([e3e9b2b](https://github.com/unraid/api/commit/e3e9b2bf404cb6f3bcae83db0395be272e4b79e3))
* always start the API and run npm link from script path
([30133ac](https://github.com/unraid/api/commit/30133acb0514a480177f563d4aee364a8a3fab1b))
* **api:** rm 2fa & t2fa from myservers config type
([#996](https://github.com/unraid/api/issues/996))
([89e791a](https://github.com/unraid/api/commit/89e791ad2e6f0395bee05e3f8bdcb2c8d72305dd))
* array iteration for restoring files
([036e97b](https://github.com/unraid/api/commit/036e97bb02e463872b3c2f4b5f1aa3b4bf525d1e))
* attempt to resolve performance issues with rm earlier in build …
([#1152](https://github.com/unraid/api/issues/1152))
([2a1aa95](https://github.com/unraid/api/commit/2a1aa95bd62ebfe42b62b8e7105c7a92b00cfca9))
* attempt to start unraid-api with background task
([2a102fc](https://github.com/unraid/api/commit/2a102fc9944f3080af66a8ebadee35059bce2009))
* begin building plugin with node instead of bash
([#1120](https://github.com/unraid/api/issues/1120))
([253b65a](https://github.com/unraid/api/commit/253b65a85ab9c5f53d53ef265b41aa132678f278))
* cleanup disclaimer and command to add users
([6be3af8](https://github.com/unraid/api/commit/6be3af8d7569d9c413dd9349df52e3fa4cb4f631))
* convert to pnpm monorepo
([#1137](https://github.com/unraid/api/issues/1137))
([8d89f8b](https://github.com/unraid/api/commit/8d89f8b20d6f3983d4e85b33827a857aa862db37))
* copy only needed files for nodejs
([acf587a](https://github.com/unraid/api/commit/acf587aa53ca25a3beae86afc608fc9ed68919ef))
* do not move upgradepkg
([ea16419](https://github.com/unraid/api/commit/ea16419929e0233e2c1ce37e2f4b79e3e64ce619))
* dont pass entire server state for privacy
([54e3f17](https://github.com/unraid/api/commit/54e3f17bd9e541f50970c696bbe8b602ec38a748))
* download nodejs and install on legacy OS versions
([2a95e4b](https://github.com/unraid/api/commit/2a95e4beb2364510003f187459e28bb610583c41))
* error when nodejs download fails
([6a9b14c](https://github.com/unraid/api/commit/6a9b14c68170d6430328cbb793d750f3177bdb32))
* extract node to usr/local/
([4c0b55b](https://github.com/unraid/api/commit/4c0b55b269f47a9d8f746344ae701e353d80509a))
* fix missing flash line
([6897aad](https://github.com/unraid/api/commit/6897aad67f5c8b38450aa81e612b8aa98a9328c7))
* fix pm2 setup and add link command
([de9500f](https://github.com/unraid/api/commit/de9500ffa6f3aa1842152e0ab26f54c8c5c6e5cb))
* move ssoenabled to a boolean flag rather than ids
([404a02b](https://github.com/unraid/api/commit/404a02b26bae6554d15e317f613ebc727c8f702f))
* move variable declarations to theme.ts
([3c82ee1](https://github.com/unraid/api/commit/3c82ee1e9acc197c9768a624cdef8c2e23c56d00))
* name package with PR number
([a642bf1](https://github.com/unraid/api/commit/a642bf15fd813dca522808765994414e4ed5a56c))
* nodejs issues with version 2
([9c6e52c](https://github.com/unraid/api/commit/9c6e52c2fa46e7504bc3fa500770373d8c1d1690))
* **plugin:** rm Date & Time format settings from Notification Settings
([e2148f3](https://github.com/unraid/api/commit/e2148f3c2eaf77ad707eddb7989cc20ec8df70ab))
* remove nghttp3 and only bundle nodejs
([8d8df15](https://github.com/unraid/api/commit/8d8df1592e5af127a992d5634ee9d344055cdf2c))
* separate install process
([b90a516](https://github.com/unraid/api/commit/b90a51600c3f70615b117f157d41585e55ef49de))
* sso login boolean
([34190a6](https://github.com/unraid/api/commit/34190a6910679b65362a9cfa0837bca7775cfda0))
* track node version in slackware
([42b010e](https://github.com/unraid/api/commit/42b010e4a141f2a338d65f4f727bf1d15521a5c6))
* **ui:** webgui-compatible web component library
([#1075](https://github.com/unraid/api/issues/1075))
([1c7b2e0](https://github.com/unraid/api/commit/1c7b2e091b0975438860a8e1fc3db5fd8d3fcf93))
* unraid single sign on with account app
([5183104](https://github.com/unraid/api/commit/5183104b322a328eea3e4b2f6d86fd9d4b1c76e3))
* update packageManager field for pnpm
([8d5db7a](https://github.com/unraid/api/commit/8d5db7a9bfdf528e2d58b20cc62434ea5929d24f))
* Update
plugin/source/dynamix.unraid.net/usr/local/emhttp/plugins/dynamix.my.servers/include/state.php
([42c0d58](https://github.com/unraid/api/commit/42c0d58da4d0570b7d865a8774964c18120ed585))
* upload files directly to cloudflare
([1982fc2](https://github.com/unraid/api/commit/1982fc238fefa1c67323bdc11ec1fb9c9f43c387))
* use plugin file for install and uninstall
([c9ac3a5](https://github.com/unraid/api/commit/c9ac3a5a0a3103fbd9c33a5d909fa475614a704a))
* validate entries correctly
([b101a69](https://github.com/unraid/api/commit/b101a695e18d71ddd170462b3d49289352166489))
* **web:** activation modal steps, updated copy
([#1079](https://github.com/unraid/api/issues/1079))
([8af9d8c](https://github.com/unraid/api/commit/8af9d8c58895010e3ddc03cc5fa075ac1e264f50))
* **web:** rm old notification bell upon plugin installation
([#979](https://github.com/unraid/api/issues/979))
([e09c07c](https://github.com/unraid/api/commit/e09c07c5070d59ac032baeff1ed253b5c00f4163))


### Bug Fixes

* add another missing symlink
([4e7f3ff](https://github.com/unraid/api/commit/4e7f3ff4d9aa0e4af417a50e2b30537dda3c759c))
* add error check to nodejs
([c8e0fe8](https://github.com/unraid/api/commit/c8e0fe87a34d7f066b7d0900dda205a40616bfb6))
* attempt to restore upgradepkg if install failed
([19c2a79](https://github.com/unraid/api/commit/19c2a79ce6c31c989f3d7f70cf7d8e2c219517b2))
* capitalize name
([31166b3](https://github.com/unraid/api/commit/31166b3483dc01847ad555618c43f8248411bdfa))
* cleanup commands
([052aea0](https://github.com/unraid/api/commit/052aea06a0d30963532f29f9961fce0ffc7fa3e8))
* delete unused line
([de4882e](https://github.com/unraid/api/commit/de4882ea17f54e788049cc5bb96b99b16822b6b4))
* delete upgradepkg
([74f0177](https://github.com/unraid/api/commit/74f0177ba0fd57722caa3ec14318d35167d3c6f7))
* dnserr on new line
([a3398a2](https://github.com/unraid/api/commit/a3398a29e15269be006e887fba6366c81b1d00f5))
* empty manifest and version alignment
([c4c44d9](https://github.com/unraid/api/commit/c4c44d92caa593795b9cb111a27828ecb8f62dbd))
* formatting issue
([42ca969](https://github.com/unraid/api/commit/42ca9691f7547a4340501863c1882efc0aee4c60))
* further resolve sso sub ids issues
([ef3d0ea](https://github.com/unraid/api/commit/ef3d0ead687d4a6071da290c0df29c12163303e1))
* install syntax error
([ec83480](https://github.com/unraid/api/commit/ec83480eb6aea09b98b9135516dc1fc8cdd6c692))
* more verbose logging for node install to find issues
([445af0c](https://github.com/unraid/api/commit/445af0c147ef641dac05ebeb2d44e63e8a4df799))
* node_txz naming
([b7c24ca](https://github.com/unraid/api/commit/b7c24ca861e92bf01118a17bc7e2322063e6a800))
* pass ssoSubIds only
([5adf13e](https://github.com/unraid/api/commit/5adf13ee070bdcd849339460b9888e51d224e765))
* PHP Warning in state.php
([#1126](https://github.com/unraid/api/issues/1126))
([c154b4e](https://github.com/unraid/api/commit/c154b4e0ad2d0627b1541a7f9ee5e55235d4dd5e))
* pkg_build
([d4bff0e](https://github.com/unraid/api/commit/d4bff0ee96e6e0974978465573e72e68d09fd829))
* proper file replacements
([e0042f3](https://github.com/unraid/api/commit/e0042f353b47cfa72a485d6a58ad0b956ea6dbc2))
* restore upgradepkg before install
([fddca27](https://github.com/unraid/api/commit/fddca2738c0ec016e744169d88b35a55dea092fa))
* strip components from tar line
([911cd5b](https://github.com/unraid/api/commit/911cd5bc0b0983df4ca8c9057bea5166f7d1c7f1))
* upgradepkg
([90cf1a8](https://github.com/unraid/api/commit/90cf1a8eea67d3dbc736ecdfba47e0025b1dc31c))
* used TGZ instead of TXZ for nghttp3
([09ad394](https://github.com/unraid/api/commit/09ad39483fed7a8155176b6568114b4e6679d587))
* variables passed properly
([e0875e7](https://github.com/unraid/api/commit/e0875e7a1b273969939d6902a55f4a9772640078))


### Miscellaneous Chores

* release 4.1.2
([dbab290](https://github.com/unraid/api/commit/dbab290b429f9eff8fa903d193de2bd02bb392bd))
</details>

---
This PR was generated with [Release
Please](https://github.com/googleapis/release-please). See
[documentation](https://github.com/googleapis/release-please#release-please).

Co-authored-by: github-actions[bot] <41898282+github-actions[bot]@users.noreply.github.com>
elibosley pushed a commit that referenced this pull request Feb 21, 2025
:robot: I have created a release *beep* *boop*
---


## [4.1.2](https://github.com/unraid/api/compare/v4.1.3...v4.1.2)
(2025-02-21)


### Features

* add api key creation logic
([81382bc](https://github.com/unraid/api/commit/81382bcf1d26364ad9c5445530f648209101cf91))
* add category.json
([c9e87e2](https://github.com/unraid/api/commit/c9e87e2e5b47a8801b7865ed586c803d0b470915))
* add command to package.json scripts
([0dfb07f](https://github.com/unraid/api/commit/0dfb07f9eb519e60441f4123423f65acfdffca3b))
* add csrf support to api & web components
([#999](https://github.com/unraid/api/issues/999))
([19241ed](https://github.com/unraid/api/commit/19241ed55f5112f878b9890d8695badf7eb1c3eb))
* add description flag, remove console log, and update readme
([c416c30](https://github.com/unraid/api/commit/c416c30951de4ed6b8d7a8c014403772db1c2015))
* add developer docs
([#1128](https://github.com/unraid/api/issues/1128))
([bb2e340](https://github.com/unraid/api/commit/bb2e340b68268d5121db650b27e8b2580c7966bb))
* add line about recommendation for sso command
([44727a8](https://github.com/unraid/api/commit/44727a8d1a7c16c566678da43119b17a6303e375))
* add log rotation
([f5c7ad9](https://github.com/unraid/api/commit/f5c7ad9221f80e4630e69f78d57f08f4c7252719))
* add logging around fixture downloads
([a1ce27b](https://github.com/unraid/api/commit/a1ce27b17c970657f52635600f0d13116523f928))
* add logrotate cron again
([4f85f66](https://github.com/unraid/api/commit/4f85f6687f920dae50277e726e2db2c3d946e867))
* add patch for auth-request.php
([ec6ec56](https://github.com/unraid/api/commit/ec6ec562f43aac9947de2e9c269181303f42b2db))
* add unraid-ui documentation
([#1142](https://github.com/unraid/api/issues/1142))
([c557806](https://github.com/unraid/api/commit/c55780680ae905558b79dfefa91b116aef22b105))
* add user with cli
([37458cd](https://github.com/unraid/api/commit/37458cd7408a1ad8aedca66a55ff13ac19ee30db))
* address log level feedback
([49774aa](https://github.com/unraid/api/commit/49774aae459797f04ef2866ca064050aa476ae91))
* allow csrf passing through querystring
([dba38c0](https://github.com/unraid/api/commit/dba38c0d149a77e4104c718c53d426330a17f2fa))
* allow deletion and creation of files with patches
([32c9524](https://github.com/unraid/api/commit/32c952402c25e8340b1c628b4d0fdc4816b28ade))
* always ensureDirectory for keys exists
([c6e9f80](https://github.com/unraid/api/commit/c6e9f804c58e44b46bce9f0da2260888544354cd))
* always start the API and run npm link from script path
([30133ac](https://github.com/unraid/api/commit/30133acb0514a480177f563d4aee364a8a3fab1b))
* **api:** graphql sandbox on unraid servers
([#1047](https://github.com/unraid/api/issues/1047))
([ec504f3](https://github.com/unraid/api/commit/ec504f39297c92b64d9d3cc2f8f482cc1f3a2e44))
* **api:** omit tz from sys time date format by default
([b2acde3](https://github.com/unraid/api/commit/b2acde3351d7afe18a2902e90b672537aadabffd))
* **api:** rm 2fa & t2fa from myservers config type
([#996](https://github.com/unraid/api/issues/996))
([89e791a](https://github.com/unraid/api/commit/89e791ad2e6f0395bee05e3f8bdcb2c8d72305dd))
* array iteration for restoring files
([036e97b](https://github.com/unraid/api/commit/036e97bb02e463872b3c2f4b5f1aa3b4bf525d1e))
* async disk mapping
([bbb27e6](https://github.com/unraid/api/commit/bbb27e686897e4f9a0c926553d75aa046d7a8323))
* async hypervisor and FIXED vm listing
([e79f4dd](https://github.com/unraid/api/commit/e79f4ddbc7061c249efb8214a311bb629628f669))
* attempt to resolve performance issues with rm earlier in build …
([#1152](https://github.com/unraid/api/issues/1152))
([2a1aa95](https://github.com/unraid/api/commit/2a1aa95bd62ebfe42b62b8e7105c7a92b00cfca9))
* attempt to start unraid-api with background task
([2a102fc](https://github.com/unraid/api/commit/2a102fc9944f3080af66a8ebadee35059bce2009))
* auto-docusaurus-prs
([#1127](https://github.com/unraid/api/issues/1127))
([1147e76](https://github.com/unraid/api/commit/1147e762ae2fed6dea198fa38d6bcc514a1e66fb))
* automatic session setup for dev
([36d630e](https://github.com/unraid/api/commit/36d630e89bbf9bc7e3ae64bdf5cf73a8536d44ab))
* back to callbackUrl
([e39b120](https://github.com/unraid/api/commit/e39b1203a315889c5b5232ecfd32c7377ae04800))
* begin building plugin with node instead of bash
([#1120](https://github.com/unraid/api/issues/1120))
([253b65a](https://github.com/unraid/api/commit/253b65a85ab9c5f53d53ef265b41aa132678f278))
* begin fixing dark mode in the webcomponents
([5f7dcdb](https://github.com/unraid/api/commit/5f7dcdb1a7e7bce87b29add7849c94a0353c2c96))
* begin nuking alpha beta gamma
([25acd4b](https://github.com/unraid/api/commit/25acd4b39fff9a0cb573f9e90c52830fef41d737))
* better patch application
([a3e7daa](https://github.com/unraid/api/commit/a3e7daa6a6565ac81004ffd13da35d8b95b429cf))
* better pm2 calls, log lines
([338ce30](https://github.com/unraid/api/commit/338ce3061310dfc42ad5f65edacbe5272de4afc7))
* bug report template
([f1ee8b2](https://github.com/unraid/api/commit/f1ee8b27b11fa969d0e6891590e44047c76eedb5))
* checkout correct branch on close
([#1123](https://github.com/unraid/api/issues/1123))
([a20b812](https://github.com/unraid/api/commit/a20b812b020adfade129ebd9fb0e6536004f8bee))
* cleanup config entries
([943e73f](https://github.com/unraid/api/commit/943e73fa696b6ecec3227be914ab4962c4fee79d))
* cleanup disclaimer and command to add users
([6be3af8](https://github.com/unraid/api/commit/6be3af8d7569d9c413dd9349df52e3fa4cb4f631))
* cli Commands
([f8e5367](https://github.com/unraid/api/commit/f8e5367f3eb47daa5bcbd7711ae5835369502a1d))
* CLI options for adding and deleting users
([16bf6d4](https://github.com/unraid/api/commit/16bf6d4c27ae8fa8d6d05ec4b28ce49a12673278))
* codeowners
([ab090b4](https://github.com/unraid/api/commit/ab090b48ec7291597a135a72b8e55c2d1bb389f3))
* coderabbit suggestion
([11ac36c](https://github.com/unraid/api/commit/11ac36c3616a90853d91467526fd39ecba17db88))
* configure PM2 on startup
([2b908f1](https://github.com/unraid/api/commit/2b908f100b9eefaccf2264d5ff9945667568acf0))
* contributing guide
([c912476](https://github.com/unraid/api/commit/c912476b431750834c64bdec80a61fda23e6c490))
* convert to pnpm monorepo
([#1137](https://github.com/unraid/api/issues/1137))
([8d89f8b](https://github.com/unraid/api/commit/8d89f8b20d6f3983d4e85b33827a857aa862db37))
* copy only needed files for nodejs
([acf587a](https://github.com/unraid/api/commit/acf587aa53ca25a3beae86afc608fc9ed68919ef))
* create key cli command logic and add to index command list
([9b2a62d](https://github.com/unraid/api/commit/9b2a62d642b0942e3787e4ddd582a66e40321ab2))
* csv validation
([84aae15](https://github.com/unraid/api/commit/84aae15a73014592c226fa3701e34e57c7b60b46))
* default value for option
([6513fc4](https://github.com/unraid/api/commit/6513fc49de61c836e1aabf32a874d7da7da18adb))
* delete unused imports
([97a3772](https://github.com/unraid/api/commit/97a3772d95aff534d85c410e58391d30494d9237))
* disable button on submit
([2ceb5da](https://github.com/unraid/api/commit/2ceb5da3c70826cc50df476decb6b117025f46c0))
* disable casbin logging
([2518e7c](https://github.com/unraid/api/commit/2518e7c506f0d3aa9f44031d61dce95d9db0a4cf))
* do not move upgradepkg
([ea16419](https://github.com/unraid/api/commit/ea16419929e0233e2c1ce37e2f4b79e3e64ce619))
* docstrings
([b836ba7](https://github.com/unraid/api/commit/b836ba72516c554ee8973d69aaaa4ed35b465fa7))
* dont pass entire server state for privacy
([54e3f17](https://github.com/unraid/api/commit/54e3f17bd9e541f50970c696bbe8b602ec38a748))
* download fixtures from the web
([1258c2b](https://github.com/unraid/api/commit/1258c2bc1813f0fa3cd52b4932302ad12b4edd01))
* download nodejs and install on legacy OS versions
([2a95e4b](https://github.com/unraid/api/commit/2a95e4beb2364510003f187459e28bb610583c41))
* eliminate all alpha beta gamma variable usage
([fbdbce9](https://github.com/unraid/api/commit/fbdbce97ec2171ec7057f0f159e73032e984705a))
* enable PR releases on non-mainline merges
([7ae8d03](https://github.com/unraid/api/commit/7ae8d03166952a602f0b7ebaf1cc65a9a8d27e7b))
* enable sandbox in dev mode
([4536d70](https://github.com/unraid/api/commit/4536d7092d77c68f5a996fd63bf74ce6e64f5efe))
* enable sandbox with developer command
([c354d48](https://github.com/unraid/api/commit/c354d482283295547afeb99c5e110b0181197c44))
* enable token sign in with comma separated subs in myservers.config
([ebed5bd](https://github.com/unraid/api/commit/ebed5bddea1445d9aaaee60d54758dc74b77271e))
* error state outside of button
([18c63e0](https://github.com/unraid/api/commit/18c63e0b0c7451c99eacabb504e18f8070ff7dc2))
* error when nodejs download fails
([6a9b14c](https://github.com/unraid/api/commit/6a9b14c68170d6430328cbb793d750f3177bdb32))
* exit cli after running command
([04bf528](https://github.com/unraid/api/commit/04bf528616fcbdf916916734a12d5fd32db9a06d))
* extensive file checking
([ab881c8](https://github.com/unraid/api/commit/ab881c8aed8dd4aa9fd71c32b50d3514d1496fa5))
* extract node to usr/local/
([4c0b55b](https://github.com/unraid/api/commit/4c0b55b269f47a9d8f746344ae701e353d80509a))
* fallback to local
([a2579c2](https://github.com/unraid/api/commit/a2579c2a7f80f54b4cc61533aec9ecc41a7e7f54))
* faster failure logic
([b439434](https://github.com/unraid/api/commit/b439434f1574e174fcf23f3a5f5b8df8e092eb1e))
* feature request template
([72a042c](https://github.com/unraid/api/commit/72a042c4fab295cf561807102c9eb9a78273bd83))
* fix docusaurus build + update snapshot
([23b27bd](https://github.com/unraid/api/commit/23b27bd63ea99f4137538eab40501daa67d7e3f5))
* fix missing flash line
([6897aad](https://github.com/unraid/api/commit/6897aad67f5c8b38450aa81e612b8aa98a9328c7))
* fix pm2 setup and add link command
([de9500f](https://github.com/unraid/api/commit/de9500ffa6f3aa1842152e0ab26f54c8c5c6e5cb))
* force linting on build
([43e6639](https://github.com/unraid/api/commit/43e663998a55e83c142067cb64ae7a331395fe68))
* generate key one time
([afe53c3](https://github.com/unraid/api/commit/afe53c30ea9987e6d8728faa2cb7291f8a126ecb))
* glob for files
([3fe281f](https://github.com/unraid/api/commit/3fe281f1ae28e3cbc089b5244a6ae2863b20adcb))
* group renovate updates
([715d556](https://github.com/unraid/api/commit/715d55682aa2ec5bf9297e02134b4302e652535f))
* hide sign in from the dropdown text
([3e68aaf](https://github.com/unraid/api/commit/3e68aaf8cdc0fb20c6e1b819a8571f419d94a811))
* hypervisor async imports
([32686ca](https://github.com/unraid/api/commit/32686ca4f0c25c43c6a9f7162bb8179b39e58f7e))
* ignore generated code
([68265a2](https://github.com/unraid/api/commit/68265a26efa588b60001310b9a11b398f04ae88f))
* ignore nuxt-custom-elements and reflect-metadata
([4a71083](https://github.com/unraid/api/commit/4a710836425f4200b7481a3deb28681588ccf8b0))
* improve packing
([9ef02d5](https://github.com/unraid/api/commit/9ef02d53666b70d41fdd186364808deac715e1ff))
* initial patcher implementation using the diff tool
([c87acbb](https://github.com/unraid/api/commit/c87acbb146c2e4e30997c964cd8be325dee68cea))
* initial setup of permissions on keys
([#1068](https://github.com/unraid/api/issues/1068))
([cf0fa85](https://github.com/unraid/api/commit/cf0fa850954ea2f018e338a132149f872b966df4))
* initial version of modification service
([b80469d](https://github.com/unraid/api/commit/b80469d38e519a7ba0e6eae636cda2a821e2d465))
* inject after form
([a4b276f](https://github.com/unraid/api/commit/a4b276f7874580bbf9827025730777715c9983da))
* kill timeout extended
([22d4026](https://github.com/unraid/api/commit/22d40264a02672a818053b5280d63a03ff7336b9))
* log size and only tar files
([731f2f8](https://github.com/unraid/api/commit/731f2f8e77a77b544a7f526c78aabfacca71eee4))
* logrotate test
([4504c39](https://github.com/unraid/api/commit/4504c39a2bbcf51385578b69a9fdc7b81a950e98))
* lots of progress on colors
([dc8b2ee](https://github.com/unraid/api/commit/dc8b2ee01b454d307e779d495dbcf11227760480))
* major release-please workflow improvements
([#1160](https://github.com/unraid/api/issues/1160))
([85aaab5](https://github.com/unraid/api/commit/85aaab501150b60764d6febf20566ab133625640))
* manually install libvirt in build process to ensure it is included in
the final build
([e695481](https://github.com/unraid/api/commit/e695481363f0d5d7add9d0e0d50d1e113b3024f6))
* more pm2 fixes
([8257bdf](https://github.com/unraid/api/commit/8257bdff3624211ee645349abdec303bf271538e))
* move fixtures into __test__ folder
([22a901d](https://github.com/unraid/api/commit/22a901de9b0c274d3f75ed4b4618cd6cd90324ba))
* move ssoenabled to a boolean flag rather than ids
([404a02b](https://github.com/unraid/api/commit/404a02b26bae6554d15e317f613ebc727c8f702f))
* move variable declarations to theme.ts
([3c82ee1](https://github.com/unraid/api/commit/3c82ee1e9acc197c9768a624cdef8c2e23c56d00))
* myservers_fb keepalive location
([e07e7f3](https://github.com/unraid/api/commit/e07e7f335c8ea4a73966ada90c26b7c82dbb025e))
* name package with PR number
([a642bf1](https://github.com/unraid/api/commit/a642bf15fd813dca522808765994414e4ed5a56c))
* nuxt config simplification and formatting
([02ffde2](https://github.com/unraid/api/commit/02ffde24d19594949faa97f9d070383b498fdcbe))
* only write config when a specific config update action occurs
([ec29778](https://github.com/unraid/api/commit/ec29778e37a50f43eb164991bcf2a6ff9c266033))
* or button on sign in page
([1433e93](https://github.com/unraid/api/commit/1433e938d7ac01af326e2875c582a6aa6d622615))
* **plugin:** rm Date & Time format settings from Notification Settings
([e2148f3](https://github.com/unraid/api/commit/e2148f3c2eaf77ad707eddb7989cc20ec8df70ab))
* properly read log level from environment
([b5151e9](https://github.com/unraid/api/commit/b5151e9ba76a6814e24e8da34e8a3c1bf1cc2144))
* public index
([f0641ea](https://github.com/unraid/api/commit/f0641ea7ca0919884dc3b8642c2e6694398e3246))
* remove apiKey from server
([b110a11](https://github.com/unraid/api/commit/b110a118fb153c0af09a74755deb468b3760ba27))
* remove nghttp3 and only bundle nodejs
([8d8df15](https://github.com/unraid/api/commit/8d8df1592e5af127a992d5634ee9d344055cdf2c))
* remove sso if disabled on Unraid-API start
([3bc407c](https://github.com/unraid/api/commit/3bc407c54e8e7aeadebd9ac223d71f21ef97fca1))
* remove sso user command
([bbd809b](https://github.com/unraid/api/commit/bbd809b83826e81eef38a06e66f3393e4f83e81e))
* remove sso user options
([e34041f](https://github.com/unraid/api/commit/e34041f86ef7ab6cf5e2fdf7efb86458d190edc1))
* remove unused config sections
([f0b9c4f](https://github.com/unraid/api/commit/f0b9c4f44ab0ee8f75bf96fde2413988ef4f6a8c))
* remove unused fields
([d2d0f7c](https://github.com/unraid/api/commit/d2d0f7cd9acb53ea2372245d7ef669c7ca24ee8a))
* remove unused vars
([0507713](https://github.com/unraid/api/commit/0507713972e344ad47bd077554d5888269669e9c))
* rename api key resource back to api_key
([ee9666b](https://github.com/unraid/api/commit/ee9666b317d7feb5c15d53e2a6b902c7771c8c7a))
* rename modification file
([70a93f2](https://github.com/unraid/api/commit/70a93f2cc63e0e62242be6fe1a717515a6fbec85))
* reorder index
([858553f](https://github.com/unraid/api/commit/858553f0debb6424ae0614640b82a050c33f175a))
* responsive notifications
([d427054](https://github.com/unraid/api/commit/d427054443176563faa3e44249219c1d938e4b07))
* restart the API when an SSO user is added
([a6b0c90](https://github.com/unraid/api/commit/a6b0c906a423df048401750943f02dfdc9bc2619))
* restoring sso error
([234bf7d](https://github.com/unraid/api/commit/234bf7dfa4b0be88b6cc13996d8f29ec819da26e))
* revert local api key value
([ff40e7a](https://github.com/unraid/api/commit/ff40e7ae392052d3d9e1b084c5f4851e8ebd529e))
* rollback if patch exists before applying
([c2f4e8d](https://github.com/unraid/api/commit/c2f4e8d4e5c758601bd20ba491fd077b434ba45e))
* secondary changes
([d75331a](https://github.com/unraid/api/commit/d75331a67e3566875ce8642fce80195e79932a4c))
* separate install process
([b90a516](https://github.com/unraid/api/commit/b90a51600c3f70615b117f157d41585e55ef49de))
* service tests for modifier service
([08c1502](https://github.com/unraid/api/commit/08c150259f2b4630d973803f4edff69c8bf0ec3a))
* session issues
([5981693](https://github.com/unraid/api/commit/5981693abd605337f9174ba4c85fd1bfc243edeb))
* set background color on webcomponents
([b66e684](https://github.com/unraid/api/commit/b66e6847c895f216a5dec42410186b81a31af1a9))
* shared call to createPatch
([eb3e263](https://github.com/unraid/api/commit/eb3e263fb32a748bfa06ec6d119ee51d242707cf))
* sidebar notification count
([694f01b](https://github.com/unraid/api/commit/694f01b6c4ab83c4131ae42bc11002d0300497c5))
* simplify docs
([d428030](https://github.com/unraid/api/commit/d428030b806f55b62421559d434fc723786b03ad))
* sso button token exchange
([f6f2390](https://github.com/unraid/api/commit/f6f2390b0169ceaf90ab88edfab3f2809bfe86b5))
* sso login boolean
([34190a6](https://github.com/unraid/api/commit/34190a6910679b65362a9cfa0837bca7775cfda0))
* sso testing page and form disable on submit
([ffc6d8a](https://github.com/unraid/api/commit/ffc6d8a286d7c6ba751894464000f9870784507c))
* state using crypto
([afce130](https://github.com/unraid/api/commit/afce13099f5018d0c39765bfdd181adc8383a105))
* style improvements
([b0f395e](https://github.com/unraid/api/commit/b0f395ef76f11047eaa13091df277df0459e9d8f))
* swap to async exit hook
([4302f31](https://github.com/unraid/api/commit/4302f316820a109c76408092994727b2dc030a15))
* switch to nest-commander
([1ab2ab5](https://github.com/unraid/api/commit/1ab2ab5b58a1f49cd6b05aaa84bfeced49d68e8e))
* try catch restart
([89abee6](https://github.com/unraid/api/commit/89abee680bdbdaa9946ddb991f0e6b5ada9ccdf7))
* **ui:** webgui-compatible web component library
([#1075](https://github.com/unraid/api/issues/1075))
([1c7b2e0](https://github.com/unraid/api/commit/1c7b2e091b0975438860a8e1fc3db5fd8d3fcf93))
* unnecessary comment
([0c52256](https://github.com/unraid/api/commit/0c5225612875b96319b28ef447db69ecab15cfda))
* unraid single sign on with account app
([5183104](https://github.com/unraid/api/commit/5183104b322a328eea3e4b2f6d86fd9d4b1c76e3))
* unraid ui component library
([#976](https://github.com/unraid/api/issues/976))
([03e2fee](https://github.com/unraid/api/commit/03e2feebc73d620b21e54912e0bbddc1826880e1))
* update based on review feedback
([4383971](https://github.com/unraid/api/commit/43839711e3365e31120e156abac3746c55e8e694))
* update packageManager field for pnpm
([8d5db7a](https://github.com/unraid/api/commit/8d5db7a9bfdf528e2d58b20cc62434ea5929d24f))
* Update
plugin/source/dynamix.unraid.net/usr/local/emhttp/plugins/dynamix.my.servers/include/state.php
([42c0d58](https://github.com/unraid/api/commit/42c0d58da4d0570b7d865a8774964c18120ed585))
* upgrade dependencies
([0a0cac3](https://github.com/unraid/api/commit/0a0cac3da74c2fe20f7100a9ad5d1caafa74b157))
* upgrade workflow and auto-assign reviewers
([58a419e](https://github.com/unraid/api/commit/58a419ed36926d121e405a3de37bcb39f26f50b1))
* use execa for start and stop
([46ab014](https://github.com/unraid/api/commit/46ab0144d41b425015487c251c1884744223ba29))
* use plugin file for install and uninstall
([c9ac3a5](https://github.com/unraid/api/commit/c9ac3a5a0a3103fbd9c33a5d909fa475614a704a))
* use state passing to validate requests
([4480c14](https://github.com/unraid/api/commit/4480c14c932fd8b42ba44989abdbecb49252e6f3))
* use text-secondary-foreground instead of gray
([463a1f7](https://github.com/unraid/api/commit/463a1f7b611599a19a23d3c75156c0a16da83312))
* use zod to parse config
([19cf1be](https://github.com/unraid/api/commit/19cf1be079f2ccb9e0cfa10f2fb97a18f15c5729))
* validate entries correctly
([b101a69](https://github.com/unraid/api/commit/b101a695e18d71ddd170462b3d49289352166489))
* validate token format in both PHP and CLI
([6ef05a3](https://github.com/unraid/api/commit/6ef05a3d7770f799e7d587c2cef8d29f6058bee1))
* viewport watch refactor
([9aefa38](https://github.com/unraid/api/commit/9aefa382ec64f08b1da8a3748ce16f637d562c8c))
* warning on missing fields
([0ef9aec](https://github.com/unraid/api/commit/0ef9aecccdde879e3be44d0b2a0fa4d8befc53b5))
* **web:** activation modal steps, updated copy
([#1079](https://github.com/unraid/api/issues/1079))
([8af9d8c](https://github.com/unraid/api/commit/8af9d8c58895010e3ddc03cc5fa075ac1e264f50))
* **web:** add an 'all' option to notification filter
([7c2a72e](https://github.com/unraid/api/commit/7c2a72e0c9537827c3c96df7b6378c03e2cc2852))
* **web:** add confirmation before archiving or deleting all
notifications
([d16f08c](https://github.com/unraid/api/commit/d16f08c266953ddb84223f90f1275d19c9d3c380))
* **web:** add count labels to notification tabs
([4caea3d](https://github.com/unraid/api/commit/4caea3dfc2c7067062f3ce8d863f9385ad030dbd))
* **web:** add delete all notifications button to archive view in
notifications sidebar
([3bda9d6](https://github.com/unraid/api/commit/3bda9d6a4ca01cc5580012b0133e72929d6dab40))
* **web:** add link to settings in notification sidebar
([f1a4d87](https://github.com/unraid/api/commit/f1a4d873481c212ffde1af7e38327a53a7e41d43))
* **web:** add loading and error states to notification sidebar
([2e9183a](https://github.com/unraid/api/commit/2e9183a479e0ec5f7cfc34bb81ccfd05e4bd2b29))
* **web:** clear notifications indicator after opening sidebar
([68958d1](https://github.com/unraid/api/commit/68958d17b78220c77c3cda4f0f4068b3ce623688))
* **web:** enhance notifications indicator in UPC
([#950](https://github.com/unraid/api/issues/950))
([6376848](https://github.com/unraid/api/commit/63768486e4ec64ab32666a26adf96f4db4a53e81))
* **web:** improve notification count syncing
([#1148](https://github.com/unraid/api/issues/1148))
([af2057c](https://github.com/unraid/api/commit/af2057c643640270e3e152ff8e08c3045e622437))
* **web:** move notification indicator icons to top-right of bell icon
([2fe4303](https://github.com/unraid/api/commit/2fe4303387023d303d7e50fc4d9a41f1eafdcc45))
* **web:** open official release notes via header os version
([54a893f](https://github.com/unraid/api/commit/54a893f396b29251b982ff1f26d376d24b962b93))
* **web:** pull date format from display/date and time settings
([b058067](https://github.com/unraid/api/commit/b058067b628ca7866a9ba0a6c4c5e4d5505d98cb))
* **web:** remove notification indicator pulse
([f320a77](https://github.com/unraid/api/commit/f320a77330c8cc7b92e170b0099d6c7f93b11c0e))
* **web:** rm api-key validation from connect sign in
([#986](https://github.com/unraid/api/issues/986))
([7b105d1](https://github.com/unraid/api/commit/7b105d18678e88a064f0643d6e857704789e0ee8))
* **web:** rm old notification bell upon plugin installation
([#979](https://github.com/unraid/api/issues/979))
([e09c07c](https://github.com/unraid/api/commit/e09c07c5070d59ac032baeff1ed253b5c00f4163))
* **web:** support markdown in notification messages
([90cbef7](https://github.com/unraid/api/commit/90cbef774962e9d8ede47df7a4c1ca06f2a6651b))
* **web:** use Markdown helper class to interact with markdown
([f9c2d35](https://github.com/unraid/api/commit/f9c2d353133b01e74fe1bfbc420df3980d944012))
* work intent
([feee4be](https://github.com/unraid/api/commit/feee4bebfe97620c73e6a6093065f22ea26ee8b9))
* work intent process
([b04a97a](https://github.com/unraid/api/commit/b04a97a493f06c450949c674629e8a787164464b))
* zod config no longer any
([c32c5f5](https://github.com/unraid/api/commit/c32c5f57127b9469bde8806d78dc364562e73d9f))


### Bug Fixes

* 12 hour timestamp logic corrected
([03be43b](https://github.com/unraid/api/commit/03be43b4579f1dcf6a666a144f75b3063576748a))
* add another missing symlink
([4e7f3ff](https://github.com/unraid/api/commit/4e7f3ff4d9aa0e4af417a50e2b30537dda3c759c))
* allow concurrent testing with a shared patcher instance
([623846e](https://github.com/unraid/api/commit/623846ef46eb24a32c62516de58e8bc5d0219833))
* **api:** append time to formatted date when a custom date format is
selected
([0ac8ed9](https://github.com/unraid/api/commit/0ac8ed9d9e7e239e471eedf466832aed0270d123))
* **api:** change log output location for diagnostic compatibility
([#1130](https://github.com/unraid/api/issues/1130))
([cba1551](https://github.com/unraid/api/commit/cba155138379d47bc3151c7c27d745ba6a345d83))
* **api:** delay pm2 start until server has booted
([bd3188e](https://github.com/unraid/api/commit/bd3188efea4d3656994ffae32bd53f821c96358d))
* **api:** exclude duplicates from legacy script in archive retrieval
([8644e13](https://github.com/unraid/api/commit/8644e130979ed8740c5a8da0b3984266e2b3684c))
* **api:** improve defaults in PM2 service
([#1116](https://github.com/unraid/api/issues/1116))
([57526de](https://github.com/unraid/api/commit/57526dede69e3a6547d05183e43c5b36dd1cae89))
* **api:** logrotate modification & permissions
([#1145](https://github.com/unraid/api/issues/1145))
([5209df2](https://github.com/unraid/api/commit/5209df2776e1a985e82bedc655fe28acf1fd0bde))
* **api:** make cookie recognition during websocket connection more
([353e012](https://github.com/unraid/api/commit/353e012db8ab5280863f32392c520b4a330c13cc))
* **api:** pm2 start script & limit auto restarts
([#1040](https://github.com/unraid/api/issues/1040))
([ebcd347](https://github.com/unraid/api/commit/ebcd3479e735724626ffc6907c338d5080898bee))
* **api:** retry mothership connection up to 3x before logout
([#1069](https://github.com/unraid/api/issues/1069))
([c27bb1b](https://github.com/unraid/api/commit/c27bb1be4c7a9ab201585586f3bc5e4afa1c7791))
* **api:** sanitize incoming user session id's
([f5e3424](https://github.com/unraid/api/commit/f5e3424b79702e8f959b5519e83370a9e1d2033b))
* **api:** slow init of unraid-api cli
([#1022](https://github.com/unraid/api/issues/1022))
([5dbbae7](https://github.com/unraid/api/commit/5dbbae796792a62234497d056eac019aa084b21c))
* **api:** update deploy-dev script to dist instead of src
([55cce09](https://github.com/unraid/api/commit/55cce09e65521762a6fe388d5b9b88ace1337c26))
* **api:** validate cookie session data
([491f680](https://github.com/unraid/api/commit/491f680607ce7244d9e47a457e44cde711fbe00c))
* apply and rollback error handling
([e22191b](https://github.com/unraid/api/commit/e22191bc77bc09f5c6c4ad57e5073829cf966ba4))
* authorization type error
([#987](https://github.com/unraid/api/issues/987))
([7a4799e](https://github.com/unraid/api/commit/7a4799e9cd4caef6acfc3661d205a377fcf499ab))
* back to default configs
([b5711c9](https://github.com/unraid/api/commit/b5711c91284072991bcf409ac6126cd4b46afc7c))
* backup restore formatting
([15210f6](https://github.com/unraid/api/commit/15210f64b0938ec884a3ef4379d245c661eab9a3))
* basic test fixed
([2f38035](https://github.com/unraid/api/commit/2f38035520ca0fe796c981d08b9136d89ffc5888))
* better js file handling
([ddf160e](https://github.com/unraid/api/commit/ddf160e878a352842e813154b607945ccc7b4081))
* better loader functionality and error handling
([8a57d2d](https://github.com/unraid/api/commit/8a57d2dccbcb9c2effc5df5d8c69ad02713de24a))
* check width before changing viewport
([f07381b](https://github.com/unraid/api/commit/f07381b243501ecc6d54063881faad77a99a7655))
* chown correctly ([#1185](https://github.com/unraid/api/issues/1185))
([11b0441](https://github.com/unraid/api/commit/11b04417371ba77eddc12699db8575988d6deafc))
* cleaner logs for starting API
([79f26ef](https://github.com/unraid/api/commit/79f26ef251cb42e7f2106d00c6c05e2bf17b8227))
* cleanup commands
([052aea0](https://github.com/unraid/api/commit/052aea06a0d30963532f29f9961fce0ffc7fa3e8))
* clearer error messaging
([e373849](https://github.com/unraid/api/commit/e37384966c5b9079bb507052dcaba56232c1c42a))
* code review feedback
([c66079e](https://github.com/unraid/api/commit/c66079e9a8e0ef47e5054118d0581bec708ac604))
* completion script registration
([05c8c9b](https://github.com/unraid/api/commit/05c8c9bf078ece2061ad8ae32497f52b8c9b94dc))
* connect breaks default css of header
([#1155](https://github.com/unraid/api/issues/1155))
([4ac9aa3](https://github.com/unraid/api/commit/4ac9aa3e409d0d89f2be61bfbafb8d7b5a5b3b00))
* connect key role
([2dcfc1c](https://github.com/unraid/api/commit/2dcfc1c19a1d085df84f0b1b50c096e3220205dd))
* create api key for connect on startup
([58329bc](https://github.com/unraid/api/commit/58329bc29521ebc26b27ee20013ac3926c5088c2))
* create api key permissions
([cefb644](https://github.com/unraid/api/commit/cefb644bd7fa513f553ca0ca4c49f0fb42a74112))
* create connect key
([6b1ab7b](https://github.com/unraid/api/commit/6b1ab7b74ae1d2938fa9105180a5f66e9604fd41))
* create PR ignored
([bdfefa8](https://github.com/unraid/api/commit/bdfefa808f5f1d85ff957a78a624edcef3afb47a))
* create releases as drafts
([0adba3a](https://github.com/unraid/api/commit/0adba3ae3566ad84626a9ab1cf7d38edb24de69f))
* cwd on ecosystem.config.json
([dfd0da4](https://github.com/unraid/api/commit/dfd0da4ca23078f6de2e54d5e5bd6cba06334abc))
* dark theme as array
([1021d0d](https://github.com/unraid/api/commit/1021d0da0d7a919dedec70656bb52775575aa9e7))
* default overwrite false test
([cf59107](https://github.com/unraid/api/commit/cf59107e568d91be684176335db5300bee9be865))
* delete .original files
([a9eb21a](https://github.com/unraid/api/commit/a9eb21aac0f373990aaa3f7a99731612540533cf))
* delete unused line
([de4882e](https://github.com/unraid/api/commit/de4882ea17f54e788049cc5bb96b99b16822b6b4))
* deprecated version warning
([89d0bd2](https://github.com/unraid/api/commit/89d0bd2e6da35fb1e8d95627d38edb54f82e0c6b))
* **deps:** update all non-major dependencies
([#1158](https://github.com/unraid/api/issues/1158))
([45ebc8b](https://github.com/unraid/api/commit/45ebc8b6e07c53ad3eee28d5cf8ac9cd0d827754))
* **deps:** update all non-major dependencies
([#1168](https://github.com/unraid/api/issues/1168))
([443b608](https://github.com/unraid/api/commit/443b608163cdb64775d6758b8207cbe745f892f4))
* **deps:** update apollo graphql packages
([7b1ee99](https://github.com/unraid/api/commit/7b1ee9940cca46e563bb79c7056996315f9decc5))
* **deps:** update dependency @apollo/client to v3.12.6
([22ce615](https://github.com/unraid/api/commit/22ce61574f862eac4cdf8c00141bfbf1ac948055))
* **deps:** update dependency @apollo/client to v3.12.6
([bb7800a](https://github.com/unraid/api/commit/bb7800a8c088705fd8310671a9896cbe9b0184e5))
* **deps:** update dependency @apollo/client to v3.12.9
([6607cf2](https://github.com/unraid/api/commit/6607cf20c10a091d466c6a8031eebc17feb3e3fc))
* **deps:** update dependency @floating-ui/dom to v1.6.13
([08798d2](https://github.com/unraid/api/commit/08798d2f77683412807d684d7a8e63f1aadb0c34))
* **deps:** update dependency @floating-ui/dom to v1.6.13
([4d4c218](https://github.com/unraid/api/commit/4d4c218ac78e82a18679ec7b4939523db032b99b))
* **deps:** update dependency @floating-ui/vue to v1.1.6
([b4b7d89](https://github.com/unraid/api/commit/b4b7d898b62f746180b7f5730b5d9b5033dcecc2))
* **deps:** update dependency @floating-ui/vue to v1.1.6
([4c07d38](https://github.com/unraid/api/commit/4c07d389523f277950b8d2d359102f889587e5ce))
* **deps:** update dependency @graphql-tools/load-files to v7.0.1
([4e5c724](https://github.com/unraid/api/commit/4e5c7242e43cc356f1c69adcfcd25b57896af476))
* **deps:** update dependency @nestjs/schedule to v4.1.2
([faf0de5](https://github.com/unraid/api/commit/faf0de5a19256efb83dc45a484e3cba65596ccd7))
* **deps:** update dependency chokidar to v4.0.3
([d63a93c](https://github.com/unraid/api/commit/d63a93c55004d17b6d17634c55ffbc5670ebbec7))
* **deps:** update dependency dockerode to v4
([#830](https://github.com/unraid/api/issues/830))
([c331ecd](https://github.com/unraid/api/commit/c331ecd50c4910fd6c35e5ad92b3f676d552febc))
* **deps:** update dependency dotenv to v16.4.7
([c66a650](https://github.com/unraid/api/commit/c66a6502b027853046d126a14ddee870ffabd10c))
* **deps:** update dependency execa to v9.5.2
([d487c90](https://github.com/unraid/api/commit/d487c90ccc20162c76f0cdf49a736c1fee4271bd))
* **deps:** update dependency express to v4.21.2
([a070306](https://github.com/unraid/api/commit/a07030684c8777e47eb4a51be0ea680b7f217e74))
* **deps:** update dependency focus-trap to v7.6.4
([41ff232](https://github.com/unraid/api/commit/41ff232a3232dda66e5cdc2d4808a820a90a5d34))
* **deps:** update dependency focus-trap to v7.6.4
([f0e3038](https://github.com/unraid/api/commit/f0e3038ee7426aafb6cef01b85b47893c2238302))
* **deps:** update dependency got to v14.4.5
([975a47c](https://github.com/unraid/api/commit/975a47c7d47841c49443f46264feb54abf53698c))
* **deps:** update dependency graphql-ws to v5.16.2
([a189a03](https://github.com/unraid/api/commit/a189a0308a734e66750fe5059f7c59d8c9532bd8))
* **deps:** update dependency graphql-ws to v5.16.2
([25d8f08](https://github.com/unraid/api/commit/25d8f085b67c2e53876d837c739214dc874116b8))
* **deps:** update dependency ini to v4.1.3
([4c88cbe](https://github.com/unraid/api/commit/4c88cbee4b2d5f6717241dadac23bfe90ce15193))
* **deps:** update dependency node-window-polyfill to v1.0.4
([8bfa88f](https://github.com/unraid/api/commit/8bfa88f4bc932eb82dd9b33a494811ea15764758))
* **deps:** update dependency openid-client to v6.1.7
([0f50517](https://github.com/unraid/api/commit/0f50517a8544e1eb9b08ad1b3f05f798491b7f23))
* **deps:** update dependency p-retry to v6.2.1
([c6f3241](https://github.com/unraid/api/commit/c6f324155019e066701723a57b642c6e3ba8332d))
* **deps:** update dependency pm2 to v5.4.3
([a754090](https://github.com/unraid/api/commit/a75409026dd4e3d9ed120802012b67b179327448))
* **deps:** update dependency radix-vue to v1.9.12
([0fd433f](https://github.com/unraid/api/commit/0fd433fe2a6b3f787624cb5a98efeae0f6c31cfd))
* **deps:** update dependency radix-vue to v1.9.13
([249feff](https://github.com/unraid/api/commit/249feff5cfe0bbb60bfa8f943b76b9c16c6c161b))
* **deps:** update dependency uuid to v11.0.5
([7e3398b](https://github.com/unraid/api/commit/7e3398b2efabf1a5407d6e20c165eb4923b3bab2))
* **deps:** update graphql-tools monorepo
([cd7e2fe](https://github.com/unraid/api/commit/cd7e2feea199276a1d431cf355e54e12e5960d9a))
* **deps:** update graphqlcodegenerator monorepo
([0446c59](https://github.com/unraid/api/commit/0446c5924a6a9dd15b875628ca0f1197cfe521c4))
* **deps:** update graphqlcodegenerator monorepo
([15c789d](https://github.com/unraid/api/commit/15c789dbb34b85bed55c2731fb8ae8260f5f311f))
* **deps:** update nest monorepo to v10.4.15
([07b1ea9](https://github.com/unraid/api/commit/07b1ea9a10634a597909ae1d237cc3b1e7f959b7))
* **deps:** update nest-graphql monorepo to v12.2.2
([91aabd9](https://github.com/unraid/api/commit/91aabd9ffbfb8c2ceb4110217dfc05de8859077d))
* dev mode
([fd64e01](https://github.com/unraid/api/commit/fd64e01e0c87db03fc2d4d0f32a0e8205fbe8b84))
* do not process.exit on restart or stop command
([933575f](https://github.com/unraid/api/commit/933575fc2badbb09b3a9d3c66724e37a9ee246f2))
* docs creation workflow
([86134c6](https://github.com/unraid/api/commit/86134c60856c130dab9f96b718d9afa5bbab1e50))
* don't check code for execa
([508a5eb](https://github.com/unraid/api/commit/508a5eb49d9514dca9953317d9fa93314fe63e4c))
* don't specify pnpm version in install
([929a4d2](https://github.com/unraid/api/commit/929a4d2deaccbdcf16c623ded62b80535c88b672))
* dont remove login file without a backup presetn
([0370e4f](https://github.com/unraid/api/commit/0370e4f7ea3e3df0d2264264324d8e53ffc0c086))
* downgrade marked to fix changelog preview issue
([cfb3a45](https://github.com/unraid/api/commit/cfb3a45533d3c1bd31c44094f7ae2912e77a673e))
* edit settings padding issue
([adf349b](https://github.com/unraid/api/commit/adf349b76560b5f1fd4c320da35b3c6f660895fb))
* empty manifest and version alignment
([c4c44d9](https://github.com/unraid/api/commit/c4c44d92caa593795b9cb111a27828ecb8f62dbd))
* ensure directory exists before making connect key
([9e27ec9](https://github.com/unraid/api/commit/9e27ec98b68a49bdd6dc4b03de8c0cc3a1470a5e))
* excess spacing in api-key.service
([1deb002](https://github.com/unraid/api/commit/1deb0023287a39d40e52e89c515a28e62352f62c))
* extra log line
([1183063](https://github.com/unraid/api/commit/1183063aa7063afd8222def18f5e1fd6077e8c88))
* extra spacing in config.ts
([f3ee7be](https://github.com/unraid/api/commit/f3ee7be80f2c60266fbb13597a70f0a389fb577f))
* file modification service fixes
([aa5b3f4](https://github.com/unraid/api/commit/aa5b3f4e47ed88df23af00dfcccb7b64786b6231))
* finalize workflow fixes
([77d427a](https://github.com/unraid/api/commit/77d427a9f3d8ef3b846236858426f558d1acc422))
* find by key, not ID
([3c3fa1e](https://github.com/unraid/api/commit/3c3fa1e27cfabbe6926c3da8870751397eed1def))
* fix libvirt bindings
([#1167](https://github.com/unraid/api/issues/1167))
([5817e5d](https://github.com/unraid/api/commit/5817e5d4b1697bbdfaa1984ccd650a232233cc15))
* format authrequest mod as other files
([180a81d](https://github.com/unraid/api/commit/180a81dbae8e749eae237fc8cee2950c790eedf0))
* formatting issue
([42ca969](https://github.com/unraid/api/commit/42ca9691f7547a4340501863c1882efc0aee4c60))
* further resolve sso sub ids issues
([ef3d0ea](https://github.com/unraid/api/commit/ef3d0ead687d4a6071da290c0df29c12163303e1))
* improve typing and format lookup
([c6097f8](https://github.com/unraid/api/commit/c6097f86e42fcc57209c1344029abe854198edca))
* initial feedback about report addressed
([5dee09c](https://github.com/unraid/api/commit/5dee09c77ad375de2eca59f650e5fea2070087b5))
* install as-integrations/fastify
([ff4546d](https://github.com/unraid/api/commit/ff4546d6692d2a4799f2dbeef0d5e5c6bac62561))
* install syntax error
([ec83480](https://github.com/unraid/api/commit/ec83480eb6aea09b98b9135516dc1fc8cdd6c692))
* integration of `unraid-ui` tailwind config in `web`
([#1074](https://github.com/unraid/api/issues/1074))
([f3cd85b](https://github.com/unraid/api/commit/f3cd85bd3f02bdbe4c44136189d1c61935015844))
* length
([83579f1](https://github.com/unraid/api/commit/83579f1fbd03ffe929d009c20d214b4de62835c6))
* lint
([0f218b8](https://github.com/unraid/api/commit/0f218b8b72e397734823efab8f2141973a3a80ce))
* lint
([82bca54](https://github.com/unraid/api/commit/82bca54c594265ddf23a298691bd7ef6d4b47f32))
* lint
([ceb443d](https://github.com/unraid/api/commit/ceb443da15d177a950c36af61b93a7126cf4ca85))
* lint
([da04e7c](https://github.com/unraid/api/commit/da04e7ce0873d7802a936952d91e9867f0868a6e))
* lint
([7d87f0e](https://github.com/unraid/api/commit/7d87f0eee23dfa0f391fd342d38ed9084f18d8d4))
* logrotate error
([8c64dd2](https://github.com/unraid/api/commit/8c64dd2f2c65aa83ce0e2d501357ee595c976e56))
* lowercase or
([386cbde](https://github.com/unraid/api/commit/386cbdef5c9158290e03c670efb992cf11d5af1b))
* main.yml release issue
([8a2a24e](https://github.com/unraid/api/commit/8a2a24eb22762034d44995580d6057186521dae5))
* make public not a part of folder structure in PR
([099a88e](https://github.com/unraid/api/commit/099a88eb4970da48e57dafbc3807e16f1987d7fc))
* marked single input
([ceacbbe](https://github.com/unraid/api/commit/ceacbbe5d46466627df0fccc5ca8e7c56fa36a37))
* missing server type
([f1b721b](https://github.com/unraid/api/commit/f1b721bd72b875d9ff8c0bca2cc1eee506ba7697))
* mock ensureDirSync
([7e012e6](https://github.com/unraid/api/commit/7e012e6a2eb96ccddf5a1f69d7580b4bdfe7a0a9))
* more color work
([c48f826](https://github.com/unraid/api/commit/c48f8268def64ef8828dea556360b375b8cb32c7))
* more generic test
([0f651db](https://github.com/unraid/api/commit/0f651dbf61a1822b492aa80030f0bc231bc6f606))
* more verbose logging for node install to find issues
([445af0c](https://github.com/unraid/api/commit/445af0c147ef641dac05ebeb2d44e63e8a4df799))
* oauth2 api prefix
([ec00add](https://github.com/unraid/api/commit/ec00adde20d4d9eca28f6b18615073305f491a73))
* only instantiate service one time
([933dc81](https://github.com/unraid/api/commit/933dc81b6c50db5a33f586f7094e1ea524b9a9fa))
* only run release if releases_create == 'true'
([e5196f9](https://github.com/unraid/api/commit/e5196f97554d5f8d8d6d3cbd574049d641621361))
* only toast unread notifications, not archived ones
([cc59be6](https://github.com/unraid/api/commit/cc59be6cb3efc71226ee50f9f04e37a2e4b50de6))
* padding and glob function issues
([1d3f2eb](https://github.com/unraid/api/commit/1d3f2eb8213115c3385ac2d29ee8f53560347ba8))
* pass ssoSubIds only
([5adf13e](https://github.com/unraid/api/commit/5adf13ee070bdcd849339460b9888e51d224e765))
* pass token to password field
([499b023](https://github.com/unraid/api/commit/499b023d359ed5181450ee9e04cbbf4531a4a680))
* patch-utils unused
([047808d](https://github.com/unraid/api/commit/047808dce0cd9e9b4b273a9124dbd45ca9446208))
* paths now correct, better download logic
([16db2d9](https://github.com/unraid/api/commit/16db2d908dcb2c65508b367712c51bf9872a95e5))
* PHP Warning in state.php
([#1126](https://github.com/unraid/api/issues/1126))
([c154b4e](https://github.com/unraid/api/commit/c154b4e0ad2d0627b1541a7f9ee5e55235d4dd5e))
* properly log error with template string
([3781f1f](https://github.com/unraid/api/commit/3781f1f41c7f0eef604daee0402ed9a2bb27cd46))
* pull token from query not params
([2e827e7](https://github.com/unraid/api/commit/2e827e7cabe4a6a069d4e8779015e5896d8a1d1d))
* re-add manifest
([d1a42ab](https://github.com/unraid/api/commit/d1a42abf9cfe58609aff2e325a4f299616984242))
* re-add type-check
([60e9d1d](https://github.com/unraid/api/commit/60e9d1d912c983cf04e3e6cf15e221c39938612a))
* recreate package-lock to fix issues
([ad5a537](https://github.com/unraid/api/commit/ad5a53793d25ac9f63bae6df6c2a30d8d2780c67))
* release-please upload single draft
([33e0e45](https://github.com/unraid/api/commit/33e0e45921dca4f19a06e3134a434201695e2372))
* remove console log
([8e75b82](https://github.com/unraid/api/commit/8e75b8254bbda93ded786750226090b769bed5c4))
* remove devDependencies from output package json
([294869b](https://github.com/unraid/api/commit/294869bbea7f8a1863f8aafae6b074330e057679))
* remove extra space
([a99ee03](https://github.com/unraid/api/commit/a99ee03fc37059b3a018db289c43fc419a634524))
* remove isNaN in favor of number.isNaN
([03e3a46](https://github.com/unraid/api/commit/03e3a46092db613281176b88cae284f6448027c6))
* remove line from or in button
([1a1bce7](https://github.com/unraid/api/commit/1a1bce7b64b1cf90505f811e11b585ff87476f72))
* remove memory key generation
([b84db13](https://github.com/unraid/api/commit/b84db1322104c7f26f7b6378f25a2757b3010c6d))
* remove unused constructor
([e0e2a7b](https://github.com/unraid/api/commit/e0e2a7b41c5e599ed4cf3bf49c7faea3b71f0b70))
* remove unused date-fns
([fe94ef5](https://github.com/unraid/api/commit/fe94ef5ba88df56aad87089081dd5fe4518fa414))
* remove unused login entries
([7833b5d](https://github.com/unraid/api/commit/7833b5db386f724318857fc31d825fb3534c84b9))
* remove usage of Role.UPC
([d1e2f6e](https://github.com/unraid/api/commit/d1e2f6e0b391cb4eca75a0997b41cb99a9953d42))
* render function fixed
([8008ab4](https://github.com/unraid/api/commit/8008ab46fb2f231b68201758a258fd43e2e1672e))
* report issues + pm2 issues
([28c383e](https://github.com/unraid/api/commit/28c383e1d111d4ac4226d7d966533ba80ca5d9a1))
* reset config to be closer to default
([b7fbb0b](https://github.com/unraid/api/commit/b7fbb0b6af0453f5f6a17087bb7e68c393b9fe3f))
* resource busy when removing all subdirectories
([29936c9](https://github.com/unraid/api/commit/29936c90938fb83bc2f154315ca63a9d7cc98552))
* restart command elegant
([296117b](https://github.com/unraid/api/commit/296117b51aac8a4c15366f2271af858868b6e071))
* restore upgradepkg before install
([fddca27](https://github.com/unraid/api/commit/fddca2738c0ec016e744169d88b35a55dea092fa))
* revert changelog
([e9d47ca](https://github.com/unraid/api/commit/e9d47caf181148317e5ba8f0d11a433b09add0e3))
* revert changes to indicator.vue
([84d2a83](https://github.com/unraid/api/commit/84d2a832c0f64e52be05670eb438b21bff2e5163))
* revert config
([c17db6d](https://github.com/unraid/api/commit/c17db6d7648c15f71cd2195530cd0c2d051db3f6))
* revert dockerode upgrade
([#1140](https://github.com/unraid/api/issues/1140))
([a74a379](https://github.com/unraid/api/commit/a74a379a93fd15a315e31191de1bf69c5879f8a6))
* revert myservers.cfg
([d0896f3](https://github.com/unraid/api/commit/d0896f3ef8aebdd9c76d805ed6a35b4a5d5a1b08))
* sandbox defaults in dev mode wrong
([2a24919](https://github.com/unraid/api/commit/2a2491936cf85013be836450ab7ed0cc11207e11))
* sequential test execution for generic-modification
([79ee1f7](https://github.com/unraid/api/commit/79ee1f7552cee47c6f5a8eb5942468292212e2f2))
* shell path to unraid-api
([15d11e4](https://github.com/unraid/api/commit/15d11e477bb2a08d785a7b22bd51900279a55508))
* shorten work intent form
([95fe671](https://github.com/unraid/api/commit/95fe671717ab856518f5b4893dfbcbade0d0f2ed))
* simplify api setup index
([701b1fb](https://github.com/unraid/api/commit/701b1fbd9096c9675475062eaf32a2cbfb0567b9))
* simplify upcoming features
([8af79b2](https://github.com/unraid/api/commit/8af79b27501b42e1c1f7697756a56a9001000d8f))
* small modal window
([#1183](https://github.com/unraid/api/issues/1183))
([2183965](https://github.com/unraid/api/commit/2183965757ded27a5a01a6e27c4b783c9434af2e))
* staging build issues
([e6bcb8d](https://github.com/unraid/api/commit/e6bcb8de7daee463f7ac0dbf977e085e108302ba))
* start command simplification
([e1faf3a](https://github.com/unraid/api/commit/e1faf3aa8db5973eb1bb0ea7a4844f820504618d))
* stop command exits
([2dbfdb6](https://github.com/unraid/api/commit/2dbfdb670a773114c0fdc68c7cf9d29fa4e28a9b))
* storybook resolution issue
([#1153](https://github.com/unraid/api/issues/1153))
([52c70b9](https://github.com/unraid/api/commit/52c70b9d85469008894d44788429ba298b082ac7))
* strip components from tar line
([911cd5b](https://github.com/unraid/api/commit/911cd5bc0b0983df4ca8c9057bea5166f7d1c7f1))
* swap to placeholder key
([d1864d0](https://github.com/unraid/api/commit/d1864d0020ed56ab2368d23b48604b55cff21ae4))
* switch to useToggle
([848233f](https://github.com/unraid/api/commit/848233f05465053876ac6f9f6ac4bfad2a48abff))
* test issues
([e4b55b1](https://github.com/unraid/api/commit/e4b55b133bb2dc4bf2ccfd6fd2fc244daadbea53))
* test simplification to ensure no redownloads
([e07dad3](https://github.com/unraid/api/commit/e07dad3a6947aa186c4ac03032b5b3813cd046b6))
* tests
([25c1c1a](https://github.com/unraid/api/commit/25c1c1a55a3fb32b76bf5cb7257a4ba44f717a89))
* tests and validate token clears screen
([7f48ddd](https://github.com/unraid/api/commit/7f48dddcd2e2ea1ae3a55ecc54d5ac274535b714))
* theme store now uses singular variables object
([5ca6e40](https://github.com/unraid/api/commit/5ca6e40b2d4942385b12a4325d6b8a551cb3f44b))
* thorw on invalid token body
([f1af763](https://github.com/unraid/api/commit/f1af763eaf0dd8215eed470293d3a7f98784f38a))
* type error on element render
([a2563eb](https://github.com/unraid/api/commit/a2563eb8e710a9ac7259c4260cad9a3454565dae))
* type for generic test
([e856535](https://github.com/unraid/api/commit/e85653592a9d6eadcd0be89bf90a96c5d313fda3))
* unit test issues
([c58f7a7](https://github.com/unraid/api/commit/c58f7a7f246902c7d354eb51d1c87c8ea3b636a3))
* unit tests updated
([9548505](https://github.com/unraid/api/commit/954850535bec6b09aaf66b01d3ee749c8a22de5d))
* unneeded await on api-key service
([0325be7](https://github.com/unraid/api/commit/0325be757ee4c04b5c23365ff592f521a492595b))
* unused imports
([a5447aa](https://github.com/unraid/api/commit/a5447aa2f4c99968651fa3750d6bf0e8d68678de))
* upc header text color
([f989026](https://github.com/unraid/api/commit/f9890260d1c4abe69dac3ac4c05ebab17aab5161))
* Update renovate.json
([7dbb9c9](https://github.com/unraid/api/commit/7dbb9c991437cfe2d891797adcc452d0064fc1a9))
* update tests
([d0696a9](https://github.com/unraid/api/commit/d0696a93810893ccd6c676df1c639ca279992428))
* upgradepkg
([90cf1a8](https://github.com/unraid/api/commit/90cf1a8eea67d3dbc736ecdfba47e0025b1dc31c))
* upload to correct tag directory on build
([c5fe723](https://github.com/unraid/api/commit/c5fe723a0abee0d0fc494a5b512c995001ae0615))
* upload to release by name
([70333a0](https://github.com/unraid/api/commit/70333a00f6eed56fbec3e922db01b1ba370b00f3))
* use an enum and defaults for sandbox value
([eb56483](https://github.com/unraid/api/commit/eb56483ba2693944d39f4409c91b75ee82a7d30b))
* use batchProcess
([ffbb9d7](https://github.com/unraid/api/commit/ffbb9d7750568bfa849d21e051503d1fcca5355f))
* use cwd when running application
([e016652](https://github.com/unraid/api/commit/e01665264b6f45366cdacf60c0f3553adfbd85d3))
* use foreground text color for UPC
([87b8165](https://github.com/unraid/api/commit/87b816550d413dc9023c5057efe18b9cb26761e7))
* use placeholder in test API key
([c6b7755](https://github.com/unraid/api/commit/c6b7755214de8bedd5c0f2735473c2a559b1e26f))
* watch all events to load keys
([59ca177](https://github.com/unraid/api/commit/59ca17787e4d36113b0a8c5ef2117acfc491c49c))
* **web:** broken modals
([aebf339](https://github.com/unraid/api/commit/aebf3392595d45c84a84668f461c632a2d62e7dd))
* **web:** display error message in sidebar when api is offline
([#984](https://github.com/unraid/api/issues/984))
([125c0a1](https://github.com/unraid/api/commit/125c0a140b4e9b5401bacf1addab1820c412917e))
* **web:** edge case where archived notifications don't appear
([0a8c574](https://github.com/unraid/api/commit/0a8c5746fc2b8f8639643f013c1f19f0d7236d41))
* **web:** env var typo
([22cf90b](https://github.com/unraid/api/commit/22cf90b27fadec3024d9d038c53683e8f8a723bc))
* **web:** escaping html-encoded symbols like apostrophes in
translations ([#1002](https://github.com/unraid/api/issues/1002))
([04a3362](https://github.com/unraid/api/commit/04a33621e1d406d75ed0ff9af9f1f945813a1e8d))
* **web:** flash of disconnected api state on page load
([a8c02f4](https://github.com/unraid/api/commit/a8c02f4c49433b440a6f9c70f269bf69076655dc))
* **web:** infinite scroll loop when there's only 1 page of
notifications
([e9f2fc4](https://github.com/unraid/api/commit/e9f2fc424c350d07c756ae7573e90f615bcae25b))
* **web:** infinite trigger at bottom of infinite scroll
([eb691d3](https://github.com/unraid/api/commit/eb691d3514d8dc079987bfa566de4aa86094ef67))
* **web:** name of toaster component
([e093242](https://github.com/unraid/api/commit/e093242d20ddd72567396f4a53238250f2199a64))
* **web:** notification styles & alignment
([#968](https://github.com/unraid/api/issues/968))
([0d65e12](https://github.com/unraid/api/commit/0d65e12cede3324261fd3b219745b1e7793a33de))
* **web:** refetch notifications for sidebar when new notifications
arrive
([591bf4a](https://github.com/unraid/api/commit/591bf4a643ccc13c151c0a8cafad833d3137043e))
* **web:** remove warn and error console log removal
([#1086](https://github.com/unraid/api/issues/1086))
([9375639](https://github.com/unraid/api/commit/9375639e4a71ecfe8d4b877301c1f9bb22800a72))
* **web:** replaces dompurify with isomorphic-dompurify to enable
server-side usage with same syntax
([27834d9](https://github.com/unraid/api/commit/27834d92375be96a7842e81cc88eb083bcf127b8))
* **web:** reset infinite scroll when notification filters change
([da6de2c](https://github.com/unraid/api/commit/da6de2ccdb710772a199c8cba8952adc247412db))
* **web:** sanitize changelog markup after parsing
([c960292](https://github.com/unraid/api/commit/c96029273283f5970a5029eea1d7f451bbd0071b))
* **web:** stop opening notification sidebar to archive tab
([325e75f](https://github.com/unraid/api/commit/325e75f5d444908a2227fbe2e94be9ab5196ad8e))
* **web:** theme header differences
([#1085](https://github.com/unraid/api/issues/1085))
([1ccdd8d](https://github.com/unraid/api/commit/1ccdd8dc71ee5e1e3aacabd113d1cf213ca7c7ae))
* **web:** track 'notification seen' state across tabs & page loads
([#1121](https://github.com/unraid/api/issues/1121))
([64cf6ec](https://github.com/unraid/api/commit/64cf6ecc6aec25cd8edee5659efb09f288bb9908))
* **web:** update unread total immediately upon archiving
([#982](https://github.com/unraid/api/issues/982))
([ff5fd8e](https://github.com/unraid/api/commit/ff5fd8e5eb8eb4803db1265e31b0c1352af20251))


### Miscellaneous Chores

* release 4.1.2
([dbab290](https://github.com/unraid/api/commit/dbab290b429f9eff8fa903d193de2bd02bb392bd))

---
This PR was generated with [Release
Please](https://github.com/googleapis/release-please). See
[documentation](https://github.com/googleapis/release-please#release-please).

Co-authored-by: github-actions[bot] <41898282+github-actions[bot]@users.noreply.github.com>
@coderabbitai coderabbitai bot mentioned this pull request Apr 9, 2025
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Labels

None yet

Projects

None yet

Development

Successfully merging this pull request may close these issues.

3 participants