2017-06-13 21:04:43 +12:00
|
|
|
<?php
|
|
|
|
|
2017-06-15 14:20:12 +12:00
|
|
|
namespace SilverStripe\Security;
|
2017-06-13 21:04:43 +12:00
|
|
|
|
2017-06-15 14:20:12 +12:00
|
|
|
use BadMethodCallException;
|
|
|
|
use InvalidArgumentException;
|
2017-06-13 21:04:43 +12:00
|
|
|
use SilverStripe\Core\Config\Configurable;
|
2017-10-16 16:43:12 +13:00
|
|
|
use SilverStripe\Core\Environment;
|
2017-06-13 21:04:43 +12:00
|
|
|
use SilverStripe\Core\Extensible;
|
2017-06-15 14:20:12 +12:00
|
|
|
use SilverStripe\Core\Injector\Injectable;
|
2017-06-13 21:04:43 +12:00
|
|
|
|
2017-06-15 14:20:12 +12:00
|
|
|
/**
|
|
|
|
* Provides access to the default admin
|
|
|
|
*/
|
2017-06-13 21:04:43 +12:00
|
|
|
class DefaultAdminService
|
|
|
|
{
|
|
|
|
use Extensible;
|
|
|
|
use Configurable;
|
2017-06-15 14:20:12 +12:00
|
|
|
use Injectable;
|
2017-06-13 21:04:43 +12:00
|
|
|
|
|
|
|
/**
|
2017-06-22 22:50:45 +12:00
|
|
|
* Can be set to explicitly true or false, or left null.
|
|
|
|
* If null, hasDefaultAdmin() will be inferred from environment.
|
|
|
|
*
|
|
|
|
* @var bool|null
|
2017-06-13 21:04:43 +12:00
|
|
|
*/
|
2017-06-22 22:50:45 +12:00
|
|
|
protected static $has_default_admin = null;
|
2017-06-13 21:04:43 +12:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @var string
|
|
|
|
*/
|
2017-06-15 14:20:12 +12:00
|
|
|
protected static $default_username = null;
|
2017-06-13 21:04:43 +12:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @var string
|
|
|
|
*/
|
2017-06-15 14:20:12 +12:00
|
|
|
protected static $default_password = null;
|
|
|
|
|
|
|
|
public function __construct()
|
|
|
|
{
|
|
|
|
}
|
2017-06-13 21:04:43 +12:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Set the default admin credentials
|
|
|
|
*
|
|
|
|
* @param string $username
|
|
|
|
* @param string $password
|
|
|
|
*/
|
|
|
|
public static function setDefaultAdmin($username, $password)
|
|
|
|
{
|
|
|
|
// don't overwrite if already set
|
2017-06-15 14:20:12 +12:00
|
|
|
if (static::hasDefaultAdmin()) {
|
|
|
|
throw new BadMethodCallException(
|
|
|
|
"Default admin already exists. Use clearDefaultAdmin() first."
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (empty($username) || empty($password)) {
|
|
|
|
throw new InvalidArgumentException("Default admin username / password cannot be empty");
|
2017-06-13 21:04:43 +12:00
|
|
|
}
|
|
|
|
|
|
|
|
static::$default_username = $username;
|
|
|
|
static::$default_password = $password;
|
2017-06-15 14:20:12 +12:00
|
|
|
static::$has_default_admin = true;
|
2017-06-13 21:04:43 +12:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return string The default admin username
|
2017-06-15 14:20:12 +12:00
|
|
|
* @throws BadMethodCallException Throws exception if there is no default admin
|
2017-06-13 21:04:43 +12:00
|
|
|
*/
|
|
|
|
public static function getDefaultAdminUsername()
|
|
|
|
{
|
2017-06-15 14:20:12 +12:00
|
|
|
if (!static::hasDefaultAdmin()) {
|
|
|
|
throw new BadMethodCallException(
|
|
|
|
"No default admin configured. Please call hasDefaultAdmin() before getting default admin username"
|
|
|
|
);
|
|
|
|
}
|
2017-10-16 16:43:12 +13:00
|
|
|
return static::$default_username ?: Environment::getEnv('SS_DEFAULT_ADMIN_USERNAME');
|
2017-06-13 21:04:43 +12:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return string The default admin password
|
2017-06-15 14:20:12 +12:00
|
|
|
* @throws BadMethodCallException Throws exception if there is no default admin
|
2017-06-13 21:04:43 +12:00
|
|
|
*/
|
|
|
|
public static function getDefaultAdminPassword()
|
|
|
|
{
|
2017-06-15 14:20:12 +12:00
|
|
|
if (!static::hasDefaultAdmin()) {
|
|
|
|
throw new BadMethodCallException(
|
|
|
|
"No default admin configured. Please call hasDefaultAdmin() before getting default admin password"
|
|
|
|
);
|
|
|
|
}
|
2017-10-16 16:43:12 +13:00
|
|
|
return static::$default_password ?: Environment::getEnv('SS_DEFAULT_ADMIN_PASSWORD');
|
2017-06-13 21:04:43 +12:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Check if there is a default admin
|
|
|
|
*
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
public static function hasDefaultAdmin()
|
|
|
|
{
|
2017-06-22 22:50:45 +12:00
|
|
|
// Check environment if not explicitly set
|
|
|
|
if (!isset(static::$has_default_admin)) {
|
2017-10-16 16:43:12 +13:00
|
|
|
return !empty(Environment::getEnv('SS_DEFAULT_ADMIN_USERNAME'))
|
|
|
|
&& !empty(Environment::getEnv('SS_DEFAULT_ADMIN_PASSWORD'));
|
2017-06-22 22:50:45 +12:00
|
|
|
}
|
2017-06-13 21:04:43 +12:00
|
|
|
return static::$has_default_admin;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2017-06-22 22:50:45 +12:00
|
|
|
* Flush the default admin credentials.
|
2017-06-13 21:04:43 +12:00
|
|
|
*/
|
|
|
|
public static function clearDefaultAdmin()
|
|
|
|
{
|
2017-06-15 14:20:12 +12:00
|
|
|
static::$has_default_admin = false;
|
|
|
|
static::$default_username = null;
|
|
|
|
static::$default_password = null;
|
2017-06-13 21:04:43 +12:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2017-06-15 14:20:12 +12:00
|
|
|
* @return Member|null
|
2017-06-13 21:04:43 +12:00
|
|
|
*/
|
|
|
|
public function findOrCreateDefaultAdmin()
|
|
|
|
{
|
2017-06-15 14:20:12 +12:00
|
|
|
$this->extend('beforeFindOrCreateDefaultAdmin');
|
2017-06-13 21:04:43 +12:00
|
|
|
|
|
|
|
// Check if we have default admins
|
2017-06-15 14:20:12 +12:00
|
|
|
if (!static::hasDefaultAdmin()) {
|
2017-06-13 21:04:43 +12:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2017-07-06 13:29:27 +12:00
|
|
|
// Create admin with default admin username
|
|
|
|
$admin = $this->findOrCreateAdmin(
|
|
|
|
static::getDefaultAdminUsername(),
|
|
|
|
_t(__CLASS__ . '.DefaultAdminFirstname', 'Default Admin')
|
|
|
|
);
|
2017-06-13 21:04:43 +12:00
|
|
|
|
2017-07-06 13:29:27 +12:00
|
|
|
$this->extend('afterFindOrCreateDefaultAdmin', $admin);
|
|
|
|
|
|
|
|
return $admin;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Find or create a Member with admin permissions
|
|
|
|
*
|
|
|
|
* @skipUpgrade
|
|
|
|
* @param string $email
|
|
|
|
* @param string $name
|
|
|
|
* @return Member
|
|
|
|
*/
|
|
|
|
public function findOrCreateAdmin($email, $name = null)
|
|
|
|
{
|
|
|
|
$this->extend('beforeFindOrCreateAdmin', $email, $name);
|
2017-06-13 21:04:43 +12:00
|
|
|
|
|
|
|
// Find member
|
2017-07-06 13:29:27 +12:00
|
|
|
/** @var Member $admin */
|
2017-06-13 21:04:43 +12:00
|
|
|
$admin = Member::get()
|
2017-07-06 13:29:27 +12:00
|
|
|
->filter('Email', $email)
|
2017-06-13 21:04:43 +12:00
|
|
|
->first();
|
2017-07-06 13:29:27 +12:00
|
|
|
|
|
|
|
// Find or create admin group
|
|
|
|
$adminGroup = $this->findOrCreateAdminGroup();
|
|
|
|
|
2017-06-13 21:04:43 +12:00
|
|
|
// If no admin is found, create one
|
2017-07-06 13:29:27 +12:00
|
|
|
if ($admin) {
|
|
|
|
$inGroup = $admin->inGroup($adminGroup);
|
|
|
|
} else {
|
|
|
|
// Note: This user won't be able to login until a password is set
|
2017-06-13 21:04:43 +12:00
|
|
|
// Set 'Email' to identify this as the default admin
|
2017-07-06 13:29:27 +12:00
|
|
|
$inGroup = false;
|
2017-06-13 21:04:43 +12:00
|
|
|
$admin = Member::create();
|
2017-07-06 13:29:27 +12:00
|
|
|
$admin->FirstName = $name ?: $email;
|
|
|
|
$admin->Email = $email;
|
2017-06-13 21:04:43 +12:00
|
|
|
$admin->write();
|
|
|
|
}
|
|
|
|
|
2017-07-06 13:29:27 +12:00
|
|
|
// Ensure this user is in an admin group
|
|
|
|
if (!$inGroup) {
|
2017-06-13 21:04:43 +12:00
|
|
|
// Add member to group instead of adding group to member
|
|
|
|
// This bypasses the privilege escallation code in Member_GroupSet
|
|
|
|
$adminGroup
|
|
|
|
->DirectMembers()
|
|
|
|
->add($admin);
|
|
|
|
}
|
|
|
|
|
2017-07-06 13:29:27 +12:00
|
|
|
$this->extend('afterFindOrCreateAdmin', $admin);
|
2017-06-13 21:04:43 +12:00
|
|
|
|
|
|
|
return $admin;
|
|
|
|
}
|
|
|
|
|
2017-07-06 13:29:27 +12:00
|
|
|
/**
|
|
|
|
* Ensure a Group exists with admin permission
|
|
|
|
*
|
|
|
|
* @return Group
|
|
|
|
*/
|
|
|
|
protected function findOrCreateAdminGroup()
|
|
|
|
{
|
|
|
|
// Check pre-existing group
|
|
|
|
$adminGroup = Permission::get_groups_by_permission('ADMIN')->first();
|
|
|
|
if ($adminGroup) {
|
|
|
|
return $adminGroup;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check if default records create the group
|
|
|
|
Group::singleton()->requireDefaultRecords();
|
|
|
|
$adminGroup = Permission::get_groups_by_permission('ADMIN')->first();
|
|
|
|
if ($adminGroup) {
|
|
|
|
return $adminGroup;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create new admin group directly
|
|
|
|
$adminGroup = Group::create();
|
|
|
|
$adminGroup->Code = 'administrators';
|
|
|
|
$adminGroup->Title = _t('SilverStripe\\Security\\Group.DefaultGroupTitleAdministrators', 'Administrators');
|
|
|
|
$adminGroup->Sort = 0;
|
|
|
|
$adminGroup->write();
|
|
|
|
Permission::grant($adminGroup->ID, 'ADMIN');
|
|
|
|
return $adminGroup;
|
|
|
|
}
|
|
|
|
|
2017-06-13 21:04:43 +12:00
|
|
|
/**
|
2017-06-15 14:20:12 +12:00
|
|
|
* Check if the user is a default admin.
|
|
|
|
* Returns false if there is no default admin.
|
|
|
|
*
|
|
|
|
* @param string $username
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
public static function isDefaultAdmin($username)
|
|
|
|
{
|
|
|
|
return static::hasDefaultAdmin()
|
|
|
|
&& $username
|
|
|
|
&& $username === static::getDefaultAdminUsername();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Check if the user credentials match the default admin.
|
|
|
|
* Returns false if there is no default admin.
|
|
|
|
*
|
2017-06-13 21:04:43 +12:00
|
|
|
* @param string $username
|
|
|
|
* @param string $password
|
2017-06-15 14:20:12 +12:00
|
|
|
* @return bool
|
2017-06-13 21:04:43 +12:00
|
|
|
*/
|
2017-06-15 14:20:12 +12:00
|
|
|
public static function isDefaultAdminCredentials($username, $password)
|
2017-06-13 21:04:43 +12:00
|
|
|
{
|
2017-06-15 14:20:12 +12:00
|
|
|
return static::isDefaultAdmin($username)
|
|
|
|
&& $password
|
|
|
|
&& $password === static::getDefaultAdminPassword();
|
2017-06-13 21:04:43 +12:00
|
|
|
}
|
2017-06-15 14:20:12 +12:00
|
|
|
}
|