2022-03-27 13:13:32 +02:00
|
|
|
// SPDX-License-Identifier: GPL-3.0-only
|
|
|
|
/*
|
|
|
|
* PolyMC - Minecraft Launcher
|
|
|
|
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
2015-05-11 22:21:37 +02:00
|
|
|
*
|
2022-03-27 13:13:32 +02:00
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation, version 3.
|
2015-05-11 22:21:37 +02:00
|
|
|
*
|
2022-03-27 13:13:32 +02:00
|
|
|
* 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.
|
2015-05-11 22:21:37 +02:00
|
|
|
*
|
2022-03-27 13:13:32 +02:00
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
2015-05-11 22:21:37 +02:00
|
|
|
*
|
2022-03-27 13:13:32 +02:00
|
|
|
* This file incorporates work covered by the following copyright and
|
|
|
|
* permission notice:
|
|
|
|
*
|
|
|
|
* Copyright 2013-2021 MultiMC Contributors
|
|
|
|
*
|
|
|
|
* Authors: Orochimarufan <orochimarufan.x3@gmail.com>
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
2015-05-11 22:21:37 +02:00
|
|
|
*/
|
|
|
|
|
2015-07-10 01:11:06 +02:00
|
|
|
#include "launch/LaunchTask.h"
|
2015-05-11 22:21:37 +02:00
|
|
|
#include "MessageLevel.h"
|
2015-06-16 09:19:55 +02:00
|
|
|
#include "MMCStrings.h"
|
|
|
|
#include "java/JavaChecker.h"
|
2015-07-04 20:02:43 +02:00
|
|
|
#include "tasks/Task.h"
|
2015-05-11 22:21:37 +02:00
|
|
|
#include <QDebug>
|
|
|
|
#include <QDir>
|
|
|
|
#include <QEventLoop>
|
2015-06-16 09:19:55 +02:00
|
|
|
#include <QRegularExpression>
|
|
|
|
#include <QCoreApplication>
|
|
|
|
#include <QStandardPaths>
|
2015-07-21 02:38:15 +02:00
|
|
|
#include <assert.h>
|
2015-07-04 20:02:43 +02:00
|
|
|
|
2015-07-10 01:11:06 +02:00
|
|
|
void LaunchTask::init()
|
2015-07-04 20:02:43 +02:00
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
m_instance->setRunning(true);
|
2015-07-04 20:02:43 +02:00
|
|
|
}
|
|
|
|
|
2019-04-07 23:59:04 +02:00
|
|
|
shared_qobject_ptr<LaunchTask> LaunchTask::create(InstancePtr inst)
|
2015-06-16 09:19:55 +02:00
|
|
|
{
|
2019-04-07 23:59:04 +02:00
|
|
|
shared_qobject_ptr<LaunchTask> proc(new LaunchTask(inst));
|
2018-07-15 14:51:05 +02:00
|
|
|
proc->init();
|
|
|
|
return proc;
|
2015-06-16 09:19:55 +02:00
|
|
|
}
|
|
|
|
|
2015-07-10 01:11:06 +02:00
|
|
|
LaunchTask::LaunchTask(InstancePtr instance): m_instance(instance)
|
2015-05-11 22:21:37 +02:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2019-04-07 23:59:04 +02:00
|
|
|
void LaunchTask::appendStep(shared_qobject_ptr<LaunchStep> step)
|
2015-06-16 09:19:55 +02:00
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
m_steps.append(step);
|
2015-07-21 02:38:15 +02:00
|
|
|
}
|
2015-06-16 09:19:55 +02:00
|
|
|
|
2019-04-07 23:59:04 +02:00
|
|
|
void LaunchTask::prependStep(shared_qobject_ptr<LaunchStep> step)
|
2015-07-21 02:38:15 +02:00
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
m_steps.prepend(step);
|
2015-07-21 02:38:15 +02:00
|
|
|
}
|
2015-06-16 09:19:55 +02:00
|
|
|
|
2015-07-21 02:38:15 +02:00
|
|
|
void LaunchTask::executeTask()
|
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
m_instance->setCrashed(false);
|
|
|
|
if(!m_steps.size())
|
|
|
|
{
|
|
|
|
state = LaunchTask::Finished;
|
|
|
|
emitSucceeded();
|
|
|
|
}
|
|
|
|
state = LaunchTask::Running;
|
|
|
|
onStepFinished();
|
2015-07-21 02:38:15 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void LaunchTask::onReadyForLaunch()
|
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
state = LaunchTask::Waiting;
|
|
|
|
emit readyForLaunch();
|
2015-07-21 02:38:15 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void LaunchTask::onStepFinished()
|
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
// initial -> just start the first step
|
|
|
|
if(currentStep == -1)
|
|
|
|
{
|
|
|
|
currentStep ++;
|
|
|
|
m_steps[currentStep]->start();
|
|
|
|
return;
|
|
|
|
}
|
2015-07-21 02:38:15 +02:00
|
|
|
|
2018-07-15 14:51:05 +02:00
|
|
|
auto step = m_steps[currentStep];
|
|
|
|
if(step->wasSuccessful())
|
|
|
|
{
|
|
|
|
// end?
|
|
|
|
if(currentStep == m_steps.size() - 1)
|
|
|
|
{
|
|
|
|
finalizeSteps(true, QString());
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
currentStep ++;
|
|
|
|
step = m_steps[currentStep];
|
|
|
|
step->start();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
finalizeSteps(false, step->failReason());
|
|
|
|
}
|
2016-11-15 02:51:22 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void LaunchTask::finalizeSteps(bool successful, const QString& error)
|
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
for(auto step = currentStep; step >= 0; step--)
|
|
|
|
{
|
|
|
|
m_steps[step]->finalize();
|
|
|
|
}
|
|
|
|
if(successful)
|
|
|
|
{
|
|
|
|
emitSucceeded();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
emitFailed(error);
|
|
|
|
}
|
2015-07-21 02:38:15 +02:00
|
|
|
}
|
2015-06-16 09:19:55 +02:00
|
|
|
|
2015-07-26 17:55:29 +02:00
|
|
|
void LaunchTask::onProgressReportingRequested()
|
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
state = LaunchTask::Waiting;
|
|
|
|
emit requestProgress(m_steps[currentStep].get());
|
2015-07-26 17:55:29 +02:00
|
|
|
}
|
|
|
|
|
2015-07-21 02:38:15 +02:00
|
|
|
void LaunchTask::setCensorFilter(QMap<QString, QString> filter)
|
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
m_censorFilter = filter;
|
2015-07-21 02:38:15 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
QString LaunchTask::censorPrivateInfo(QString in)
|
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
auto iter = m_censorFilter.begin();
|
|
|
|
while (iter != m_censorFilter.end())
|
|
|
|
{
|
|
|
|
in.replace(iter.key(), iter.value());
|
|
|
|
iter++;
|
|
|
|
}
|
|
|
|
return in;
|
2015-06-16 09:19:55 +02:00
|
|
|
}
|
|
|
|
|
2015-07-21 02:38:15 +02:00
|
|
|
void LaunchTask::proceed()
|
2015-06-16 09:19:55 +02:00
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
if(state != LaunchTask::Waiting)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
m_steps[currentStep]->proceed();
|
2015-06-16 09:19:55 +02:00
|
|
|
}
|
|
|
|
|
2016-11-26 18:06:08 +01:00
|
|
|
bool LaunchTask::canAbort() const
|
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
switch(state)
|
|
|
|
{
|
|
|
|
case LaunchTask::Aborted:
|
|
|
|
case LaunchTask::Failed:
|
|
|
|
case LaunchTask::Finished:
|
|
|
|
return false;
|
|
|
|
case LaunchTask::NotStarted:
|
|
|
|
return true;
|
|
|
|
case LaunchTask::Running:
|
|
|
|
case LaunchTask::Waiting:
|
|
|
|
{
|
|
|
|
auto step = m_steps[currentStep];
|
|
|
|
return step->canAbort();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
2016-11-26 18:06:08 +01:00
|
|
|
}
|
|
|
|
|
2015-07-21 02:38:15 +02:00
|
|
|
bool LaunchTask::abort()
|
2015-06-16 09:19:55 +02:00
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
switch(state)
|
|
|
|
{
|
|
|
|
case LaunchTask::Aborted:
|
|
|
|
case LaunchTask::Failed:
|
|
|
|
case LaunchTask::Finished:
|
|
|
|
return true;
|
|
|
|
case LaunchTask::NotStarted:
|
|
|
|
{
|
|
|
|
state = LaunchTask::Aborted;
|
|
|
|
emitFailed("Aborted");
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
case LaunchTask::Running:
|
|
|
|
case LaunchTask::Waiting:
|
|
|
|
{
|
|
|
|
auto step = m_steps[currentStep];
|
|
|
|
if(!step->canAbort())
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if(step->abort())
|
|
|
|
{
|
|
|
|
state = LaunchTask::Aborted;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return false;
|
2015-07-04 20:02:43 +02:00
|
|
|
}
|
2015-06-16 09:19:55 +02:00
|
|
|
|
2016-08-18 21:31:37 +02:00
|
|
|
shared_qobject_ptr<LogModel> LaunchTask::getLogModel()
|
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
if(!m_logModel)
|
|
|
|
{
|
|
|
|
m_logModel.reset(new LogModel());
|
|
|
|
m_logModel->setMaxLines(m_instance->getConsoleMaxLines());
|
|
|
|
m_logModel->setStopOnOverflow(m_instance->shouldStopOnConsoleOverflow());
|
|
|
|
// FIXME: should this really be here?
|
2022-03-26 22:21:08 +01:00
|
|
|
m_logModel->setOverflowMessage(tr("Stopped watching the game log because the log length surpassed %1 lines.\n"
|
2018-07-15 14:51:05 +02:00
|
|
|
"You may have to fix your mods because the game is still logging to files and"
|
|
|
|
" likely wasting harddrive space at an alarming rate!").arg(m_logModel->getMaxLines()));
|
|
|
|
}
|
|
|
|
return m_logModel;
|
2016-08-18 21:31:37 +02:00
|
|
|
}
|
|
|
|
|
2015-07-21 02:38:15 +02:00
|
|
|
void LaunchTask::onLogLines(const QStringList &lines, MessageLevel::Enum defaultLevel)
|
2015-05-11 22:21:37 +02:00
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
for (auto & line: lines)
|
|
|
|
{
|
|
|
|
onLogLine(line, defaultLevel);
|
|
|
|
}
|
2015-05-11 22:21:37 +02:00
|
|
|
}
|
|
|
|
|
2015-07-21 02:38:15 +02:00
|
|
|
void LaunchTask::onLogLine(QString line, MessageLevel::Enum level)
|
2015-05-11 22:21:37 +02:00
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
// if the launcher part set a log level, use it
|
|
|
|
auto innerLevel = MessageLevel::fromLine(line);
|
|
|
|
if(innerLevel != MessageLevel::Unknown)
|
|
|
|
{
|
|
|
|
level = innerLevel;
|
|
|
|
}
|
2015-05-11 22:21:37 +02:00
|
|
|
|
2018-07-15 14:51:05 +02:00
|
|
|
// If the level is still undetermined, guess level
|
|
|
|
if (level == MessageLevel::StdErr || level == MessageLevel::StdOut || level == MessageLevel::Unknown)
|
|
|
|
{
|
|
|
|
level = m_instance->guessLevel(line, level);
|
|
|
|
}
|
2015-05-11 22:21:37 +02:00
|
|
|
|
2018-07-15 14:51:05 +02:00
|
|
|
// censor private user info
|
|
|
|
line = censorPrivateInfo(line);
|
2015-05-11 22:21:37 +02:00
|
|
|
|
2018-07-15 14:51:05 +02:00
|
|
|
auto &model = *getLogModel();
|
|
|
|
model.append(level, line);
|
2015-05-11 22:21:37 +02:00
|
|
|
}
|
|
|
|
|
2015-07-10 01:11:06 +02:00
|
|
|
void LaunchTask::emitSucceeded()
|
2015-07-05 01:54:30 +02:00
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
m_instance->setRunning(false);
|
|
|
|
Task::emitSucceeded();
|
2015-07-05 01:54:30 +02:00
|
|
|
}
|
|
|
|
|
2015-07-10 01:11:06 +02:00
|
|
|
void LaunchTask::emitFailed(QString reason)
|
2015-07-05 01:54:30 +02:00
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
m_instance->setRunning(false);
|
|
|
|
m_instance->setCrashed(true);
|
|
|
|
Task::emitFailed(reason);
|
2015-07-05 01:54:30 +02:00
|
|
|
}
|
|
|
|
|
2022-05-02 19:10:45 +02:00
|
|
|
void LaunchTask::substituteVariables(const QStringList &args) const
|
|
|
|
{
|
|
|
|
auto variables = m_instance->getVariables();
|
|
|
|
auto envVariables = QProcessEnvironment::systemEnvironment();
|
|
|
|
|
|
|
|
for (auto arg : args) {
|
|
|
|
for (auto key : variables)
|
|
|
|
{
|
|
|
|
arg.replace("$" + key, variables.value(key));
|
|
|
|
}
|
|
|
|
for (auto env : envVariables.keys())
|
|
|
|
{
|
|
|
|
arg.replace("$" + env, envVariables.value(env));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-07-10 01:11:06 +02:00
|
|
|
QString LaunchTask::substituteVariables(const QString &cmd) const
|
2015-05-11 22:21:37 +02:00
|
|
|
{
|
2018-07-15 14:51:05 +02:00
|
|
|
QString out = cmd;
|
|
|
|
auto variables = m_instance->getVariables();
|
|
|
|
for (auto it = variables.begin(); it != variables.end(); ++it)
|
|
|
|
{
|
|
|
|
out.replace("$" + it.key(), it.value());
|
|
|
|
}
|
|
|
|
auto env = QProcessEnvironment::systemEnvironment();
|
|
|
|
for (auto var : env.keys())
|
|
|
|
{
|
|
|
|
out.replace("$" + var, env.value(var));
|
|
|
|
}
|
|
|
|
return out;
|
2015-05-11 22:21:37 +02:00
|
|
|
}
|