Newer
Older
<?php
namespace QUI\Memberships\Users;
use QUI;
use QUI\CRUD\Child;
use QUI\Memberships\Handler as MembershipsHandler;
use QUI\Memberships\Users\Handler as MembershipUsersHandler;
use QUI\Memberships\Utils;
use QUI\Mail\Mailer;

Patrick Müller
committed
use QUI\Permissions\Permission;
use QUI\ERP\Products\Handler\Products as ProductsHandler;

Patrick Müller
committed
use QUI\ERP\Accounting\Contracts\Handler as ContractsHandler;
use QUI\Interfaces\Users\User as QUIUserInterface;
/**
* Class MembershipUser
*
* Represents a user that is assigned to a specific membership
*
* @package QUI\Memberships\Users
*/
class MembershipUser extends Child
{
/**
* User that is editing this MembershipUser in the current runtime
*
* @var QUIUserInterface
*/
protected $EditUser = null;
/**
* Set User that is editing this MembershipUser in the current runtime
*
* @param QUIUserInterface $EditUser
*/
public function setEditUser(QUIUserInterface $EditUser)
{
$this->EditUser = $EditUser;
}
* @param bool $withPermission - check permissions on update [default: true]
public function update()
Permission::checkPermission(MembershipUsersHandler::PERMISSION_EDIT_USERS, $this->EditUser);

Patrick Müller
committed
if (!$this->getMembership()->isInfinite()) {
$beginDate = strtotime($this->getAttribute('beginDate'));
$endDate = strtotime($this->getAttribute('endDate'));
if ($beginDate === false
|| $endDate === false
) {
throw new QUI\Memberships\Exception([
'quiqqer/memberships',
'exception.users.membershipuser.wrong.dates'
throw new QUI\Memberships\Exception([
'quiqqer/memberships',
'exception.users.membershipuser.begin.after.end'
}
// check dates
foreach ($this->getAttributes() as $k => $v) {
switch ($k) {
case 'addedDate':
case 'beginDate':
case 'endDate':
case 'cancelDate':
case 'archiveDate':
if (empty($v) || $v === '0000-00-00 00:00:00') {
$this->setAttribute($k, null);
}
break;
case 'cancelled':
$this->setAttribute($k, $v ? 1 : 0);
break;
* Extend the current membership cycle of this membership user
* @param bool $auto (optional) - Used if the membership is automatically extended.
* If set to false, the setting membershipusers.extendMode is used [default: true]
public function extend($auto = true)

Patrick Müller
committed
$extendMode = MembershipUsersHandler::getSetting('extendMode');
// Calculate new start and/or end time
if ($auto || $extendMode === 'reset') {
$start = time();
$extendCounter = $this->getAttribute('extendCounter');
$this->setAttributes([
'beginDate' => Utils::getFormattedTimestamp($start),
'endDate' => $Membership->calcEndDate($start),
'extendCounter' => $extendCounter + 1
} else {
$endDate = $this->getAttribute('endDate');
$this->setAttributes([
'endDate' => $Membership->calcEndDate(strtotime($endDate))
$historyData = [

Patrick Müller
committed
'start' => $this->getAttribute('beginDate'),
'end' => $this->getAttribute('endDate'),
'auto' => $auto ? '1' : '0'

Patrick Müller
committed
$this->addHistoryEntry(MembershipUsersHandler::HISTORY_TYPE_EXTENDED, json_encode($historyData));
// send mail
if ($auto) {
$this->sendAutoExtendMail();
} else {
$this->sendManualExtendMail();
}
}
/**
* Send mail to the user if the membership is extended automatically
*
* @return void
*/
protected function sendAutoExtendMail()
{
$sendMail = MembershipUsersHandler::getSetting('sendAutoExtendMail');
if (!$sendMail) {
return;
}
$subject = $this->getUser()->getLocale()->get(
'quiqqer/memberships',
'templates.mail.autoextend.subject'
$this->sendMail($subject, dirname(__FILE__, 5).'/templates/mail_autoextend.html');
/**
* Send mail to the user if the membership is extended manually
*
* Manually = Either by admin edit or if the user is re-added to the membership
* although he already is a member
*
* @return void
*/
public function sendManualExtendMail()
{
$sendMail = MembershipUsersHandler::getSetting('sendManualExtendMail');
if (!$sendMail) {
return;
}
$subject = $this->getUser()->getLocale()->get(
'quiqqer/memberships',
'templates.mail.manualextend.subject'
$this->sendMail($subject, dirname(__FILE__, 5).'/templates/mail_manualextend.html');
/**
* Expires this memberships user
*
* @return void

Patrick Müller
committed
* @throws \QUI\Exception
*/
public function expire()
{
$this->addHistoryEntry(MembershipUsersHandler::HISTORY_TYPE_EXPIRED);
$this->archive(MembershipUsersHandler::ARCHIVE_REASON_EXPIRED);
// send expire mail
$subject = $this->getUser()->getLocale()->get('quiqqer/memberships', 'templates.mail.expired.subject');
$this->sendMail($subject, dirname(__FILE__, 5).'/templates/mail_expired.html');

Patrick Müller
committed
QUI::getEvents()->fireEvent('quiqqerMembershipsExpired', [$this]);
* Start the manual membership cancellation process
*
* Generates a random hash and sends an email to the user
*
* @throws QUI\Memberships\Exception

Patrick Müller
committed
* @throws QUI\Verification\Exception
* @throws QUI\Exception
public function startManualCancel()
// check cancel permission
if ((int)QUI::getUserBySession()->getId() !== (int)$this->getUserId()) {
throw new QUI\Memberships\Exception([
'quiqqer/memberships',
'exception.users.membershipuser.manualcancel.no.permission'
return;
}
$Membership = $this->getMembership();
// cannot manually cancel infinite memberships
if ($Membership->isInfinite()) {
return;
}
// cannot manually cancel default membership
if ($Membership->isDefault()) {

Patrick Müller
committed
$userEmail = $this->getUser()->getAttribute('email');
if (empty($userEmail)) {
throw new QUI\Memberships\Exception([

Patrick Müller
committed
'quiqqer/memberships',
'exception.users.membershipuser.manualcancel.no_email_address'

Patrick Müller
committed
}
$cancelUrl = Verifier::startVerification($this->getCancelVerification(), true);
$cancelDate = Utils::getFormattedTimestamp();
$this->setAttributes([
'cancelStatus' => MembershipUsersHandler::CANCEL_STATUS_CANCEL_CONFIRM_PENDING,
'cancelDate' => $cancelDate
$this->addHistoryEntry(MembershipUsersHandler::HISTORY_TYPE_CANCEL_START);
$this->setEditUser(QUI::getUsers()->getSystemUser());
$this->update();
$this->sendMail(
QUI::getLocale()->get('quiqqer/memberships', 'templates.mail.startcancel.subject'),
dirname(__FILE__, 5).'/templates/mail_startcancel.html',
[

Patrick Müller
committed
'cancelDate' => $cancelDate,
'cancelUrl' => $cancelUrl,
'cancelEndDate' => $this->getCurrentCancelEndDate()->format('Y-m-d H:i:s')
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
/**
* Automatic cancellation of a MembershipUser.
*
* HINT: This is not supposed to be executed by the user, but programmatically only if
* a membership needs to be cancelled for other reasons than a manual cancellation by the user.
*
* A user CANNOT revoke a cancellation executed this way!
*
* @return void
* @throws \Exception
*/
public function autoCancel()
{
if ($this->isCancelled()) {
return;
}
$Membership = $this->getMembership();
// cannot cancel infinite memberships
if ($Membership->isInfinite()) {
return;
}
// cannot cancel default membership
if ($Membership->isDefault()) {
return;
}
$cancelDate = Utils::getFormattedTimestamp();
$this->setAttributes([
'cancelStatus' => MembershipUsersHandler::CANCEL_STATUS_CANCELLED_BY_SYSTEM,
'cancelDate' => $cancelDate,
'cancelled' => true,
'cancelEndDate' => $this->getCurrentCancelEndDate()->format('Y-m-d H:i:s')
]);
$this->addHistoryEntry(MembershipUsersHandler::HISTORY_TYPE_CANCEL_START_SYSTEM);
// save cancel hash and date to database
$this->setEditUser(QUI::getUsers()->getSystemUser());
try {
$this->update();
QUI::getEvents()->fireEvent('quiqqerMembershipsAutoCancel', [$this]);
} catch (\Exception $Exception) {
QUI\System\Log::writeException($Exception);
}
}
* Start to abort a manually started cancellation process
*
* @return void
* @throws QUI\Memberships\Exception

Patrick Müller
committed
* @throws QUI\Verification\Exception
* @throws QUI\Exception
public function startAbortCancel()
{
// check cancel permission
if ((int)QUI::getUserBySession()->getId() !== (int)$this->getUserId()) {
throw new QUI\Memberships\Exception([
'quiqqer/memberships',
'exception.users.membershipuser.manualcancel.no.permission'
$cancelStatus = (int)$this->getAttribute('cancelStatus');
// If cancellation was initiated programmatically (by system), a user cannot undo this
if ($cancelStatus === MembershipUsersHandler::CANCEL_STATUS_CANCELLED_BY_SYSTEM) {
throw new QUI\Memberships\Exception([
'quiqqer/memberships',
'exception.users.membershipuser.manualcancel.no_system_uncancel'
]);
}
if ($cancelStatus !== MembershipUsersHandler::CANCEL_STATUS_CANCEL_CONFIRM_PENDING
&& $cancelStatus !== MembershipUsersHandler::CANCEL_STATUS_CANCELLED

Patrick Müller
committed
$userEmail = $this->getUser()->getAttribute('email');
if (empty($userEmail)) {
throw new QUI\Memberships\Exception([

Patrick Müller
committed
'quiqqer/memberships',
'exception.users.membershipuser.abortcancel.no_email_address'

Patrick Müller
committed
}
$abortCancelUrl = Verifier::startVerification($this->getAbortCancelVerification(), true);
$this->setAttributes([
'cancelStatus' => MembershipUsersHandler::CANCEL_STATUS_ABORT_CANCEL_CONFIRM_PENDING,
$this->addHistoryEntry(MembershipUsersHandler::HISTORY_TYPE_CANCEL_ABORT_START);
$this->setEditUser(QUI::getUsers()->getSystemUser());
$this->update();
// send abort cancellation mail
$this->sendMail(
QUI::getLocale()->get('quiqqer/memberships', 'templates.mail.startabortcancel.subject'),
dirname(__FILE__, 5).'/templates/mail_startabortcancel.html',
[
'abortCancelUrl' => $abortCancelUrl
);
}
/**
* Confirm abortion of cancellation
*
* @return void
*/
public function confirmAbortCancel()
{
$this->setAttributes([

Patrick Müller
committed
'cancelDate' => null,
'cancelStatus' => MembershipUsersHandler::CANCEL_STATUS_NOT_CANCELLED,
'cancelled' => false,
'cancelEndDate' => null
try {
Verifier::removeVerification($this->getAbortCancelVerification());
} catch (\Exception $Exception) {
QUI\System\Log::writeException($Exception);
}
$this->addHistoryEntry(MembershipUsersHandler::HISTORY_TYPE_CANCEL_ABORT_CONFIRM);
$this->setEditUser(QUI::getUsers()->getSystemUser());
try {
$this->update();
QUI::getEvents()->fireEvent('quiqqerMembershipsCancelAbort', [$this]);
} catch (\Exception $Exception) {
QUI\System\Log::writeException($Exception);
}

Patrick Müller
committed
* Confirm membership cancellation by user
*
* @return void
* @throws QUI\Memberships\Exception

Patrick Müller
committed
* @throws QUI\ExceptionStack|QUI\Exception
public function confirmManualCancel()
$this->setAttributes([

Patrick Müller
committed
'cancelled' => true,
'cancelStatus' => MembershipUsersHandler::CANCEL_STATUS_CANCELLED,
'cancelEndDate' => $this->getCurrentCancelEndDate()->format('Y-m-d H:i:s')
$this->addHistoryEntry(MembershipUsersHandler::HISTORY_TYPE_CANCEL_CONFIRM);
// send confirm cancel mail
$this->sendConfirmCancelMail();

Patrick Müller
committed
QUI::getEvents()->fireEvent('quiqqerMembershipsCancelConfirm', [$this]);
}
/**
* Send mail to user to confirm cancellation
*
* @return void
*/
public function sendConfirmCancelMail()
{

Patrick Müller
committed
try {
$subject = $this->getUser()->getLocale()->get(
'quiqqer/memberships',
'templates.mail.confirmcancel.subject'
);
$this->sendMail($subject, dirname(__FILE__, 5).'/templates/mail_confirmcancel.html');
} catch (\Exception $Exception) {
QUI\System\Log::writeException($Exception);
}
/**
* Send e-mail to remind user of an outstanding cancellation confirmation.
*
* @return bool - success
*/
public function sendConfirmCancelReminderMail()
{
try {
$subject = $this->getUser()->getLocale()->get(
'quiqqer/memberships',
'templates.mail.confirmcancel_reminder.subject'
);
$this->sendMail(
$subject,
dirname(__FILE__, 5).'/templates/mail_confirmcancel_reminder.html',
[
'cancelUrl' => Verifier::getVerificationUrl($this->getCancelVerification())
]
);
return true;
} catch (\Exception $Exception) {
QUI\System\Log::writeException($Exception);
return false;

Patrick Müller
committed
* @throws \QUI\Exception
$this->archive(MembershipUsersHandler::ARCHIVE_REASON_CANCELLED);
// send expired mail
$subject = $this->getUser()->getLocale()->get('quiqqer/memberships', 'templates.mail.expired.subject');

Patrick Müller
committed
$this->sendMail($subject, dirname(__FILE__, 5).'/templates/mail_cancelled.html');
QUI::getEvents()->fireEvent('quiqqerMembershipsCancelled', [$this]);
}
/**
* Check if this user has cancelled his membership
*
* @return mixed
*/
public function isCancelled()
{
return boolval($this->getAttribute('cancelled'));
/**
* Delete membership user and remove QUIQQER user from all unique groups
*
* A deleted membership user is not removed from the database but set to "archived".
*
* @return void
*/
public function delete()
Permission::checkPermission(MembershipUsersHandler::PERMISSION_EDIT_USERS, $this->EditUser);

Patrick Müller
committed
$this->addHistoryEntry(MembershipUsersHandler::HISTORY_TYPE_DELETED);
// do not delete, just set to archived
$this->archive(MembershipUsersHandler::ARCHIVE_REASON_DELETED);

Patrick Müller
committed
QUI::getEvents()->fireEvent('quiqqerMembershipsUserDelete', [$this]);
}
/**
* Set User to all membership QUIQQER groups
*
* @return void
*/
public function addToGroups()
{
$groupIds = $this->getMembership()->getGroupIds();
$User = $this->getUser();
foreach ($groupIds as $groupId) {
$User->addToGroup($groupId);
}
$User->save(QUI::getUsers()->getSystemUser());
}
/**
* Removes the membership user from all quiqqer groups (that he is part of because of
* his membership)
*
* @return void

Patrick Müller
committed
* @throws QUI\Exception

Patrick Müller
committed
/**
* Check if the user exists first. If he does NOT, then he does not need to be removed
* from QUIQQER groups (anymore).
*/
try {
$User = QUI::getUsers()->get($this->getUserId());
} catch (\Exception $Exception) {
if ($Exception->getCode() === 404) {
return;
}
QUI\System\Log::writeException($Exception);
return;
}
$Groups = QUI::getGroups();
$Memberships = MembershipsHandler::getInstance();
$Membership = $this->getMembership();
$membershipGroupIds = $Membership->getGroupIds();
// remove user from unique group ids
foreach ($Membership->getUniqueGroupIds() as $groupId) {
$Groups->get($groupId)->removeUser($User);
$k = array_search($groupId, $membershipGroupIds);
if ($k !== false) {
unset($membershipGroupIds[$k]);
}
// remove user from all non-unique group ids where the user is not part of
// the membership
foreach ($membershipGroupIds as $groupId) {
foreach ($Memberships->getMembershipIdsByGroupIds([$groupId]) as $membershipId) {
$OtherMembership = $Memberships->getChild($membershipId);
if (!$OtherMembership->hasMembershipUserId($User->getId())) {
$User->removeGroup($groupId);
}
}
}
$User->save(QUI::getUsers()->getSystemUser());
* @param string $reason - The reason why this user is archived
public function archive($reason)
$this->addHistoryEntry(MembershipUsersHandler::HISTORY_TYPE_ARCHIVED, [

Patrick Müller
committed
'reason' => $reason
]);
$this->setAttributes([
'archived' => 1,
'archiveDate' => Utils::getFormattedTimestamp(),
'archiveReason' => $reason
$this->update();
}
/**
* Checks if this membership user is archived
*
* @retun bool
*/
public function isArchived()
{
return boolval($this->getAttribute('archived'));
/**
* Get the Membership this membership user is assigned to
*
* @return QUI\Memberships\Membership
*/
public function getMembership()
{
return MembershipsHandler::getInstance()->getChild(
$this->getAttribute('membershipId')
);
}
/**
* Get QUIQQER User ID of membership user
*
* @return int
*/
public function getUserId()
{
return (int)$this->getAttribute('userId');
}
/**
* Get QUIQQER User
*
* @return QUI\Users\User

Patrick Müller
committed
* @throws \QUI\Exception
*/
public function getUser()
{
return QUI::getUsers()->get($this->getUserId());
}

Patrick Müller
committed
/**
* Get ID of the Contract if this MembershipUser was created due to a
* contract.
*
* @return int|false
*/
public function getContractId()
{
$contractId = $this->getAttribute('contractId');
if (empty($contractId)) {
return false;
}
return (int)$contractId;
}
/**
* Add an entry to the membership user history
*
* @param string $type - History entry type (see \QUI\Memberships\Users\Handler)
* @param string $msg (optional) - additional custom message
*/
public function addHistoryEntry($type, $msg = "")
$history = $this->getHistory();
if (empty($msg)) {
$msg = "";
}
if (is_array($msg)) {
$msg = json_encode($msg);
}
$history[] = [
'type' => $type,
'time' => Utils::getFormattedTimestamp(),
'user' => $User->getName().' ('.$User->getId().')',
$this->setAttribute('history', json_encode($history));
}
/**
* Get history data of this MembershipUser
*
* @return array
*/
public function getHistory()
{
$history = $this->getAttribute('history');
if (empty($history)) {
$history = [];
} else {
$history = json_decode($history, true);
}
return $history;
}

Patrick Müller
committed
/**
* Format date based on User Locale and duration mode
*
* @param string $date - Formatted date YYYY-MM-DD HH:MM:SS
* @return string|false - formatted date or false on error

Patrick Müller
committed
* @throws \QUI\Exception

Patrick Müller
committed
*/
protected function formatDate($date)
{
if (empty($date)
|| $date === '0000-00-00 00:00:00'
) {
return false;
}

Patrick Müller
committed
$Locale = $this->getUser()->getLocale();

Patrick Müller
committed
$durationMode = MembershipsHandler::getSetting('durationMode');
$Conf = QUI::getPackage('quiqqer/memberships')->getConfig();

Patrick Müller
committed
switch ($durationMode) {
case 'day':
$dateFormat = $Conf->get('date_formats_short', $lang);
// fallback to default value
if (empty($dateFormat)) {
$dateFormat = '%D';
}

Patrick Müller
committed
break;
default:
$dateFormat = $Conf->get('date_formats_long', $lang);
// fallback to default value
if (empty($dateFormat)) {
$dateFormat = '%D %H:%M';
}

Patrick Müller
committed
}
return $Locale->formatDate(strtotime($date), $dateFormat);

Patrick Müller
committed
}
/**
* Get membership data for frontend view/edit purposes with correctly formatted dates
*
* @return array
*/
public function getFrontendViewData()
{
$QuiqqerUser = $this->getUser();
$Membership = $this->getMembership();

Patrick Müller
committed
$Locale = QUI::getLocale();

Patrick Müller
committed
// determine source of title, short and content
$viewDataMode = MembershipUsersHandler::getSetting('viewDataMode');
$productId = $this->getAttribute('productId');
if ($viewDataMode === 'product'
&& !empty($productId)
&& Utils::isQuiqqerProductsInstalled()
) {
$Product = ProductsHandler::getProduct($productId);
$title = $Product->getTitle($Locale);
$description = $Product->getDescription($Locale);
$content = '';
} else {
$title = $Membership->getTitle($Locale);
$description = $Membership->getDescription($Locale);
$content = $Membership->getContent($Locale);
}
return [
'id' => $this->getId(),
'userId' => $QuiqqerUser->getId(),
'membershipId' => $Membership->getId(),
'membershipTitle' => $title,
'membershipShort' => $description,
'membershipContent' => $content,
'username' => $QuiqqerUser->getUsername(),
'fullName' => $QuiqqerUser->getName(),
'addedDate' => $this->formatDate($this->getAttribute('addedDate')),
'beginDate' => $this->formatDate($this->getAttribute('beginDate')),
'endDate' => $this->formatDate($this->getAttribute('endDate')),

Patrick Müller
committed
'cancelEndDate' => $this->formatDate($this->getCurrentCancelEndDate()->format('Y-m-d H:i:s')),
'cancelDate' => $this->formatDate($this->getAttribute('cancelDate')),
'cancelStatus' => $this->getAttribute('cancelStatus'),
// 'archived' => $this->isArchived(),
// 'archiveReason' => $this->getAttribute('archiveReason'),
'cancelled' => $this->isCancelled(),
'autoExtend' => $Membership->isAutoExtend(),
'infinite' => $Membership->isInfinite()

Patrick Müller
committed
}
/**
* Get membership data for backend view/edit purposes
*
* @return array
*/
public function getBackendViewData()
{
$QuiqqerUser = $this->getUser();
$Membership = $this->getMembership();
return [
'id' => $this->getId(),
'userId' => $QuiqqerUser->getId(),
'membershipId' => $Membership->getId(),
'membershipTitle' => $Membership->getTitle(),
'username' => $QuiqqerUser->getUsername(),
'firstname' => $QuiqqerUser->getAttribute('firstname'),
'lastname' => $QuiqqerUser->getAttribute('lastname'),
'fullName' => $QuiqqerUser->getName(),
'addedDate' => $this->getAttribute('addedDate'),
'beginDate' => $this->getAttribute('beginDate'),
'endDate' => $this->getAttribute('endDate'),
'archived' => $this->isArchived(),
'archiveReason' => $this->getAttribute('archiveReason'),
'archiveDate' => $this->getAttribute('archiveDate'),
'cancelled' => $this->isCancelled(),

Patrick Müller
committed
'infinite' => $Membership->isInfinite(),
'contractId' => $this->getContractId()
/**
* Get Verification object for MembershipUser cancellation
*
* @return CancelVerification
*/
protected function getCancelVerification()
{
return new CancelVerification($this->id);
}
/**
* Get Verification object for MembershipUser cancel abort
*
* @return AbortCancelVerification
*/
protected function getAbortCancelVerification()
{
return new AbortCancelVerification($this->id);
}
/**
* Send an email to the membership user
*
* @param string $subject - mail subject
* @param string $templateFile
* @param array $templateVars (optional) - additional template variables (besides $this)
* @return void
* @throws \QUI\Exception
public function sendMail($subject, $templateFile, $templateVars = [])
$User = $this->getUser();
$email = $User->getAttribute('email');
if (empty($email)) {
QUI\System\Log::addError(
'Could not send mail to user #'.$User->getId().' because the user has'
.' no email address!'
);
$Engine = QUI::getTemplateManager()->getEngine();
$Engine->assign(array_merge(

Patrick Müller
committed
'Locale' => $this->getUser()->getLocale(),
'data' => $this->getFrontendViewData()
$templateVars
));
$template = $Engine->fetch($templateFile);
$Mailer = new Mailer();
$Mailer->setSubject($subject);
$Mailer->setBody($template);
$Mailer->send();
}
/**
* Set any extra text data to the MembershipUser
*
* This is meant for extra information that is not already covered by the history.
*
* @param string $key
* @param string $value
*/
public function setExtraData($key, $value)
{
$extraData = $this->getExtraData();
$User = QUI::getUserBySession();
$userString = $User->getName().' ('.$User->getId().')';
$editString = Utils::getFormattedTimestamp().' - '.$userString;
if (isset($extraData[$key])) {
$extraData[$key]['edit'] = $editString;
$extraData[$key]['value'] = $value;
} else {
$extraData[$key] = [
'value' => $value,
'add' => $editString,
'edit' => '-'
}
$this->setAttribute('extraData', json_encode($extraData));
}
/**
* Get extra data of this MembershipUser
*
* @param string $key (optional) - If omitted return all extra data
* @return array|string|false
*/
public function getExtraData($key = null)
{
$extraData = $this->getAttribute('extraData');
if (empty($extraData)) {
$extraData = [];
} else {
$extraData = json_decode($extraData, true);
}
if (is_null($key)) {
return $extraData;
}
if (!array_key_exists($key, $extraData)) {
return false;
}
return $extraData[$key]['value'];
}

Patrick Müller
committed
/**
* Calculates the date the membership for this user would end
* if it was cancelled NOW
*
* @return \DateTime
* @throws \Exception
*/
public function getCurrentCancelEndDate()
{
$endDate = $this->getAttribute('endDate');