User removal support.
This commit is contained in:
parent
21224c0966
commit
3ab7993d04
app/DoctrineMigrations
src/Skobkin/Bundle/PointToolsBundle
Command
DataFixtures/ORM
Entity
Repository
Service
tests/Skobkin/PointToolsBundle/Event
34
app/DoctrineMigrations/Version20170116224555.php
Normal file
34
app/DoctrineMigrations/Version20170116224555.php
Normal file
|
@ -0,0 +1,34 @@
|
|||
<?php
|
||||
|
||||
namespace Application\Migrations;
|
||||
|
||||
use Doctrine\DBAL\Migrations\AbstractMigration;
|
||||
use Doctrine\DBAL\Schema\Schema;
|
||||
|
||||
/**
|
||||
* User removal support
|
||||
*/
|
||||
class Version20170116224555 extends AbstractMigration
|
||||
{
|
||||
/**
|
||||
* @param Schema $schema
|
||||
*/
|
||||
public function up(Schema $schema)
|
||||
{
|
||||
// this up() migration is auto-generated, please modify it to your needs
|
||||
$this->abortIf($this->connection->getDatabasePlatform()->getName() !== 'postgresql', 'Migration can only be executed safely on \'postgresql\'.');
|
||||
|
||||
$this->addSql('ALTER TABLE users.users ADD is_removed BOOLEAN DEFAULT FALSE NOT NULL');
|
||||
}
|
||||
|
||||
/**
|
||||
* @param Schema $schema
|
||||
*/
|
||||
public function down(Schema $schema)
|
||||
{
|
||||
// this down() migration is auto-generated, please modify it to your needs
|
||||
$this->abortIf($this->connection->getDatabasePlatform()->getName() !== 'postgresql', 'Migration can only be executed safely on \'postgresql\'.');
|
||||
|
||||
$this->addSql('ALTER TABLE users.users DROP is_removed');
|
||||
}
|
||||
}
|
|
@ -75,15 +75,9 @@ class ImportUsersCommand extends ContainerAwareCommand
|
|||
continue;
|
||||
}
|
||||
|
||||
$createdAt = \DateTime::createFromFormat('Y-m-d_H:i:s', $row[3]);
|
||||
$createdAt = \DateTime::createFromFormat('Y-m-d_H:i:s', $row[3]) ?: new \DateTime();
|
||||
|
||||
if (!$createdAt) {
|
||||
$createdAt = new \DateTime();
|
||||
}
|
||||
|
||||
$user = (new User($row[0], $row[1], $row[2]))
|
||||
->setCreatedAt($createdAt)
|
||||
;
|
||||
$user = new User($row[0], $createdAt, $row[1], $row[2]);
|
||||
|
||||
if (!$input->getOption('check-only')) {
|
||||
$em->persist($user);
|
||||
|
|
|
@ -6,6 +6,7 @@ use Doctrine\ORM\EntityManagerInterface;
|
|||
use Psr\Log\LoggerInterface;
|
||||
use Skobkin\Bundle\PointToolsBundle\Entity\Subscription;
|
||||
use Skobkin\Bundle\PointToolsBundle\Entity\User;
|
||||
use Skobkin\Bundle\PointToolsBundle\Exception\Api\UserNotFoundException;
|
||||
use Skobkin\Bundle\PointToolsBundle\Repository\UserRepository;
|
||||
use Skobkin\Bundle\PointToolsBundle\Service\SubscriptionsManager;
|
||||
use Skobkin\Bundle\PointToolsBundle\Service\Api\UserApi;
|
||||
|
@ -185,6 +186,11 @@ class UpdateSubscriptionsCommand extends ContainerAwareCommand
|
|||
|
||||
try {
|
||||
$userCurrentSubscribers = $this->api->getUserSubscribersById($user->getId());
|
||||
} catch (UserNotFoundException $e) {
|
||||
$this->logger->warning('User not found. Marking as removed', ['login' => $user->getLogin(), 'user_id' => $user->getId()]);
|
||||
$user->markAsRemoved();
|
||||
|
||||
continue;
|
||||
} catch (\Exception $e) {
|
||||
$this->logger->error(
|
||||
'Error while getting subscribers. Skipping.',
|
||||
|
@ -222,14 +228,16 @@ class UpdateSubscriptionsCommand extends ContainerAwareCommand
|
|||
|
||||
private function getUsersForUpdate(int $appUserId): array
|
||||
{
|
||||
$usersForUpdate = [];
|
||||
|
||||
if ($this->input->getOption('all-users')) {
|
||||
$usersForUpdate = $this->userRepo->findAll();
|
||||
$usersForUpdate = $this->userRepo->findBy(['removed' => false]);
|
||||
} else {
|
||||
/** @var User $serviceUser */
|
||||
$serviceUser = $this->userRepo->find($appUserId);
|
||||
$serviceUser = $this->userRepo->findActiveUserWithSubscribers($appUserId);
|
||||
|
||||
if (!$serviceUser) {
|
||||
$this->logger->info('Service user not found');
|
||||
$this->logger->critical('Service user not found');
|
||||
// @todo Retrieving user
|
||||
|
||||
throw new \RuntimeException('Service user not found in the database');
|
||||
|
@ -239,6 +247,10 @@ class UpdateSubscriptionsCommand extends ContainerAwareCommand
|
|||
|
||||
try {
|
||||
$usersForUpdate = $this->api->getUserSubscribersById($appUserId);
|
||||
} catch (UserNotFoundException $e) {
|
||||
$this->logger->critical('Service user deleted or API response is invalid');
|
||||
|
||||
throw $e;
|
||||
} catch (\Exception $e) {
|
||||
$this->logger->warning(
|
||||
'Error while getting service subscribers. Fallback to local list.',
|
||||
|
@ -251,8 +263,6 @@ class UpdateSubscriptionsCommand extends ContainerAwareCommand
|
|||
]
|
||||
);
|
||||
|
||||
$usersForUpdate = [];
|
||||
|
||||
/** @var Subscription $subscription */
|
||||
foreach ((array) $serviceUser->getSubscribers() as $subscription) {
|
||||
$usersForUpdate[] = $subscription->getSubscriber();
|
||||
|
|
|
@ -27,9 +27,7 @@ class LoadUserData extends AbstractFixture implements OrderedFixtureInterface
|
|||
$userId = 99999;
|
||||
|
||||
foreach ($this->users as $userData) {
|
||||
$user = (new User($userId--, $userData['login'], $userData['name']))
|
||||
->setCreatedAt(new \DateTime())
|
||||
;
|
||||
$user = new User($userId--, new \DateTime(), $userData['login'], $userData['name']);
|
||||
|
||||
$om->persist($user);
|
||||
|
||||
|
|
|
@ -68,17 +68,25 @@ class User
|
|||
|
||||
/**
|
||||
* @var ArrayCollection|SubscriptionEvent[]
|
||||
*
|
||||
* @ORM\OneToMany(targetEntity="SubscriptionEvent", mappedBy="author", fetch="EXTRA_LAZY")
|
||||
*/
|
||||
private $newSubscriberEvents;
|
||||
|
||||
/**
|
||||
* @var bool
|
||||
*
|
||||
* @ORM\Column(name="is_removed", type="boolean", options={"default": false})
|
||||
*/
|
||||
private $removed = false;
|
||||
|
||||
public function __construct(int $id, string $login = null, string $name = null)
|
||||
|
||||
public function __construct(int $id, \DateTime $createdAt = null, string $login = null, string $name = null)
|
||||
{
|
||||
$this->id = $id;
|
||||
$this->login = $login;
|
||||
$this->name = $name;
|
||||
$this->createdAt = new \DateTime();
|
||||
$this->createdAt = $createdAt ?: new \DateTime();
|
||||
|
||||
$this->subscribers = new ArrayCollection();
|
||||
$this->subscriptions = new ArrayCollection();
|
||||
|
@ -98,30 +106,25 @@ class User
|
|||
return $this->id;
|
||||
}
|
||||
|
||||
public function setLogin(string $login): self
|
||||
{
|
||||
$this->login = $login;
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
public function getLogin(): string
|
||||
{
|
||||
return $this->login;
|
||||
}
|
||||
|
||||
public function setName(?string $name): self
|
||||
{
|
||||
$this->name = $name;
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
public function getName(): ?string
|
||||
{
|
||||
return $this->name;
|
||||
}
|
||||
|
||||
public function updateLoginAndName(string $login, string $name): self
|
||||
{
|
||||
$this->login = $login;
|
||||
$this->name = $name;
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
public function addSubscriber(Subscription $subscribers): self
|
||||
{
|
||||
$this->subscribers[] = $subscribers;
|
||||
|
@ -170,15 +173,18 @@ class User
|
|||
return $this->createdAt;
|
||||
}
|
||||
|
||||
public function setCreatedAt(\DateTime $createdAt): self
|
||||
{
|
||||
$this->createdAt = $createdAt;
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
public function getUpdatedAt(): ?\DateTime
|
||||
{
|
||||
return $this->updatedAt;
|
||||
}
|
||||
|
||||
public function isRemoved(): bool
|
||||
{
|
||||
return $this->isRemoved();
|
||||
}
|
||||
|
||||
public function markAsRemoved(): void
|
||||
{
|
||||
$this->removed = true;
|
||||
}
|
||||
}
|
||||
|
|
|
@ -13,6 +13,19 @@ class UserRepository extends EntityRepository
|
|||
$this->getEntityManager()->persist($entity);
|
||||
}
|
||||
|
||||
public function findActiveUserWithSubscribers(int $id): ?User
|
||||
{
|
||||
$qb = $this->createQueryBuilder('u');
|
||||
|
||||
return $qb
|
||||
->select(['u', ''])
|
||||
->innerJoin('u.subscribers', 's')
|
||||
->innerJoin('s.subscriber', 'us')
|
||||
->where('u.removed = FALSE')
|
||||
->getQuery()->getOneOrNullResult()
|
||||
;
|
||||
}
|
||||
|
||||
/**
|
||||
* Case-insensitive user search
|
||||
*/
|
||||
|
|
|
@ -51,22 +51,6 @@ class AbstractApi
|
|||
$this->logger = $logger;
|
||||
}
|
||||
|
||||
/**
|
||||
* Make GET request and return response body
|
||||
*/
|
||||
public function getGetResponseBody($path, array $parameters = []): StreamInterface
|
||||
{
|
||||
return $this->sendGetRequest($path, $parameters)->getBody();
|
||||
}
|
||||
|
||||
/**
|
||||
* Make POST request and return response body
|
||||
*/
|
||||
public function getPostResponseBody(string $path, array $parameters = []): StreamInterface
|
||||
{
|
||||
return $this->sendPostRequest($path, $parameters)->getBody();
|
||||
}
|
||||
|
||||
/**
|
||||
* Make GET request and return DTO objects
|
||||
*
|
||||
|
@ -97,6 +81,22 @@ class AbstractApi
|
|||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Make GET request and return response body
|
||||
*/
|
||||
public function getGetResponseBody($path, array $parameters = []): StreamInterface
|
||||
{
|
||||
return $this->sendGetRequest($path, $parameters)->getBody();
|
||||
}
|
||||
|
||||
/**
|
||||
* Make POST request and return response body
|
||||
*/
|
||||
public function getPostResponseBody(string $path, array $parameters = []): StreamInterface
|
||||
{
|
||||
return $this->sendPostRequest($path, $parameters)->getBody();
|
||||
}
|
||||
|
||||
/**
|
||||
* @param string $path Request path
|
||||
* @param array $parameters Key => Value array of query parameters
|
||||
|
@ -151,6 +151,12 @@ class AbstractApi
|
|||
$code = $response->getStatusCode();
|
||||
$reason = $response->getReasonPhrase();
|
||||
|
||||
// @todo remove after fix
|
||||
// Temporary fix until @arts fixes this bug
|
||||
if ('{"error": "UserNotFound"}' === (string) $response->getBody()) {
|
||||
throw new NotFoundException($reason, $code);
|
||||
}
|
||||
|
||||
switch ($code) {
|
||||
case SymfonyResponse::HTTP_UNAUTHORIZED:
|
||||
throw new UnauthorizedException($reason, $code);
|
||||
|
|
|
@ -39,16 +39,14 @@ class UserFactory extends AbstractFactory
|
|||
|
||||
/** @var User $user */
|
||||
if (null === ($user = $this->userRepository->find($userData->getId()))) {
|
||||
// Creating new user
|
||||
$user = new User($userData->getId());
|
||||
$user = new User(
|
||||
$userData->getId(),
|
||||
\DateTime::createFromFormat('Y-m-d_H:i:s', $userData->getCreated()) ?: new \DateTime()
|
||||
);
|
||||
$this->userRepository->add($user);
|
||||
}
|
||||
|
||||
// Updating data
|
||||
$user
|
||||
->setLogin($userData->getLogin())
|
||||
->setName($userData->getName())
|
||||
;
|
||||
$user->updateLoginAndName($userData->getLogin(), $userData->getName());
|
||||
|
||||
return $user;
|
||||
}
|
||||
|
|
|
@ -52,6 +52,7 @@ class SubscriptionsManager
|
|||
*/
|
||||
public function updateUserSubscribers(User $user, $newSubscribersList = []): void
|
||||
{
|
||||
// @todo optimize
|
||||
$tmpOldSubscribers = $user->getSubscribers();
|
||||
|
||||
$oldSubscribersList = [];
|
||||
|
|
|
@ -14,15 +14,15 @@ class UserSubscribersUpdatedEventTest extends \PHPUnit_Framework_TestCase
|
|||
|
||||
public function testCreate()
|
||||
{
|
||||
$user = new User(99999, 'testuser', 'Test User 1');
|
||||
$user = new User(99999, new \DateTime(), 'testuser', 'Test User 1');
|
||||
|
||||
$subscribed = [
|
||||
new User(99998, 'testuser2', 'Test User 2'),
|
||||
new User(99998, new \DateTime(), 'testuser2', 'Test User 2'),
|
||||
];
|
||||
|
||||
$unsubscribed = [
|
||||
new User(99997, 'testuser3', 'Test User 3'),
|
||||
new User(99996, 'testuser4', 'Test User 4'),
|
||||
new User(99997, new \DateTime(), 'testuser3', 'Test User 3'),
|
||||
new User(99996, new \DateTime(), 'testuser4', 'Test User 4'),
|
||||
];
|
||||
|
||||
return new UserSubscribersUpdatedEvent($user, $subscribed, $unsubscribed);
|
||||
|
|
|
@ -15,7 +15,7 @@ class UsersRenamedEventTest extends \PHPUnit_Framework_TestCase
|
|||
|
||||
public function testGetRenames()
|
||||
{
|
||||
$user = new User(99999, 'testuser', 'Test User 1');
|
||||
$user = new User(99999, new \DateTime(), 'testuser', 'Test User 1');
|
||||
$renameRecords = [
|
||||
new UserRenameEvent($user, 'testuser_old1'),
|
||||
new UserRenameEvent($user, 'testuser_old2'),
|
||||
|
|
Loading…
Reference in a new issue