diff --git a/PluginManager.Test/PluginManager.Test.csproj b/PluginManager.Test/PluginManager.Test.csproj
deleted file mode 100644
index 5e0573b..0000000
--- a/PluginManager.Test/PluginManager.Test.csproj
+++ /dev/null
@@ -1,21 +0,0 @@
-
-
-  
-    netcoreapp3.1
-
-    false
-  
-
-  
-    
-    
-    
-    
-  
-
-  
-    
-    
-  
-
-
diff --git a/PluginManager.Test/PluginManager.cs b/PluginManager.Test/PluginManager.cs
deleted file mode 100644
index ec7b5af..0000000
--- a/PluginManager.Test/PluginManager.cs
+++ /dev/null
@@ -1,38 +0,0 @@
-using System;
-using Xunit;
-using PluginManager.Manager;
-using TestPlugin;
-
-namespace PluginManager.Test
-{
-    public class PDM : PluginDataManager
-    {
-        public override PluginData LoadData(string pluginDataFolder)
-        {
-            return null;
-        }
-
-        public override void SaveData(string yes, PluginData data)
-        {
-            
-        }
-    }
-
-    class PPD : PluginProcessData
-    {
-
-    }
-
-    public class PluginManagerTest
-    {
-        [Fact]
-        public void Test1()
-        {
-            PluginManager pluginManager = new PluginManager();
-            PluginManagerConfig.PluginFolderPath = "Plugins";
-            pluginManager.Load();
-            pluginManager.DoProcessing(new PPD());
-            pluginManager.Unload();
-        }
-    }
-}
diff --git a/PluginManager/PluginManager.csproj b/PluginManager.csproj
similarity index 100%
rename from PluginManager/PluginManager.csproj
rename to PluginManager.csproj
diff --git a/PluginManager/Manager/PluginManager.cs b/PluginManager/Manager/PluginManager.cs
deleted file mode 100644
index 6da8f9b..0000000
--- a/PluginManager/Manager/PluginManager.cs
+++ /dev/null
@@ -1,162 +0,0 @@
-using Crayon;
-using System;
-using System.Collections.Generic;
-using System.IO;
-using System.Linq;
-using System.Reflection;
-using System.Text;
-
-namespace PluginManager.Manager
-{
-    /// 
-    /// Class that handles all plugin handling
-    /// 
-    /// Type of the Plugin that is going to get loaded
-    /// Type of PluginDataManager that is going to get used to load data
-    public class PluginManager where T : Plugin where M : PluginDataManager
-    {
-        /// 
-        /// A list of plugins based on their name
-        /// 
-        public List Plugins { get; private set; }
-
-        private int pluginsSize;
-
-        private PluginDataManager dataManager;
-
-        /// 
-        /// Specifies if you are using the plugin system or not
-        /// 
-        public bool Enabled { get; set; }
-
-        public PluginManager()
-        {
-            Plugins = new List();
-            dataManager = (PluginDataManager)Activator.CreateInstance(typeof(M), null);
-            Enabled = true; // By default it's true
-        }
-
-        /// 
-        /// Loads all plugins and their data from the plugin folder
-        /// 
-        public void Load()
-        {
-            if (Enabled)
-            {
-                if (string.IsNullOrEmpty(PluginManagerConfig.PluginFolderPath))
-                    throw new ArgumentNullException("Plugins folder path", "You need to set a path for your plugins folder");
-
-                // First run
-                if (!Directory.Exists(PluginManagerConfig.PluginFolderPath))
-                {
-                    Directory.CreateDirectory(PluginManagerConfig.PluginFolderPath);
-                    return; // Let's return here, no plugins available
-                }
-
-                // Ignores files in this folder
-                var pluginDirectories = Directory.GetDirectories(PluginManagerConfig.PluginFolderPath);
-
-                foreach (var pluginPath in pluginDirectories)
-                {
-                    // Should contain the plugin file
-                    var files = Directory.GetFiles(pluginPath, "*.dll");
-                    foreach (var dll in files)
-                    {
-                        try
-                        {
-                            // We try to load them if they have at least one class that inherits Plugin
-                            var assembly = Assembly.LoadFrom(dll);
-
-                            var types = assembly.GetTypes();
-
-                            foreach (var type in types)
-                            {
-                                if (type.IsSubclassOf(typeof(T)))
-                                    addPlugin(type);
-                            }
-                        }
-                        catch (ReflectionTypeLoadException ex)
-                        {
-                            Console.Error.WriteLine(Output.Red($"Failed to load {Path.GetFileName(dll)}. We will continue to load more assemblies"), ex);
-                        }
-                        catch (Exception ex)
-                        {
-                            Console.Error.WriteLine(Output.Red($"Failed to load {Path.GetFileName(dll)}. We will continue to load more assemblies"), ex);
-                        }
-                    }
-                }
-
-                pluginsSize = Plugins.Count;
-            }
-            else
-            {
-                // Throw here?
-            }
-        }
-
-        /// 
-        /// Saves plugin information to their data files
-        /// 
-        public void Unload()
-        {
-            foreach (var plugin in Plugins)
-            {
-                try
-                {
-                    var dataFilePath = plugin.DataFilePath();
-
-                    // If it's null it means it's not using a file
-                    if (!string.IsNullOrEmpty(dataFilePath))
-                        dataManager.SaveData(dataFilePath, plugin.Data);
-                }
-                catch (Exception)
-                {
-                    Console.Error.WriteLine(Output.Red($"Couldn't save data for {plugin.Name()}"));
-                }
-            }
-        }
-
-        /// 
-        /// Internal function that handles plugin instancing
-        /// 
-        /// 
-        internal void addPlugin(Type plugin)
-        {
-            Plugin pluginInstance = (T)Activator.CreateInstance(plugin);
-
-            try
-            {
-                Plugins.Add((T)pluginInstance);
-
-                var pluginDataFilePath = pluginInstance.DataFilePath();
-
-                // If it's null we are not using plugin data
-                if (!string.IsNullOrEmpty(PluginManagerConfig.PluginDataFolderPath) && !string.IsNullOrEmpty(pluginDataFilePath))
-                {
-                    var pluginData = dataManager.LoadData(Path.Combine(PluginManagerConfig.PluginDataFolderPath, pluginDataFilePath));
-
-                    // This one shouldn't throw, loadData() will always exist
-                    pluginInstance.loadData(pluginData);
-                }
-
-                Console.WriteLine($"Loaded {pluginInstance.Name()}");
-            }
-            catch (ArgumentException)
-            {
-                Console.Error.WriteLine(Output.Red($"{pluginInstance.Name()} is already loaded"));
-            }
-        }
-
-        /// 
-        /// Executes DoProcessing on all plugins
-        /// 
-        /// 
-        public void DoProcessing(PluginProcessData processData)
-        {
-            for (int i = 0; i < pluginsSize; i++)
-            {
-                Plugins[i].DoProcessing(processData);
-            }
-        }
-    }
-}
diff --git a/TestPlugin/Class1.cs b/TestPlugin/Class1.cs
deleted file mode 100644
index 7023502..0000000
--- a/TestPlugin/Class1.cs
+++ /dev/null
@@ -1,50 +0,0 @@
-using PluginManager;
-using System;
-
-namespace TestPlugin
-{
-    public abstract class Class1 : Plugin
-    {
-        public Class1()
-        {
-        }
-    }
-
-    public class asd : Class1
-    {
-        public asd()
-        {
-            base.OnLoaded += Asd_OnLoaded;
-        }
-
-        private void Asd_OnLoaded(PluginData data)
-        {
-            Console.Out.WriteLine("I'm loaded man");
-        }
-
-        public override void DoProcessing(PluginProcessData data)
-        {
-            //Console.WriteLine("working");
-        }
-
-        public override string Name()
-        {
-            return "asd";
-        }
-
-        public override Version Version()
-        {
-            return new Version("1.0.0.0");
-        }
-
-        public override void BuildDataFile()
-        {
-            // Leave this empty if you are not using a data file
-        }
-
-        public override string DataFilePath()
-        {
-            return null;
-        }
-    }
-}
diff --git a/TestPlugin/TestPlugin.csproj b/TestPlugin/TestPlugin.csproj
deleted file mode 100644
index 87fe537..0000000
--- a/TestPlugin/TestPlugin.csproj
+++ /dev/null
@@ -1,15 +0,0 @@
-
-
-  
-    netcoreapp3.1
-  
-
-  
-    ..\PluginManager.Test\bin\Debug\Plugins\Yep\
-  
-
-  
-    
-  
-
-
diff --git a/src/Manager/PluginClassManager.cs b/src/Manager/PluginClassManager.cs
new file mode 100644
index 0000000..7d7348f
--- /dev/null
+++ b/src/Manager/PluginClassManager.cs
@@ -0,0 +1,39 @@
+using Crayon;
+using System;
+using System.Collections.Generic;
+using System.IO;
+using System.Linq;
+using System.Reflection;
+using System.Text;
+
+namespace PluginManager.Manager
+{
+    public class PluginClassManager : PluginManagerBase where T : Plugin where M : PluginDataManager
+                                           where R : PluginResponseData where P : PluginProcessData
+    {
+        /// 
+        /// Loads all plugins and their data from the plugin folder
+        /// 
+        public override void Load()
+        {
+            if (Enabled)
+            {
+                // Is this slow?
+                // I mean.... will this work
+                Assembly[] assemblies = AppDomain.CurrentDomain.GetAssemblies();
+                foreach (var assembly in assemblies)
+                {
+                    var types = assembly.GetTypes();
+                    foreach (Type type in types)
+                    {
+                        if (type.IsSubclassOf(typeof(T)))
+                            addPlugin(type);
+                    }
+                }
+
+                // It's faster if we store them instead of accessing the property
+                pluginsSize = Plugins.Count;
+            }
+        }
+    }
+}
diff --git a/PluginManager/Manager/PluginDataManager.cs b/src/Manager/PluginDataManager.cs
similarity index 100%
rename from PluginManager/Manager/PluginDataManager.cs
rename to src/Manager/PluginDataManager.cs
diff --git a/src/Manager/PluginDllManager.cs b/src/Manager/PluginDllManager.cs
new file mode 100644
index 0000000..b93e4af
--- /dev/null
+++ b/src/Manager/PluginDllManager.cs
@@ -0,0 +1,74 @@
+using Crayon;
+using System;
+using System.Collections.Generic;
+using System.IO;
+using System.Linq;
+using System.Reflection;
+using System.Text;
+
+namespace PluginManager.Manager
+{
+    /// 
+    /// Class that handles all plugin handling
+    /// 
+    /// Type of the Plugin that is going to get loaded
+    /// Type of PluginDataManager that is going to get used to load data
+    /// Type of PluginResponseData that is going to get used to return a response from the plugin list
+    /// Type of PluginProcess data that is going to get used to process data
+    public class PluginDllManager : PluginManagerBase where T : Plugin where M : PluginDataManager 
+                                           where R : PluginResponseData where P : PluginProcessData
+    {
+        /// 
+        /// Loads all plugins and their data from the plugin folder
+        /// 
+        public override void Load()
+        {
+            if (Enabled)
+            {
+                if (string.IsNullOrEmpty(PluginManagerConfig.PluginFolderPath))
+                    throw new ArgumentNullException("Plugins folder path", "You need to set a path for your plugins folder");
+
+                // First run
+                if (!Directory.Exists(PluginManagerConfig.PluginFolderPath))
+                {
+                    Directory.CreateDirectory(PluginManagerConfig.PluginFolderPath);
+                    return; // Let's return here, no plugins available
+                }
+
+                // Ignores files in this folder
+                var pluginDirectories = Directory.GetDirectories(PluginManagerConfig.PluginFolderPath);
+
+                foreach (var pluginPath in pluginDirectories)
+                {
+                    // Should contain the plugin file
+                    var files = Directory.GetFiles(pluginPath, "*.dll");
+
+                    // If it doesn't contain anything, this will not run
+                    foreach (var dll in files)
+                    {
+                        try
+                        {
+                            // We try to load them if they have at least one class that inherits Plugin
+                            var assembly = Assembly.LoadFrom(dll);
+
+                            var types = assembly.GetTypes();
+
+                            foreach (var type in types)
+                            {
+                                if (type.IsSubclassOf(typeof(T)))
+                                    addPlugin(type);
+                            }
+                        }
+                        catch (ReflectionTypeLoadException ex)
+                        {
+                            Console.Error.WriteLine(Output.Red($"Failed to load {Path.GetFileName(dll)}. We will continue to load more assemblies"), ex);
+                        }
+                    }
+                }
+
+                // It's faster if we store them instead of accessing the property
+                pluginsSize = Plugins.Count;
+            }
+        }
+    }
+}
diff --git a/src/Manager/PluginManagerBase.cs b/src/Manager/PluginManagerBase.cs
new file mode 100644
index 0000000..650867d
--- /dev/null
+++ b/src/Manager/PluginManagerBase.cs
@@ -0,0 +1,112 @@
+using Crayon;
+using System;
+using System.Collections.Generic;
+using System.IO;
+using System.Text;
+
+namespace PluginManager.Manager
+{
+    public abstract class PluginManagerBase where T : Plugin where M : PluginDataManager
+                                           where R : PluginResponseData where P : PluginProcessData
+    {
+        public PluginManagerBase()
+        {
+            DataManager = (M)Activator.CreateInstance(typeof(M), null);
+            Plugins = new List();
+            Enabled = true; // By default it's true
+        }
+
+        /// 
+        /// Data manager object
+        /// 
+        internal M DataManager { get; set; }
+
+        /// 
+        /// A list of plugins based on their name
+        /// 
+        public List Plugins { get; internal set; }
+
+        internal int pluginsSize;
+
+        /// 
+        /// Specifies if you are using the plugin system or not
+        /// 
+        public bool Enabled { get; set; }
+
+        public abstract void Load();
+
+        public virtual void Unload()
+        {
+            foreach (var plugin in Plugins)
+            {
+                try
+                {
+                    plugin.InvokeOnUnload();
+
+                    if (plugin.UsingDataFile())
+                    {
+                        var dataFilePath = plugin.DataFileName();
+                        DataManager.SaveData(dataFilePath, plugin.Data);
+                    }
+                }
+                catch (Exception)
+                {
+                    Console.Error.WriteLine(Output.Red($"Couldn't save data for {plugin.Name()}"));
+                }
+            }
+        }
+
+        /// 
+        /// Executes DoProcessing on all plugins
+        /// 
+        /// 
+        /// The response object
+        public virtual R DoProcessing(P processData)
+        {
+            R response = (R)Activator.CreateInstance(typeof(R));
+            for (int i = 0; i < pluginsSize; i++)
+            {
+                Plugins[i].DoProcessing(processData, ref response);
+            }
+
+            return response;
+        }
+
+        /// 
+        /// Internal function that handles plugin instancing
+        /// 
+        /// 
+        internal void addPlugin(Type type)
+        {
+            Plugin plugin = (T)Activator.CreateInstance(type);
+
+            var pluginName = plugin.Name();
+
+            // check versions too?
+            // and load the most recent one?
+            for (int i = 0; i < pluginsSize; i++)
+            {
+                if (Plugins[i].Name() == pluginName)
+                {
+                    Console.Error.WriteLine(Output.Red($"{pluginName} is already loaded"));
+                    return;
+                }
+            }
+
+            Plugins.Add((T)plugin);
+
+            if (plugin.UsingDataFile())
+            {
+                var pluginDataFilePath = plugin.DataFileName();
+                if (string.IsNullOrEmpty(pluginDataFilePath))
+                    throw new ArgumentNullException("Plugin Data File Path", "To use data files you must specify a name for the file");
+
+                var pluginData = DataManager.LoadData(Path.Combine(PluginManagerConfig.PluginDataFolderPath, pluginDataFilePath));
+
+                plugin.loadData(pluginData);
+            }
+
+            Console.WriteLine($"Loaded {pluginName}");
+        }
+    }
+}
diff --git a/PluginManager/Plugin/Plugin.cs b/src/Plugin/Plugin.cs
similarity index 58%
rename from PluginManager/Plugin/Plugin.cs
rename to src/Plugin/Plugin.cs
index da53ba7..a012a45 100644
--- a/PluginManager/Plugin/Plugin.cs
+++ b/src/Plugin/Plugin.cs
@@ -4,18 +4,30 @@ using System.Text;
 
 namespace PluginManager
 {
+    public delegate void OnLoadEvent();
     public delegate void OnLoadedEvent(PluginData data);
+    public delegate void OnUnloadEvent();
 
     /// 
     /// Base class to create plugins
     /// 
-    public abstract class Plugin
+    public abstract class Plugin where R : PluginResponseData where P : PluginProcessData
     {
+        /// 
+        /// Event executed when plugin object is instanciated
+        /// 
+        public event OnLoadEvent OnLoad;
+
         /// 
         /// Event executed when the plugin manager finishes loading this plugin
         /// 
         public event OnLoadedEvent OnLoaded;
 
+        /// 
+        /// Event executed when DataManager starts unloading the plugin
+        /// 
+        public event OnUnloadEvent OnUnload;
+
         /// 
         /// The plugin data object
         /// 
@@ -23,7 +35,12 @@ namespace PluginManager
 
         public Plugin()
         {
-            // Need a constructor to use Activator.CreateInstance()
+            OnLoad?.Invoke();
+        }
+
+        internal void InvokeOnUnload()
+        {
+            OnUnload?.Invoke();
         }
 
         /// 
@@ -32,8 +49,6 @@ namespace PluginManager
         /// Data object that is going to get loaded
         internal void loadData(PluginData data)
         {
-            if (data == null)
-                BuildDataFile();
             this.Data = data;
             OnLoaded?.Invoke(data);
         }
@@ -55,16 +70,24 @@ namespace PluginManager
         /// The name of the plugin
         public abstract string Name();
 
+        /// 
+        /// Specifies whether we are using a data file or not
+        /// 
+        /// Whether we are using a data file or not
+        public abstract bool UsingDataFile();
+
         /// 
         /// The path to the plugin data file
         /// 
         /// The path to the plugin data file
-        public abstract string DataFilePath();
+        public abstract string DataFileName();
 
         /// 
         /// Base function that should get called everytime this plugin needs to process data
         /// 
-        /// 
-        public abstract void DoProcessing(PluginProcessData data);
+        /// The data the plugin needs to do processing
+        /// Response object that the plugin will append to
+        /// The same object that is passed to the response parameter
+        public abstract ref R DoProcessing(P data, ref R response);
     }
 }
