mirror of
https://github.com/inspircd/inspircd.git
synced 2025-03-09 18:49:03 -04:00
159 lines
4.2 KiB
C++
159 lines
4.2 KiB
C++
/*
|
|
* InspIRCd -- Internet Relay Chat Daemon
|
|
*
|
|
* Copyright (C) 2013, 2015, 2019-2024 Sadie Powell <sadie@witchery.services>
|
|
* Copyright (C) 2013 Adam <Adam@anope.org>
|
|
* Copyright (C) 2012-2013, 2015 Attila Molnar <attilamolnar@hush.com>
|
|
* Copyright (C) 2012 Robby <robby@chatbelgie.be>
|
|
* Copyright (C) 2009-2010 Daniel De Graaf <danieldg@inspircd.org>
|
|
*
|
|
* This file is part of InspIRCd. InspIRCd is free software: you can
|
|
* redistribute it and/or modify it under the terms of the GNU General Public
|
|
* License as published by the Free Software Foundation, version 2.
|
|
*
|
|
* This program is distributed in the hope that it will be useful, but WITHOUT
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
|
|
* FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
|
|
* details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
|
|
#include <filesystem>
|
|
|
|
#include <fmt/color.h>
|
|
|
|
#include "inspircd.h"
|
|
#include "dynamic.h"
|
|
|
|
bool ModuleManager::Load(const std::string& modname, bool defer)
|
|
{
|
|
/* Don't allow people to specify paths for modules, it doesn't work as expected */
|
|
if (modname.find_first_of("\\/") != std::string::npos)
|
|
{
|
|
LastModuleError = "You can't load modules with a path: " + modname;
|
|
return false;
|
|
}
|
|
|
|
const std::string filename = ExpandModName(modname);
|
|
const std::string moduleFile = ServerInstance->Config->Paths.PrependModule(filename);
|
|
|
|
std::error_code ec;
|
|
if (!std::filesystem::is_regular_file(moduleFile, ec))
|
|
{
|
|
LastModuleError = "Module file could not be found: " + filename;
|
|
ServerInstance->Logs.Critical("MODULE", LastModuleError);
|
|
return false;
|
|
}
|
|
|
|
if (Modules.find(filename) != Modules.end())
|
|
{
|
|
LastModuleError = "Module " + filename + " is already loaded, cannot load a module twice!";
|
|
ServerInstance->Logs.Critical("MODULE", LastModuleError);
|
|
return false;
|
|
}
|
|
|
|
Module* newmod = nullptr;
|
|
auto* newhandle = new DLLManager(moduleFile);
|
|
ServiceList newservices;
|
|
if (!defer)
|
|
this->NewServices = &newservices;
|
|
|
|
try
|
|
{
|
|
newmod = newhandle->CallInit();
|
|
this->NewServices = nullptr;
|
|
|
|
if (newmod)
|
|
{
|
|
newmod->ModuleFile = filename;
|
|
newmod->ModuleDLL = newhandle;
|
|
Modules[filename] = newmod;
|
|
|
|
if (!defer)
|
|
{
|
|
AttachAll(newmod);
|
|
AddServices(newservices);
|
|
|
|
ConfigStatus confstatus;
|
|
newmod->init();
|
|
newmod->ReadConfig(confstatus);
|
|
}
|
|
|
|
ServerInstance->Logs.Normal("MODULE", "New module introduced: {} (version {}, properties {})",
|
|
filename, newmod->GetVersion(), newmod->GetPropertyString());
|
|
}
|
|
else
|
|
{
|
|
LastModuleError = "Unable to load " + filename + ": " + newhandle->LastError();
|
|
ServerInstance->Logs.Critical("MODULE", LastModuleError);
|
|
delete newhandle;
|
|
return false;
|
|
}
|
|
}
|
|
catch (const CoreException& modexcept)
|
|
{
|
|
this->NewServices = nullptr;
|
|
|
|
// failure in module constructor
|
|
if (newmod)
|
|
{
|
|
DoSafeUnload(newmod);
|
|
ServerInstance->GlobalCulls.AddItem(newhandle);
|
|
}
|
|
else
|
|
delete newhandle;
|
|
LastModuleError = "Unable to load " + filename + ": " + modexcept.GetReason();
|
|
ServerInstance->Logs.Critical("MODULE", LastModuleError);
|
|
return false;
|
|
}
|
|
|
|
if (defer)
|
|
return true;
|
|
|
|
FOREACH_MOD(OnLoadModule, (newmod));
|
|
PrioritizeHooks();
|
|
return true;
|
|
}
|
|
|
|
/* We must load the modules AFTER initializing the socket engine, now */
|
|
void ModuleManager::LoadCoreModules(std::map<std::string, ServiceList>& servicemap)
|
|
{
|
|
fmt::print("Loading core modules ");
|
|
fflush(stdout);
|
|
|
|
try
|
|
{
|
|
for (const auto& entry : std::filesystem::directory_iterator(ServerInstance->Config->Paths.Module))
|
|
{
|
|
if (!entry.is_regular_file())
|
|
continue;
|
|
|
|
const std::string name = entry.path().filename().string();
|
|
if (!InspIRCd::Match(name, "core_*" DLL_EXTENSION))
|
|
continue;
|
|
|
|
fmt::print(".");
|
|
fflush(stdout);
|
|
this->NewServices = &servicemap[name];
|
|
|
|
if (!Load(name, true))
|
|
{
|
|
fmt::println("");
|
|
fmt::println("[{}] {}", fmt::styled("*", fmt::emphasis::bold | fmt::fg(fmt::terminal_color::red)), LastError());
|
|
fmt::println("");
|
|
ServerInstance->Exit(EXIT_FAILURE);
|
|
}
|
|
}
|
|
}
|
|
catch (const std::filesystem::filesystem_error& err)
|
|
{
|
|
fmt::println("failed: {}", err.what());
|
|
ServerInstance->Exit(EXIT_FAILURE);
|
|
}
|
|
|
|
fmt::println("");
|
|
}
|