Make DCC support opt-in for now
[quassel.git] / src / common / main.cpp
index 5ce1ac9..8ed40da 100644 (file)
@@ -1,5 +1,5 @@
 /***************************************************************************
- *   Copyright (C) 2005-08 by the Quassel Project                          *
+ *   Copyright (C) 2005-2013 by the Quassel Project                        *
  *   devel@quassel-irc.org                                                 *
  *                                                                         *
  *   This program is free software; you can redistribute it and/or modify  *
  *   You should have received a copy of the GNU General Public License     *
  *   along with this program; if not, write to the                         *
  *   Free Software Foundation, Inc.,                                       *
- *   59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.             *
+ *   51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.         *
  ***************************************************************************/
 
-#include <QDateTime>
-#include <QString>
-#include <QTimer>
-#include <QTranslator>
-#include <QFile>
-
-#include "global.h"
-#include "logger.h"
-#include "network.h"
-#include "settings.h"
-#include "cliparser.h"
-
-#if defined BUILD_CORE
-#include <QDir>
-#include "core.h"
-#include "message.h"
+#include <cstdlib>
 
+#ifdef BUILD_CORE
+#  include "coreapplication.h"
 #elif defined BUILD_QTUI
-#include "client.h"
-#include "qtuiapplication.h"
-#include "qtui.h"
-
+#  include "qtuiapplication.h"
 #elif defined BUILD_MONO
-#include "client.h"
-#include "core.h"
-#include "coresession.h"
-#include "qtuiapplication.h"
-#include "qtui.h"
+#  include "monoapplication.h"
 
 #else
 #error "Something is wrong - you need to #define a build mode!"
 #endif
 
-
-#include <signal.h>
-
-#if not defined(Q_OS_WIN32) && not defined(MAC_10_4_SDK)
-#include <execinfo.h>
-#include <dlfcn.h>
-#include <cxxabi.h>
+// We don't want quasselcore to depend on KDE
+#if defined HAVE_KDE && defined BUILD_CORE
+#  undef HAVE_KDE
 #endif
 