diff --git a/PluginManager/Plugin/PluginData.cs b/src/Plugin/PluginData.cs
similarity index 100%
rename from PluginManager/Plugin/PluginData.cs
rename to src/Plugin/PluginData.cs
diff --git a/PluginManager/Plugin/PluginProcessData.cs b/src/Plugin/PluginProcessData.cs
similarity index 100%
rename from PluginManager/Plugin/PluginProcessData.cs
rename to src/Plugin/PluginProcessData.cs
diff --git a/src/Plugin/PluginResponseData.cs b/src/Plugin/PluginResponseData.cs
new file mode 100644
index 0000000..faf25a5
--- /dev/null
+++ b/src/Plugin/PluginResponseData.cs
@@ -0,0 +1,10 @@
+using System;
+using System.Collections.Generic;
+using System.Text;
+
+namespace PluginManager
+{
+    public abstract class PluginResponseData
+    {
+    }
+}
diff --git a/PluginManager/PluginConfig.cs b/src/PluginConfig.cs
similarity index 54%
rename from PluginManager/PluginConfig.cs
rename to src/PluginConfig.cs
index 8120d3e..aa65f26 100644
--- a/PluginManager/PluginConfig.cs
+++ b/src/PluginConfig.cs
@@ -6,7 +6,14 @@ namespace PluginManager
 {
     public static class PluginManagerConfig
     {
+        /// 
+        /// Path to the folder where plugin data is going to be stored
+        /// 
         public static string PluginDataFolderPath { get; set; }
+
+        /// 
+        /// Path to the folder where plugins are going to be stored
+        /// 
         public static string PluginFolderPath { get; set; }
     }
 }