main.cpp 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286
  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:" << Version::GetVersionString() << "build date:" << Version::GetBuildDate();
  167. // Quit app and apply update if we find one.
  168. if (UpdateManager::CheckForUpdates())
  169. {
  170. app.quit();
  171. return 0;
  172. }
  173. #ifdef Q_OS_WIN32
  174. initD3DDevice();
  175. #endif
  176. #ifdef Q_OS_UNIX
  177. setlocale(LC_NUMERIC, "C");
  178. #endif
  179. // Initialize all the components. This needs to be done
  180. // early since most everything else relies on it
  181. //
  182. ComponentManager::Get().initialize();
  183. // enable remote inspection if we have the correct setting for it.
  184. if (SettingsComponent::Get().value(SETTINGS_SECTION_MAIN, "remoteInspector").toBool())
  185. qputenv("QTWEBENGINE_REMOTE_DEBUGGING", "0.0.0.0:9992");
  186. QtWebEngine::initialize();
  187. // Qt and QWebEngineProfile set the locale, which breaks parsing and
  188. // formatting float numbers in a few countries.
  189. #ifdef Q_OS_UNIX
  190. setlocale(LC_NUMERIC, "C");
  191. #endif
  192. // start our helper
  193. HelperLauncher::Get().connectToHelper();
  194. // load QtWebChannel so that we can register our components with it.
  195. QQmlApplicationEngine *engine = KonvergoEngine::Get();
  196. KonvergoWindow::RegisterClass();
  197. engine->rootContext()->setContextProperty("components", &ComponentManager::Get().getQmlPropertyMap());
  198. // the only way to detect if QML parsing fails is to hook to this signal and then see
  199. // if we get a valid object passed to it. Any error messages will be reported on stderr
  200. // but since no normal user should ever see this it should be fine
  201. //
  202. QObject::connect(engine, &QQmlApplicationEngine::objectCreated, [=](QObject* object, const QUrl& url)
  203. {
  204. Q_UNUSED(url);
  205. if (object == 0)
  206. throw FatalException(QObject::tr("Failed to parse application engine script."));
  207. QObject* rootObject = engine->rootObjects().first();
  208. QObject* webChannel = qvariant_cast<QObject*>(rootObject->property("webChannel"));
  209. Q_ASSERT(webChannel);
  210. ComponentManager::Get().setWebChannel(qobject_cast<QWebChannel*>(webChannel));
  211. KonvergoWindow* window = qobject_cast<KonvergoWindow*>(rootObject);
  212. Q_ASSERT(window);
  213. QObject::connect(uniqueApp, &UniqueApplication::otherApplicationStarted, window, &KonvergoWindow::otherAppFocus);
  214. });
  215. engine->load(QUrl(QStringLiteral("qrc:/ui/webview.qml")));
  216. // run our application
  217. int ret = app.exec();
  218. delete KonvergoEngine::Get();
  219. delete uniqueApp;
  220. return ret;
  221. }
  222. catch (FatalException& e)
  223. {
  224. QLOG_FATAL() << "Unhandled FatalException:" << qPrintable(e.message());
  225. QGuiApplication app(argc, argv);
  226. QString text = e.message() + "<br>" + QObject::tr("Please visit Plex support forums for support.");
  227. QQmlApplicationEngine* engine = new QQmlApplicationEngine(NULL);
  228. engine->rootContext()->setContextProperty("errorTitle", QObject::tr("A critical error occurred."));
  229. engine->rootContext()->setContextProperty("errorText", text);
  230. engine->load(QUrl(QStringLiteral("qrc:/ui/errormessage.qml")));
  231. app.exec();
  232. return 1;
  233. }
  234. }