Added support for up to 100,000 enterprise configuration slots (#782)
Some checks are pending
Build and Release / Determine run mode (push) Waiting to run
Build and Release / Read metadata (push) Blocked by required conditions
Build and Release / Build app (${{ matrix.dotnet_runtime }}) (-aarch64-apple-darwin, osx-arm64, macos-latest, aarch64-apple-darwin, dmg,app,updater, dmg) (push) Blocked by required conditions
Build and Release / Build app (${{ matrix.dotnet_runtime }}) (-aarch64-pc-windows-msvc.exe, win-arm64, windows-latest, aarch64-pc-windows-msvc, nsis,updater, nsis) (push) Blocked by required conditions
Build and Release / Build app (${{ matrix.dotnet_runtime }}) (-aarch64-unknown-linux-gnu, linux-arm64, ubuntu-22.04-arm, aarch64-unknown-linux-gnu, appimage,updater, appimage) (push) Blocked by required conditions
Build and Release / Build app (${{ matrix.dotnet_runtime }}) (-x86_64-apple-darwin, osx-x64, macos-latest, x86_64-apple-darwin, dmg,app,updater, dmg) (push) Blocked by required conditions
Build and Release / Build app (${{ matrix.dotnet_runtime }}) (-x86_64-pc-windows-msvc.exe, win-x64, windows-latest, x86_64-pc-windows-msvc, nsis,updater, nsis) (push) Blocked by required conditions
Build and Release / Build app (${{ matrix.dotnet_runtime }}) (-x86_64-unknown-linux-gnu, linux-x64, ubuntu-22.04, x86_64-unknown-linux-gnu, appimage,updater, appimage) (push) Blocked by required conditions
Build and Release / Prepare & create release (push) Blocked by required conditions
Build and Release / Publish release (push) Blocked by required conditions

This commit is contained in:
Thorsten Sommer 2026-05-31 12:11:09 +02:00 committed by GitHub
parent a15c47b56d
commit def685d2c2
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
12 changed files with 668 additions and 174 deletions

View File

@ -6046,9 +6046,15 @@ UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T1137744461"] = "ID mismatch: the
-- This is a private AI Studio installation. It runs without an enterprise configuration. -- This is a private AI Studio installation. It runs without an enterprise configuration.
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T1209549230"] = "This is a private AI Studio installation. It runs without an enterprise configuration." UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T1209549230"] = "This is a private AI Studio installation. It runs without an enterprise configuration."
-- Copies the configuration origin to the clipboard
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T125850635"] = "Copies the configuration origin to the clipboard"
-- Unknown configuration plugin -- Unknown configuration plugin
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T1290340974"] = "Unknown configuration plugin" UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T1290340974"] = "Unknown configuration plugin"
-- Copies the configuration slot to the clipboard
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T1347508205"] = "Copies the configuration slot to the clipboard"
-- This library is used to read PDF files. This is necessary, e.g., for using PDFs as a data source for a chat. -- This library is used to read PDF files. This is necessary, e.g., for using PDFs as a data source for a chat.
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T1388816916"] = "This library is used to read PDF files. This is necessary, e.g., for using PDFs as a data source for a chat." UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T1388816916"] = "This library is used to read PDF files. This is necessary, e.g., for using PDFs as a data source for a chat."
@ -6145,6 +6151,12 @@ UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T2371107659"] = "installation pro
-- Installed Pandoc version: Pandoc is not installed or not available. -- Installed Pandoc version: Pandoc is not installed or not available.
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T2374031539"] = "Installed Pandoc version: Pandoc is not installed or not available." UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T2374031539"] = "Installed Pandoc version: Pandoc is not installed or not available."
-- Configuration origin:
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T2435772109"] = "Configuration origin:"
-- Configuration slot:
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T254943559"] = "Configuration slot:"
-- This library is used to determine the language of the operating system. This is necessary to set the language of the user interface. -- This library is used to determine the language of the operating system. This is necessary to set the language of the user interface.
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T2557014401"] = "This library is used to determine the language of the operating system. This is necessary to set the language of the user interface." UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T2557014401"] = "This library is used to determine the language of the operating system. This is necessary to set the language of the user interface."
@ -6199,6 +6211,9 @@ UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T2868174483"] = "The .NET backend
-- AI Studio runs with an enterprise configuration and configuration servers. The configuration plugins are not yet available. -- AI Studio runs with an enterprise configuration and configuration servers. The configuration plugins are not yet available.
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T2924964415"] = "AI Studio runs with an enterprise configuration and configuration servers. The configuration plugins are not yet available." UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T2924964415"] = "AI Studio runs with an enterprise configuration and configuration servers. The configuration plugins are not yet available."
-- Copies the configuration source to the clipboard
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T2929232062"] = "Copies the configuration source to the clipboard"
-- Changelog -- Changelog
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T3017574265"] = "Changelog" UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T3017574265"] = "Changelog"
@ -6259,6 +6274,9 @@ UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T3722989559"] = "This library is
-- Username provided by the OS -- Username provided by the OS
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T3764549776"] = "Username provided by the OS" UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T3764549776"] = "Username provided by the OS"
-- Configuration source:
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T3801531724"] = "Configuration source:"
-- this version does not met the requirements -- this version does not met the requirements
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T3813932670"] = "this version does not met the requirements" UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T3813932670"] = "this version does not met the requirements"

View File

@ -89,18 +89,7 @@
{ {
<ConfigPluginInfoCard HeaderIcon="@Icons.Material.Filled.HourglassBottom" <ConfigPluginInfoCard HeaderIcon="@Icons.Material.Filled.HourglassBottom"
HeaderText="@T("Waiting for the configuration plugin...")" HeaderText="@T("Waiting for the configuration plugin...")"
Items="@([ Items="@this.BuildEnterpriseConfigurationItems(env)"/>
new ConfigInfoRowItem(Icons.Material.Filled.ArrowRightAlt,
$"{T("Enterprise configuration ID:")} {env.ConfigurationId}",
env.ConfigurationId.ToString(),
T("Copies the config ID to the clipboard")),
new ConfigInfoRowItem(Icons.Material.Filled.ArrowRightAlt,
$"{T("Configuration server:")} {env.ConfigurationServerUrl}",
env.ConfigurationServerUrl,
T("Copies the server URL to the clipboard"),
"margin-top: 4px;")
])"/>
} }
<EncryptionSecretInfo IsConfigured="@(PluginFactory.EnterpriseEncryption?.IsAvailable is true)" <EncryptionSecretInfo IsConfigured="@(PluginFactory.EnterpriseEncryption?.IsAvailable is true)"
@ -130,41 +119,13 @@
{ {
<ConfigPluginInfoCard HeaderIcon="@Icons.Material.Filled.HourglassBottom" <ConfigPluginInfoCard HeaderIcon="@Icons.Material.Filled.HourglassBottom"
HeaderText="@T("Waiting for the configuration plugin...")" HeaderText="@T("Waiting for the configuration plugin...")"
Items="@([ Items="@this.BuildEnterpriseConfigurationItems(env)"/>
new ConfigInfoRowItem(Icons.Material.Filled.ArrowRightAlt,
$"{T("Enterprise configuration ID:")} {env.ConfigurationId}",
env.ConfigurationId.ToString(),
T("Copies the config ID to the clipboard")),
new ConfigInfoRowItem(Icons.Material.Filled.ArrowRightAlt,
$"{T("Configuration server:")} {env.ConfigurationServerUrl}",
env.ConfigurationServerUrl,
T("Copies the server URL to the clipboard"),
"margin-top: 4px;")
])"/>
continue; continue;
} }
<ConfigPluginInfoCard HeaderIcon="@Icons.Material.Filled.Extension" <ConfigPluginInfoCard HeaderIcon="@Icons.Material.Filled.Extension"
HeaderText="@matchingPlugin.Name" HeaderText="@matchingPlugin.Name"
Items="@([ Items="@this.BuildEnterpriseConfigurationItems(env, matchingPlugin)"
new ConfigInfoRowItem(Icons.Material.Filled.ArrowRightAlt,
$"{T("Enterprise configuration ID:")} {env.ConfigurationId}",
env.ConfigurationId.ToString(),
T("Copies the config ID to the clipboard")),
new ConfigInfoRowItem(Icons.Material.Filled.ArrowRightAlt,
$"{T("Configuration server:")} {env.ConfigurationServerUrl}",
env.ConfigurationServerUrl,
T("Copies the server URL to the clipboard"),
"margin-top: 4px;"),
new ConfigInfoRowItem(Icons.Material.Filled.ArrowRightAlt,
$"{T("Configuration plugin ID:")} {matchingPlugin.Id}",
matchingPlugin.Id.ToString(),
T("Copies the configuration plugin ID to the clipboard"),
"margin-top: 4px;")
])"
ShowWarning="@this.IsManagedConfigurationIdMismatch(matchingPlugin, env.ConfigurationId)" ShowWarning="@this.IsManagedConfigurationIdMismatch(matchingPlugin, env.ConfigurationId)"
WarningText="@T("ID mismatch: the plugin ID differs from the enterprise configuration ID.")"/> WarningText="@T("ID mismatch: the plugin ID differs from the enterprise configuration ID.")"/>
} }

