Logo Search packages:      
Sourcecode: fatrat version File versions

fatrat.cpp

/*
FatRat download manager
http://fatrat.dolezel.info

Copyright (C) 2006-2008 Lubos Dolezel <lubos a dolezel.info>

This program is free software; you can redistribute it and/or
modify it under the terms of the GNU General Public License
version 2 as published by the Free Software Foundation.

This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
GNU General Public License for more details.

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., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.

In addition, as a special exemption, Luboš Doležel gives permission
to link the code of FatRat with the OpenSSL project's
"OpenSSL" library (or with modified versions of it that use the; same
license as the "OpenSSL" library), and distribute the linked
executables. You must obey the GNU General Public License in all
respects for all of the code used other than "OpenSSL".
*/

#include "config.h"
#include "fatrat.h"

#include <QApplication>
#include <QMessageBox>
#include <QTranslator>
#include <QLocale>
#include <QtDebug>
#include <QVariant>
#include <QSettings>
#include <QMap>
#include <QDir>
#include <QTextCodec>

#include <dlfcn.h>
#include <cstdlib>
#include <signal.h>
#include <iostream>

#include "MainWindow.h"
#include "QueueMgr.h"
#include "Queue.h"
#include "Transfer.h"
#include "AppTools.h"
#include "Settings.h"
#include "config.h"
#include "RuntimeException.h"
#include "dbus/DbusAdaptor.h"
#include "dbus/DbusImpl.h"
#include "rss/RssFetcher.h"
#include "MyApplication.h"
#include "Scheduler.h"

#ifdef WITH_WEBINTERFACE
#     include "remote/HttpService.h"
#endif

#ifdef WITH_JABBER
#     include "remote/JabberService.h"
#endif

using namespace std;

MainWindow* g_wndMain = 0;
QMap<QString,PluginInfo> g_plugins;

extern QVector<EngineEntry> g_enginesDownload;
extern QVector<EngineEntry> g_enginesUpload;
extern QSettings* g_settings;

const char* USER_PROFILE_PATH = "/.local/share/fatrat";

static void runEngines(bool init = true);
static QString argsToArg(int argc,char** argv);
static void processSession(QString arg);
static void loadPlugins();
static void loadPlugins(const char* dir);
static void showHelp();
static void installSignalHandler();

static bool m_bForceNewInstance = false;
static bool m_bStartHidden = false;
static bool m_bStartGUI = true;

class MyApplication;

int main(int argc,char** argv)
{
      QApplication* app = 0;
      int rval;
      QueueMgr* qmgr;
      QString arg = argsToArg(argc, argv);
      
      app = new MyApplication(argc, argv, m_bStartGUI);
      
      QCoreApplication::setOrganizationName("Dolezel");
      QCoreApplication::setOrganizationDomain("dolezel.info");
      QCoreApplication::setApplicationName("fatrat");
      QTextCodec::setCodecForCStrings( QTextCodec::codecForName("UTF-8") );
      
      installSignalHandler();
      
      if(!m_bForceNewInstance)
            processSession(arg);
      
#ifdef WITH_NLS
      QTranslator translator;
      {
            QString fname = QString("fatrat_") + QLocale::system().name();
            qDebug() << "Current locale" << QLocale::system().name();
            translator.load(fname, getDataFileDir("/lang", fname));
            QCoreApplication::installTranslator(&translator);
      }
#endif
      
      // Init download engines (let them load settings)
      initSettingsDefaults();
      
      if(m_bStartGUI)
            initSettingsPages();
      
      initTransferClasses();
      loadPlugins();
      runEngines();
      Queue::loadQueues();
      initAppTools();
      
      qRegisterMetaType<QString*>("QString*");
      qRegisterMetaType<QByteArray*>("QByteArray*");
      
      qmgr = new QueueMgr;

#ifdef WITH_WEBINTERFACE
      new HttpService;
#endif
      
      if(m_bStartGUI)
            g_wndMain = new MainWindow(m_bStartHidden);
      else
            qDebug() << "FatRat is up and running now";
      
      new RssFetcher;
      
      DbusImpl* impl = new DbusImpl;
      new FatratAdaptor(impl);
      
      QDBusConnection::sessionBus().registerObject("/", impl);
      QDBusConnection::sessionBus().registerService("info.dolezel.fatrat");
      
      if(!arg.isEmpty() && m_bStartGUI)
            g_wndMain->addTransfer(arg);
      
#ifdef WITH_JABBER
      new JabberService;
#endif
      new Scheduler;
      
      if(m_bStartGUI)
            QApplication::setQuitOnLastWindowClosed(false);
      rval = app->exec();
      
#ifdef WITH_JABBER
      delete JabberService::instance();
#endif
#ifdef WITH_WEBINTERFACE
      delete HttpService::instance();
#endif
      delete RssFetcher::instance();
      delete Scheduler::instance();
      delete g_wndMain;
      delete app;
      
      Queue::stopQueues();
      Queue::saveQueues();
      qmgr->exit();
      Queue::unloadQueues();
      
      runEngines(false);
      
      delete qmgr;
      exitSettings();
      
      return rval;
}

