geekbot/Geekbot.net/Program.cs

243 lines
10 KiB
C#
Raw Normal View History

2017-04-26 14:12:42 +02:00
using System;
using System.Linq;
using System.Net;
2017-04-26 14:12:42 +02:00
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;
using Discord;
using Discord.Commands;
using Discord.WebSocket;
using Geekbot.net.Lib;
using Geekbot.net.Lib.Media;
2017-09-15 22:56:03 +02:00
using Microsoft.Extensions.DependencyInjection;
using Nancy.Hosting.Self;
2017-04-26 14:12:42 +02:00
using StackExchange.Redis;
using WikipediaApi;
2017-04-26 14:12:42 +02:00
namespace Geekbot.net
{
2017-09-15 22:56:03 +02:00
internal class Program
2017-04-26 14:12:42 +02:00
{
private DiscordSocketClient client;
2017-09-15 22:56:03 +02:00
private CommandService commands;
private IDatabase redis;
private IServiceCollection services;
private IServiceProvider servicesProvider;
2017-09-15 22:56:03 +02:00
private RedisValue token;
private IGeekbotLogger logger;
private IUserRepository userRepository;
private string[] args;
private bool firstStart = false;
2017-04-26 14:12:42 +02:00
2017-09-26 13:02:38 +02:00
private static void Main(string[] args)
2017-04-26 14:12:42 +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(@" \____|_____|_____|_|\_\____/ \___/ |_|");
logo.AppendLine("=========================================");
Console.WriteLine(logo.ToString());
var logger = new GeekbotLogger();
logger.Information("Geekbot", "Starting...");
try
{
new Program().MainAsync(args, logger).GetAwaiter().GetResult();
}
catch (Exception e)
{
logger.Error("Geekbot", "RIP", e);
}
2017-04-26 14:12:42 +02:00
}
private async Task MainAsync(string[] args, IGeekbotLogger logger)
2017-04-26 14:12:42 +02:00
{
2017-09-26 13:02:38 +02:00
this.logger = logger;
this.args = args;
logger.Information("Geekbot", "Initing Stuff");
2017-09-26 22:09:57 +02:00
client = new DiscordSocketClient(new DiscordSocketConfig
{
LogLevel = LogSeverity.Verbose,
2017-11-10 21:37:39 +01:00
MessageCacheSize = 1000
2017-09-26 22:09:57 +02:00
});
client.Log += DiscordLogger;
2017-04-26 14:12:42 +02:00
commands = new CommandService();
try
{
var redisMultiplexer = ConnectionMultiplexer.Connect("127.0.0.1:6379");
redis = redisMultiplexer.GetDatabase(6);
logger.Information("Redis", $"Connected to db {redis.Database}");
}
catch (Exception e)
{
logger.Error("Redis", "Redis Connection Failed", e);
2017-09-19 20:39:49 +02:00
Environment.Exit(102);
}
token = redis.StringGet("discordToken");
2017-04-26 14:12:42 +02:00
if (token.IsNullOrEmpty)
{
Console.Write("Your bot Token: ");
var newToken = Console.ReadLine();
redis.StringSet("discordToken", newToken);
redis.StringSet("Game", "Ping Pong");
2017-04-26 14:12:42 +02:00
token = newToken;
firstStart = true;
2017-09-26 22:09:57 +02:00
}
2017-04-26 14:12:42 +02:00
services = new ServiceCollection();
userRepository = new UserRepository(redis, logger);
var fortunes = new FortunesProvider(logger);
var mediaProvider = new MediaProvider(logger);
2017-10-01 23:41:25 +02:00
var malClient = new MalClient(redis, logger);
var levelCalc = new LevelCalc();
2017-11-06 23:55:28 +01:00
var emojiConverter = new EmojiConverter();
var audioUtils = new AudioUtils();
2018-02-19 21:35:45 +01:00
var mtgManaConverter = new MtgManaConverter();
var wikipediaClient = new WikipediaClient();
2017-09-15 22:56:03 +02:00
services.AddSingleton(redis);
services.AddSingleton<IGeekbotLogger>(logger);
services.AddSingleton<IUserRepository>(userRepository);
services.AddSingleton<ILevelCalc>(levelCalc);
2017-11-06 23:55:28 +01:00
services.AddSingleton<IEmojiConverter>(emojiConverter);
services.AddSingleton<IAudioUtils>(audioUtils);
2017-09-15 12:58:49 +02:00
services.AddSingleton<IFortunesProvider>(fortunes);
services.AddSingleton<IMediaProvider>(mediaProvider);
2017-10-01 23:41:25 +02:00
services.AddSingleton<IMalClient>(malClient);
2018-02-19 21:35:45 +01:00
services.AddSingleton<IMtgManaConverter>(mtgManaConverter);
services.AddSingleton<IWikipediaClient>(wikipediaClient);
2017-04-26 14:12:42 +02:00
logger.Information("Geekbot", "Connecting to Discord");
2017-04-26 14:12:42 +02:00
await Login();
await Task.Delay(-1);
}
2017-09-26 22:09:57 +02:00
private async Task Login()
2017-04-26 14:12:42 +02:00
{
try
{
await client.LoginAsync(TokenType.Bot, token);
await client.StartAsync();
2017-04-27 19:50:03 +02:00
var isConneted = await isConnected();
if (isConneted)
{
await client.SetGameAsync(redis.StringGet("Game"));
logger.Information("Geekbot", $"Now Connected as {client.CurrentUser.Username} to {client.Guilds.Count} Servers");
2017-04-27 19:50:03 +02:00
logger.Information("Geekbot", "Registering Stuff");
2017-11-15 01:08:20 +01:00
var translationHandler = new TranslationHandler(client.Guilds, redis, logger);
var errorHandler = new ErrorHandler(logger, translationHandler, args.Contains("--expose-errors"));
2018-02-19 22:31:40 +01:00
var reactionListener = new ReactionListener(redis);
2017-04-27 19:50:03 +02:00
await commands.AddModulesAsync(Assembly.GetEntryAssembly());
2017-09-15 22:56:03 +02:00
services.AddSingleton(commands);
services.AddSingleton<IErrorHandler>(errorHandler);
2017-11-15 01:08:20 +01:00
services.AddSingleton<ITranslationHandler>(translationHandler);
services.AddSingleton<DiscordSocketClient>(client);
2018-02-19 22:31:40 +01:00
services.AddSingleton<IReactionListener>(reactionListener);
servicesProvider = services.BuildServiceProvider();
2018-02-19 22:31:40 +01:00
var handlers = new Handlers(client, logger, redis, servicesProvider, commands, userRepository, reactionListener);
client.MessageReceived += handlers.RunCommand;
client.MessageReceived += handlers.UpdateStats;
client.MessageDeleted += handlers.MessageDeleted;
client.UserJoined += handlers.UserJoined;
client.UserUpdated += handlers.UserUpdated;
client.UserLeft += handlers.UserLeft;
2018-02-19 22:31:40 +01:00
client.ReactionAdded += handlers.ReactionAdded;
client.ReactionRemoved += handlers.ReactionRemoved;
if (firstStart || args.Contains("--reset"))
{
logger.Information("Geekbot", "Finishing setup");
await FinishSetup();
logger.Information("Geekbot", "Setup finished");
}
if (!args.Contains("--disable-api"))
{
2017-10-12 16:34:10 +02:00
startWebApi();
}
logger.Information("Geekbot", "Done and ready for use");
2017-04-27 19:50:03 +02:00
}
2017-04-26 14:12:42 +02:00
}
catch (Exception e)
2017-04-26 14:12:42 +02:00
{
logger.Error("Discord", "Could not connect...", e);
2017-09-19 20:39:49 +02:00
Environment.Exit(103);
2017-04-26 14:12:42 +02:00
}
}
2017-09-26 22:09:57 +02:00
private async Task<bool> isConnected()
2017-04-27 19:50:03 +02:00
{
while (!client.ConnectionState.Equals(ConnectionState.Connected))
await Task.Delay(25);
return true;
}
private void startWebApi()
2017-10-12 16:34:10 +02:00
{
logger.Information("API", "Starting Webserver");
2017-10-12 16:34:10 +02:00
var webApiUrl = new Uri("http://localhost:12995");
new NancyHost(webApiUrl).Start();
logger.Information("API", $"Webserver now running on {webApiUrl}");
2017-10-12 16:34:10 +02:00
}
private async Task<Task> FinishSetup()
{
var appInfo = await client.GetApplicationInfoAsync();
logger.Information("Setup", $"Just a moment while i setup everything {appInfo.Owner.Username}");
2017-11-10 21:36:41 +01:00
try
{
redis.StringSet("botOwner", appInfo.Owner.Id);
var req = HttpWebRequest.Create(appInfo.IconUrl);
using (var stream = req.GetResponse().GetResponseStream())
2017-11-10 21:36:41 +01:00
{
await client.CurrentUser.ModifyAsync(User =>
{
User.Avatar = new Image(stream);
User.Username = appInfo.Name.ToString();
});
}
logger.Information("Setup", "Everything done, enjoy!");
2017-11-10 21:36:41 +01:00
}
catch (Exception e)
{
logger.Warning("Setup", "Oha, it seems like something went wrong while running the setup, geekbot will work never the less though", e);
}
return Task.CompletedTask;
2017-04-26 14:12:42 +02:00
}
2017-09-26 22:09:57 +02:00
private Task DiscordLogger(LogMessage message)
{
var logMessage = $"[{message.Source}] {message.Message}";
switch (message.Severity)
{
case LogSeverity.Verbose:
case LogSeverity.Debug:
logger.Debug(message.Source, message.Message);
2017-09-26 22:09:57 +02:00
break;
case LogSeverity.Info:
logger.Information(message.Source, message.Message);
2017-09-26 22:09:57 +02:00
break;
case LogSeverity.Critical:
case LogSeverity.Error:
case LogSeverity.Warning:
2017-11-16 00:51:36 +01:00
if (logMessage.Contains("VOICE_STATE_UPDATE")) break;
logger.Error(message.Source, message.Message, message.Exception);
2017-09-26 22:09:57 +02:00
break;
default:
logger.Information(message.Source, $"{logMessage} --- {message.Severity}");
2017-09-26 22:09:57 +02:00
break;
}
return Task.CompletedTask;
}
2017-04-26 14:12:42 +02:00
}
2017-09-15 22:56:03 +02:00
}