-//! Signal handler for graceful shutdown.
-void handle_signal(int sig) {
-  qWarning("%s", qPrintable(QString("Caught signal %1 - exiting.").arg(sig)));
-  QCoreApplication::quit();
-}
-
-#if not defined(Q_OS_WIN32) && not defined(MAC_10_4_SDK)
-void handle_crash(int sig) {
-  void* callstack[128];
-  int i, frames = backtrace(callstack, 128);
-
-  QFile dumpFile(QString("Quassel-Crash-%1").arg(QDateTime::currentDateTime().toString("yyyyMMdd-hhmm.log")));
-  dumpFile.open(QIODevice::WriteOnly);
-  QTextStream dumpStream(&dumpFile);
-
-  for (i = 0; i < frames; ++i) {
-    Dl_info info;
-    dladdr (callstack[i], &info);
-    // as a reference:
-    //     typedef struct
-    //     {
-    //       __const char *dli_fname;  /* File name of defining object.  */
-    //       void *dli_fbase;          /* Load address of that object.  */
-    //       __const char *dli_sname;  /* Name of nearest symbol.  */
-    //       void *dli_saddr;          /* Exact value of nearest symbol.  */
-    //     } Dl_info;
-
-#if __LP64__
-    int addrSize = 16;
-#else
-    int addrSize = 8;
+#ifdef HAVE_KDE
+#  include <KAboutData>
+#  include "kcmdlinewrapper.h"
 #endif
 
-    QString funcName;
-    if(info.dli_sname) {
-      char *func = abi::__cxa_demangle(info.dli_sname, 0, 0, 0);
-      if(func) {
-       funcName = QString(func);
-       free(func);
-      } else {
-       funcName = QString(info.dli_sname);
-      }
-    } else {
-      funcName = QString("0x%1").arg((long)info.dli_saddr, addrSize, QLatin1Char('0'));
-    }
-
-    // prettificating the filename
-    QString fileName("???");
-    if(info.dli_fname) {
-      fileName = QString(info.dli_fname);
-      int slashPos = fileName.lastIndexOf('/');
-      if(slashPos != -1)
-       fileName = fileName.mid(slashPos + 1);
-      if(fileName.count() < 20)
-       fileName += QString(20 - fileName.count(), ' ');
-    }
-
-    QString debugLine = QString("#%1 %2 0x%3 %4").arg(i, 3, 10)
-      .arg(fileName)
-      .arg((long)(callstack[i]), addrSize, 16, QLatin1Char('0'))
-      .arg(funcName);
-
-    dumpStream << debugLine << "\n";
-    qDebug() << qPrintable(debugLine);
-  }
-  dumpFile.close();
-  exit(27);
-}
-#endif // ifndef Q_OS_WIN32
-
-
-int main(int argc, char **argv) {
-  // We catch SIGTERM and SIGINT (caused by Ctrl+C) to graceful shutdown Quassel.
-  signal(SIGTERM, handle_signal);
-  signal(SIGINT, handle_signal);
-
-#if not defined(Q_OS_WIN32) && not defined(MAC_10_4_SDK)
-  signal(SIGABRT, handle_crash);
-  signal(SIGBUS, handle_crash);
-  signal(SIGSEGV, handle_crash);
-#endif // ndef Q_OS_WIN32
-  
-  Global::registerMetaTypes();
-  Global::setupVersion();
-
-/*
-#if defined BUILD_CORE
-  Global::runMode = Global::CoreOnly;
-  QCoreApplication app(argc, argv);
-#elif defined BUILD_QTUI
-  Global::runMode = Global::ClientOnly;
-  QApplication app(argc, argv);
-#else
-  Global::runMode = Global::Monolithic;
-  QApplication app(argc, argv);
+#if !defined(BUILD_CORE) && defined(STATIC)
+#include <QtPlugin>
+Q_IMPORT_PLUGIN(qjpeg)
+Q_IMPORT_PLUGIN(qgif)
 #endif
-*/
-#if defined BUILD_CORE
-  Global::runMode = Global::CoreOnly;
-  QCoreApplication app(argc, argv);
-#elif defined BUILD_QTUI
-  Global::runMode = Global::ClientOnly;
-  QtUiApplication app(argc, argv);
+
+#include "cliparser.h"
+#include "quassel.h"
+
+int main(int argc, char **argv)
+{
+    // Setup build information and version string
+  # include "version.gen"
+    buildinfo.append(QString(",%1,%2").arg(__DATE__, __TIME__));
+    Quassel::setupBuildInfo(buildinfo);
+    QCoreApplication::setApplicationName(Quassel::buildInfo().applicationName);
+    QCoreApplication::setOrganizationName(Quassel::buildInfo().organizationName);
+    QCoreApplication::setOrganizationDomain(Quassel::buildInfo().organizationDomain);
+
+    AbstractCliParser *cliParser;
+
+#ifdef HAVE_KDE
+    // We need to init KCmdLineArgs first
+    // TODO: build an AboutData compat class to replace our aboutDlg strings
+    KAboutData aboutData("quassel", "kdelibs4", ki18n("Quassel IRC"), Quassel::buildInfo().plainVersionString.toUtf8(),
+        ki18n("A modern, distributed IRC client"));
+    aboutData.addLicense(KAboutData::License_GPL_V2);
+    aboutData.addLicense(KAboutData::License_GPL_V3);
+    aboutData.setBugAddress("http://bugs.quassel-irc.org/projects/quassel-irc/issues/new");
+    aboutData.setOrganizationDomain(Quassel::buildInfo().organizationDomain.toUtf8());
+    KCmdLineArgs::init(argc, argv, &aboutData);
+
+    cliParser = new KCmdLineWrapper();
 #else
-  Global::runMode = Global::Monolithic;
-  QtUiApplication app(argc, argv);
+    cliParser = new CliParser();
 #endif
+    Quassel::setCliParser(cliParser);
 
+    // Initialize CLI arguments
+    // NOTE: We can't use tr() at this point, since app is not yet created
 
-
-  Global::parser = CliParser(QCoreApplication::arguments());
-
-#ifndef BUILD_QTUI
-// put core-only arguments here
-  Global::parser.addOption("port",'p',"The port quasselcore will listen at",QString("4242"));
-  Global::parser.addSwitch("norestore", 'n', "Don't restore last core's state");
-  Global::parser.addOption("logfile",'l',"Path to logfile");
-  Global::parser.addOption("loglevel",'L',"Loglevel Debug|Info|Warning|Error","Info");
-  Global::parser.addOption("datadir", 0, "Specify the directory holding datafiles like the Sqlite DB and the SSL Cert");
-#endif // BUILD_QTUI
-#ifndef BUILD_CORE
-// put client-only arguments here
-  Global::parser.addSwitch("debugbufferswitches",0,"Enables debugging for bufferswitches");
-  Global::parser.addSwitch("debugmodel",0,"Enables debugging for models");
-#endif // BUILD_QTCORE
-// put shared client&core arguments here
-  Global::parser.addSwitch("debug",'d',"Enable debug output");
-  Global::parser.addSwitch("help",'h', "Display this help and exit");
-
-  if(!Global::parser.parse() || Global::parser.isSet("help")) {
-    Global::parser.usage();
-    return 1;
-  }
-
-  /*
-   This is an initial check if logfile is writable since the warning would spam stdout if done
-   in current Logger implementation. Can be dropped whenever the logfile is only opened once.
-  */
-  if(Global::runMode != Global::ClientOnly) {
-    QFile logFile;
-    if(!Global::parser.value("logfile").isEmpty()) {
-      logFile.setFileName(Global::parser.value("logfile"));
-      if(!logFile.open(QIODevice::Append | QIODevice::Text))
-        qWarning("Warning: Couldn't open logfile '%s' - will log to stdout instead",qPrintable(logFile.fileName()));
-      else logFile.close();
-    }
-  }
-
-  qsrand(QTime(0,0,0).secsTo(QTime::currentTime()));
-
-  // Set up i18n support
-  QLocale locale = QLocale::system();
-
-  QTranslator qtTranslator(&app);
-  qtTranslator.setObjectName("QtTr");
-  qtTranslator.load(QString(":i18n/qt_%1").arg(locale.name()));
-  app.installTranslator(&qtTranslator);
-
-  QTranslator quasselTranslator(&app);
-  quasselTranslator.setObjectName("QuasselTr");
-  quasselTranslator.load(QString(":i18n/quassel_%1").arg(locale.name()));
-  app.installTranslator(&quasselTranslator);
-
-  Network::setDefaultCodecForServer("ISO-8859-1");
-  Network::setDefaultCodecForEncoding("UTF-8");
-  Network::setDefaultCodecForDecoding("ISO-8859-15");
-
-  QCoreApplication::setOrganizationDomain("quassel-irc.org");
-  QCoreApplication::setApplicationName("Quassel IRC");
-  QCoreApplication::setOrganizationName("Quassel Project");
-
-  
-#ifndef BUILD_QTUI
-  Core::instance();  // create and init the core
+    // put shared client&core arguments here
+    cliParser->addSwitch("debug", 'd', "Enable debug output");
+    cliParser->addSwitch("help", 'h', "Display this help and exit");
+    cliParser->addSwitch("version", 'v', "Display version information");
+#ifdef BUILD_QTUI
+    cliParser->addOption("configdir <path>", 'c', "Specify the directory holding the client configuration");
+#else
+    cliParser->addOption("configdir <path>", 'c', "Specify the directory holding configuration files, the SQlite database and the SSL certificate");
 #endif
-
-  //Settings::init();
+    cliParser->addOption("datadir <path>", 0, "DEPRECATED - Use --configdir instead");
 
 #ifndef BUILD_CORE
-  // session resume
-  QtUi *gui = new QtUi();
-  Client::init(gui);
-  // init gui only after the event loop has started
-  QTimer::singleShot(0, gui, SLOT(init()));
-  //gui->init();
+    // put client-only arguments here
+    cliParser->addOption("qss <file.qss>", 0, "Load a custom application stylesheet");
+    cliParser->addSwitch("debugbufferswitches", 0, "Enables debugging for bufferswitches");
+    cliParser->addSwitch("debugmodel", 0, "Enables debugging for models");
+    cliParser->addSwitch("hidewindow", 0, "Start the client minimized to the system tray");
 #endif
-
 #ifndef BUILD_QTUI
-  if(!Global::parser.isSet("norestore")) {
-    Core::restoreState();
-  }
+    // put core-only arguments here
+    cliParser->addOption("listen <address>[,<address[,...]]>", 0, "The address(es) quasselcore will listen on", "::,0.0.0.0");
+    cliParser->addOption("port <port>", 'p', "The port quasselcore will listen at", QString("4242"));
+    cliParser->addSwitch("norestore", 'n', "Don't restore last core's state");
+    cliParser->addOption("loglevel <level>", 'L', "Loglevel Debug|Info|Warning|Error", "Info");
+#ifdef HAVE_SYSLOG
+    cliParser->addSwitch("syslog", 0, "Log to syslog");
 #endif
-
-#ifndef BUILD_CORE 
-  app.resumeSessionIfPossible();
+    cliParser->addOption("logfile <path>", 'l', "Log to a file");
+    cliParser->addOption("select-backend <backendidentifier>", 0, "Switch storage backend (migrating data if possible)");
+    cliParser->addSwitch("add-user", 0, "Starts an interactive session to add a new core user");
+    cliParser->addOption("change-userpass <username>", 0, "Starts an interactive session to change the password of the user identified by username");
+    cliParser->addSwitch("oidentd", 0, "Enable oidentd integration");
+    cliParser->addOption("oidentd-conffile <file>", 0, "Set path to oidentd configuration file");
+#ifdef HAVE_SSL
+    cliParser->addSwitch("require-ssl", 0, "Require SSL for client connections");
 #endif
-  
-  int exitCode = app.exec();
-
-#ifndef BUILD_QTUI
-  Core::saveState();
+    cliParser->addSwitch("enable-experimental-dcc", 0, "Enable highly experimental and unfinished support for CTCP DCC (DANGEROUS)");
 #endif
 
-#ifndef BUILD_CORE
-  // the mainWin has to be deleted before the Core
-  // if not Quassel will crash on exit under certain conditions since the gui
-  // still wants to access clientdata
-  delete gui;
-  Client::destroy();
+#ifdef HAVE_KDE
+    // the KDE version needs this extra call to parse argc/argv before app is instantiated
+    if (!cliParser->init()) {
+        cliParser->usage();
+        return EXIT_FAILURE;
+    }
 #endif
-#ifndef BUILD_QTUI
-  Core::destroy();
+
+#  if defined BUILD_CORE
+    CoreApplication app(argc, argv);
+#  elif defined BUILD_QTUI
+    QtUiApplication app(argc, argv);
+#  elif defined BUILD_MONO
+    MonolithicApplication app(argc, argv);
+#  endif
+
+#ifndef HAVE_KDE
+    // the non-KDE version parses after app has been instantiated
+    if (!cliParser->init(app.arguments())) {
+        cliParser->usage();
+        return false;
+    }
 #endif
 
-  return exitCode;
+    if (!app.init()) return EXIT_FAILURE;
+    return app.exec();
 }