View File

@ -324,6 +324,55 @@ public partial class Information : MSGComponentBase
?? this.configPlugins.FirstOrDefault(plugin => plugin.ManagedConfigurationId is null && plugin.Id == configurationId); ?? this.configPlugins.FirstOrDefault(plugin => plugin.ManagedConfigurationId is null && plugin.Id == configurationId);
} }
private IReadOnlyList<ConfigInfoRowItem> BuildEnterpriseConfigurationItems(EnterpriseEnvironment environment, IAvailablePlugin? plugin = null)
{
var items = new List<ConfigInfoRowItem>
{
new(Icons.Material.Filled.ArrowRightAlt,
$"{T("Enterprise configuration ID:")} {environment.ConfigurationId}",
environment.ConfigurationId.ToString(),
T("Copies the config ID to the clipboard")),
new(Icons.Material.Filled.ArrowRightAlt,
$"{T("Configuration server:")} {environment.ConfigurationServerUrl}",
environment.ConfigurationServerUrl,
T("Copies the server URL to the clipboard"),
"margin-top: 4px;"),
new(Icons.Material.Filled.ArrowRightAlt,
$"{T("Configuration source:")} {environment.Source}",
environment.Source,
T("Copies the configuration source to the clipboard"),
"margin-top: 4px;"),
};
if (!string.IsNullOrWhiteSpace(environment.SourceDetail))
{
items.Add(new ConfigInfoRowItem(Icons.Material.Filled.ArrowRightAlt,
$"{T("Configuration origin:")} {environment.SourceDetail}",
environment.SourceDetail,
T("Copies the configuration origin to the clipboard"),
"margin-top: 4px;"));
}
items.Add(new ConfigInfoRowItem(Icons.Material.Filled.ArrowRightAlt,
$"{T("Configuration slot:")} {environment.Slot}",
environment.Slot,
T("Copies the configuration slot to the clipboard"),
"margin-top: 4px;"));
if (plugin is not null)
{
items.Add(new ConfigInfoRowItem(Icons.Material.Filled.ArrowRightAlt,
$"{T("Configuration plugin ID:")} {plugin.Id}",
plugin.Id.ToString(),
T("Copies the configuration plugin ID to the clipboard"),
"margin-top: 4px;"));
}
return items;
}
private bool IsManagedConfigurationIdMismatch(IAvailablePlugin plugin, Guid configurationId) private bool IsManagedConfigurationIdMismatch(IAvailablePlugin plugin, Guid configurationId)
{ {
return plugin.ManagedConfigurationId == configurationId && plugin.Id != configurationId; return plugin.ManagedConfigurationId == configurationId && plugin.Id != configurationId;

View File

@ -6048,9 +6048,15 @@ UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T1137744461"] = "ID-Konflikt: Die
-- This is a private AI Studio installation. It runs without an enterprise configuration. -- This is a private AI Studio installation. It runs without an enterprise configuration.
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T1209549230"] = "Dies ist eine private AI Studio-Installation. Sie läuft ohne Unternehmenskonfiguration." UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T1209549230"] = "Dies ist eine private AI Studio-Installation. Sie läuft ohne Unternehmenskonfiguration."
-- Copies the configuration origin to the clipboard
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T125850635"] = "Kopiert den Ursprung der Konfiguration in die Zwischenablage"
-- Unknown configuration plugin -- Unknown configuration plugin
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T1290340974"] = "Unbekanntes Konfigurations-Plugin" UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T1290340974"] = "Unbekanntes Konfigurations-Plugin"
-- Copies the configuration slot to the clipboard
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T1347508205"] = "Kopiert den Slot der Konfiguration in die Zwischenablage"
-- This library is used to read PDF files. This is necessary, e.g., for using PDFs as a data source for a chat. -- This library is used to read PDF files. This is necessary, e.g., for using PDFs as a data source for a chat.
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T1388816916"] = "Diese Bibliothek wird verwendet, um PDF-Dateien zu lesen. Das ist zum Beispiel notwendig, um PDFs als Datenquelle für einen Chat zu nutzen." UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T1388816916"] = "Diese Bibliothek wird verwendet, um PDF-Dateien zu lesen. Das ist zum Beispiel notwendig, um PDFs als Datenquelle für einen Chat zu nutzen."
@ -6147,6 +6153,12 @@ UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T2371107659"] = "Installation vom
-- Installed Pandoc version: Pandoc is not installed or not available. -- Installed Pandoc version: Pandoc is not installed or not available.
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T2374031539"] = "Installierte Pandoc-Version: Pandoc ist nicht installiert oder nicht verfügbar." UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T2374031539"] = "Installierte Pandoc-Version: Pandoc ist nicht installiert oder nicht verfügbar."
-- Configuration origin:
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T2435772109"] = "Ursprung der Konfiguration:"
-- Configuration slot:
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T254943559"] = "Slot der Konfiguration:"
-- This library is used to determine the language of the operating system. This is necessary to set the language of the user interface. -- This library is used to determine the language of the operating system. This is necessary to set the language of the user interface.
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T2557014401"] = "Diese Bibliothek wird verwendet, um die Sprache des Betriebssystems zu erkennen. Dies ist notwendig, um die Sprache der Benutzeroberfläche einzustellen." UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T2557014401"] = "Diese Bibliothek wird verwendet, um die Sprache des Betriebssystems zu erkennen. Dies ist notwendig, um die Sprache der Benutzeroberfläche einzustellen."
@ -6201,6 +6213,9 @@ UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T2868174483"] = "Das .NET-Backend
-- AI Studio runs with an enterprise configuration and configuration servers. The configuration plugins are not yet available. -- AI Studio runs with an enterprise configuration and configuration servers. The configuration plugins are not yet available.
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T2924964415"] = "AI Studio wird mit Unternehmenskonfigurationen und Konfigurationsservern betrieben. Die Konfigurations-Plugins sind noch nicht verfügbar." UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T2924964415"] = "AI Studio wird mit Unternehmenskonfigurationen und Konfigurationsservern betrieben. Die Konfigurations-Plugins sind noch nicht verfügbar."
-- Copies the configuration source to the clipboard
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T2929232062"] = "Kopiert die Quelle der Konfiguration in die Zwischenablage"
-- Changelog -- Changelog
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T3017574265"] = "Änderungsprotokoll" UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T3017574265"] = "Änderungsprotokoll"
@ -6261,6 +6276,9 @@ UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T3722989559"] = "Diese Bibliothek
-- Username provided by the OS -- Username provided by the OS
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T3764549776"] = "Vom Betriebssystem bereitgestellter Benutzername" UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T3764549776"] = "Vom Betriebssystem bereitgestellter Benutzername"
-- Configuration source:
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T3801531724"] = "Quelle der Konfiguration:"
-- this version does not met the requirements -- this version does not met the requirements
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T3813932670"] = "diese Version erfüllt die Anforderungen nicht" UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T3813932670"] = "diese Version erfüllt die Anforderungen nicht"

View File

