logger: Refactor the logging framework
[quassel.git] / src / common / main.cpp
1 /***************************************************************************
2  *   Copyright (C) 2005-2018 by the Quassel Project                        *
3  *   devel@quassel-irc.org                                                 *
4  *                                                                         *
5  *   This program is free software; you can redistribute it and/or modify  *
6  *   it under the terms of the GNU General Public License as published by  *
7  *   the Free Software Foundation; either version 2 of the License, or     *
8  *   (at your option) version 3.                                           *
9  *                                                                         *
10  *   This program is distributed in the hope that it will be useful,       *
11  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        *
12  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         *
13  *   GNU General Public License for more details.                          *
14  *                                                                         *
15  *   You should have received a copy of the GNU General Public License     *
16  *   along with this program; if not, write to the                         *
17  *   Free Software Foundation, Inc.,                                       *
18  *   51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.         *
19  ***************************************************************************/
20
21 #include <cstdlib>
22 #include <memory>
23
24 #ifdef HAVE_UMASK
25 #  include <sys/types.h>
26 #  include <sys/stat.h>
27 #endif /* HAVE_UMASK */
28
29 #include <QCoreApplication>
30 #include <QTextCodec>
31
32 #ifdef BUILD_CORE
33 #  include "coreapplication.h"
34 #elif defined BUILD_QTUI
35 #  include "aboutdata.h"
36 #  include "qtuiapplication.h"
37 #elif defined BUILD_MONO
38 #  include "aboutdata.h"
39 #  include "monoapplication.h"
40
41 #else
42 #error "Something is wrong - you need to #define a build mode!"
43 #endif
44
45 // We don't want quasselcore to depend on KDE
46 #if defined HAVE_KDE4 && defined BUILD_CORE
47 #  undef HAVE_KDE4
48 #endif
49 // We don't want quasselcore to depend on KDE
50 #if defined HAVE_KF5 && defined BUILD_CORE
51 #  undef HAVE_KF5
52 #endif
53
54 #ifdef HAVE_KDE4
55 #  include <KAboutData>
56 #  include "kcmdlinewrapper.h"
57 #elif defined HAVE_KF5
58 #  include <KCoreAddons/KAboutData>
59 #  include <KCoreAddons/Kdelibs4ConfigMigrator>
60 #  include "qt5cliparser.h"
61 #elif defined HAVE_QT5
62 #  include "qt5cliparser.h"
63 #else
64 #  include "cliparser.h"
65 #endif
66
67 #if !defined(BUILD_CORE) && defined(STATIC)
68 #include <QtPlugin>
69 Q_IMPORT_PLUGIN(qjpeg)
70 Q_IMPORT_PLUGIN(qgif)
71 #endif
72
73 #include "quassel.h"
74
75 int main(int argc, char **argv)
76 {
77 #ifdef HAVE_UMASK
78     umask(S_IRWXG | S_IRWXO);
79 #endif
80
81     // Instantiate early, so log messages are handled
82     Quassel::instance();
83
84 #if QT_VERSION < 0x050000
85     // All our source files are in UTF-8, and Qt5 even requires that
86     QTextCodec::setCodecForCStrings(QTextCodec::codecForName("UTF-8"));
87     QTextCodec::setCodecForTr(QTextCodec::codecForName("UTF-8"));
88 #endif
89
90     Quassel::setupBuildInfo();
91     QCoreApplication::setApplicationName(Quassel::buildInfo().applicationName);
92     QCoreApplication::setApplicationVersion(Quassel::buildInfo().plainVersionString);
93     QCoreApplication::setOrganizationName(Quassel::buildInfo().organizationName);
94     QCoreApplication::setOrganizationDomain(Quassel::buildInfo().organizationDomain);
95
96     // on OSX with Qt4, raster seems to fix performance issues
97 #if QT_VERSION < 0x050000 && defined Q_OS_MAC && !defined BUILD_CORE
98     QApplication::setGraphicsSystem("raster");
99 #endif
100 //Setup the High-DPI settings
101 # if QT_VERSION >= 0x050600 && defined(Q_OS_WIN)
102     QCoreApplication::setAttribute(Qt::AA_EnableHighDpiScaling); //Added in Qt 5.6
103 #endif
104 # if QT_VERSION >= 0x050400
105    //Added in the early Qt5 versions (5.0?)- use 5.4 as the cutoff since lots of high-DPI work was added then
106     QCoreApplication::setAttribute(Qt::AA_UseHighDpiPixmaps);
107 # endif
108     // We need to explicitly initialize the required resources when linking statically
109 #ifndef BUILD_QTUI
110     Q_INIT_RESOURCE(sql);
111 #endif
112 #ifndef BUILD_CORE
113     Q_INIT_RESOURCE(pics);
114     Q_INIT_RESOURCE(hicolor_icons);
115 #endif
116
117 #ifdef EMBED_DATA
118     Q_INIT_RESOURCE(i18n);
119 # ifndef BUILD_CORE
120     Q_INIT_RESOURCE(data);
121     Q_INIT_RESOURCE(breeze_icons);
122     Q_INIT_RESOURCE(breeze_dark_icons);
123 #  ifdef WITH_OXYGEN_ICONS
124     Q_INIT_RESOURCE(oxygen_icons);
125 #  endif
126 #  ifdef WITH_BUNDLED_ICONS
127       Q_INIT_RESOURCE(breeze_icon_theme);
128       Q_INIT_RESOURCE(breeze_dark_icon_theme);
129 #   ifdef WITH_OXYGEN_ICONS
130       Q_INIT_RESOURCE(oxygen_icon_theme);
131 #   endif
132 #  endif
133 # endif
134 #endif
135
136     std::shared_ptr<AbstractCliParser> cliParser;
137
138 #ifdef HAVE_KDE4
139     // We need to init KCmdLineArgs first
140     KAboutData aboutData("quassel", "kdelibs4", ki18n("Quassel IRC"), Quassel::buildInfo().plainVersionString.toUtf8(),
141         ki18n("A modern, distributed IRC client"));
142     aboutData.addLicense(KAboutData::License_GPL_V2);
143     aboutData.addLicense(KAboutData::License_GPL_V3);
144     aboutData.setBugAddress("https://bugs.quassel-irc.org/projects/quassel-irc/issues/new");
145     aboutData.setOrganizationDomain(Quassel::buildInfo().organizationDomain.toUtf8());
146     KCmdLineArgs::init(argc, argv, &aboutData);
147
148     cliParser = std::make_shared<KCmdLineWrapper>();
149 #elif defined HAVE_QT5
150     cliParser = std::make_shared<Qt5CliParser>();
151 #else
152     cliParser = std::make_shared<CliParser>();
153 #endif
154     Quassel::setCliParser(cliParser);
155
156     // Initialize CLI arguments
157     // NOTE: We can't use tr() at this point, since app is not yet created
158     // TODO: Change this once we get rid of KDE4 and can initialize the parser after creating the app
159
160     // put shared client&core arguments here
161     cliParser->addSwitch("debug", 'd', "Enable debug output");
162     cliParser->addSwitch("help", 'h', "Display this help and exit");
163     cliParser->addSwitch("version", 'v', "Display version information");
164 #ifdef BUILD_QTUI
165     cliParser->addOption("configdir", 'c', "Specify the directory holding the client configuration", "path");
166 #else
167     cliParser->addOption("configdir", 'c', "Specify the directory holding configuration files, the SQlite database and the SSL certificate", "path");
168 #endif
169     cliParser->addOption("datadir", 0, "DEPRECATED - Use --configdir instead", "path");
170     cliParser->addOption("loglevel", 'L', "Loglevel Debug|Info|Warning|Error", "level", "Info");
171 #ifdef HAVE_SYSLOG
172     cliParser->addSwitch("syslog", 0, "Log to syslog");
173 #endif
174     cliParser->addOption("logfile", 'l', "Log to a file", "path");
175
176 #ifndef BUILD_CORE
177     // put client-only arguments here
178     cliParser->addOption("icontheme", 0, "Override the system icon theme ('breeze' is recommended)", "theme");
179     cliParser->addOption("qss", 0, "Load a custom application stylesheet", "file.qss");
180     cliParser->addSwitch("debugbufferswitches", 0, "Enables debugging for bufferswitches");
181     cliParser->addSwitch("debugmodel", 0, "Enables debugging for models");
182     cliParser->addSwitch("hidewindow", 0, "Start the client minimized to the system tray");
183 #endif
184 #ifndef BUILD_QTUI
185     // put core-only arguments here
186     cliParser->addOption("listen", 0, "The address(es) quasselcore will listen on", "<address>[,<address>[,...]]", "::,0.0.0.0");
187     cliParser->addOption("port", 'p', "The port quasselcore will listen at", "port", "4242");
188     cliParser->addSwitch("norestore", 'n', "Don't restore last core's state");
189     cliParser->addSwitch("config-from-environment", 0, "Load configuration from environment variables");
190     cliParser->addOption("select-backend", 0, "Switch storage backend (migrating data if possible)", "backendidentifier");
191     cliParser->addOption("select-authenticator", 0, "Select authentication backend", "authidentifier");
192     cliParser->addSwitch("add-user", 0, "Starts an interactive session to add a new core user");
193     cliParser->addOption("change-userpass", 0, "Starts an interactive session to change the password of the user identified by <username>", "username");
194     cliParser->addSwitch("oidentd", 0, "Enable oidentd integration.  In most cases you should also enable --strict-ident");
195     cliParser->addOption("oidentd-conffile", 0, "Set path to oidentd configuration file", "file");
196     cliParser->addSwitch("strict-ident", 0, "Use users' quasselcore username as ident reply. Ignores each user's configured ident setting.");
197     cliParser->addSwitch("ident-daemon", 0, "Enable internal ident daemon");
198     cliParser->addOption("ident-port", 0, "The port quasselcore will listen at for ident requests. Only meaningful with --ident-daemon", "port", "10113");
199 #ifdef HAVE_SSL
200     cliParser->addSwitch("require-ssl", 0, "Require SSL for remote (non-loopback) client connections");
201     cliParser->addOption("ssl-cert", 0, "Specify the path to the SSL Certificate", "path", "configdir/quasselCert.pem");
202     cliParser->addOption("ssl-key", 0, "Specify the path to the SSL key", "path", "ssl-cert-path");
203 #endif
204     cliParser->addSwitch("enable-experimental-dcc", 0, "Enable highly experimental and unfinished support for CTCP DCC (DANGEROUS)");
205 #endif
206
207 #ifdef HAVE_KDE4
208     // the KDE version needs this extra call to parse argc/argv before app is instantiated
209     if (!cliParser->init()) {
210         cliParser->usage();
211         return EXIT_FAILURE;
212     }
213 #endif
214
215 #if defined BUILD_CORE
216     CoreApplication app(argc, argv);
217 #elif defined BUILD_QTUI
218     QtUiApplication app(argc, argv);
219 #elif defined BUILD_MONO
220     MonolithicApplication app(argc, argv);
221 #endif
222
223 #ifndef HAVE_KDE4
224     // the non-KDE version parses after app has been instantiated
225     if (!cliParser->init(app.arguments())) {
226         cliParser->usage();
227         return EXIT_FAILURE;
228     }
229 #endif
230
231 // Migrate settings from KDE4 to KF5 if appropriate
232 #ifdef HAVE_KF5
233     Kdelibs4ConfigMigrator migrator(QCoreApplication::applicationName());
234     migrator.setConfigFiles(QStringList() << "quasselrc" << "quassel.notifyrc");
235     migrator.migrate();
236 #endif
237
238 #ifdef HAVE_KF5
239     // FIXME: This should be done after loading the translation catalogue, but still in main()
240     AboutData aboutData;
241     AboutData::setQuasselPersons(&aboutData);
242     KAboutData::setApplicationData(aboutData.kAboutData());
243 #endif
244     if (!app.init())
245         return EXIT_FAILURE;
246
247     return app.exec();
248 }