QString argsToArg(int argc,char** argv)
{
      QString arg;
      
      for(int i=1;i<argc;i++)
      {
            if(!strcasecmp(argv[i], "--force") || !strcasecmp(argv[i], "-f"))
                  m_bForceNewInstance = true;
            else if(!strcasecmp(argv[i], "--hidden") || !strcasecmp(argv[i], "-i"))
                  m_bStartHidden = true;
            else if(!strcasecmp(argv[i], "--nogui") || !strcasecmp(argv[i], "-n"))
                  m_bStartGUI = false;
            else if(!strcasecmp(argv[i], "--help") || !strcasecmp(argv[i], "-h"))
                  showHelp();
            else if(argv[i][0] == '-')
                  i++;
            else
            {
                  if(!arg.isEmpty())
                        arg += '\n';
                  arg += argv[i];
            }
      }
      
      return arg;
}

void processSession(QString arg)
{
      QDBusConnection conn = QDBusConnection::sessionBus();
      QDBusConnectionInterface* bus = conn.interface();
      
      if(bus->isServiceRegistered("info.dolezel.fatrat"))
      {
            qDebug() << "FatRat is already running";
            
            if(!arg.isEmpty())
            {
                  qDebug() << "Passing arguments to an existing instance.";
                  QDBusInterface iface("info.dolezel.fatrat", "/", "info.dolezel.fatrat", conn);
                  
                  iface.call("addTransfers", arg);
            }
            else
            {
                  QMessageBox::critical(0, "FatRat", QObject::tr("There is already a running instance.\n"
                              "If you want to start FatRat anyway, pass --force among arguments."));
            }
            
            exit(0);
      }
}

static void runEngines(bool init)
{
      for(int i=0;i<g_enginesDownload.size();i++)
      {
            if(init)
            {
                  if(g_enginesDownload[i].lpfnInit)
                        g_enginesDownload[i].lpfnInit();
            }
            else
            {
                  if(g_enginesDownload[i].lpfnExit)
                        g_enginesDownload[i].lpfnExit();
            }
      }
      
      for(int i=0;i<g_enginesUpload.size();i++)
      {
            if(init)
            {
                  if(g_enginesUpload[i].lpfnInit)
                        g_enginesUpload[i].lpfnInit();
            }
            else
            {
                  if(g_enginesUpload[i].lpfnExit)
                        g_enginesUpload[i].lpfnExit();
            }
      }
}

QWidget* getMainWindow()
{
      return g_wndMain;
}

QString formatSize(qulonglong size, bool persec)
{
      QString rval;
      
      if(size < 1024)
            rval = QString("%L1 B").arg(size);
      else if(size < 1024*1024)
            rval = QString("%L1 KB").arg(size/1024);
      else if(size < 1024*1024*1024)
            rval = QString("%L1 MB").arg(double(size)/1024.0/1024.0, 0, 'f', 1);
      else
            rval = QString("%L1 GB").arg(double(size)/1024.0/1024.0/1024.0, 0, 'f', 1);
      
      if(persec) rval += "/s";
      return rval;
}

QString formatTime(qulonglong inval)
{
      QString result;
      qulonglong days,hrs,mins,secs;
      days = inval/(60*60*24);
      inval %= 60*60*24;
      
      hrs = inval/(60*60);
      inval %= 60*60;
      
      mins = inval/60;
      secs = inval%60;
      
      if(days)
            result = QString("%1d ").arg(days);
      if(hrs)
            result += QString("%1h ").arg(hrs);
      if(mins)
            result += QString("%1m ").arg(mins);
      if(secs)
            result += QString("%1s").arg(secs);
      
      return result;
}


/////////////////////////////////////////////////////////

