main.cpp 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288
  1. #include <locale.h>
  2. #include <QGuiApplication>
  3. #include <QFileInfo>
  4. #include <QIcon>
  5. #include <QtQml>
  6. #include <QtWebEngine/qtwebengineglobal.h>
  7. #include <shared/Names.h>
  8. #include "system/SystemComponent.h"
  9. #include "system/UpdateManager.h"
  10. #include "QsLog.h"
  11. #include "Paths.h"
  12. #include "player/PlayerComponent.h"
  13. #include "breakpad/CrashDumps.h"
  14. #include "Version.h"
  15. #include "settings/SettingsComponent.h"
  16. #include "ui/KonvergoWindow.h"
  17. #include "ui/KonvergoEngine.h"
  18. #include "UniqueApplication.h"
  19. #include "utils/HelperLauncher.h"
  20. #if defined(Q_OS_MAC) || defined(Q_OS_LINUX)
  21. #include "SignalManager.h"
  22. #endif
  23. using namespace QsLogging;
  24. /////////////////////////////////////////////////////////////////////////////////////////
  25. void initQt(QGuiApplication* app)
  26. {
  27. QCoreApplication::setApplicationName(Names::MainName());
  28. QCoreApplication::setApplicationVersion(Version::GetVersionString());
  29. QCoreApplication::setOrganizationDomain("plex.tv");
  30. app->setWindowIcon(QIcon(":/images/icon.png"));
  31. }
  32. /////////////////////////////////////////////////////////////////////////////////////////
  33. static void qtMessageOutput(QtMsgType type, const QMessageLogContext& context, const QString& msg)
  34. {
  35. QByteArray localMsg = msg.toLocal8Bit();
  36. QString prefix;
  37. if (context.line)
  38. prefix = QString("%1:%2:%3: ").arg(context.file).arg(context.line).arg(context.function);
  39. QString text = prefix + msg;
  40. switch (type)
  41. {
  42. case QtDebugMsg:
  43. QLOG_DEBUG() << text;
  44. break;
  45. case QtInfoMsg:
  46. QLOG_INFO() << text;
  47. break;
  48. case QtWarningMsg:
  49. QLOG_WARN() << text;
  50. break;
  51. case QtCriticalMsg:
  52. QLOG_ERROR() << text;
  53. break;
  54. case QtFatalMsg:
  55. QLOG_FATAL() << text;
  56. break;
  57. }
  58. }
  59. /////////////////////////////////////////////////////////////////////////////////////////
  60. static void elidePattern(QString& msg, const QString& substring, int chars)
  61. {
  62. int start = 0;
  63. while (true)
  64. {
  65. start = msg.indexOf(substring, start);
  66. if (start < 0 || start + substring.length() + chars > msg.length())
  67. break;
  68. start += substring.length();
  69. for (int n = 0; n < chars; n++)
  70. msg[start + n] = QChar('x');
  71. }
  72. }
  73. /////////////////////////////////////////////////////////////////////////////////////////
  74. static void processLog(QString& msg)
  75. {
  76. elidePattern(msg, "X-Plex-Token=", 20);
  77. }
  78. /////////////////////////////////////////////////////////////////////////////////////////
  79. void initLogger()
  80. {
  81. // Note where the logfile is going to be
  82. qDebug("Logging to %s", qPrintable(Paths::logDir(Names::MainName() + ".log")));
  83. // init logging.
  84. DestinationPtr dest = DestinationFactory::MakeFileDestination(
  85. Paths::logDir(Names::MainName() + ".log"),
  86. EnableLogRotationOnOpen,
  87. MaxSizeBytes(1024 * 1024),
  88. MaxOldLogCount(9));
  89. Logger::instance().addDestination(dest);
  90. Logger::instance().setLoggingLevel(DebugLevel);
  91. Logger::instance().setProcessingCallback(processLog);
  92. qInstallMessageHandler(qtMessageOutput);
  93. }
  94. /////////////////////////////////////////////////////////////////////////////////////////
  95. char** appendCommandLineArguments(int *argc, char **argv)
  96. {
  97. static char *newArgs[16];
  98. QList<QString> argList;
  99. // Copy argv list to our StringList
  100. for (int i=0; i < *argc; i++)
  101. {
  102. argList << QString(argv[i]);
  103. }
  104. // add any required additionnal commandline argument
  105. #if KONVERGO_OPENELEC
  106. // on RPI with webengine, OpenGL contexts are shared statically with webengine
  107. // which avoids proper reset when switching display mode
  108. // On OE we also need that because there is a crash with OZONE otherwise
  109. argList << "--disable-gpu";
  110. #endif
  111. // with webengine we need those to have a proper scaling of the webview in the window
  112. argList << "--enable-viewport";
  113. argList << "--enable-viewport-meta";
  114. // Now rebuild our argc, argv list
  115. *argc = argList.size();
  116. for(int iarg=0; iarg < argList.size(); iarg++)
  117. {
  118. newArgs[iarg] = (char*)malloc(256);
  119. strcpy(newArgs[iarg], argList.value(iarg).toStdString().c_str());
  120. }
  121. return (char**)newArgs;
  122. }
  123. /////////////////////////////////////////////////////////////////////////////////////////
  124. int main(int argc, char *argv[])
  125. {
  126. try
  127. {
  128. for (int n = 1; n < argc; n++) {
  129. if (strcmp(argv[n], "--licenses") == 0) {
  130. QFile licenses(":/misc/licenses.txt");
  131. licenses.open(QIODevice::ReadOnly | QIODevice::Text);
  132. QByteArray contents = licenses.readAll();
  133. printf("%.*s\n", (int)contents.size(), contents.data());
  134. return 0;
  135. }
  136. }
  137. int newArgc = argc;
  138. char **newArgv = appendCommandLineArguments(&newArgc, argv);
  139. // Supress SSL related warnings on OSX
  140. // See https://bugreports.qt.io/browse/QTBUG-43173 for more info
  141. //
  142. #ifdef Q_OS_MAC
  143. qputenv("QT_LOGGING_RULES", "qt.network.ssl.warning=false");
  144. // Request OpenGL 4.1 if possible on OSX, otherwise it defaults to 2.0
  145. // This needs to be done before we create the QGuiApplication
  146. //
  147. QSurfaceFormat format = QSurfaceFormat::defaultFormat();
  148. format.setMajorVersion(3);
  149. format.setMinorVersion(2);
  150. format.setProfile(QSurfaceFormat::CoreProfile);
  151. QSurfaceFormat::setDefaultFormat(format);
  152. #endif
  153. QGuiApplication app(newArgc, newArgv);
  154. initQt(&app);
  155. // init breakpad.
  156. setupCrashDumper();
  157. UniqueApplication* uniqueApp = new UniqueApplication();
  158. if (!uniqueApp->ensureUnique())
  159. return EXIT_SUCCESS;
  160. #ifdef Q_OS_UNIX
  161. // install signals handlers for proper app closing.
  162. SignalManager signalManager(&app);
  163. Q_UNUSED(signalManager);
  164. #endif
  165. initLogger();
  166. QLOG_INFO() << "Starting Plex Media Player version:" << qPrintable(Version::GetVersionString()) << "build date:" << qPrintable(Version::GetBuildDate());
  167. QLOG_INFO() << qPrintable(QString(" Running on: %1 [%2] arch %3").arg(QSysInfo::prettyProductName()).arg(QSysInfo::kernelVersion()).arg(QSysInfo::currentCpuArchitecture()));
  168. QLOG_INFO() << " Qt Version:" << QT_VERSION_STR << qPrintable(QString("[%1]").arg(QSysInfo::buildAbi()));
  169. // Quit app and apply update if we find one.
  170. if (UpdateManager::CheckForUpdates())
  171. {
  172. app.quit();
  173. return 0;
  174. }
  175. #ifdef Q_OS_WIN32
  176. initD3DDevice();
  177. #endif
  178. #ifdef Q_OS_UNIX
  179. setlocale(LC_NUMERIC, "C");
  180. #endif
  181. // Initialize all the components. This needs to be done
  182. // early since most everything else relies on it
  183. //
  184. ComponentManager::Get().initialize();
  185. // enable remote inspection if we have the correct setting for it.
  186. if (SettingsComponent::Get().value(SETTINGS_SECTION_MAIN, "remoteInspector").toBool())
  187. qputenv("QTWEBENGINE_REMOTE_DEBUGGING", "0.0.0.0:9992");
  188. QtWebEngine::initialize();
  189. // Qt and QWebEngineProfile set the locale, which breaks parsing and
  190. // formatting float numbers in a few countries.
  191. #ifdef Q_OS_UNIX
  192. setlocale(LC_NUMERIC, "C");
  193. #endif
  194. // start our helper
  195. HelperLauncher::Get().connectToHelper();
  196. // load QtWebChannel so that we can register our components with it.
  197. QQmlApplicationEngine *engine = KonvergoEngine::Get();
  198. KonvergoWindow::RegisterClass();
  199. engine->rootContext()->setContextProperty("components", &ComponentManager::Get().getQmlPropertyMap());
  200. // the only way to detect if QML parsing fails is to hook to this signal and then see
  201. // if we get a valid object passed to it. Any error messages will be reported on stderr
  202. // but since no normal user should ever see this it should be fine
  203. //
  204. QObject::connect(engine, &QQmlApplicationEngine::objectCreated, [=](QObject* object, const QUrl& url)
  205. {
  206. Q_UNUSED(url);
  207. if (object == 0)
  208. throw FatalException(QObject::tr("Failed to parse application engine script."));
  209. QObject* rootObject = engine->rootObjects().first();
  210. QObject* webChannel = qvariant_cast<QObject*>(rootObject->property("webChannel"));
  211. Q_ASSERT(webChannel);
  212. ComponentManager::Get().setWebChannel(qobject_cast<QWebChannel*>(webChannel));
  213. KonvergoWindow* window = qobject_cast<KonvergoWindow*>(rootObject);
  214. Q_ASSERT(window);
  215. QObject::connect(uniqueApp, &UniqueApplication::otherApplicationStarted, window, &KonvergoWindow::otherAppFocus);
  216. });
  217. engine->load(QUrl(QStringLiteral("qrc:/ui/webview.qml")));
  218. // run our application
  219. int ret = app.exec();
  220. delete KonvergoEngine::Get();
  221. delete uniqueApp;
  222. return ret;
  223. }
  224. catch (FatalException& e)
  225. {
  226. QLOG_FATAL() << "Unhandled FatalException:" << qPrintable(e.message());
  227. QGuiApplication app(argc, argv);
  228. QString text = e.message() + "<br>" + QObject::tr("Please visit Plex support forums for support.");
  229. QQmlApplicationEngine* engine = new QQmlApplicationEngine(NULL);
  230. engine->rootContext()->setContextProperty("errorTitle", QObject::tr("A critical error occurred."));
  231. engine->rootContext()->setContextProperty("errorText", text);
  232. engine->load(QUrl(QStringLiteral("qrc:/ui/errormessage.qml")));
  233. app.exec();
  234. return 1;
  235. }
  236. }