@ -6048,9 +6048,15 @@ UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T1137744461"] = "ID mismatch: the
-- This is a private AI Studio installation. It runs without an enterprise configuration. -- This is a private AI Studio installation. It runs without an enterprise configuration.
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T1209549230"] = "This is a private AI Studio installation. It runs without an enterprise configuration." UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T1209549230"] = "This is a private AI Studio installation. It runs without an enterprise configuration."
-- Copies the configuration origin to the clipboard
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T125850635"] = "Copies the configuration origin to the clipboard"
-- Unknown configuration plugin -- Unknown configuration plugin
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T1290340974"] = "Unknown configuration plugin" UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T1290340974"] = "Unknown configuration plugin"
-- Copies the configuration slot to the clipboard
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T1347508205"] = "Copies the configuration slot to the clipboard"
-- This library is used to read PDF files. This is necessary, e.g., for using PDFs as a data source for a chat. -- This library is used to read PDF files. This is necessary, e.g., for using PDFs as a data source for a chat.
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T1388816916"] = "This library is used to read PDF files. This is necessary, e.g., for using PDFs as a data source for a chat." UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T1388816916"] = "This library is used to read PDF files. This is necessary, e.g., for using PDFs as a data source for a chat."
@ -6147,6 +6153,12 @@ UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T2371107659"] = "installation pro
-- Installed Pandoc version: Pandoc is not installed or not available. -- Installed Pandoc version: Pandoc is not installed or not available.
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T2374031539"] = "Installed Pandoc version: Pandoc is not installed or not available." UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T2374031539"] = "Installed Pandoc version: Pandoc is not installed or not available."
-- Configuration origin:
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T2435772109"] = "Configuration origin:"
-- Configuration slot:
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T254943559"] = "Configuration slot:"
-- This library is used to determine the language of the operating system. This is necessary to set the language of the user interface. -- This library is used to determine the language of the operating system. This is necessary to set the language of the user interface.
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T2557014401"] = "This library is used to determine the language of the operating system. This is necessary to set the language of the user interface." UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T2557014401"] = "This library is used to determine the language of the operating system. This is necessary to set the language of the user interface."
@ -6201,6 +6213,9 @@ UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T2868174483"] = "The .NET backend
-- AI Studio runs with an enterprise configuration and configuration servers. The configuration plugins are not yet available. -- AI Studio runs with an enterprise configuration and configuration servers. The configuration plugins are not yet available.
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T2924964415"] = "AI Studio runs with an enterprise configuration and configuration servers. The configuration plugins are not yet available." UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T2924964415"] = "AI Studio runs with an enterprise configuration and configuration servers. The configuration plugins are not yet available."
-- Copies the configuration source to the clipboard
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T2929232062"] = "Copies the configuration source to the clipboard"
-- Changelog -- Changelog
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T3017574265"] = "Changelog" UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T3017574265"] = "Changelog"
@ -6261,6 +6276,9 @@ UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T3722989559"] = "This library is
-- Username provided by the OS -- Username provided by the OS
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T3764549776"] = "Username provided by the OS" UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T3764549776"] = "Username provided by the OS"
-- Configuration source:
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T3801531724"] = "Configuration source:"
-- this version does not met the requirements -- this version does not met the requirements
UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T3813932670"] = "this version does not met the requirements" UI_TEXT_CONTENT["AISTUDIO::PAGES::INFORMATION::T3813932670"] = "this version does not met the requirements"

View File

@ -2,7 +2,7 @@ using System.Net.Http.Headers;
namespace AIStudio.Tools; namespace AIStudio.Tools;
public readonly record struct EnterpriseEnvironment(string ConfigurationServerUrl, Guid ConfigurationId, EntityTagHeaderValue? ETag) public readonly record struct EnterpriseEnvironment(string ConfigurationServerUrl, Guid ConfigurationId, string Source, string SourceDetail, string Slot, EntityTagHeaderValue? ETag)
{ {
public bool IsActive => !string.IsNullOrWhiteSpace(this.ConfigurationServerUrl) && this.ConfigurationId != Guid.Empty; public bool IsActive => !string.IsNullOrWhiteSpace(this.ConfigurationServerUrl) && this.ConfigurationId != Guid.Empty;
} }

View File

@ -1,3 +1,3 @@
namespace AIStudio.Tools.Rust; namespace AIStudio.Tools.Rust;
public sealed record EnterpriseConfig(string Id, string ServerUrl); public sealed record EnterpriseConfig(string Id, string ServerUrl, string Source, string SourceDetail, string Slot);

View File