class RecursiveRemove : public QThread
{
public:
      RecursiveRemove(QString what) : m_what(what)
      {
            start();
      }
      void run()
      {
            connect(this, SIGNAL(finished()), this, SLOT(deleteLater()));
            work(m_what);
      }
      static bool work(QString what)
      {
            qDebug() << "recursiveRemove" << what;
            if(!QFile::exists(what))
                  return true; // silently ignore
            if(!QFile::remove(what))
            {
                  QDir dir(what);
                  if(!dir.exists())
                  {
                        qDebug() << "Not a directory:" << what;
                        return false;
                  }
                  
                  QStringList contents;
                  contents = dir.entryList();
                  
                  foreach(QString item, contents)
                  {
                        if(item != "." && item != "..")
                        {
                              if(!work(dir.filePath(item)))
                                    return false;
                        }
                  }
                  
                  QString name = dir.dirName();
                  if(!dir.cdUp())
                  {
                        qDebug() << "Cannot cdUp:" << what;
                        return false;
                  }
                  if(!dir.rmdir(name))
                  {
                        qDebug() << "Cannot rmdir:" << name;
                        return false;
                  }
            }
            return true;
      }
private:
      QString m_what;
};

void recursiveRemove(QString what)
{
      new RecursiveRemove(what);
}

bool openDataFile(QFile* file, QString filePath)
{
      if(filePath[0] != '/')
            filePath.prepend('/');
      
      file->setFileName(QDir::homePath() + QLatin1String(USER_PROFILE_PATH) + filePath);
      if(file->open(QIODevice::ReadOnly))
            return true;
      file->setFileName(QLatin1String(DATA_LOCATION) + filePath);
      if(!file->open(QIODevice::ReadOnly))
      {
            Logger::global()->enterLogMessage(QObject::tr("Unable to load a data file:") + ' ' + filePath);
            return false;
      }
      else
            return true;
}

QString getDataFileDir(QString dir, QString fileName)
{
      if(fileName.isEmpty())
            return QLatin1String(DATA_LOCATION) + dir;
      
      QString f = QDir::homePath() + QLatin1String(USER_PROFILE_PATH) + dir;
      if(fileName[0] != '/')
            fileName.prepend('/');
      if(QFile::exists(f + fileName))
            return f;
      else
            return QLatin1String(DATA_LOCATION) + dir;
}

void loadPlugins()
{
      loadPlugins(PLUGIN_LOCATION);
      
      if(const char* p = getenv("PLUGIN_PATH"))
            loadPlugins(p);
}

void loadPlugins(const char* p)
{
      QDir dir(p);
      QStringList plugins = dir.entryList(QStringList("*.so"));
      
      foreach(QString pl, plugins)
      {
            QByteArray p = dir.filePath(pl).toUtf8();
            qDebug() << "dlopen" << p;
            if(void* l = dlopen(p.constData(), RTLD_NOW))
            {
                  Logger::global()->enterLogMessage(QObject::tr("Loaded a plugin:") + ' ' + pl);
                  
                  PluginInfo (*info)() = (PluginInfo(*)()) dlsym(l, "getInfo");
                  if(info != 0)
                  {
                        PluginInfo i = info();
                        g_plugins[pl] = i;
                        
                        if(strcmp(i.version, VERSION))
                        {
                              qDebug() << "WARNING! Version conflict." << pl << "is" << i.version << "but FatRat is" << VERSION;
                              Logger::global()->enterLogMessage(QObject::tr("WARNING: the plugin is incompatible:") + ' ' + pl);
                        }
                  }
            }
            else
                  Logger::global()->enterLogMessage(QObject::tr("Failed to load a plugin: %1: %2").arg(pl).arg(dlerror()));
      }
}

bool programHasGUI()
{
      return m_bStartGUI;
}

void showHelp()
{
      std::cout << "FatRat download manager ("VERSION")\n\n"
                  "Copyright (C) 2006-2008 Lubos Dolezel\n"
                  "Licensed under the terms of the GNU GPL version 2 as published by the Free Software Foundation\n\n"
                  "-f, --force \tRun the program even if an instance already exists\n"
                  "-i, --hidden\tHide the GUI at startup (only if the tray icon exists)\n"
                  "-n, --nogui \tStart with no GUI at all\n"
                  "-h, --help  \tShow this help\n\n"
                  "If started in the GUI mode, you may pass transfers as arguments and they will be presented to the user\n";
      exit(0);
}

static void terminateHandler(int s)
{
      static int callno = 0;
      if(!callno++)
            QCoreApplication::exit();
      else
            exit(s);
}

void installSignalHandler()
{
      struct sigaction act;

      act.sa_handler = terminateHandler;
      act.sa_flags = SA_NOMASK;
      act.sa_restorer = 0;

      sigaction(SIGINT, &act, 0);
      sigaction(SIGTERM, &act, 0);
}


Generated by  Doxygen 1.6.0   Back to index