mirror of
https://github.com/silverstripe/silverstripe-framework
synced 2024-10-22 12:05:37 +00:00
API CHANGE Removed "auto-merging" of member records from Member->onBeforeWrite() due to security reasons - please use DataObject->merge() explicitly if this is desired behaviour (from r100705)
git-svn-id: svn://svn.silverstripe.com/silverstripe/open/modules/sapphire/branches/2.4@100718 467b73ca-7a2a-4603-9d3b-597d59a354a9
This commit is contained in:
parent
d61f45ea61
commit
90e8171536
@ -564,27 +564,33 @@ class Member extends DataObject {
|
||||
*/
|
||||
function onBeforeWrite() {
|
||||
if($this->SetPassword) $this->Password = $this->SetPassword;
|
||||
|
||||
// If a member with the same "unique identifier" already exists with a different ID, don't allow merging.
|
||||
// Note: This does not a full replacement for safeguards in the controller layer (e.g. in a registration form),
|
||||
// but rather a last line of defense against data inconsistencies.
|
||||
$identifierField = self::$unique_identifier_field;
|
||||
if($this->$identifierField) {
|
||||
$idClause = ($this->ID) ? " AND \"Member\".\"ID\" <> $this->ID" : '';
|
||||
$SQL_identifierField = Convert::raw2sql($this->$identifierField);
|
||||
|
||||
$existingRecord = DataObject::get_one('Member', "\"$identifierField\" = '{$SQL_identifierField}'{$idClause}");
|
||||
// Note: Same logic as Member_Validator class
|
||||
$idClause = ($this->ID) ? sprintf(" AND \"Member\".\"ID\" <> %d", (int)$this->ID) : '';
|
||||
$existingRecord = DataObject::get_one(
|
||||
'Member',
|
||||
sprintf(
|
||||
"\"%s\" = '%s' %s",
|
||||
$identifierField,
|
||||
Convert::raw2sql($this->$identifierField),
|
||||
$idClause
|
||||
)
|
||||
);
|
||||
if($existingRecord) {
|
||||
$newID = $existingRecord->ID;
|
||||
if($this->ID) {
|
||||
DB::query("UPDATE \"Group_Members\" SET \"MemberID\" = $newID WHERE \"MemberID\" = $this->ID");
|
||||
}
|
||||
$this->ID = $newID;
|
||||
// Merge existing data into the local record
|
||||
|
||||
foreach($existingRecord->getAllFields() as $k => $v) {
|
||||
if(!$this->isChanged($k)) $this->record[$k] = $v;
|
||||
}
|
||||
$existingRecord->destroy();
|
||||
throw new ValidationException(sprintf(
|
||||
'Can\'t overwrite existing member #%d with identical $unique_identifier_field (%s = %s))',
|
||||
$existingRecord->ID,
|
||||
$identifierField,
|
||||
$this->$identifierField
|
||||
));
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
// We don't send emails out on dev/tests sites to prevent accidentally spamming users.
|
||||
// However, if TestMailer is in use this isn't a risk.
|
||||
if(
|
||||
@ -595,7 +601,7 @@ class Member extends DataObject {
|
||||
) {
|
||||
$this->sendInfo('changePassword');
|
||||
}
|
||||
|
||||
|
||||
// The test on $this->ID is used for when records are initially created.
|
||||
// Note that this only works with cleartext passwords, as we can't rehash
|
||||
// existing passwords.
|
||||
@ -611,7 +617,7 @@ class Member extends DataObject {
|
||||
$this->Password = $encryption_details['password'];
|
||||
$this->Salt = $encryption_details['salt'];
|
||||
$this->PasswordEncryption = $encryption_details['algorithm'];
|
||||
|
||||
|
||||
// If we haven't manually set a password expiry
|
||||
if(!$this->isChanged('PasswordExpiry')) {
|
||||
// then set it for us
|
||||
@ -622,7 +628,7 @@ class Member extends DataObject {
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
// save locale
|
||||
if(!$this->Locale) {
|
||||
$this->Locale = i18n::get_locale();
|
||||
|
@ -6,12 +6,51 @@
|
||||
class MemberTest extends FunctionalTest {
|
||||
static $fixture_file = 'sapphire/tests/security/MemberTest.yml';
|
||||
|
||||
protected $orig = array();
|
||||
|
||||
function setUp() {
|
||||
parent::setUp();
|
||||
|
||||
$this->orig['Member_unique_identifier_field'] = Member::get_unique_identifier_field();
|
||||
Member::set_unique_identifier_field('Email');
|
||||
Member::set_password_validator(null);
|
||||
}
|
||||
|
||||
function tearDown() {
|
||||
Member::set_unique_identifier_field($this->orig['Member_unique_identifier_field']);
|
||||
|
||||
parent::tearDown();
|
||||
}
|
||||
|
||||
/**
|
||||
* @expectedException ValidationException
|
||||
*/
|
||||
function testWriteDoesntMergeNewRecordWithExistingMember() {
|
||||
$m1 = new Member();
|
||||
$m1->Email = 'member@test.com';
|
||||
$m1->write();
|
||||
|
||||
$m2 = new Member();
|
||||
$m2->Email = 'member@test.com';
|
||||
$m2->write();
|
||||
}
|
||||
|
||||
/**
|
||||
* @expectedException ValidationException
|
||||
*/
|
||||
function testWriteDoesntMergeExistingMemberOnIdentifierChange() {
|
||||
$m1 = new Member();
|
||||
$m1->Email = 'member@test.com';
|
||||
$m1->write();
|
||||
|
||||
$m2 = new Member();
|
||||
$m2->Email = 'member_new@test.com';
|
||||
$m2->write();
|
||||
|
||||
$m2->Email = 'member@test.com';
|
||||
$m2->write();
|
||||
}
|
||||
|
||||
function testDefaultPasswordEncryptionOnMember() {
|
||||
$member = new Member();
|
||||
$member->Password = 'mypassword';
|
||||
|
Loading…
x
Reference in New Issue
Block a user