2013-11-14 00:08:28 +05:30
|
|
|
/* Copyright 2013 MultiMC Contributors
|
2013-11-12 00:29:59 +05:30
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <QObject>
|
|
|
|
#include <QString>
|
2013-11-15 02:02:43 +05:30
|
|
|
#include <QList>
|
2013-11-19 00:28:03 +05:30
|
|
|
#include <QJsonObject>
|
2013-12-01 06:30:42 +05:30
|
|
|
#include <QPair>
|
2013-12-13 07:17:59 +05:30
|
|
|
#include <QMap>
|
2013-11-15 02:02:43 +05:30
|
|
|
|
2013-11-18 23:35:35 +05:30
|
|
|
#include <memory>
|
2013-11-15 02:02:43 +05:30
|
|
|
|
2013-12-08 22:04:45 +05:30
|
|
|
class Task;
|
2013-12-05 07:09:52 +05:30
|
|
|
class YggdrasilTask;
|
2013-11-19 00:28:03 +05:30
|
|
|
class MojangAccount;
|
|
|
|
|
|
|
|
typedef std::shared_ptr<MojangAccount> MojangAccountPtr;
|
|
|
|
Q_DECLARE_METATYPE(MojangAccountPtr)
|
|
|
|
|
2013-11-15 02:02:43 +05:30
|
|
|
/**
|
2013-12-05 07:09:52 +05:30
|
|
|
* A profile within someone's Mojang account.
|
2013-11-15 02:02:43 +05:30
|
|
|
*
|
|
|
|
* Currently, the profile system has not been implemented by Mojang yet,
|
|
|
|
* but we might as well add some things for it in MultiMC right now so
|
|
|
|
* we don't have to rip the code to pieces to add it later.
|
|
|
|
*/
|
2013-12-05 07:09:52 +05:30
|
|
|
struct AccountProfile
|
2013-11-15 02:02:43 +05:30
|
|
|
{
|
2013-12-05 07:09:52 +05:30
|
|
|
QString id;
|
|
|
|
QString name;
|
2013-12-13 07:17:59 +05:30
|
|
|
bool legacy;
|
2013-11-15 02:02:43 +05:30
|
|
|
};
|
|
|
|
|
2013-12-01 06:30:42 +05:30
|
|
|
struct User
|
|
|
|
{
|
|
|
|
QString id;
|
2013-12-13 07:17:59 +05:30
|
|
|
QMultiMap<QString,QString> properties;
|
2013-12-01 06:30:42 +05:30
|
|
|
};
|
2013-11-12 00:29:59 +05:30
|
|
|
|
2013-12-05 07:09:52 +05:30
|
|
|
enum AccountStatus
|
|
|
|
{
|
|
|
|
NotVerified,
|
|
|
|
Verified,
|
|
|
|
Online
|
|
|
|
};
|
|
|
|
|
2013-11-12 00:29:59 +05:30
|
|
|
/**
|
|
|
|
* Object that stores information about a certain Mojang account.
|
|
|
|
*
|
2013-12-01 06:30:42 +05:30
|
|
|
* Said information may include things such as that account's username, client token, and access
|
2013-11-12 00:29:59 +05:30
|
|
|
* token if the user chose to stay logged in.
|
|
|
|
*/
|
|
|
|
class MojangAccount : public QObject
|
|
|
|
{
|
2013-12-01 06:30:42 +05:30
|
|
|
Q_OBJECT
|
2013-12-05 07:09:52 +05:30
|
|
|
public: /* construction */
|
|
|
|
//! Do not copy accounts. ever.
|
|
|
|
explicit MojangAccount(const MojangAccount &other, QObject *parent) = delete;
|
2013-11-12 00:29:59 +05:30
|
|
|
|
2013-12-05 07:09:52 +05:30
|
|
|
//! Default constructor
|
|
|
|
explicit MojangAccount(QObject *parent = 0) : QObject(parent) {};
|
2013-11-12 00:29:59 +05:30
|
|
|
|
2013-12-05 07:09:52 +05:30
|
|
|
//! Creates an empty account for the specified user name.
|
|
|
|
static MojangAccountPtr createFromUsername(const QString &username);
|
2013-11-18 23:35:35 +05:30
|
|
|
|
2013-12-05 07:09:52 +05:30
|
|
|
//! Loads a MojangAccount from the given JSON object.
|
2013-12-01 06:30:42 +05:30
|
|
|
static MojangAccountPtr loadFromJson(const QJsonObject &json);
|
2013-11-19 00:28:03 +05:30
|
|
|
|
2013-12-05 07:09:52 +05:30
|
|
|
//! Saves a MojangAccount to a JSON object and returns it.
|
|
|
|
QJsonObject saveToJson() const;
|
2013-11-19 00:28:03 +05:30
|
|
|
|
2013-12-05 07:09:52 +05:30
|
|
|
public: /* manipulation */
|
2013-12-01 06:30:42 +05:30
|
|
|
/**
|
2013-12-05 07:09:52 +05:30
|
|
|
* Sets the currently selected profile to the profile with the given ID string.
|
|
|
|
* If profileId is not in the list of available profiles, the function will simply return
|
|
|
|
* false.
|
2013-11-12 00:29:59 +05:30
|
|
|
*/
|
2013-12-05 07:09:52 +05:30
|
|
|
bool setCurrentProfile(const QString &profileId);
|
2013-11-12 00:29:59 +05:30
|
|
|
|
|
|
|
/**
|
2013-12-05 07:09:52 +05:30
|
|
|
* Attempt to login. Empty password means we use the token.
|
|
|
|
* If the attempt fails because we already are performing some task, it returns false.
|
2013-11-12 00:29:59 +05:30
|
|
|
*/
|
2013-12-08 22:04:45 +05:30
|
|
|
std::shared_ptr<Task> login(QString password = QString());
|
2013-11-12 00:29:59 +05:30
|
|
|
|
2013-12-09 02:36:04 +05:30
|
|
|
void downgrade()
|
|
|
|
{
|
|
|
|
m_online = false;
|
|
|
|
}
|
2013-12-05 07:09:52 +05:30
|
|
|
public: /* queries */
|
|
|
|
const QString &username() const
|
|
|
|
{
|
|
|
|
return m_username;
|
|
|
|
}
|
2013-11-15 02:02:43 +05:30
|
|
|
|
2013-12-05 07:09:52 +05:30
|
|
|
const QString &clientToken() const
|
|
|
|
{
|
|
|
|
return m_clientToken;
|
|
|
|
}
|
2013-11-24 23:11:35 +05:30
|
|
|
|
2013-12-05 07:09:52 +05:30
|
|
|
const QString &accessToken() const
|
|
|
|
{
|
|
|
|
return m_accessToken;
|
|
|
|
}
|
2013-11-15 02:02:43 +05:30
|
|
|
|
2013-12-05 07:09:52 +05:30
|
|
|
const QList<AccountProfile> &profiles() const
|
|
|
|
{
|
|
|
|
return m_profiles;
|
|
|
|
}
|
2013-11-24 23:11:35 +05:30
|
|
|
|
2013-12-05 07:09:52 +05:30
|
|
|
//! Get the session ID required for legacy Minecraft versions
|
|
|
|
QString sessionId() const
|
|
|
|
{
|
|
|
|
if (m_currentProfile != -1 && !m_accessToken.isEmpty())
|
|
|
|
return "token:" + m_accessToken + ":" + m_profiles[m_currentProfile].id;
|
|
|
|
return "-";
|
|
|
|
}
|
2013-11-15 02:02:43 +05:30
|
|
|
|
2013-12-05 07:09:52 +05:30
|
|
|
//! Returns the currently selected profile (if none, returns nullptr)
|
2013-12-01 06:30:42 +05:30
|
|
|
const AccountProfile *currentProfile() const;
|
2013-11-15 02:02:43 +05:30
|
|
|
|
2013-12-05 07:09:52 +05:30
|
|
|
//! Returns whether the account is NotVerified, Verified or Online
|
|
|
|
AccountStatus accountStatus() const;
|
2013-11-15 02:02:43 +05:30
|
|
|
|
2013-12-01 06:30:42 +05:30
|
|
|
signals:
|
|
|
|
/**
|
2013-12-05 07:09:52 +05:30
|
|
|
* This signal is emitted when the account changes
|
2013-12-01 06:30:42 +05:30
|
|
|
*/
|
|
|
|
void changed();
|
2013-11-12 00:29:59 +05:30
|
|
|
|
2013-12-05 07:09:52 +05:30
|
|
|
// TODO: better signalling for the various possible state changes - especially errors
|
|
|
|
|
|
|
|
protected: /* variables */
|
2013-11-12 00:29:59 +05:30
|
|
|
QString m_username;
|
2013-12-05 07:09:52 +05:30
|
|
|
|
|
|
|
// Used to identify the client - the user can have multiple clients for the same account
|
|
|
|
// Think: different launchers, all connecting to the same account/profile
|
2013-11-12 00:29:59 +05:30
|
|
|
QString m_clientToken;
|
2013-12-05 07:09:52 +05:30
|
|
|
|
|
|
|
// Blank if not logged in.
|
|
|
|
QString m_accessToken;
|
|
|
|
|
|
|
|
// Index of the selected profile within the list of available
|
|
|
|
// profiles. -1 if nothing is selected.
|
|
|
|
int m_currentProfile = -1;
|
|
|
|
|
|
|
|
// List of available profiles.
|
|
|
|
QList<AccountProfile> m_profiles;
|
|
|
|
|
|
|
|
// the user structure, whatever it is.
|
|
|
|
User m_user;
|
|
|
|
|
|
|
|
// true when the account is verified
|
|
|
|
bool m_online = false;
|
|
|
|
|
|
|
|
// current task we are executing here
|
|
|
|
std::shared_ptr<YggdrasilTask> m_currentTask;
|
|
|
|
|
|
|
|
private slots:
|
|
|
|
void authSucceeded();
|
|
|
|
void authFailed(QString reason);
|
|
|
|
|
|
|
|
public:
|
|
|
|
friend class YggdrasilTask;
|
|
|
|
friend class AuthenticateTask;
|
|
|
|
friend class ValidateTask;
|
|
|
|
friend class RefreshTask;
|
2013-11-12 00:29:59 +05:30
|
|
|
};
|