You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

194 lines
6.2 KiB

11 years ago
/**
* Copyright (C) 2013-2016 Jessica James.
11 years ago
*
* Permission to use, copy, modify, and/or distribute this software for any
* purpose with or without fee is hereby granted, provided that the above
* copyright notice and this permission notice appear in all copies.
11 years ago
*
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
* SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION
* OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
* CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
11 years ago
*
* Written by Jessica James <jessica.aj@outlook.com>
11 years ago
*/
#include <cstdlib>
#include <cstdio>
#include <cstring>
#include <exception>
#include <thread>
#include <mutex>
11 years ago
#include "Jupiter/Functions.h"
#include "Jupiter/INIFile.h"
#include "Jupiter/Queue.h"
#include "Jupiter/Socket.h"
#include "Jupiter/Plugin.h"
#include "Jupiter/Timer.h"
#include "IRC_Bot.h"
#include "ServerManager.h"
#include "Console_Command.h"
#include "IRC_Command.h"
using namespace Jupiter::literals;
#define INPUT_BUFFER_SIZE 2048
struct ConsoleInput
{
Jupiter::String input;
std::mutex input_mutex;
bool awaiting_processing = false;
ConsoleInput() : input(INPUT_BUFFER_SIZE) {}
} console_input;
11 years ago
void onTerminate()
{
puts("Terminate signal received...");
}
void onExit()
{
puts("Exit signal received; Cleaning up...");
Jupiter::Socket::cleanup();
puts("Clean-up complete. Closing...");
}
void inputLoop()
{
char input[INPUT_BUFFER_SIZE];
size_t input_length;
11 years ago
while (ftell(stdin) != -1) // This can be expanded later to check for EBADF specifically.
{
fgets(input, sizeof(input), stdin);
input_length = strcspn(input, "\r\n");
check_input_processing:
console_input.input_mutex.lock();
if (console_input.awaiting_processing == false)
{
console_input.input.set(input, input_length);
console_input.awaiting_processing = true;
console_input.input_mutex.unlock();
}
else // User input received before previous input was processed.
{
std::this_thread::sleep_for((std::chrono::seconds(1)));
goto check_input_processing;
}
11 years ago
}
}
int main(int argc, const char **args)
{
atexit(onExit);
std::set_terminate(onTerminate);
std::thread inputThread(inputLoop);
Jupiter::ReferenceString command;
srand(static_cast<unsigned int>(std::chrono::duration_cast<std::chrono::milliseconds>(std::chrono::system_clock::now().time_since_epoch()).count()));
11 years ago
puts(Jupiter::copyright);
const char *configFileName = CONFIG_INI;
for (int i = 1; i < argc; i++)
{
if ("-help"_jrs.equalsi(args[i]))
11 years ago
{
puts("Help coming soon, to a theatre near you!");
return 0;
}
else if ("-config"_jrs.equalsi(args[i]) && ++i < argc)
configFileName = args[i];
else if ("-pluginsdir"_jrs.equalsi(args[i]) && ++i < argc)
Jupiter::setPluginDirectory(Jupiter::ReferenceString(args[i]));
else if ("-configFormat"_jrs.equalsi(args[i]) && ++i < argc)
puts("Feature not yet supported!");
else
printf("Warning: Unknown command line argument \"%s\" specified. Ignoring...", args[i]);
11 years ago
}
puts("Loading config file...");
if (!Jupiter::IRC::Client::Config->readFile(configFileName))
{
puts("Unable to read config file. Closing...");
exit(0);
}
fputs("Config loaded. ", stdout);
const Jupiter::ReadableString &pDir = Jupiter::IRC::Client::Config->get(STRING_LITERAL_AS_REFERENCE("Config"), STRING_LITERAL_AS_REFERENCE("PluginsDirectory"));
if (pDir.isNotEmpty())
11 years ago
{
Jupiter::setPluginDirectory(pDir);
printf("Plugins will be loaded from \"%.*s\"." ENDL, pDir.size(), pDir.ptr());
}
puts("Loading plugins...");
const Jupiter::ReadableString &pluginList = Jupiter::IRC::Client::Config->get(STRING_LITERAL_AS_REFERENCE("Config"), STRING_LITERAL_AS_REFERENCE("Plugins"));
if (pluginList.isEmpty())
puts("No plugins to load!");
else
{
unsigned int nPlugins = pluginList.wordCount(WHITESPACE);
printf("Attempting to load %u plugins..." ENDL, nPlugins);
for (unsigned int i = 0; i < nPlugins; i++)
{
Jupiter::ReferenceString plugin = Jupiter::ReferenceString::getWord(pluginList, i, WHITESPACE);
if (Jupiter::loadPlugin(plugin) == nullptr) fprintf(stderr, "WARNING: Failed to load plugin \"%.*s\"!" ENDL, plugin.size(), plugin.ptr());
else printf("\"%.*s\" loaded successfully." ENDL, plugin.size(), plugin.ptr());
}
}
if (consoleCommands->size() > 0)
printf("%u Console Commands have been initialized%s" ENDL, consoleCommands->size(), getConsoleCommand(STRING_LITERAL_AS_REFERENCE("help")) == nullptr ? "." : "; type \"help\" for more information.");
if (IRCMasterCommandList->size() > 0)
printf("%u IRC Commands have been loaded into the master list." ENDL, IRCMasterCommandList->size());
puts("Retreiving network list...");
const Jupiter::ReadableString &serverList = Jupiter::IRC::Client::Config->get(STRING_LITERAL_AS_REFERENCE("Config"), STRING_LITERAL_AS_REFERENCE("Servers"));
if (serverList == nullptr)
puts("Unable to find network list.");
else
{
unsigned int nServers = serverList.wordCount(WHITESPACE);
printf("Attempting to connect to %u servers..." ENDL, nServers);
for (unsigned int i = 0; i < nServers; i++)
serverManager->addServer(Jupiter::ReferenceString::getWord(serverList, i, WHITESPACE));
}
puts("Sockets established.");
size_t index;
11 years ago
while (1)
{
index = 0;
while (index < Jupiter::plugins->size())
if (Jupiter::plugins->get(index)->shouldRemove() || Jupiter::plugins->get(index)->think() != 0)
Jupiter::freePlugin(index);
else
++index;
11 years ago
Jupiter_checkTimers();
serverManager->think();
if (console_input.input_mutex.try_lock())
11 years ago
{
if (console_input.awaiting_processing)
{
console_input.awaiting_processing = false;
command = console_input.input.getWord(0, WHITESPACE);
ConsoleCommand *cmd = getConsoleCommand(command);
if (cmd != nullptr)
cmd->trigger(console_input.input.gotoWord(1, WHITESPACE));
else
printf("Error: Command \"%.*s\" not found." ENDL, command.size(), command.ptr());
}
console_input.input_mutex.unlock();
11 years ago
}
std::this_thread::sleep_for((std::chrono::milliseconds(1)));
}
return 0;
}