2017-04-26 14:12:42 +02:00
|
|
|
|
using System;
|
|
|
|
|
using System.Reflection;
|
2017-09-26 13:02:38 +02:00
|
|
|
|
using System.Text;
|
2017-04-26 14:12:42 +02:00
|
|
|
|
using System.Threading.Tasks;
|
2018-05-02 20:19:11 +02:00
|
|
|
|
using CommandLine;
|
2017-04-26 14:12:42 +02:00
|
|
|
|
using Discord;
|
|
|
|
|
using Discord.Commands;
|
|
|
|
|
using Discord.WebSocket;
|
2020-08-08 22:24:01 +02:00
|
|
|
|
using Geekbot.Bot.Handlers;
|
|
|
|
|
using Geekbot.Core;
|
|
|
|
|
using Geekbot.Core.Converters;
|
|
|
|
|
using Geekbot.Core.Database;
|
|
|
|
|
using Geekbot.Core.DiceParser;
|
|
|
|
|
using Geekbot.Core.ErrorHandling;
|
|
|
|
|
using Geekbot.Core.GlobalSettings;
|
|
|
|
|
using Geekbot.Core.GuildSettingsManager;
|
|
|
|
|
using Geekbot.Core.Highscores;
|
|
|
|
|
using Geekbot.Core.KvInMemoryStore;
|
|
|
|
|
using Geekbot.Core.Levels;
|
|
|
|
|
using Geekbot.Core.Logger;
|
|
|
|
|
using Geekbot.Core.MalClient;
|
|
|
|
|
using Geekbot.Core.Media;
|
|
|
|
|
using Geekbot.Core.RandomNumberGenerator;
|
|
|
|
|
using Geekbot.Core.ReactionListener;
|
|
|
|
|
using Geekbot.Core.UserRepository;
|
|
|
|
|
using Geekbot.Core.WikipediaClient;
|
|
|
|
|
using Geekbot.Web;
|
2018-08-29 21:16:01 +02:00
|
|
|
|
using Microsoft.EntityFrameworkCore;
|
2017-09-15 22:56:03 +02:00
|
|
|
|
using Microsoft.Extensions.DependencyInjection;
|
2017-04-26 14:12:42 +02:00
|
|
|
|
|
2020-08-08 22:24:01 +02:00
|
|
|
|
namespace Geekbot.Bot
|
2017-04-26 14:12:42 +02:00
|
|
|
|
{
|
2017-09-15 22:56:03 +02:00
|
|
|
|
internal class Program
|
2017-04-26 14:12:42 +02:00
|
|
|
|
{
|
2018-04-30 23:44:19 +02:00
|
|
|
|
private DiscordSocketClient _client;
|
|
|
|
|
private CommandService _commands;
|
2018-08-29 21:16:01 +02:00
|
|
|
|
private DatabaseInitializer _databaseInitializer;
|
2018-05-13 17:49:13 +02:00
|
|
|
|
private IGlobalSettings _globalSettings;
|
2018-04-30 23:44:19 +02:00
|
|
|
|
private IServiceProvider _servicesProvider;
|
2018-05-03 00:56:06 +02:00
|
|
|
|
private GeekbotLogger _logger;
|
2018-04-30 23:44:19 +02:00
|
|
|
|
private IUserRepository _userRepository;
|
2018-05-02 20:19:11 +02:00
|
|
|
|
private RunParameters _runParameters;
|
2020-06-19 03:34:37 +02:00
|
|
|
|
private IReactionListener _reactionListener;
|
2020-06-19 04:10:26 +02:00
|
|
|
|
private IGuildSettingsManager _guildSettingsManager;
|
2017-04-26 14:12:42 +02:00
|
|
|
|
|
2020-07-15 03:11:36 +02:00
|
|
|
|
private static async Task Main(string[] args)
|
2017-04-26 14:12:42 +02:00
|
|
|
|
{
|
2018-05-02 20:19:11 +02:00
|
|
|
|
RunParameters runParameters = null;
|
|
|
|
|
Parser.Default.ParseArguments<RunParameters>(args)
|
|
|
|
|
.WithParsed(e => runParameters = e)
|
2018-05-06 02:00:45 +02:00
|
|
|
|
.WithNotParsed(_ => Environment.Exit(GeekbotExitCode.InvalidArguments.GetHashCode()));
|
2018-05-02 20:19:11 +02:00
|
|
|
|
|
2017-09-26 22:09:57 +02:00
|
|
|
|
var logo = new StringBuilder();
|
2017-09-26 13:02:38 +02:00
|
|
|
|
logo.AppendLine(@" ____ _____ _____ _ ______ ___ _____");
|
|
|
|
|
logo.AppendLine(@" / ___| ____| ____| |/ / __ ) / _ \\_ _|");
|
|
|
|
|
logo.AppendLine(@"| | _| _| | _| | ' /| _ \| | | || |");
|
|
|
|
|
logo.AppendLine(@"| |_| | |___| |___| . \| |_) | |_| || |");
|
|
|
|
|
logo.AppendLine(@" \____|_____|_____|_|\_\____/ \___/ |_|");
|
2018-05-26 02:33:45 +02:00
|
|
|
|
logo.AppendLine($"Version {Constants.BotVersion()} ".PadRight(41, '='));
|
2017-09-26 13:02:38 +02:00
|
|
|
|
Console.WriteLine(logo.ToString());
|
2020-06-20 00:20:00 +02:00
|
|
|
|
var logger = new GeekbotLogger(runParameters);
|
2018-05-06 01:47:13 +02:00
|
|
|
|
logger.Information(LogSource.Geekbot, "Starting...");
|
2017-10-26 00:55:04 +02:00
|
|
|
|
try
|
|
|
|
|
{
|
2020-07-15 03:11:36 +02:00
|
|
|
|
await new Program().Start(runParameters, logger);
|
2017-10-26 00:55:04 +02:00
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
2018-05-06 01:47:13 +02:00
|
|
|
|
logger.Error(LogSource.Geekbot, "RIP", e);
|
2017-10-26 00:55:04 +02:00
|
|
|
|
}
|
2017-04-26 14:12:42 +02:00
|
|
|
|
}
|
|
|
|
|
|
2020-07-15 03:11:36 +02:00
|
|
|
|
private async Task Start(RunParameters runParameters, GeekbotLogger logger)
|
2017-04-26 14:12:42 +02:00
|
|
|
|
{
|
2018-04-30 23:44:19 +02:00
|
|
|
|
_logger = logger;
|
2018-05-02 20:19:11 +02:00
|
|
|
|
_runParameters = runParameters;
|
2020-06-19 03:34:37 +02:00
|
|
|
|
|
|
|
|
|
logger.Information(LogSource.Geekbot, "Connecting to Database");
|
|
|
|
|
var database = ConnectToDatabase();
|
|
|
|
|
_globalSettings = new GlobalSettings(database);
|
2017-09-26 22:09:57 +02:00
|
|
|
|
|
2020-06-19 03:34:37 +02:00
|
|
|
|
logger.Information(LogSource.Geekbot, "Connecting to Discord");
|
|
|
|
|
SetupDiscordClient();
|
|
|
|
|
await Login();
|
|
|
|
|
_logger.Information(LogSource.Geekbot, $"Now Connected as {_client.CurrentUser.Username} to {_client.Guilds.Count} Servers");
|
|
|
|
|
await _client.SetGameAsync(_globalSettings.GetKey("Game"));
|
|
|
|
|
|
|
|
|
|
_logger.Information(LogSource.Geekbot, "Loading Dependencies and Handlers");
|
|
|
|
|
RegisterDependencies();
|
|
|
|
|
await RegisterHandlers();
|
|
|
|
|
|
|
|
|
|
_logger.Information(LogSource.Api, "Starting Web API");
|
|
|
|
|
StartWebApi();
|
|
|
|
|
|
|
|
|
|
_logger.Information(LogSource.Geekbot, "Done and ready for use");
|
|
|
|
|
|
|
|
|
|
await Task.Delay(-1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private async Task Login()
|
|
|
|
|
{
|
|
|
|
|
try
|
2017-09-26 22:09:57 +02:00
|
|
|
|
{
|
2020-06-19 03:34:37 +02:00
|
|
|
|
var token = await GetToken();
|
|
|
|
|
await _client.LoginAsync(TokenType.Bot, token);
|
|
|
|
|
await _client.StartAsync();
|
|
|
|
|
while (!_client.ConnectionState.Equals(ConnectionState.Connected)) await Task.Delay(25);
|
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
|
|
|
|
_logger.Error(LogSource.Geekbot, "Could not connect to Discord", e);
|
|
|
|
|
Environment.Exit(GeekbotExitCode.CouldNotLogin.GetHashCode());
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-04-26 14:12:42 +02:00
|
|
|
|
|
2020-06-19 03:34:37 +02:00
|
|
|
|
private DatabaseContext ConnectToDatabase()
|
|
|
|
|
{
|
|
|
|
|
_databaseInitializer = new DatabaseInitializer(_runParameters, _logger);
|
2020-04-06 15:51:28 +02:00
|
|
|
|
var database = _databaseInitializer.Initialize();
|
|
|
|
|
database.Database.EnsureCreated();
|
|
|
|
|
if(!_runParameters.InMemory) database.Database.Migrate();
|
2020-06-19 03:34:37 +02:00
|
|
|
|
|
|
|
|
|
return database;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private async Task<string> GetToken()
|
|
|
|
|
{
|
|
|
|
|
var token = _runParameters.Token ?? _globalSettings.GetKey("DiscordToken");
|
|
|
|
|
if (string.IsNullOrEmpty(token))
|
2017-04-26 14:12:42 +02:00
|
|
|
|
{
|
|
|
|
|
Console.Write("Your bot Token: ");
|
|
|
|
|
var newToken = Console.ReadLine();
|
2018-07-28 16:31:18 +02:00
|
|
|
|
await _globalSettings.SetKey("DiscordToken", newToken);
|
|
|
|
|
await _globalSettings.SetKey("Game", "Ping Pong");
|
2020-06-19 03:34:37 +02:00
|
|
|
|
token = newToken;
|
2017-09-26 22:09:57 +02:00
|
|
|
|
}
|
2017-04-26 14:12:42 +02:00
|
|
|
|
|
2020-06-19 03:34:37 +02:00
|
|
|
|
return token;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void SetupDiscordClient()
|
|
|
|
|
{
|
|
|
|
|
_client = new DiscordSocketClient(new DiscordSocketConfig
|
|
|
|
|
{
|
|
|
|
|
LogLevel = LogSeverity.Verbose,
|
|
|
|
|
MessageCacheSize = 1000,
|
|
|
|
|
ExclusiveBulkDelete = true
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
var discordLogger = new DiscordLogger(_logger);
|
|
|
|
|
_client.Log += discordLogger.Log;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void RegisterDependencies()
|
|
|
|
|
{
|
|
|
|
|
var services = new ServiceCollection();
|
2017-09-30 01:38:10 +02:00
|
|
|
|
|
2020-06-19 03:34:37 +02:00
|
|
|
|
_userRepository = new UserRepository(_databaseInitializer.Initialize(), _logger);
|
|
|
|
|
_reactionListener = new ReactionListener(_databaseInitializer.Initialize());
|
2020-06-19 04:10:26 +02:00
|
|
|
|
_guildSettingsManager = new GuildSettingsManager(_databaseInitializer.Initialize());
|
2020-06-19 03:34:37 +02:00
|
|
|
|
var fortunes = new FortunesProvider(_logger);
|
|
|
|
|
var malClient = new MalClient(_globalSettings, _logger);
|
2017-10-25 00:58:59 +02:00
|
|
|
|
var levelCalc = new LevelCalc();
|
2017-11-06 23:55:28 +01:00
|
|
|
|
var emojiConverter = new EmojiConverter();
|
2018-02-19 21:35:45 +01:00
|
|
|
|
var mtgManaConverter = new MtgManaConverter();
|
2018-04-28 01:01:48 +02:00
|
|
|
|
var wikipediaClient = new WikipediaClient();
|
2019-05-11 01:18:22 +02:00
|
|
|
|
var randomNumberGenerator = new RandomNumberGenerator();
|
2020-06-25 15:12:41 +02:00
|
|
|
|
var mediaProvider = new MediaProvider(_logger, randomNumberGenerator);
|
2020-05-30 17:02:17 +02:00
|
|
|
|
var kvMemoryStore = new KvInInMemoryStore();
|
2020-08-14 23:15:11 +02:00
|
|
|
|
var errorHandler = new ErrorHandler(_logger, _runParameters, () => Localization.Internal.SomethingWentWrong);
|
2020-06-21 03:33:05 +02:00
|
|
|
|
var diceParser = new DiceParser(randomNumberGenerator);
|
2017-09-27 22:48:09 +02:00
|
|
|
|
|
2020-06-19 03:34:37 +02:00
|
|
|
|
services.AddSingleton(_userRepository);
|
|
|
|
|
services.AddSingleton<IGeekbotLogger>(_logger);
|
|
|
|
|
services.AddSingleton<ILevelCalc>(levelCalc);
|
|
|
|
|
services.AddSingleton<IEmojiConverter>(emojiConverter);
|
|
|
|
|
services.AddSingleton<IFortunesProvider>(fortunes);
|
|
|
|
|
services.AddSingleton<IMediaProvider>(mediaProvider);
|
|
|
|
|
services.AddSingleton<IMalClient>(malClient);
|
|
|
|
|
services.AddSingleton<IMtgManaConverter>(mtgManaConverter);
|
|
|
|
|
services.AddSingleton<IWikipediaClient>(wikipediaClient);
|
|
|
|
|
services.AddSingleton<IRandomNumberGenerator>(randomNumberGenerator);
|
|
|
|
|
services.AddSingleton<IKvInMemoryStore>(kvMemoryStore);
|
|
|
|
|
services.AddSingleton<IGlobalSettings>(_globalSettings);
|
|
|
|
|
services.AddSingleton<IErrorHandler>(errorHandler);
|
2020-06-21 03:33:05 +02:00
|
|
|
|
services.AddSingleton<IDiceParser>(diceParser);
|
2020-06-19 03:34:37 +02:00
|
|
|
|
services.AddSingleton<IReactionListener>(_reactionListener);
|
2020-06-19 04:10:26 +02:00
|
|
|
|
services.AddSingleton<IGuildSettingsManager>(_guildSettingsManager);
|
2020-06-19 03:34:37 +02:00
|
|
|
|
services.AddSingleton(_client);
|
|
|
|
|
services.AddTransient<IHighscoreManager>(e => new HighscoreManager(_databaseInitializer.Initialize(), _userRepository));
|
|
|
|
|
services.AddTransient(e => _databaseInitializer.Initialize());
|
|
|
|
|
|
|
|
|
|
_servicesProvider = services.BuildServiceProvider();
|
2017-04-26 14:12:42 +02:00
|
|
|
|
}
|
2020-06-19 03:34:37 +02:00
|
|
|
|
|
|
|
|
|
private async Task RegisterHandlers()
|
2017-04-26 14:12:42 +02:00
|
|
|
|
{
|
2020-06-19 03:34:37 +02:00
|
|
|
|
var applicationInfo = await _client.GetApplicationInfoAsync();
|
|
|
|
|
|
|
|
|
|
_commands = new CommandService();
|
|
|
|
|
await _commands.AddModulesAsync(Assembly.GetEntryAssembly(), _servicesProvider);
|
|
|
|
|
|
2020-06-19 04:10:26 +02:00
|
|
|
|
var commandHandler = new CommandHandler(_databaseInitializer.Initialize(), _client, _logger, _servicesProvider, _commands, applicationInfo, _guildSettingsManager);
|
2020-06-19 03:34:37 +02:00
|
|
|
|
var userHandler = new UserHandler(_userRepository, _logger, _databaseInitializer.Initialize(), _client);
|
|
|
|
|
var reactionHandler = new ReactionHandler(_reactionListener);
|
|
|
|
|
var statsHandler = new StatsHandler(_logger, _databaseInitializer.Initialize());
|
|
|
|
|
var messageDeletedHandler = new MessageDeletedHandler(_databaseInitializer.Initialize(), _logger, _client);
|
2020-04-17 23:48:50 +02:00
|
|
|
|
|
2020-06-19 03:34:37 +02:00
|
|
|
|
_client.MessageReceived += commandHandler.RunCommand;
|
|
|
|
|
_client.MessageDeleted += messageDeletedHandler.HandleMessageDeleted;
|
|
|
|
|
_client.UserJoined += userHandler.Joined;
|
|
|
|
|
_client.UserUpdated += userHandler.Updated;
|
|
|
|
|
_client.UserLeft += userHandler.Left;
|
|
|
|
|
_client.ReactionAdded += reactionHandler.Added;
|
|
|
|
|
_client.ReactionRemoved += reactionHandler.Removed;
|
|
|
|
|
if (!_runParameters.InMemory) _client.MessageReceived += statsHandler.UpdateStats;
|
2017-04-27 19:50:03 +02:00
|
|
|
|
}
|
|
|
|
|
|
2020-06-19 03:34:37 +02:00
|
|
|
|
private void StartWebApi()
|
2017-10-12 16:34:10 +02:00
|
|
|
|
{
|
2020-06-19 03:34:37 +02:00
|
|
|
|
if (_runParameters.DisableApi)
|
|
|
|
|
{
|
|
|
|
|
_logger.Warning(LogSource.Api, "Web API is disabled");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2020-04-06 15:51:28 +02:00
|
|
|
|
var highscoreManager = new HighscoreManager(_databaseInitializer.Initialize(), _userRepository);
|
|
|
|
|
WebApiStartup.StartWebApi(_logger, _runParameters, _commands, _databaseInitializer.Initialize(), _client, _globalSettings, highscoreManager);
|
2017-10-12 16:34:10 +02:00
|
|
|
|
}
|
2017-04-26 14:12:42 +02:00
|
|
|
|
}
|
2017-09-15 22:56:03 +02:00
|
|
|
|
}
|