View source
<?php
namespace Drupal\simple_oauth\Entity;
use Drupal\Core\Entity\EntityStorageInterface;
use Drupal\Core\Field\BaseFieldDefinition;
use Drupal\Core\Entity\ContentEntityBase;
use Drupal\Core\Entity\EntityChangedTrait;
use Drupal\Core\Entity\EntityTypeInterface;
use Drupal\simple_oauth\AccessTokenInterface;
use Drupal\simple_oauth\AccessTokenValue;
use Drupal\user\UserInterface;
class AccessToken extends ContentEntityBase implements AccessTokenInterface {
use EntityChangedTrait;
const DEFAULT_EXPIRATION_PERIOD = 300;
const REFRESH_EXTENSION_TIME = 86400;
public static function preCreate(EntityStorageInterface $storage_controller, array &$values) {
parent::preCreate($storage_controller, $values);
$values += array(
'user_id' => \Drupal::currentUser()
->id(),
);
}
public static function baseFieldDefinitions(EntityTypeInterface $entity_type) {
$fields['id'] = BaseFieldDefinition::create('integer')
->setLabel(t('ID'))
->setDescription(t('The ID of the Access Token entity.'))
->setReadOnly(TRUE);
$fields['uuid'] = BaseFieldDefinition::create('uuid')
->setLabel(t('UUID'))
->setDescription(t('The UUID of the Access Token entity.'))
->setReadOnly(TRUE);
$fields['user_id'] = BaseFieldDefinition::create('entity_reference')
->setLabel(t('Creator'))
->setDescription(t('The user ID of author of the Access Token entity.'))
->setRevisionable(TRUE)
->setSetting('target_type', 'user')
->setSetting('handler', 'default')
->setDefaultValueCallback('Drupal\\node\\Entity\\Node::getCurrentUserId')
->setTranslatable(TRUE)
->setDisplayOptions('view', array(
'label' => 'inline',
'type' => 'author',
'weight' => 0,
))
->setDisplayOptions('form', array(
'type' => 'hidden',
))
->setDisplayConfigurable('form', TRUE)
->setDisplayConfigurable('view', TRUE);
$fields['auth_user_id'] = BaseFieldDefinition::create('entity_reference')
->setLabel(t('User'))
->setDescription(t('The user ID of the user this access token is authenticating.'))
->setRevisionable(TRUE)
->setSetting('target_type', 'user')
->setSetting('handler', 'default')
->setDefaultValueCallback('Drupal\\node\\Entity\\Node::getCurrentUserId')
->setTranslatable(FALSE)
->setDisplayOptions('view', array(
'label' => 'inline',
'type' => 'author',
'weight' => 0,
))
->setCardinality(1)
->setDisplayOptions('form', array(
'type' => 'entity_reference_autocomplete',
'weight' => 0,
'settings' => array(
'match_operator' => 'CONTAINS',
'size' => '60',
'autocomplete_type' => 'tags',
'placeholder' => '',
),
))
->setDisplayConfigurable('form', TRUE)
->setDisplayConfigurable('view', TRUE)
->setPropertyConstraints('target_id', [
'OwnOrAdmin' => [
'permission' => 'administer access token entities',
],
]);
$fields['resource'] = BaseFieldDefinition::create('entity_reference')
->setLabel(t('Resource'))
->setDescription(t('The resource for this Access Token.'))
->setRevisionable(TRUE)
->setSetting('target_type', 'access_token_resource')
->setSetting('handler', 'default')
->setDefaultValue('global')
->setTranslatable(FALSE)
->setDisplayOptions('view', array(
'label' => 'inline',
'type' => 'entity_reference_label',
'weight' => 4,
))
->setDisplayOptions('form', array(
'type' => 'entity_reference_autocomplete',
'weight' => 4,
'settings' => array(
'match_operator' => 'CONTAINS',
'size' => '60',
'autocomplete_type' => 'tags',
'placeholder' => '',
),
))
->setDisplayConfigurable('form', TRUE)
->setDisplayConfigurable('view', TRUE);
$fields['access_token_id'] = BaseFieldDefinition::create('entity_reference')
->setLabel(t('Refresh Token'))
->setDescription(t('The Refresh Token to re-create an Access Token.'))
->setRevisionable(TRUE)
->setSetting('target_type', 'access_token')
->setSetting('handler', 'default')
->setTranslatable(FALSE)
->setDisplayOptions('form', array(
'type' => 'hidden',
))
->setDisplayConfigurable('form', TRUE)
->setDisplayConfigurable('view', TRUE);
$fields['value'] = BaseFieldDefinition::create('string')
->setLabel(t('Token'))
->setDescription(t('The token value.'))
->setSettings(array(
'max_length' => 128,
'text_processing' => 0,
))
->setRequired(TRUE)
->setDefaultValue('')
->setDisplayOptions('view', array(
'label' => 'inline',
'type' => 'string',
'weight' => -4,
))
->setDisplayOptions('form', array(
'type' => 'hidden',
))
->setDisplayConfigurable('form', FALSE)
->setDisplayConfigurable('view', TRUE);
$fields['created'] = BaseFieldDefinition::create('created')
->setLabel(t('Created'))
->setDescription(t('The time that the entity was created.'))
->setDisplayOptions('view', array(
'label' => 'inline',
'type' => 'timestamp',
'weight' => 1,
))
->setDisplayConfigurable('view', TRUE);
$fields['changed'] = BaseFieldDefinition::create('changed')
->setLabel(t('Changed'))
->setDescription(t('The time that the entity was last edited.'));
$fields['expire'] = BaseFieldDefinition::create('timestamp')
->setLabel(t('Expire'))
->setDefaultValueCallback(__CLASS__ . '::defaultExpiration')
->setDescription(t('The time when the token expires.'))
->setDisplayOptions('form', array(
'type' => 'datetime_timestamp',
'weight' => 1,
))
->setDisplayOptions('view', array(
'label' => 'inline',
'type' => 'timestamp',
'weight' => 1,
))
->setRequired(TRUE)
->setDisplayConfigurable('form', TRUE)
->setDisplayConfigurable('view', TRUE);
return $fields;
}
public function getCreatedTime() {
return $this
->get('created')->value;
}
public function getOwner() {
return $this
->get('user_id')->entity;
}
public function getOwnerId() {
return $this
->get('user_id')->target_id;
}
public function setOwnerId($uid) {
$this
->set('user_id', $uid);
return $this;
}
public function setOwner(UserInterface $account) {
$this
->set('user_id', $account
->id());
return $this;
}
public function preSave(EntityStorageInterface $storage) {
parent::preSave($storage);
if ($this
->get('value')
->isEmpty()) {
$value = AccessTokenValue::createFromValues($this
->normalize())
->digest();
$this
->set('value', $value);
}
}
public function postSave(EntityStorageInterface $storage, $update = TRUE) {
parent::postSave($storage, $update);
if (!$this
->isRefreshToken()) {
$this
->addRefreshToken();
}
}
public function isRefreshToken() {
return !$this
->get('access_token_id')
->isEmpty() && $this
->get('resource')->target_id == 'authentication';
}
protected function addRefreshToken() {
$has_refresh_token = (bool) $this
->entityManager()
->getStorage($this
->getEntityTypeId())
->getQuery()
->condition('access_token_id', $this
->id())
->count()
->execute();
if ($has_refresh_token) {
return;
}
$extension = \Drupal::config('simple_oauth.settings')
->get('refresh_extension') ?: static::REFRESH_EXTENSION_TIME;
$values = [
'expire' => $this
->get('expire')->value + $extension,
'auth_user_id' => $this
->get('auth_user_id')->target_id,
'access_token_id' => $this
->id(),
'resource' => 'authentication',
'created' => $this
->getCreatedTime(),
'changed' => $this
->getChangedTime(),
];
$refresh_token = $this
->entityManager()
->getStorage($this
->getEntityType()
->id())
->create($values);
$refresh_token
->save();
}
protected function normalize() {
$keys = [
'auth_user_id',
'expire',
'created',
'resource',
];
$values = array_map(function ($item) {
return $this
->get($item)
->getValue();
}, $keys);
return $values;
}
public static function defaultExpiration() {
$expiration = \Drupal::config('simple_oauth.settings')
->get('expiration') ?: static::DEFAULT_EXPIRATION_PERIOD;
return [
REQUEST_TIME + $expiration,
];
}
public function hasPermission($permission) {
if ($permission == 'refresh access token') {
return $this
->isRefreshToken();
}
$resource = $this
->get('resource')->entity;
$token_permissions = $resource
->get('permissions') ?: [];
return $resource
->id() == 'global' || in_array($permission, $token_permissions);
}
public function refresh() {
if (!$this
->isRefreshToken()) {
return NULL;
}
$current_access_token = $this
->get('access_token_id')->entity;
if ($current_access_token && $current_access_token
->get('expire')->value > REQUEST_TIME) {
return $current_access_token;
}
$values = [
'expire' => static::defaultExpiration(),
'user_id' => $this
->get('user_id')->target_id,
'auth_user_id' => $this
->get('auth_user_id')->target_id,
'resource' => $current_access_token ? $current_access_token
->get('resource')->target_id : 'global',
'created' => REQUEST_TIME,
'changed' => REQUEST_TIME,
];
$store = \Drupal::entityManager()
->getStorage('access_token');
$access_token = $store
->create($values);
$this
->delete();
if ($current_access_token) {
$current_access_token
->delete();
}
$access_token
->save();
return $access_token;
}
}