2025-04-23 12:07:22 +00:00
using System.Text ;
2025-06-01 19:14:21 +00:00
using AIStudio.Settings.DataModel ;
2025-04-23 12:07:22 +00:00
using Lua ;
using Lua.Standard ;
namespace AIStudio.Tools.PluginSystem ;
public static partial class PluginFactory
{
private static readonly List < IAvailablePlugin > AVAILABLE_PLUGINS = [ ] ;
private static readonly SemaphoreSlim PLUGIN_LOAD_SEMAPHORE = new ( 1 , 1 ) ;
/// <summary>
/// A list of all available plugins.
/// </summary>
public static IReadOnlyCollection < IPluginMetadata > AvailablePlugins = > AVAILABLE_PLUGINS ;
/// <summary>
/// Try to load all plugins from the plugins directory.
/// </summary>
/// <remarks>
/// Loading plugins means:<br/>
/// - Parsing and checking the plugin code<br/>
/// - Check for forbidden plugins<br/>
/// - Creating a new instance of the allowed plugin<br/>
/// - Read the plugin metadata<br/>
2025-04-27 07:06:05 +00:00
/// - Start the plugin<br/>
2025-04-23 12:07:22 +00:00
/// </remarks>
public static async Task LoadAll ( CancellationToken cancellationToken = default )
{
if ( ! IS_INITIALIZED )
{
LOG . LogError ( "PluginFactory is not initialized. Please call Setup() before using it." ) ;
return ;
}
if ( ! await PLUGIN_LOAD_SEMAPHORE . WaitAsync ( 0 , cancellationToken ) )
return ;
try
{
LOG . LogInformation ( "Start loading plugins." ) ;
if ( ! Directory . Exists ( PLUGINS_ROOT ) )
{
LOG . LogInformation ( "No plugins found." ) ;
return ;
}
AVAILABLE_PLUGINS . Clear ( ) ;
//
// The easiest way to load all plugins is to find all `plugin.lua` files and load them.
// By convention, each plugin is enforced to have a `plugin.lua` file.
//
var pluginMainFiles = Directory . EnumerateFiles ( PLUGINS_ROOT , "plugin.lua" , SearchOption . AllDirectories ) ;
foreach ( var pluginMainFile in pluginMainFiles )
{
2025-04-27 14:13:15 +00:00
try
{
if ( cancellationToken . IsCancellationRequested )
2025-06-01 19:14:21 +00:00
{
LOG . LogWarning ( "Was not able to load all plugins, because the operation was cancelled. It seems to be a timeout." ) ;
2025-04-27 14:13:15 +00:00
break ;
2025-06-01 19:14:21 +00:00
}
2025-04-27 14:13:15 +00:00
LOG . LogInformation ( $"Try to load plugin: {pluginMainFile}" ) ;
var fileInfo = new FileInfo ( pluginMainFile ) ;
string code ;
await using ( var fileStream = fileInfo . Open ( FileMode . Open , FileAccess . Read , FileShare . ReadWrite ) )
{
using var reader = new StreamReader ( fileStream , Encoding . UTF8 ) ;
code = await reader . ReadToEndAsync ( cancellationToken ) ;
}
var pluginPath = Path . GetDirectoryName ( pluginMainFile ) ! ;
var plugin = await Load ( pluginPath , code , cancellationToken ) ;
2025-04-23 12:07:22 +00:00
2025-04-27 14:13:15 +00:00
switch ( plugin )
{
case NoPlugin noPlugin when noPlugin . Issues . Any ( ) :
LOG . LogError ( $"Was not able to load plugin: '{pluginMainFile}'. Reason: {noPlugin.Issues.First()}" ) ;
continue ;
2025-04-23 12:07:22 +00:00
2025-04-27 14:13:15 +00:00
case NoPlugin :
LOG . LogError ( $"Was not able to load plugin: '{pluginMainFile}'. Reason: Unknown." ) ;
continue ;
2025-04-23 12:07:22 +00:00
2025-04-27 14:13:15 +00:00
case { IsValid : false } :
LOG . LogError ( $"Was not able to load plugin '{pluginMainFile}', because the Lua code is not a valid AI Studio plugin. There are {plugin.Issues.Count()} issues to fix. First issue is: {plugin.Issues.FirstOrDefault()}" ) ;
#if DEBUG
foreach ( var pluginIssue in plugin . Issues )
LOG . LogError ( $"Plugin issue: {pluginIssue}" ) ;
#endif
continue ;
2025-04-23 12:07:22 +00:00
2025-04-27 14:13:15 +00:00
case { IsMaintained : false } :
LOG . LogWarning ( $"The plugin '{pluginMainFile}' is not maintained anymore. Please consider to disable it." ) ;
break ;
}
2025-04-23 12:07:22 +00:00
2025-04-27 14:13:15 +00:00
LOG . LogInformation ( $"Successfully loaded plugin: '{pluginMainFile}' (Id='{plugin.Id}', Type='{plugin.Type}', Name='{plugin.Name}', Version='{plugin.Version}', Authors='{string.Join(" , ", plugin.Authors)}')" ) ;
AVAILABLE_PLUGINS . Add ( new PluginMetadata ( plugin , pluginPath ) ) ;
}
catch ( Exception e )
{
LOG . LogError ( $"Was not able to load plugin '{pluginMainFile}'. Issue: {e.Message}" ) ;
LOG . LogDebug ( e . StackTrace ) ;
}
2025-04-23 12:07:22 +00:00
}
// Start or restart all plugins:
await RestartAllPlugins ( cancellationToken ) ;
}
finally
{
PLUGIN_LOAD_SEMAPHORE . Release ( ) ;
LOG . LogInformation ( "Finished loading plugins." ) ;
}
2025-06-01 19:14:21 +00:00
//
// Next, we have to clean up our settings. It is possible that a configuration plugin was removed.
// We have to remove the related settings as well:
//
var wasConfigurationChanged = false ;
//
// Check LLM providers:
//
#pragma warning disable MWAIS0001
var configuredProviders = SETTINGS_MANAGER . ConfigurationData . Providers . ToList ( ) ;
foreach ( var configuredProvider in configuredProviders )
{
if ( ! configuredProvider . IsEnterpriseConfiguration )
continue ;
var providerSourcePluginId = configuredProvider . EnterpriseConfigurationPluginId ;
if ( providerSourcePluginId = = Guid . Empty )
continue ;
var providerSourcePlugin = AVAILABLE_PLUGINS . FirstOrDefault ( plugin = > plugin . Id = = providerSourcePluginId ) ;
if ( providerSourcePlugin is null )
{
LOG . LogWarning ( $"The configured LLM provider '{configuredProvider.InstanceName}' (id={configuredProvider.Id}) is based on a plugin that is not available anymore. Removing the provider from the settings." ) ;
SETTINGS_MANAGER . ConfigurationData . Providers . Remove ( configuredProvider ) ;
wasConfigurationChanged = true ;
}
}
#pragma warning restore MWAIS0001
//
// Check all possible settings:
//
if ( SETTINGS_LOCKER . GetConfigurationPluginId < DataApp > ( x = > x . UpdateBehavior ) is var updateBehaviorPluginId & & updateBehaviorPluginId ! = Guid . Empty )
{
var sourcePlugin = AVAILABLE_PLUGINS . FirstOrDefault ( plugin = > plugin . Id = = updateBehaviorPluginId ) ;
if ( sourcePlugin is null )
{
// Remove the locked state:
SETTINGS_LOCKER . Remove < DataApp > ( x = > x . UpdateBehavior ) ;
// Reset the setting to the default value:
SETTINGS_MANAGER . ConfigurationData . App . UpdateBehavior = UpdateBehavior . HOURLY ;
LOG . LogWarning ( $"The configured update behavior is based on a plugin that is not available anymore. Resetting the setting to the default value: {SETTINGS_MANAGER.ConfigurationData.App.UpdateBehavior}." ) ;
wasConfigurationChanged = true ;
}
}
if ( wasConfigurationChanged )
{
await SETTINGS_MANAGER . StoreSettings ( ) ;
await MessageBus . INSTANCE . SendMessage < bool > ( null , Event . CONFIGURATION_CHANGED ) ;
}
2025-04-23 12:07:22 +00:00
}
2025-04-26 16:55:23 +00:00
public static async Task < PluginBase > Load ( string? pluginPath , string code , CancellationToken cancellationToken = default )
2025-04-23 12:07:22 +00:00
{
if ( ForbiddenPlugins . Check ( code ) is { IsForbidden : true } forbiddenState )
return new NoPlugin ( $"This plugin is forbidden: {forbiddenState.Message}" ) ;
var state = LuaState . Create ( ) ;
2025-04-26 16:55:23 +00:00
if ( ! string . IsNullOrWhiteSpace ( pluginPath ) )
{
// Add the module loader so that the plugin can load other Lua modules:
state . ModuleLoader = new PluginLoader ( pluginPath ) ;
}
2025-04-23 12:07:22 +00:00
// Add some useful libraries:
state . OpenModuleLibrary ( ) ;
state . OpenStringLibrary ( ) ;
state . OpenTableLibrary ( ) ;
state . OpenMathLibrary ( ) ;
state . OpenBitwiseLibrary ( ) ;
state . OpenCoroutineLibrary ( ) ;
try
{
await state . DoStringAsync ( code , cancellationToken : cancellationToken ) ;
}
catch ( LuaParseException e )
{
return new NoPlugin ( $"Was not able to parse the plugin: {e.Message}" ) ;
}
catch ( LuaRuntimeException e )
{
return new NoPlugin ( $"Was not able to run the plugin: {e.Message}" ) ;
}
if ( ! state . Environment [ "TYPE" ] . TryRead < string > ( out var typeText ) )
return new NoPlugin ( "TYPE does not exist or is not a valid string." ) ;
if ( ! Enum . TryParse < PluginType > ( typeText , out var type ) )
return new NoPlugin ( $"TYPE is not a valid plugin type. Valid types are: {CommonTools.GetAllEnumValues<PluginType>()}" ) ;
if ( type is PluginType . NONE )
return new NoPlugin ( $"TYPE is not a valid plugin type. Valid types are: {CommonTools.GetAllEnumValues<PluginType>()}" ) ;
2025-04-26 16:55:23 +00:00
var isInternal = ! string . IsNullOrWhiteSpace ( pluginPath ) & & pluginPath . StartsWith ( INTERNAL_PLUGINS_ROOT , StringComparison . OrdinalIgnoreCase ) ;
2025-06-01 19:14:21 +00:00
switch ( type )
2025-04-23 12:07:22 +00:00
{
2025-06-01 19:14:21 +00:00
case PluginType . LANGUAGE :
return new PluginLanguage ( isInternal , state , type ) ;
case PluginType . CONFIGURATION :
var configPlug = new PluginConfiguration ( isInternal , state , type ) ;
await configPlug . InitializeAsync ( ) ;
return configPlug ;
2025-04-23 12:07:22 +00:00
2025-06-01 19:14:21 +00:00
default :
return new NoPlugin ( "This plugin type is not supported yet. Please try again with a future version of AI Studio." ) ;
}
2025-04-23 12:07:22 +00:00
}
}