@ -14,7 +14,7 @@ public sealed class EnterpriseEnvironmentService(ILogger<EnterpriseEnvironmentSe
private static EnterpriseSecretSnapshot CURRENT_SECRET_SNAPSHOT; private static EnterpriseSecretSnapshot CURRENT_SECRET_SNAPSHOT;
private readonly record struct EnterpriseEnvironmentSnapshot(Guid ConfigurationId, string ConfigurationServerUrl, string? ETag); private readonly record struct EnterpriseEnvironmentSnapshot(Guid ConfigurationId, string ConfigurationServerUrl, string Source, string SourceDetail, string Slot, string? ETag);
private readonly record struct EnterpriseSecretSnapshot(bool HasSecret, string Fingerprint); private readonly record struct EnterpriseSecretSnapshot(bool HasSecret, string Fingerprint);
@ -224,6 +224,9 @@ public sealed class EnterpriseEnvironmentService(ILogger<EnterpriseEnvironmentSe
.Select(environment => new EnterpriseEnvironmentSnapshot( .Select(environment => new EnterpriseEnvironmentSnapshot(
environment.ConfigurationId, environment.ConfigurationId,
NormalizeServerUrl(environment.ConfigurationServerUrl), NormalizeServerUrl(environment.ConfigurationServerUrl),
environment.Source,
environment.SourceDetail,
environment.Slot,
environment.ETag?.ToString())) environment.ETag?.ToString()))
.OrderBy(environment => environment.ConfigurationId) .OrderBy(environment => environment.ConfigurationId)
.ToList(); .ToList();

View File

@ -47,7 +47,7 @@ public sealed partial class RustService
foreach (var config in configs) foreach (var config in configs)
{ {
if (Guid.TryParse(config.Id, out var id)) if (Guid.TryParse(config.Id, out var id))
environments.Add(new EnterpriseEnvironment(config.ServerUrl, id, null)); environments.Add(new EnterpriseEnvironment(config.ServerUrl, id, config.Source, config.SourceDetail, config.Slot, null));
else else
this.logger!.LogWarning($"Skipping enterprise config with invalid ID: '{config.Id}'."); this.logger!.LogWarning($"Skipping enterprise config with invalid ID: '{config.Id}'.");
} }

View File

@ -1 +1,3 @@
# v26.6.1, build 241 (2026-06-xx xx:xx UTC) # v26.6.1, build 241 (2026-06-xx xx:xx UTC)
- Added support for up to 100 thousand enterprise configuration slots, using fixed-width slot names such as `config_00000` while keeping the existing first ten slot names compatible.
- Improved the enterprise configuration details on the information page by showing where each configuration comes from and which configuration slot was used.

View File

@ -39,13 +39,15 @@ AI Studio supports loading multiple enterprise configurations simultaneously. Th
The preferred format is a fixed set of indexed pairs: The preferred format is a fixed set of indexed pairs:
- Registry values `config_id0` to `config_id9` together with `config_server_url0` to `config_server_url9` - Registry values `config_id_00000` to `config_id_99999` together with `config_server_url_00000` to `config_server_url_99999`
- Environment variables `MINDWORK_AI_STUDIO_ENTERPRISE_CONFIG_ID0` to `MINDWORK_AI_STUDIO_ENTERPRISE_CONFIG_ID9` together with `MINDWORK_AI_STUDIO_ENTERPRISE_CONFIG_SERVER_URL0` to `MINDWORK_AI_STUDIO_ENTERPRISE_CONFIG_SERVER_URL9` - Environment variables `MINDWORK_AI_STUDIO_ENTERPRISE_CONFIG_ID_00000` to `MINDWORK_AI_STUDIO_ENTERPRISE_CONFIG_ID_99999` together with `MINDWORK_AI_STUDIO_ENTERPRISE_CONFIG_SERVER_URL_00000` to `MINDWORK_AI_STUDIO_ENTERPRISE_CONFIG_SERVER_URL_99999`
- Policy files `config0.yaml` to `config9.yaml` - Policy files `config_00000.yaml` to `config_99999.yaml`
Each configuration ID must be a valid [GUID](https://en.wikipedia.org/wiki/Universally_unique_identifier#Globally_unique_identifier). Up to ten configurations are supported per device. Each configuration ID must be a valid [GUID](https://en.wikipedia.org/wiki/Universally_unique_identifier#Globally_unique_identifier). Up to 100,000 indexed configuration slots are supported per device.
If multiple configurations define the same setting, the first definition wins. For indexed pairs and policy files, the order is slot `0`, then `1`, and so on up to `9`. If multiple configurations define the same setting, the first definition wins. For indexed pairs and policy files, the order is slot `00000`, then `00001`, and so on up to `99999`.
For backwards compatibility, the older slot names `0` to `9` without an underscore are still supported. AI Studio also accepts other numeric slot suffixes with up to five digits. Slot suffixes are matched exactly, so `config_id_1`, `config_id_01`, and `config_id_00001` are treated as separate slots. Use the five-digit format with an underscore for new deployments.
### Windows registry example ### Windows registry example
@ -55,10 +57,10 @@ The Windows registry path is:
Example values: Example values:
- `config_id0` = `9072b77d-ca81-40da-be6a-861da525ef7b` - `config_id_00000` = `9072b77d-ca81-40da-be6a-861da525ef7b`
- `config_server_url0` = `https://intranet.example.org/ai-studio/configuration` - `config_server_url_00000` = `https://intranet.example.org/ai-studio/configuration`
- `config_id1` = `a1b2c3d4-e5f6-7890-abcd-ef1234567890` - `config_id_10503` = `a1b2c3d4-e5f6-7890-abcd-ef1234567890`
- `config_server_url1` = `https://intranet.example.org/ai-studio/department-config` - `config_server_url_10503` = `https://intranet.example.org/ai-studio/department-config`
- `config_encryption_secret` = `BASE64...` - `config_encryption_secret` = `BASE64...`
This approach works well with GPOs because each slot can be managed independently without rewriting a shared combined string. This approach works well with GPOs because each slot can be managed independently without rewriting a shared combined string.
@ -85,10 +87,10 @@ The directories from `$XDG_CONFIG_DIRS` are processed in order.
Configuration files: Configuration files:
- `config0.yaml` - `config_00000.yaml`
- `config1.yaml` - `config_00001.yaml`
- ... - ...
- `config9.yaml` - `config_99999.yaml`
Each configuration file contains one configuration ID and one server URL: Each configuration file contains one configuration ID and one server URL:
@ -110,10 +112,10 @@ config_encryption_secret: "BASE64..."
If you need the fallback environment-variable format, configure the values like this: If you need the fallback environment-variable format, configure the values like this:
```bash ```bash
MINDWORK_AI_STUDIO_ENTERPRISE_CONFIG_ID0=9072b77d-ca81-40da-be6a-861da525ef7b MINDWORK_AI_STUDIO_ENTERPRISE_CONFIG_ID_00000=9072b77d-ca81-40da-be6a-861da525ef7b
MINDWORK_AI_STUDIO_ENTERPRISE_CONFIG_SERVER_URL0=https://intranet.example.org/ai-studio/configuration MINDWORK_AI_STUDIO_ENTERPRISE_CONFIG_SERVER_URL_00000=https://intranet.example.org/ai-studio/configuration
MINDWORK_AI_STUDIO_ENTERPRISE_CONFIG_ID1=a1b2c3d4-e5f6-7890-abcd-ef1234567890 MINDWORK_AI_STUDIO_ENTERPRISE_CONFIG_ID_10503=a1b2c3d4-e5f6-7890-abcd-ef1234567890
MINDWORK_AI_STUDIO_ENTERPRISE_CONFIG_SERVER_URL1=https://intranet.example.org/ai-studio/department-config MINDWORK_AI_STUDIO_ENTERPRISE_CONFIG_SERVER_URL_10503=https://intranet.example.org/ai-studio/department-config
MINDWORK_AI_STUDIO_ENTERPRISE_CONFIG_ENCRYPTION_SECRET=BASE64... MINDWORK_AI_STUDIO_ENTERPRISE_CONFIG_ENCRYPTION_SECRET=BASE64...
``` ```

View File

@ -11,13 +11,22 @@ use sys_locale::get_locale;
const DEFAULT_LANGUAGE: &str = "en-US"; const DEFAULT_LANGUAGE: &str = "en-US";
const ENTERPRISE_CONFIG_SLOT_COUNT: usize = 10; const ENTERPRISE_CONFIG_SLOT_MAX: u32 = 99_999;
const ENTERPRISE_CONFIG_SLOT_WIDTH: usize = 5;
const ENTERPRISE_CONFIG_ID_KEY_PREFIX: &str = "config_id";
const ENTERPRISE_CONFIG_SERVER_URL_KEY_PREFIX: &str = "config_server_url";
#[cfg(target_os = "windows")] #[cfg(target_os = "windows")]
const ENTERPRISE_REGISTRY_KEY_PATH: &str = r"Software\github\MindWork AI Studio\Enterprise IT"; const ENTERPRISE_REGISTRY_KEY_PATH: &str = r"Software\github\MindWork AI Studio\Enterprise IT";
const ENTERPRISE_POLICY_SECRET_FILE_NAME: &str = "config_encryption_secret.yaml"; const ENTERPRISE_POLICY_SECRET_FILE_NAME: &str = "config_encryption_secret.yaml";
const ENTERPRISE_ENV_CONFIG_ID_PREFIX: &str = "MINDWORK_AI_STUDIO_ENTERPRISE_CONFIG_ID";
const ENTERPRISE_ENV_CONFIG_SERVER_URL_PREFIX: &str = "MINDWORK_AI_STUDIO_ENTERPRISE_CONFIG_SERVER_URL";
const ENTERPRISE_ENV_CONFIGS: &str = "MINDWORK_AI_STUDIO_ENTERPRISE_CONFIGS";
const ENTERPRISE_ENV_CONFIG_ENCRYPTION_SECRET: &str = "MINDWORK_AI_STUDIO_ENTERPRISE_CONFIG_ENCRYPTION_SECRET";
/// The data directory where the application stores its data. /// The data directory where the application stores its data.
pub static DATA_DIRECTORY: OnceLock<String> = OnceLock::new(); pub static DATA_DIRECTORY: OnceLock<String> = OnceLock::new();
@ -187,8 +196,53 @@ pub async fn read_user_language(_token: APIToken) -> String {
pub struct EnterpriseConfig { pub struct EnterpriseConfig {
pub id: String, pub id: String,
pub server_url: String, pub server_url: String,
pub source: String,
pub source_detail: String,
pub slot: String,
} }
#[derive(Clone, Debug, PartialEq, Eq)]
struct EnterpriseSourceValue {
value: String,
source_detail: String,
}
impl EnterpriseSourceValue {
fn new(value: String, source_detail: String) -> Self {
Self {
value,
source_detail,
}
}
}
trait EnterpriseSourceValueAccess {
fn value(&self) -> &str;
fn source_detail(&self) -> &str;
}
impl EnterpriseSourceValueAccess for EnterpriseSourceValue {
fn value(&self) -> &str {
&self.value
}
fn source_detail(&self) -> &str {
&self.source_detail
}
}
impl EnterpriseSourceValueAccess for String {
fn value(&self) -> &str {
self
}
fn source_detail(&self) -> &str {
""
}
}
type EnterpriseSourceValues = HashMap<String, EnterpriseSourceValue>;
#[derive(Clone, Debug, Default, PartialEq, Eq)] #[derive(Clone, Debug, Default, PartialEq, Eq)]
struct EnterpriseSourceData { struct EnterpriseSourceData {
source_name: String, source_name: String,
@ -292,7 +346,7 @@ fn load_registry_enterprise_source() -> EnterpriseSourceData {
info!(r"Trying to read enterprise configuration metadata from 'HKEY_CURRENT_USER\{}'.", ENTERPRISE_REGISTRY_KEY_PATH); info!(r"Trying to read enterprise configuration metadata from 'HKEY_CURRENT_USER\{}'.", ENTERPRISE_REGISTRY_KEY_PATH);
let mut values = HashMap::new(); let mut values = EnterpriseSourceValues::new();
let key = match CURRENT_USER.open(ENTERPRISE_REGISTRY_KEY_PATH) { let key = match CURRENT_USER.open(ENTERPRISE_REGISTRY_KEY_PATH) {
Ok(key) => key, Ok(key) => key,
Err(_) => { Err(_) => {
@ -304,32 +358,40 @@ fn load_registry_enterprise_source() -> EnterpriseSourceData {
} }
}; };
for index in 0..ENTERPRISE_CONFIG_SLOT_COUNT { match key.values() {
insert_registry_value(&mut values, &key, &format!("config_id{index}")); Ok(registry_values) => {
insert_registry_value(&mut values, &key, &format!("config_server_url{index}")); for (key_name, value) in registry_values {
} let Some(source_key_name) = enterprise_registry_value_key_name(&key_name) else {
continue;
};
for key_name in [ match String::try_from(value) {
"configs", Ok(value) => {
"config_id", values.insert(source_key_name, EnterpriseSourceValue::new(value, String::new()));
"config_server_url", },
"config_encryption_secret",
] { Err(error) => {
insert_registry_value(&mut values, &key, key_name); warn!(r"Could not read enterprise registry value 'HKEY_CURRENT_USER\{}\{}' as string: {}.", ENTERPRISE_REGISTRY_KEY_PATH, key_name, error);
},
}
}
},
Err(error) => {
warn!(r"Could not enumerate enterprise registry values from 'HKEY_CURRENT_USER\{}': {}.", ENTERPRISE_REGISTRY_KEY_PATH, error);
},
} }
parse_enterprise_source_values("Windows registry", &values) parse_enterprise_source_values("Windows registry", &values)
} }
#[cfg(target_os = "windows")] #[cfg(target_os = "windows")]
fn insert_registry_value( fn enterprise_registry_value_key_name(key_name: &str) -> Option<String> {
values: &mut HashMap<String, String>, if is_legacy_enterprise_source_key(key_name) {
key: &windows_registry::Key, return Some(String::from(key_name));
key_name: &str,
) {
if let Ok(value) = key.get_string(key_name) {
values.insert(String::from(key_name), value);
} }
enterprise_indexed_source_key_name(key_name)
} }
fn load_policy_file_enterprise_source() -> EnterpriseSourceData { fn load_policy_file_enterprise_source() -> EnterpriseSourceData {
@ -342,26 +404,85 @@ fn load_policy_file_enterprise_source() -> EnterpriseSourceData {
fn load_environment_enterprise_source() -> EnterpriseSourceData { fn load_environment_enterprise_source() -> EnterpriseSourceData {
info!("Trying to read enterprise configuration metadata from environment variables."); info!("Trying to read enterprise configuration metadata from environment variables.");
let mut values = HashMap::new(); let mut values = EnterpriseSourceValues::new();
for index in 0..ENTERPRISE_CONFIG_SLOT_COUNT { for (env_name, value) in env::vars() {
insert_env_value(&mut values, &format!("MINDWORK_AI_STUDIO_ENTERPRISE_CONFIG_ID{index}"), &format!("config_id{index}")); if let Some(source_key_name) = enterprise_environment_key_name(&env_name) {
insert_env_value(&mut values, &format!("MINDWORK_AI_STUDIO_ENTERPRISE_CONFIG_SERVER_URL{index}"), &format!("config_server_url{index}")); let source_detail = enterprise_environment_source_detail(&source_key_name, &env_name);
values.insert(source_key_name, EnterpriseSourceValue::new(value, source_detail));
}
} }
insert_env_value(&mut values, "MINDWORK_AI_STUDIO_ENTERPRISE_CONFIGS", "configs");
insert_env_value(&mut values, "MINDWORK_AI_STUDIO_ENTERPRISE_CONFIG_ID", "config_id");
insert_env_value(&mut values, "MINDWORK_AI_STUDIO_ENTERPRISE_CONFIG_SERVER_URL", "config_server_url");
insert_env_value(&mut values, "MINDWORK_AI_STUDIO_ENTERPRISE_CONFIG_ENCRYPTION_SECRET", "config_encryption_secret");
parse_enterprise_source_values("environment variables", &values) parse_enterprise_source_values("environment variables", &values)
} }
fn insert_env_value(values: &mut HashMap<String, String>, env_name: &str, key_name: &str) { fn enterprise_environment_source_detail(source_key_name: &str, env_name: &str) -> String {
if let Ok(value) = env::var(env_name) { if source_key_name == "config_id"
values.insert(String::from(key_name), value); || enterprise_source_key_suffix(source_key_name, ENTERPRISE_CONFIG_ID_KEY_PREFIX).is_some() {
String::from(env_name)
} else {
String::new()
} }
} }
fn enterprise_environment_key_name(env_name: &str) -> Option<String> {
if enterprise_env_key_equals(env_name, ENTERPRISE_ENV_CONFIGS) {
return Some(String::from("configs"));
}
if enterprise_env_key_equals(env_name, ENTERPRISE_ENV_CONFIG_ID_PREFIX) {
return Some(String::from("config_id"));
}
if enterprise_env_key_equals(env_name, ENTERPRISE_ENV_CONFIG_SERVER_URL_PREFIX) {
return Some(String::from("config_server_url"));
}
if enterprise_env_key_equals(env_name, ENTERPRISE_ENV_CONFIG_ENCRYPTION_SECRET) {
return Some(String::from("config_encryption_secret"));
}
if let Some(suffix) = enterprise_env_key_suffix(env_name, ENTERPRISE_ENV_CONFIG_ID_PREFIX) {
return Some(format!("config_id{suffix}"));
}
if let Some(suffix) = enterprise_env_key_suffix(env_name, ENTERPRISE_ENV_CONFIG_SERVER_URL_PREFIX) {
return Some(format!("config_server_url{suffix}"));
}
None
}
#[cfg(target_os = "windows")]
fn enterprise_env_key_equals(env_name: &str, expected: &str) -> bool {
env_name.eq_ignore_ascii_case(expected)
}
#[cfg(not(target_os = "windows"))]
fn enterprise_env_key_equals(env_name: &str, expected: &str) -> bool {
env_name == expected
}
#[cfg(target_os = "windows")]
fn enterprise_env_key_suffix<'a>(env_name: &'a str, prefix: &str) -> Option<&'a str> {
if env_name.len() < prefix.len() {
return None;
}
let (raw_prefix, suffix) = env_name.split_at(prefix.len());
if raw_prefix.eq_ignore_ascii_case(prefix) {
normalize_enterprise_slot_suffix(suffix)
} else {
None
}
}
#[cfg(not(target_os = "windows"))]
fn enterprise_env_key_suffix<'a>(env_name: &'a str, prefix: &str) -> Option<&'a str> {
env_name
.strip_prefix(prefix)
.and_then(normalize_enterprise_slot_suffix)
}
#[cfg(target_os = "windows")] #[cfg(target_os = "windows")]
fn enterprise_policy_directories() -> Vec<PathBuf> { fn enterprise_policy_directories() -> Vec<PathBuf> {
let base = env::var_os("ProgramData") let base = env::var_os("ProgramData")
@ -406,19 +527,49 @@ fn linux_policy_directories_from_xdg(xdg_config_dirs: Option<&str>) -> Vec<PathB
directories directories
} }
fn load_policy_values_from_directories(directories: &[PathBuf]) -> HashMap<String, String> { fn load_policy_values_from_directories(directories: &[PathBuf]) -> EnterpriseSourceValues {
let mut values = HashMap::new(); let mut values = EnterpriseSourceValues::new();
for directory in directories { for directory in directories {
info!("Checking enterprise policy directory '{}'.", directory.display()); info!("Checking enterprise policy directory '{}'.", directory.display());
for index in 0..ENTERPRISE_CONFIG_SLOT_COUNT { let entries = match fs::read_dir(directory) {
let path = directory.join(format!("config{index}.yaml")); Ok(entries) => entries,
Err(error) => {
info!("Could not enumerate enterprise policy directory '{}': {}.", directory.display(), error);
continue;
},
};
for entry in entries {
let entry = match entry {
Ok(entry) => entry,
Err(error) => {
warn!("Could not read an entry from enterprise policy directory '{}': {}.", directory.display(), error);
continue;
},
};
let file_name = entry.file_name();
let Some(file_name) = file_name.to_str() else {
continue;
};
let Some(suffix) = enterprise_policy_file_slot_suffix(file_name) else {
continue;
};
let path = entry.path();
if let Some(config_values) = read_policy_yaml_mapping(&path) { if let Some(config_values) = read_policy_yaml_mapping(&path) {
let source_detail = path
.canonicalize()
.unwrap_or_else(|_| path.clone())
.to_string_lossy()
.into_owned();
if let Some(id) = config_values.get("id") { if let Some(id) = config_values.get("id") {
insert_first_non_empty_value(&mut values, &format!("config_id{index}"), id); insert_first_non_empty_value(&mut values, &format!("config_id{suffix}"), id, &source_detail);
} }
if let Some(server_url) = config_values.get("server_url") { if let Some(server_url) = config_values.get("server_url") {
insert_first_non_empty_value(&mut values, &format!("config_server_url{index}"), server_url); insert_first_non_empty_value(&mut values, &format!("config_server_url{suffix}"), server_url, &source_detail);
} }
} }
} }
@ -426,13 +577,21 @@ fn load_policy_values_from_directories(directories: &[PathBuf]) -> HashMap<Strin
let secret_path = directory.join(ENTERPRISE_POLICY_SECRET_FILE_NAME); let secret_path = directory.join(ENTERPRISE_POLICY_SECRET_FILE_NAME);
if let Some(secret_values) = read_policy_yaml_mapping(&secret_path) if let Some(secret_values) = read_policy_yaml_mapping(&secret_path)
&& let Some(secret) = secret_values.get("config_encryption_secret") { && let Some(secret) = secret_values.get("config_encryption_secret") {
insert_first_non_empty_value(&mut values, "config_encryption_secret", secret); insert_first_non_empty_value(&mut values, "config_encryption_secret", secret, "");
} }
} }
values values
} }
fn enterprise_policy_file_slot_suffix(file_name: &str) -> Option<&str> {
let suffix = file_name
.strip_prefix("config")?
.strip_suffix(".yaml")?;
normalize_enterprise_slot_suffix(suffix)
}
fn read_policy_yaml_mapping(path: &Path) -> Option<HashMap<String, String>> { fn read_policy_yaml_mapping(path: &Path) -> Option<HashMap<String, String>> {
if !path.exists() { if !path.exists() {
return None; return None;
@ -516,27 +675,118 @@ fn parse_policy_yaml_value(raw_value: &str) -> Option<String> {
Some(String::from(trimmed)) Some(String::from(trimmed))
} }
fn insert_first_non_empty_value(values: &mut HashMap<String, String>, key: &str, raw_value: &str) { fn insert_first_non_empty_value(values: &mut EnterpriseSourceValues, key: &str, raw_value: &str, source_detail: &str) {
if let Some(value) = normalize_enterprise_value(raw_value) { if let Some(value) = normalize_enterprise_value(raw_value) {
values.entry(String::from(key)).or_insert(value); values
.entry(String::from(key))
.or_insert_with(|| EnterpriseSourceValue::new(value, String::from(source_detail)));
} }
} }
fn parse_enterprise_source_values( #[cfg(target_os = "windows")]
fn is_legacy_enterprise_source_key(key_name: &str) -> bool {
matches!(
key_name,
"configs" | "config_id" | "config_server_url" | "config_encryption_secret"
)
}
#[cfg(target_os = "windows")]
fn enterprise_indexed_source_key_name(key_name: &str) -> Option<String> {
if let Some(suffix) = enterprise_source_key_suffix(key_name, ENTERPRISE_CONFIG_ID_KEY_PREFIX) {
return Some(format!("config_id{suffix}"));
}
if let Some(suffix) = enterprise_source_key_suffix(key_name, ENTERPRISE_CONFIG_SERVER_URL_KEY_PREFIX) {
return Some(format!("config_server_url{suffix}"));
}
None
}
fn enterprise_source_key_suffix<'a>(key_name: &'a str, prefix: &str) -> Option<&'a str> {
key_name
.strip_prefix(prefix)
.and_then(normalize_enterprise_slot_suffix)
}
fn normalize_enterprise_slot_suffix(raw_suffix: &str) -> Option<&str> {
let suffix = raw_suffix.strip_prefix('_').unwrap_or(raw_suffix);
if is_enterprise_slot_suffix(suffix) {
Some(suffix)
} else {
None
}
}
fn is_enterprise_slot_suffix(suffix: &str) -> bool {
!suffix.is_empty()
&& suffix.len() <= ENTERPRISE_CONFIG_SLOT_WIDTH
&& suffix.chars().all(|c| c.is_ascii_digit())
&& suffix.parse::<u32>().is_ok_and(|index| index <= ENTERPRISE_CONFIG_SLOT_MAX)
}
fn collect_enterprise_config_slots<T: EnterpriseSourceValueAccess>(values: &HashMap<String, T>) -> Vec<String> {
let mut slots = HashSet::new();
for key_name in values.keys() {
if let Some(suffix) = enterprise_source_key_suffix(key_name, ENTERPRISE_CONFIG_ID_KEY_PREFIX)
&& is_enterprise_slot_suffix(suffix) {
slots.insert(String::from(suffix));
continue;
}
if let Some(suffix) = enterprise_source_key_suffix(key_name, ENTERPRISE_CONFIG_SERVER_URL_KEY_PREFIX)
&& is_enterprise_slot_suffix(suffix) {
slots.insert(String::from(suffix));
}
}
let mut slots: Vec<String> = slots.into_iter().collect();
slots.sort_by(|left, right| {
let left_index = left.parse::<u32>().unwrap_or(ENTERPRISE_CONFIG_SLOT_MAX);
let right_index = right.parse::<u32>().unwrap_or(ENTERPRISE_CONFIG_SLOT_MAX);
left_index
.cmp(&right_index)
.then_with(|| enterprise_slot_width_rank(left).cmp(&enterprise_slot_width_rank(right)))
.then_with(|| left.len().cmp(&right.len()))
.then_with(|| left.cmp(right))
});
slots
}
fn enterprise_slot_width_rank(suffix: &str) -> u8 {
if suffix.len() == ENTERPRISE_CONFIG_SLOT_WIDTH {
0
} else {
1
}
}
fn indexed_enterprise_source_value<'a, T: EnterpriseSourceValueAccess>(
values: &'a HashMap<String, T>,
prefix: &str,
suffix: &str,
) -> Option<&'a T> {
let separated_key = format!("{prefix}_{suffix}");
values
.get(&separated_key)
.or_else(|| values.get(&format!("{prefix}{suffix}")))
}
fn parse_enterprise_source_values<T: EnterpriseSourceValueAccess>(
source_name: &str, source_name: &str,
values: &HashMap<String, String>, values: &HashMap<String, T>,
) -> EnterpriseSourceData { ) -> EnterpriseSourceData {
let mut configs = Vec::new(); let mut configs = Vec::new();
let mut seen_ids = HashSet::new(); let mut seen_ids = HashSet::new();
for index in 0..ENTERPRISE_CONFIG_SLOT_COUNT { for suffix in collect_enterprise_config_slots(values) {
let id_key = format!("config_id{index}");
let server_url_key = format!("config_server_url{index}");
add_enterprise_config_pair( add_enterprise_config_pair(
source_name, source_name,
&format!("indexed slot {index}"), &format!("indexed slot {suffix}"),
values.get(&id_key).map(String::as_str), indexed_enterprise_source_value(values, ENTERPRISE_CONFIG_ID_KEY_PREFIX, &suffix),
values.get(&server_url_key).map(String::as_str), indexed_enterprise_source_value(values, ENTERPRISE_CONFIG_SERVER_URL_KEY_PREFIX, &suffix),
&mut configs, &mut configs,
&mut seen_ids, &mut seen_ids,
); );
@ -544,7 +794,7 @@ fn parse_enterprise_source_values(
if let Some(combined) = values if let Some(combined) = values
.get("configs") .get("configs")
.and_then(|value| normalize_enterprise_value(value)) .and_then(|value| normalize_enterprise_value(value.value()))
{ {
add_combined_enterprise_configs(source_name, &combined, &mut configs, &mut seen_ids); add_combined_enterprise_configs(source_name, &combined, &mut configs, &mut seen_ids);
} }
@ -552,15 +802,15 @@ fn parse_enterprise_source_values(
add_enterprise_config_pair( add_enterprise_config_pair(
source_name, source_name,
"legacy single configuration", "legacy single configuration",
values.get("config_id").map(String::as_str), values.get("config_id"),
values.get("config_server_url").map(String::as_str), values.get("config_server_url"),
&mut configs, &mut configs,
&mut seen_ids, &mut seen_ids,
); );
let encryption_secret = values let encryption_secret = values
.get("config_encryption_secret") .get("config_encryption_secret")
.and_then(|value| normalize_enterprise_value(value)) .and_then(|value| normalize_enterprise_value(value.value()))
.unwrap_or_default(); .unwrap_or_default();
EnterpriseSourceData { EnterpriseSourceData {
@ -572,26 +822,32 @@ fn parse_enterprise_source_values(
fn add_enterprise_config_pair( fn add_enterprise_config_pair(
source_name: &str, source_name: &str,
context: &str, slot: &str,
raw_id: Option<&str>, raw_id: Option<&impl EnterpriseSourceValueAccess>,
raw_server_url: Option<&str>, raw_server_url: Option<&impl EnterpriseSourceValueAccess>,
configs: &mut Vec<EnterpriseConfig>, configs: &mut Vec<EnterpriseConfig>,
seen_ids: &mut HashSet<String>, seen_ids: &mut HashSet<String>,
) { ) {
let id = raw_id.and_then(normalize_enterprise_config_id); let id = raw_id.and_then(|value| normalize_enterprise_config_id(value.value()));
let server_url = raw_server_url.and_then(normalize_enterprise_value); let server_url = raw_server_url.and_then(|value| normalize_enterprise_value(value.value()));
match (id, server_url) { match (id, server_url) {
(Some(id), Some(server_url)) => { (Some(id), Some(server_url)) => {
if seen_ids.insert(id.clone()) { if seen_ids.insert(id.clone()) {
configs.push(EnterpriseConfig { id, server_url }); configs.push(EnterpriseConfig {
id,
server_url,
source: String::from(source_name),
source_detail: raw_id.map(|value| String::from(value.source_detail())).unwrap_or_default(),
slot: String::from(slot),
});
} else { } else {
info!("Ignoring duplicate enterprise configuration '{}' from {} in '{}'.", id, source_name, context); info!("Ignoring duplicate enterprise configuration '{}' from {} in '{}'.", id, source_name, slot);
} }
} }
(Some(_), None) | (None, Some(_)) => { (Some(_), None) | (None, Some(_)) => {
warn!("Ignoring incomplete enterprise configuration from {} in '{}'.", source_name, context); warn!("Ignoring incomplete enterprise configuration from {} in '{}'.", source_name, slot);
} }
(None, None) => {} (None, None) => {}
@ -615,11 +871,13 @@ fn add_combined_enterprise_configs(
continue; continue;
}; };
let id = EnterpriseSourceValue::new(String::from(raw_id), String::new());
let server_url = EnterpriseSourceValue::new(String::from(raw_server_url), String::new());
add_enterprise_config_pair( add_enterprise_config_pair(
source_name, source_name,
&format!("combined legacy entry {}", index + 1), &format!("combined legacy entry {}", index + 1),
Some(raw_id), Some(&id),
Some(raw_server_url), Some(&server_url),
configs, configs,
seen_ids, seen_ids,
); );
@ -642,10 +900,11 @@ fn normalize_enterprise_config_id(value: &str) -> Option<String> {
#[cfg(test)] #[cfg(test)]
mod tests { mod tests {
use super::{ use super::{
enterprise_environment_key_name, enterprise_policy_file_slot_suffix,
linux_policy_directories_from_xdg, load_policy_values_from_directories, linux_policy_directories_from_xdg, load_policy_values_from_directories,
normalize_locale_tag, parse_enterprise_source_values, normalize_locale_tag, parse_enterprise_source_values,
select_effective_enterprise_config_source, select_effective_enterprise_secret_source, select_effective_enterprise_config_source, select_effective_enterprise_secret_source,
EnterpriseConfig, EnterpriseSourceData, EnterpriseConfig, EnterpriseSourceData, EnterpriseSourceValue, EnterpriseSourceValues,
}; };
use std::collections::HashMap; use std::collections::HashMap;
use std::fs; use std::fs;
@ -656,6 +915,30 @@ mod tests {
const TEST_ID_B: &str = "a1b2c3d4-e5f6-7890-abcd-ef1234567890"; const TEST_ID_B: &str = "a1b2c3d4-e5f6-7890-abcd-ef1234567890";
const TEST_ID_C: &str = "11111111-2222-3333-4444-555555555555"; const TEST_ID_C: &str = "11111111-2222-3333-4444-555555555555";
fn enterprise_config(
id: &str,
server_url: &str,
source: &str,
source_detail: &str,
slot: &str,
) -> EnterpriseConfig {
EnterpriseConfig {
id: String::from(id),
server_url: String::from(server_url),
source: String::from(source),
source_detail: String::from(source_detail),
slot: String::from(slot),
}
}
fn policy_path(path: PathBuf) -> String {
path
.canonicalize()
.unwrap_or(path)
.to_string_lossy()
.into_owned()
}
#[test] #[test]
fn normalize_locale_tag_supports_common_linux_formats() { fn normalize_locale_tag_supports_common_linux_formats() {
assert_eq!( assert_eq!(
@ -707,18 +990,9 @@ mod tests {
assert_eq!( assert_eq!(
source.configs, source.configs,
vec![ vec![
EnterpriseConfig { enterprise_config("9072b77d-ca81-40da-be6a-861da525ef7b", "https://indexed.example.org", "test", "", "indexed slot 0"),
id: String::from("9072b77d-ca81-40da-be6a-861da525ef7b"), enterprise_config(TEST_ID_B, "https://combined.example.org", "test", "", "combined legacy entry 2"),
server_url: String::from("https://indexed.example.org"), enterprise_config(TEST_ID_C, "https://legacy.example.org", "test", "", "legacy single configuration"),
},
EnterpriseConfig {
id: String::from(TEST_ID_B),
server_url: String::from("https://combined.example.org"),
},
EnterpriseConfig {
id: String::from(TEST_ID_C),
server_url: String::from("https://legacy.example.org"),
},
] ]
); );
assert_eq!(source.encryption_secret, "secret"); assert_eq!(source.encryption_secret, "secret");
@ -743,35 +1017,164 @@ mod tests {
assert_eq!( assert_eq!(
source.configs, source.configs,
vec![ vec![
EnterpriseConfig { enterprise_config("9072b77d-ca81-40da-be6a-861da525ef7b", "https://slot0.example.org", "test", "", "indexed slot 0"),
id: String::from("9072b77d-ca81-40da-be6a-861da525ef7b"), enterprise_config(TEST_ID_B, "https://slot4.example.org", "test", "", "indexed slot 4"),
server_url: String::from("https://slot0.example.org"),
},
EnterpriseConfig {
id: String::from(TEST_ID_B),
server_url: String::from("https://slot4.example.org"),
},
] ]
); );
} }
#[test]
fn parse_enterprise_source_values_supports_padded_and_high_indexed_slots() {
let mut values = HashMap::new();
values.insert(String::from("config_id_00000"), String::from(TEST_ID_A));
values.insert(
String::from("config_server_url_00000"),
String::from("https://slot0.example.org"),
);
values.insert(String::from("config_id_10503"), String::from(TEST_ID_B));
values.insert(
String::from("config_server_url_10503"),
String::from("https://slot10503.example.org"),
);
let source = parse_enterprise_source_values("test", &values);
assert_eq!(
source.configs,
vec![
enterprise_config("9072b77d-ca81-40da-be6a-861da525ef7b", "https://slot0.example.org", "test", "", "indexed slot 00000"),
enterprise_config(TEST_ID_B, "https://slot10503.example.org", "test", "", "indexed slot 10503"),
]
);
}
#[test]
fn parse_enterprise_source_values_treats_slot_widths_as_distinct_slots() {
let mut values = HashMap::new();
values.insert(String::from("config_id_00001"), String::from(TEST_ID_A));
values.insert(
String::from("config_server_url_00001"),
String::from("https://padded.example.org"),
);
values.insert(String::from("config_id1"), String::from(TEST_ID_B));
values.insert(
String::from("config_server_url1"),
String::from("https://legacy-slot.example.org"),
);
let source = parse_enterprise_source_values("test", &values);
assert_eq!(
source.configs,
vec![
enterprise_config("9072b77d-ca81-40da-be6a-861da525ef7b", "https://padded.example.org", "test", "", "indexed slot 00001"),
enterprise_config(TEST_ID_B, "https://legacy-slot.example.org", "test", "", "indexed slot 1"),
]
);
}
#[test]
fn parse_enterprise_source_values_ignores_invalid_slot_suffixes() {
let mut values = HashMap::new();
values.insert(String::from("config_id_99999"), String::from(TEST_ID_A));
values.insert(
String::from("config_server_url_99999"),
String::from("https://valid.example.org"),
);
values.insert(String::from("config_id_100000"), String::from(TEST_ID_B));
values.insert(
String::from("config_server_url_100000"),
String::from("https://too-high.example.org"),
);
values.insert(String::from("config_id_abc"), String::from(TEST_ID_C));
values.insert(
String::from("config_server_url_abc"),
String::from("https://letters.example.org"),
);
let source = parse_enterprise_source_values("test", &values);
assert_eq!(
source.configs,
vec![enterprise_config("9072b77d-ca81-40da-be6a-861da525ef7b", "https://valid.example.org", "test", "", "indexed slot 99999")]
);
}
#[test]
fn enterprise_environment_key_name_maps_indexed_and_legacy_names() {
assert_eq!(
enterprise_environment_key_name("MINDWORK_AI_STUDIO_ENTERPRISE_CONFIG_ID_10503"),
Some(String::from("config_id10503"))
);
assert_eq!(
enterprise_environment_key_name("MINDWORK_AI_STUDIO_ENTERPRISE_CONFIG_SERVER_URL_00000"),
Some(String::from("config_server_url00000"))
);
assert_eq!(
enterprise_environment_key_name("MINDWORK_AI_STUDIO_ENTERPRISE_CONFIGS"),
Some(String::from("configs"))
);
assert_eq!(
enterprise_environment_key_name("MINDWORK_AI_STUDIO_ENTERPRISE_CONFIG_ID_100000"),
None
);
}
#[test]
fn parse_enterprise_source_values_keeps_environment_id_variable_as_source_detail() {
let mut values = EnterpriseSourceValues::new();
values.insert(
String::from("config_id00000"),
EnterpriseSourceValue::new(
String::from(TEST_ID_A),
String::from("MINDWORK_AI_STUDIO_ENTERPRISE_CONFIG_ID_00000"),
),
);
values.insert(
String::from("config_server_url00000"),
EnterpriseSourceValue::new(String::from("https://env.example.org"), String::new()),
);
let source = parse_enterprise_source_values("environment variables", &values);
assert_eq!(
source.configs,
vec![enterprise_config(
"9072b77d-ca81-40da-be6a-861da525ef7b",
"https://env.example.org",
"environment variables",
"MINDWORK_AI_STUDIO_ENTERPRISE_CONFIG_ID_00000",
"indexed slot 00000"
)]
);
}
#[test]
fn enterprise_policy_file_slot_suffix_accepts_valid_slot_file_names() {
assert_eq!(enterprise_policy_file_slot_suffix("config0.yaml"), Some("0"));
assert_eq!(
enterprise_policy_file_slot_suffix("config_00000.yaml"),
Some("00000")
);
assert_eq!(
enterprise_policy_file_slot_suffix("config_10503.yaml"),
Some("10503")
);
assert_eq!(enterprise_policy_file_slot_suffix("config_100000.yaml"), None);
assert_eq!(enterprise_policy_file_slot_suffix("config_abc.yaml"), None);
}
#[test] #[test]
fn select_effective_enterprise_config_source_uses_first_source_with_configs_only() { fn select_effective_enterprise_config_source_uses_first_source_with_configs_only() {
let selected = select_effective_enterprise_config_source(vec![ let selected = select_effective_enterprise_config_source(vec![
EnterpriseSourceData { EnterpriseSourceData {
source_name: String::from("registry"), source_name: String::from("registry"),
configs: vec![EnterpriseConfig { configs: vec![enterprise_config(&TEST_ID_A.to_lowercase(), "https://registry.example.org", "registry", "", "indexed slot 0")],
id: TEST_ID_A.to_lowercase(),
server_url: String::from("https://registry.example.org"),
}],
encryption_secret: String::new(), encryption_secret: String::new(),
}, },
EnterpriseSourceData { EnterpriseSourceData {
source_name: String::from("environment"), source_name: String::from("environment"),
configs: vec![EnterpriseConfig { configs: vec![enterprise_config(TEST_ID_B, "https://env.example.org", "environment", "", "indexed slot 0")],
id: String::from(TEST_ID_B),
server_url: String::from("https://env.example.org"),
}],
encryption_secret: String::from("ENV-SECRET"), encryption_secret: String::from("ENV-SECRET"),
}, },
]); ]);
@ -791,10 +1194,7 @@ mod tests {
}, },
EnterpriseSourceData { EnterpriseSourceData {
source_name: String::from("environment"), source_name: String::from("environment"),
configs: vec![EnterpriseConfig { configs: vec![enterprise_config(TEST_ID_B, "https://env.example.org", "environment", "", "indexed slot 0")],
id: String::from(TEST_ID_B),
server_url: String::from("https://env.example.org"),
}],
encryption_secret: String::new(), encryption_secret: String::new(),
}, },
]); ]);
@ -809,10 +1209,7 @@ mod tests {
let selected = select_effective_enterprise_secret_source(vec![ let selected = select_effective_enterprise_secret_source(vec![
EnterpriseSourceData { EnterpriseSourceData {
source_name: String::from("registry"), source_name: String::from("registry"),
configs: vec![EnterpriseConfig { configs: vec![enterprise_config(&TEST_ID_A.to_lowercase(), "https://registry.example.org", "registry", "", "indexed slot 0")],
id: TEST_ID_A.to_lowercase(),
server_url: String::from("https://registry.example.org"),
}],
encryption_secret: String::new(), encryption_secret: String::new(),
}, },
EnterpriseSourceData { EnterpriseSourceData {
@ -918,19 +1315,19 @@ mod tests {
]); ]);
assert_eq!( assert_eq!(
values.get("config_id0").map(String::as_str), values.get("config_id0").map(|value| value.value.as_str()),
Some("9072b77d-ca81-40da-be6a-861da525ef7b") Some("9072b77d-ca81-40da-be6a-861da525ef7b")
); );
assert_eq!( assert_eq!(
values.get("config_server_url0").map(String::as_str), values.get("config_server_url0").map(|value| value.value.as_str()),
Some("https://org.example.org") Some("https://org.example.org")
); );
assert_eq!( assert_eq!(
values.get("config_id1").map(String::as_str), values.get("config_id1").map(|value| value.value.as_str()),
Some("bbbbbbbb-bbbb-bbbb-bbbb-bbbbbbbbbbbb") Some("bbbbbbbb-bbbb-bbbb-bbbb-bbbbbbbbbbbb")
); );
assert_eq!( assert_eq!(
values.get("config_encryption_secret").map(String::as_str), values.get("config_encryption_secret").map(|value| value.value.as_str()),
Some("SECRET-A") Some("SECRET-A")
); );
} }
@ -956,14 +1353,40 @@ mod tests {
assert_eq!( assert_eq!(
source.configs, source.configs,
vec![ vec![
EnterpriseConfig { enterprise_config("9072b77d-ca81-40da-be6a-861da525ef7b", "https://slot0.example.org", "policy files", &policy_path(directory.path().join("config0.yaml")), "indexed slot 0"),
id: String::from("9072b77d-ca81-40da-be6a-861da525ef7b"), enterprise_config(TEST_ID_B, "https://slot4.example.org", "policy files", &policy_path(directory.path().join("config4.yaml")), "indexed slot 4"),
server_url: String::from("https://slot0.example.org"), ]
}, );
EnterpriseConfig { }
id: String::from(TEST_ID_B),
server_url: String::from("https://slot4.example.org"), #[test]
}, fn load_policy_values_from_directories_supports_padded_and_high_policy_slots() {
let directory = tempdir().unwrap();
fs::write(
directory.path().join("config_00000.yaml"),
"id: \"9072b77d-ca81-40da-be6a-861da525ef7b\"\nserver_url: \"https://slot0.example.org\"",
)
.unwrap();
fs::write(
directory.path().join("config_10503.yaml"),
"id: \"a1b2c3d4-e5f6-7890-abcd-ef1234567890\"\nserver_url: \"https://slot10503.example.org\"",
)
.unwrap();
fs::write(
directory.path().join("config_100000.yaml"),
"id: \"11111111-2222-3333-4444-555555555555\"\nserver_url: \"https://ignored.example.org\"",
)
.unwrap();
let values = load_policy_values_from_directories(&[directory.path().to_path_buf()]);
let source = parse_enterprise_source_values("policy files", &values);
assert_eq!(
source.configs,
vec![
enterprise_config("9072b77d-ca81-40da-be6a-861da525ef7b", "https://slot0.example.org", "policy files", &policy_path(directory.path().join("config_00000.yaml")), "indexed slot 00000"),
enterprise_config(TEST_ID_B, "https://slot10503.example.org", "policy files", &policy_path(directory.path().join("config_10503.yaml")), "indexed slot 10503"),
] ]
); );
} }