2020-08-25 16:05:56 +02:00
|
|
|
/* === This file is part of Calamares - <https://calamares.io> ===
|
2020-07-25 12:47:01 +02:00
|
|
|
*
|
|
|
|
* SPDX-FileCopyrightText: 2020 Adriaan de Groot <groot@kde.org>
|
|
|
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
|
|
|
*
|
2020-08-25 16:05:56 +02:00
|
|
|
* Calamares is Free Software: see the License-Identifier above.
|
2020-07-25 12:47:01 +02:00
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef USERS_CONFIG_H
|
|
|
|
#define USERS_CONFIG_H
|
|
|
|
|
2020-08-05 13:03:18 +02:00
|
|
|
#include "CheckPWQuality.h"
|
|
|
|
|
2020-08-18 11:31:32 +02:00
|
|
|
#include "Job.h"
|
2021-03-12 13:54:06 +01:00
|
|
|
#include "modulesystem/Config.h"
|
2020-08-05 10:29:13 +02:00
|
|
|
#include "utils/NamedEnum.h"
|
|
|
|
|
2020-10-13 17:35:07 +02:00
|
|
|
#include <QList>
|
2020-07-25 12:47:01 +02:00
|
|
|
#include <QObject>
|
|
|
|
#include <QVariantMap>
|
|
|
|
|
2020-08-05 10:29:13 +02:00
|
|
|
enum HostNameAction
|
|
|
|
{
|
|
|
|
None = 0x0,
|
|
|
|
EtcHostname = 0x1, // Write to /etc/hostname directly
|
|
|
|
SystemdHostname = 0x2, // Set via hostnamed(1)
|
|
|
|
WriteEtcHosts = 0x4 // Write /etc/hosts (127.0.1.1 is this host)
|
|
|
|
};
|
|
|
|
Q_DECLARE_FLAGS( HostNameActions, HostNameAction )
|
|
|
|
Q_DECLARE_OPERATORS_FOR_FLAGS( HostNameActions )
|
|
|
|
|
|
|
|
const NamedEnumTable< HostNameAction >& hostNameActionNames();
|
|
|
|
|
2020-10-13 17:35:07 +02:00
|
|
|
/** @brief Settings for a single group
|
|
|
|
*
|
|
|
|
* The list of defaultgroups from the configuration can be
|
|
|
|
* set up in a fine-grained way, with both user- and system-
|
|
|
|
* level groups; this class stores a configuration for each.
|
|
|
|
*/
|
|
|
|
class GroupDescription
|
|
|
|
{
|
|
|
|
public:
|
2020-10-13 22:58:02 +02:00
|
|
|
// TODO: still too-weakly typed, add a macro to define strongly-typed bools
|
2020-10-13 23:30:28 +02:00
|
|
|
class MustExist : public std::true_type
|
|
|
|
{
|
|
|
|
};
|
|
|
|
class CreateIfNeeded : public std::false_type
|
|
|
|
{
|
|
|
|
};
|
|
|
|
class SystemGroup : public std::true_type
|
|
|
|
{
|
|
|
|
};
|
|
|
|
class UserGroup : public std::false_type
|
|
|
|
{
|
|
|
|
};
|
2020-10-13 22:58:02 +02:00
|
|
|
|
2020-10-13 17:35:07 +02:00
|
|
|
///@brief An invalid, empty group
|
|
|
|
GroupDescription() {}
|
|
|
|
|
|
|
|
///@brief A group with full details
|
2020-10-13 23:30:28 +02:00
|
|
|
GroupDescription( const QString& name, bool mustExistAlready = CreateIfNeeded {}, bool isSystem = UserGroup {} )
|
2020-10-13 17:35:07 +02:00
|
|
|
: m_name( name )
|
|
|
|
, m_isValid( !name.isEmpty() )
|
|
|
|
, m_mustAlreadyExist( mustExistAlready )
|
|
|
|
, m_isSystem( isSystem )
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
bool isValid() const { return m_isValid; }
|
|
|
|
bool isSystemGroup() const { return m_isSystem; }
|
2020-10-22 14:07:40 +02:00
|
|
|
bool mustAlreadyExist() const { return m_mustAlreadyExist; }
|
2020-10-13 17:35:07 +02:00
|
|
|
QString name() const { return m_name; }
|
|
|
|
|
|
|
|
///@brief Equality of groups depends only on name and kind
|
|
|
|
bool operator==( const GroupDescription& rhs ) const
|
|
|
|
{
|
|
|
|
return rhs.name() == name() && rhs.isSystemGroup() == isSystemGroup();
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
QString m_name;
|
|
|
|
bool m_isValid = false;
|
|
|
|
bool m_mustAlreadyExist = false;
|
|
|
|
bool m_isSystem = false;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2021-03-12 13:54:06 +01:00
|
|
|
class PLUGINDLLEXPORT Config : public Calamares::ModuleSystem::Config
|
2020-07-25 12:47:01 +02:00
|
|
|
{
|
2020-07-25 15:39:19 +02:00
|
|
|
Q_OBJECT
|
|
|
|
|
|
|
|
Q_PROPERTY( QString userShell READ userShell WRITE setUserShell NOTIFY userShellChanged )
|
|
|
|
|
2021-03-15 00:21:26 +01:00
|
|
|
Q_PROPERTY( QString autoLoginGroup READ autoLoginGroup WRITE setAutoLoginGroup NOTIFY autoLoginGroupChanged )
|
2020-07-25 16:39:13 +02:00
|
|
|
Q_PROPERTY( QString sudoersGroup READ sudoersGroup WRITE setSudoersGroup NOTIFY sudoersGroupChanged )
|
|
|
|
|
2020-07-28 11:41:52 +02:00
|
|
|
Q_PROPERTY( bool doAutoLogin READ doAutoLogin WRITE setAutoLogin NOTIFY autoLoginChanged )
|
|
|
|
|
2020-07-27 17:52:46 +02:00
|
|
|
Q_PROPERTY( QString fullName READ fullName WRITE setFullName NOTIFY fullNameChanged )
|
2020-07-25 16:54:15 +02:00
|
|
|
Q_PROPERTY( QString loginName READ loginName WRITE setLoginName NOTIFY loginNameChanged )
|
2020-07-28 10:21:23 +02:00
|
|
|
Q_PROPERTY( QString loginNameStatus READ loginNameStatus NOTIFY loginNameStatusChanged )
|
2020-07-25 16:54:15 +02:00
|
|
|
|
2020-07-27 15:54:52 +02:00
|
|
|
Q_PROPERTY( QString hostName READ hostName WRITE setHostName NOTIFY hostNameChanged )
|
2020-07-28 10:45:38 +02:00
|
|
|
Q_PROPERTY( QString hostNameStatus READ hostNameStatus NOTIFY hostNameStatusChanged )
|
2020-08-05 10:29:13 +02:00
|
|
|
Q_PROPERTY( HostNameActions hostNameActions READ hostNameActions CONSTANT )
|
2020-07-27 15:54:52 +02:00
|
|
|
|
2020-08-05 13:29:06 +02:00
|
|
|
Q_PROPERTY( QString userPassword READ userPassword WRITE setUserPassword NOTIFY userPasswordChanged )
|
|
|
|
Q_PROPERTY( QString userPasswordSecondary READ userPasswordSecondary WRITE setUserPasswordSecondary NOTIFY
|
|
|
|
userPasswordSecondaryChanged )
|
2020-08-17 11:41:04 +02:00
|
|
|
Q_PROPERTY( int userPasswordValidity READ userPasswordValidity NOTIFY userPasswordStatusChanged STORED false )
|
2020-08-17 14:20:54 +02:00
|
|
|
Q_PROPERTY( QString userPasswordMessage READ userPasswordMessage NOTIFY userPasswordStatusChanged STORED false )
|
2020-08-17 11:41:04 +02:00
|
|
|
|
2020-08-05 13:29:06 +02:00
|
|
|
Q_PROPERTY( QString rootPassword READ rootPassword WRITE setRootPassword NOTIFY rootPasswordChanged )
|
|
|
|
Q_PROPERTY( QString rootPasswordSecondary READ rootPasswordSecondary WRITE setRootPasswordSecondary NOTIFY
|
|
|
|
rootPasswordSecondaryChanged )
|
2020-08-17 11:41:04 +02:00
|
|
|
Q_PROPERTY( int rootPasswordValidity READ rootPasswordValidity NOTIFY rootPasswordStatusChanged STORED false )
|
2020-08-17 14:20:54 +02:00
|
|
|
Q_PROPERTY( QString rootPasswordMessage READ rootPasswordMessage NOTIFY rootPasswordStatusChanged STORED false )
|
2020-08-05 13:29:06 +02:00
|
|
|
|
2020-08-04 22:16:48 +02:00
|
|
|
Q_PROPERTY( bool writeRootPassword READ writeRootPassword CONSTANT )
|
|
|
|
Q_PROPERTY( bool reuseUserPasswordForRoot READ reuseUserPasswordForRoot WRITE setReuseUserPasswordForRoot NOTIFY
|
|
|
|
reuseUserPasswordForRootChanged )
|
|
|
|
|
|
|
|
Q_PROPERTY( bool permitWeakPasswords READ permitWeakPasswords CONSTANT )
|
|
|
|
Q_PROPERTY( bool requireStrongPasswords READ requireStrongPasswords WRITE setRequireStrongPasswords NOTIFY
|
|
|
|
requireStrongPasswordsChanged )
|
|
|
|
|
2020-08-18 11:21:53 +02:00
|
|
|
Q_PROPERTY( bool ready READ isReady NOTIFY readyChanged STORED false )
|
|
|
|
|
2020-07-25 12:47:01 +02:00
|
|
|
public:
|
2020-08-17 11:41:04 +02:00
|
|
|
/** @brief Validity (status) of a password
|
|
|
|
*
|
|
|
|
* Valid passwords are:
|
|
|
|
* - primary and secondary are equal **and**
|
|
|
|
* - all the password-strength checks pass
|
|
|
|
* Weak passwords:
|
|
|
|
* - primary and secondary are equal **and**
|
|
|
|
* - not all the checks pass **and**
|
|
|
|
* - permitWeakPasswords is @c true **and**
|
|
|
|
* - requireStrongPasswords is @c false
|
|
|
|
* Invalid passwords (all other cases):
|
|
|
|
* - the primary and secondary values are not equal **or**
|
|
|
|
* - not all the checks pass and weak passwords are not permitted
|
|
|
|
*/
|
|
|
|
enum PasswordValidity
|
|
|
|
{
|
|
|
|
Valid = 0,
|
|
|
|
Weak = 1,
|
|
|
|
Invalid = 2
|
|
|
|
};
|
|
|
|
|
2020-08-17 14:20:54 +02:00
|
|
|
/** @brief Full password status
|
|
|
|
*
|
|
|
|
* A password's status is in two parts:
|
|
|
|
* - a validity (valid, weak or invalid)
|
|
|
|
* - a message describing that validity
|
|
|
|
* The message is empty when the password is valid, but
|
|
|
|
* weak and invalid passwords have an explanatory message.
|
|
|
|
*/
|
|
|
|
using PasswordStatus = QPair< PasswordValidity, QString >;
|
|
|
|
|
2020-07-25 12:47:01 +02:00
|
|
|
Config( QObject* parent = nullptr );
|
2020-09-22 22:34:38 +02:00
|
|
|
~Config() override;
|
2020-07-25 12:47:01 +02:00
|
|
|
|
2021-03-12 13:54:06 +01:00
|
|
|
void setConfigurationMap( const QVariantMap& ) override;
|
2020-07-25 15:39:19 +02:00
|
|
|
|
2020-08-17 15:30:09 +02:00
|
|
|
/** @brief Fill Global Storage with some settings
|
|
|
|
*
|
|
|
|
* This should be called when moving on from the view step,
|
|
|
|
* and copies some things to GS that otherwise would not.
|
|
|
|
*/
|
|
|
|
void finalizeGlobalStorage() const;
|
|
|
|
|
2020-08-18 11:31:32 +02:00
|
|
|
/** @brief Jobs for creating user, setting passwords
|
|
|
|
*
|
|
|
|
* If the Config object isn't ready yet, returns an empty list.
|
|
|
|
*/
|
|
|
|
Calamares::JobList createJobs() const;
|
|
|
|
|
2020-07-25 15:39:19 +02:00
|
|
|
/** @brief Full path to the user's shell executable
|
|
|
|
*
|
|
|
|
* Typically this will be /bin/bash, but it can be set from
|
|
|
|
* the config file with the *userShell* setting.
|
|
|
|
*/
|
|
|
|
QString userShell() const { return m_userShell; }
|
|
|
|
|
2020-07-25 16:39:13 +02:00
|
|
|
/// The group of which auto-login users must be a member
|
2021-03-15 00:21:26 +01:00
|
|
|
QString autoLoginGroup() const { return m_autoLoginGroup; }
|
2022-02-21 15:49:10 +01:00
|
|
|
|
|
|
|
enum class SudoStyle
|
|
|
|
{
|
|
|
|
UserOnly,
|
|
|
|
UserAndGroup
|
|
|
|
};
|
2020-07-25 16:39:13 +02:00
|
|
|
/// The group of which users who can "sudo" must be a member
|
|
|
|
QString sudoersGroup() const { return m_sudoersGroup; }
|
2022-02-21 15:49:10 +01:00
|
|
|
SudoStyle sudoStyle() const { return m_sudoStyle; }
|
2020-07-25 16:39:13 +02:00
|
|
|
|
2020-07-25 16:54:15 +02:00
|
|
|
/// The full (GECOS) name of the user
|
2020-07-27 17:52:46 +02:00
|
|
|
QString fullName() const { return m_fullName; }
|
2020-07-25 16:54:15 +02:00
|
|
|
/// The login name of the user
|
|
|
|
QString loginName() const { return m_loginName; }
|
2020-07-28 10:21:23 +02:00
|
|
|
/// Status message about login -- empty for "ok"
|
|
|
|
QString loginNameStatus() const;
|
2020-07-25 16:54:15 +02:00
|
|
|
|
2020-07-27 15:54:52 +02:00
|
|
|
/// The host name (name for the system)
|
|
|
|
QString hostName() const { return m_hostName; }
|
2020-07-28 10:45:38 +02:00
|
|
|
/// Status message about hostname -- empty for "ok"
|
|
|
|
QString hostNameStatus() const;
|
2020-08-05 10:29:13 +02:00
|
|
|
/// How to write the hostname
|
|
|
|
HostNameActions hostNameActions() const { return m_hostNameActions; }
|
2020-07-27 15:54:52 +02:00
|
|
|
|
2020-07-28 11:41:52 +02:00
|
|
|
/// Should the user be automatically logged-in?
|
|
|
|
bool doAutoLogin() const { return m_doAutoLogin; }
|
2020-07-28 11:59:53 +02:00
|
|
|
/// Should the root password be written (if false, no password is set and the root account is disabled for login)
|
|
|
|
bool writeRootPassword() const { return m_writeRootPassword; }
|
2020-08-04 22:16:48 +02:00
|
|
|
/// Should the user's password be used for root, too? (if root is written at all)
|
|
|
|
bool reuseUserPasswordForRoot() const { return m_reuseUserPasswordForRoot; }
|
|
|
|
/// Show UI to change the "require strong password" setting?
|
|
|
|
bool permitWeakPasswords() const { return m_permitWeakPasswords; }
|
|
|
|
/// Current setting for "require strong password"?
|
|
|
|
bool requireStrongPasswords() const { return m_requireStrongPasswords; }
|
2020-07-28 11:41:52 +02:00
|
|
|
|
2020-10-13 17:35:07 +02:00
|
|
|
const QList< GroupDescription >& defaultGroups() const { return m_defaultGroups; }
|
2020-10-22 14:07:40 +02:00
|
|
|
/** @brief the names of all the groups for the current user
|
|
|
|
*
|
2021-03-15 00:21:26 +01:00
|
|
|
* Takes into account defaultGroups and autoLogin behavior.
|
2020-10-22 14:07:40 +02:00
|
|
|
*/
|
|
|
|
QStringList groupsForThisUser() const;
|
2020-07-29 12:18:25 +02:00
|
|
|
|
2020-08-05 13:42:18 +02:00
|
|
|
// The user enters a password (and again in a separate UI element)
|
2020-08-05 13:29:06 +02:00
|
|
|
QString userPassword() const { return m_userPassword; }
|
|
|
|
QString userPasswordSecondary() const { return m_userPasswordSecondary; }
|
2020-08-17 11:41:04 +02:00
|
|
|
int userPasswordValidity() const;
|
2020-08-17 14:20:54 +02:00
|
|
|
QString userPasswordMessage() const;
|
|
|
|
PasswordStatus userPasswordStatus() const;
|
2020-08-17 11:41:04 +02:00
|
|
|
|
2020-08-05 13:42:18 +02:00
|
|
|
// The root password **may** be entered in the UI, or may be suppressed
|
|
|
|
// entirely when writeRootPassword is off, or may be equal to
|
|
|
|
// the user password when reuseUserPasswordForRoot is on.
|
|
|
|
QString rootPassword() const;
|
|
|
|
QString rootPasswordSecondary() const;
|
2020-08-17 11:41:04 +02:00
|
|
|
int rootPasswordValidity() const;
|
2020-08-17 14:20:54 +02:00
|
|
|
QString rootPasswordMessage() const;
|
|
|
|
PasswordStatus rootPasswordStatus() const;
|
2020-08-05 13:29:06 +02:00
|
|
|
|
2020-08-18 11:21:53 +02:00
|
|
|
bool isReady() const;
|
|
|
|
|
2020-07-28 10:21:23 +02:00
|
|
|
static const QStringList& forbiddenLoginNames();
|
2020-07-28 10:45:38 +02:00
|
|
|
static const QStringList& forbiddenHostNames();
|
2020-07-28 10:21:23 +02:00
|
|
|
|
2020-07-25 15:39:19 +02:00
|
|
|
public Q_SLOTS:
|
|
|
|
/** @brief Sets the user's shell if possible
|
|
|
|
*
|
|
|
|
* If the path is empty, that's ok: no shell will be explicitly set,
|
|
|
|
* so the user will get whatever shell is set to default in the target.
|
|
|
|
*
|
|
|
|
* The given non-empty @p path must be an absolute path (for use inside
|
|
|
|
* the target system!); if it is not, the shell is not changed.
|
|
|
|
*/
|
|
|
|
void setUserShell( const QString& path );
|
|
|
|
|
2021-03-15 00:21:26 +01:00
|
|
|
/// Sets the autoLogin group; empty is ignored
|
|
|
|
void setAutoLoginGroup( const QString& group );
|
2020-07-25 16:39:13 +02:00
|
|
|
/// Sets the sudoer group; empty is ignored
|
|
|
|
void setSudoersGroup( const QString& group );
|
|
|
|
|
2020-07-25 16:54:15 +02:00
|
|
|
/// Sets the full name, may guess a loginName
|
2020-07-27 17:52:46 +02:00
|
|
|
void setFullName( const QString& name );
|
2020-07-27 15:34:59 +02:00
|
|
|
/// Sets the login name (flags it as "custom")
|
2020-07-25 16:54:15 +02:00
|
|
|
void setLoginName( const QString& login );
|
|
|
|
|
2020-07-27 15:54:52 +02:00
|
|
|
/// Sets the host name (flags it as "custom")
|
|
|
|
void setHostName( const QString& host );
|
|
|
|
|
2021-03-15 00:21:26 +01:00
|
|
|
/// Sets the autoLogin flag
|
2020-07-28 11:41:52 +02:00
|
|
|
void setAutoLogin( bool b );
|
|
|
|
|
2020-08-04 22:16:48 +02:00
|
|
|
/// Set to true to use the user password, unchanged, for root too
|
|
|
|
void setReuseUserPasswordForRoot( bool reuse );
|
|
|
|
/// Change setting for "require strong password"
|
|
|
|
void setRequireStrongPasswords( bool strong );
|
|
|
|
|
2020-08-05 13:29:06 +02:00
|
|
|
void setUserPassword( const QString& );
|
|
|
|
void setUserPasswordSecondary( const QString& );
|
|
|
|
void setRootPassword( const QString& );
|
|
|
|
void setRootPasswordSecondary( const QString& );
|
|
|
|
|
2020-07-25 15:39:19 +02:00
|
|
|
signals:
|
|
|
|
void userShellChanged( const QString& );
|
2021-03-15 00:21:26 +01:00
|
|
|
void autoLoginGroupChanged( const QString& );
|
2020-07-25 16:39:13 +02:00
|
|
|
void sudoersGroupChanged( const QString& );
|
2020-07-27 17:52:46 +02:00
|
|
|
void fullNameChanged( const QString& );
|
2020-07-25 16:54:15 +02:00
|
|
|
void loginNameChanged( const QString& );
|
2020-07-28 10:21:23 +02:00
|
|
|
void loginNameStatusChanged( const QString& );
|
2020-07-27 15:54:52 +02:00
|
|
|
void hostNameChanged( const QString& );
|
2020-07-28 10:45:38 +02:00
|
|
|
void hostNameStatusChanged( const QString& );
|
2020-07-28 11:41:52 +02:00
|
|
|
void autoLoginChanged( bool );
|
2020-08-04 22:16:48 +02:00
|
|
|
void reuseUserPasswordForRootChanged( bool );
|
|
|
|
void requireStrongPasswordsChanged( bool );
|
2020-08-05 13:29:06 +02:00
|
|
|
void userPasswordChanged( const QString& );
|
|
|
|
void userPasswordSecondaryChanged( const QString& );
|
2020-08-17 14:08:59 +02:00
|
|
|
void userPasswordStatusChanged( int, const QString& );
|
2020-08-05 13:29:06 +02:00
|
|
|
void rootPasswordChanged( const QString& );
|
|
|
|
void rootPasswordSecondaryChanged( const QString& );
|
2020-08-17 14:08:59 +02:00
|
|
|
void rootPasswordStatusChanged( int, const QString& );
|
2020-08-18 11:21:53 +02:00
|
|
|
void readyChanged( bool ) const;
|
2020-07-25 15:39:19 +02:00
|
|
|
|
|
|
|
private:
|
2020-08-17 14:20:54 +02:00
|
|
|
PasswordStatus passwordStatus( const QString&, const QString& ) const;
|
2020-08-18 11:21:53 +02:00
|
|
|
void checkReady();
|
2020-08-17 11:41:04 +02:00
|
|
|
|
2020-10-13 17:35:07 +02:00
|
|
|
QList< GroupDescription > m_defaultGroups;
|
2020-07-25 15:39:19 +02:00
|
|
|
QString m_userShell;
|
2021-03-15 00:21:26 +01:00
|
|
|
QString m_autoLoginGroup;
|
2020-07-25 16:39:13 +02:00
|
|
|
QString m_sudoersGroup;
|
2022-02-21 15:49:10 +01:00
|
|
|
SudoStyle m_sudoStyle = SudoStyle::UserOnly;
|
2020-07-25 16:54:15 +02:00
|
|
|
QString m_fullName;
|
|
|
|
QString m_loginName;
|
2020-07-27 15:54:52 +02:00
|
|
|
QString m_hostName;
|
2020-08-05 13:29:06 +02:00
|
|
|
|
|
|
|
QString m_userPassword;
|
|
|
|
QString m_userPasswordSecondary; // enter again to be sure
|
|
|
|
QString m_rootPassword;
|
|
|
|
QString m_rootPasswordSecondary;
|
|
|
|
|
2020-07-28 11:41:52 +02:00
|
|
|
bool m_doAutoLogin = false;
|
2020-08-04 22:16:48 +02:00
|
|
|
|
2020-07-28 11:59:53 +02:00
|
|
|
bool m_writeRootPassword = true;
|
2020-08-04 22:16:48 +02:00
|
|
|
bool m_reuseUserPasswordForRoot = false;
|
|
|
|
|
|
|
|
bool m_permitWeakPasswords = false;
|
|
|
|
bool m_requireStrongPasswords = true;
|
2020-07-28 11:41:52 +02:00
|
|
|
|
2020-07-27 15:34:59 +02:00
|
|
|
bool m_customLoginName = false;
|
2020-07-27 15:54:52 +02:00
|
|
|
bool m_customHostName = false;
|
2020-08-05 10:29:13 +02:00
|
|
|
|
2020-08-18 11:21:53 +02:00
|
|
|
bool m_isReady = false; ///< Used to reduce readyChanged signals
|
|
|
|
|
2020-08-05 10:29:13 +02:00
|
|
|
HostNameActions m_hostNameActions;
|
2020-08-05 13:03:18 +02:00
|
|
|
PasswordCheckList m_passwordChecks;
|
2020-07-25 12:47:01 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
#endif
|