2010-05-02 13:04:25 +02:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace Doctrine\Tests\ORM\Functional\Locking;
|
|
|
|
|
|
|
|
use Doctrine\Tests\Models\CMS\CmsArticle,
|
|
|
|
Doctrine\Tests\Models\CMS\CmsUser,
|
2010-05-15 11:48:20 +02:00
|
|
|
Doctrine\DBAL\LockMode,
|
2010-05-02 13:04:25 +02:00
|
|
|
Doctrine\ORM\EntityManager;
|
|
|
|
|
|
|
|
require_once __DIR__ . '/../../../TestInit.php';
|
|
|
|
|
2011-06-18 22:47:21 +02:00
|
|
|
/**
|
|
|
|
* @group locking_functional
|
|
|
|
*/
|
2010-05-02 13:04:25 +02:00
|
|
|
class GearmanLockTest extends \Doctrine\Tests\OrmFunctionalTestCase
|
|
|
|
{
|
|
|
|
private $gearman = null;
|
|
|
|
private $maxRunTime = 0;
|
|
|
|
private $articleId;
|
|
|
|
|
|
|
|
protected function setUp()
|
|
|
|
{
|
|
|
|
if (!class_exists('GearmanClient', false)) {
|
|
|
|
$this->markTestSkipped('pecl/gearman is required for this test to run.');
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->useModelSet('cms');
|
|
|
|
parent::setUp();
|
|
|
|
$this->tasks = array();
|
|
|
|
|
|
|
|
$this->gearman = new \GearmanClient();
|
|
|
|
$this->gearman->addServer();
|
|
|
|
$this->gearman->setCompleteCallback(array($this, "gearmanTaskCompleted"));
|
|
|
|
|
|
|
|
$article = new CmsArticle();
|
|
|
|
$article->text = "my article";
|
|
|
|
$article->topic = "Hello";
|
|
|
|
|
|
|
|
$this->_em->persist($article);
|
|
|
|
$this->_em->flush();
|
|
|
|
|
|
|
|
$this->articleId = $article->id;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function gearmanTaskCompleted($task)
|
|
|
|
{
|
|
|
|
$this->maxRunTime = max($this->maxRunTime, $task->data());
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testFindWithLock()
|
|
|
|
{
|
|
|
|
$this->asyncFindWithLock('Doctrine\Tests\Models\CMS\CmsArticle', $this->articleId, LockMode::PESSIMISTIC_WRITE);
|
|
|
|
$this->asyncFindWithLock('Doctrine\Tests\Models\CMS\CmsArticle', $this->articleId, LockMode::PESSIMISTIC_WRITE);
|
2011-12-19 22:56:19 +01:00
|
|
|
|
2010-05-02 13:04:25 +02:00
|
|
|
$this->assertLockWorked();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testFindWithWriteThenReadLock()
|
|
|
|
{
|
|
|
|
$this->asyncFindWithLock('Doctrine\Tests\Models\CMS\CmsArticle', $this->articleId, LockMode::PESSIMISTIC_WRITE);
|
|
|
|
$this->asyncFindWithLock('Doctrine\Tests\Models\CMS\CmsArticle', $this->articleId, LockMode::PESSIMISTIC_READ);
|
|
|
|
|
|
|
|
$this->assertLockWorked();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testFindWithReadThenWriteLock()
|
|
|
|
{
|
|
|
|
$this->asyncFindWithLock('Doctrine\Tests\Models\CMS\CmsArticle', $this->articleId, LockMode::PESSIMISTIC_READ);
|
|
|
|
$this->asyncFindWithLock('Doctrine\Tests\Models\CMS\CmsArticle', $this->articleId, LockMode::PESSIMISTIC_WRITE);
|
|
|
|
|
|
|
|
$this->assertLockWorked();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testFindWithOneLock()
|
|
|
|
{
|
|
|
|
$this->asyncFindWithLock('Doctrine\Tests\Models\CMS\CmsArticle', $this->articleId, LockMode::PESSIMISTIC_WRITE);
|
|
|
|
$this->asyncFindWithLock('Doctrine\Tests\Models\CMS\CmsArticle', $this->articleId, LockMode::NONE);
|
|
|
|
|
|
|
|
$this->assertLockDoesNotBlock();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testDqlWithLock()
|
|
|
|
{
|
|
|
|
$this->asyncDqlWithLock('SELECT a FROM Doctrine\Tests\Models\CMS\CmsArticle a', array(), LockMode::PESSIMISTIC_WRITE);
|
|
|
|
$this->asyncFindWithLock('Doctrine\Tests\Models\CMS\CmsArticle', $this->articleId, LockMode::PESSIMISTIC_WRITE);
|
|
|
|
|
|
|
|
$this->assertLockWorked();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testLock()
|
|
|
|
{
|
|
|
|
$this->asyncFindWithLock('Doctrine\Tests\Models\CMS\CmsArticle', $this->articleId, LockMode::PESSIMISTIC_WRITE);
|
|
|
|
$this->asyncLock('Doctrine\Tests\Models\CMS\CmsArticle', $this->articleId, LockMode::PESSIMISTIC_WRITE);
|
|
|
|
|
|
|
|
$this->assertLockWorked();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testLock2()
|
|
|
|
{
|
|
|
|
$this->asyncFindWithLock('Doctrine\Tests\Models\CMS\CmsArticle', $this->articleId, LockMode::PESSIMISTIC_WRITE);
|
|
|
|
$this->asyncLock('Doctrine\Tests\Models\CMS\CmsArticle', $this->articleId, LockMode::PESSIMISTIC_READ);
|
|
|
|
|
|
|
|
$this->assertLockWorked();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testLock3()
|
|
|
|
{
|
|
|
|
$this->asyncFindWithLock('Doctrine\Tests\Models\CMS\CmsArticle', $this->articleId, LockMode::PESSIMISTIC_READ);
|
|
|
|
$this->asyncLock('Doctrine\Tests\Models\CMS\CmsArticle', $this->articleId, LockMode::PESSIMISTIC_WRITE);
|
|
|
|
|
|
|
|
$this->assertLockWorked();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testLock4()
|
|
|
|
{
|
|
|
|
$this->asyncFindWithLock('Doctrine\Tests\Models\CMS\CmsArticle', $this->articleId, LockMode::NONE);
|
|
|
|
$this->asyncLock('Doctrine\Tests\Models\CMS\CmsArticle', $this->articleId, LockMode::PESSIMISTIC_WRITE);
|
|
|
|
|
|
|
|
$this->assertLockDoesNotBlock();
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function assertLockDoesNotBlock()
|
|
|
|
{
|
|
|
|
$this->assertLockWorked($onlyForSeconds = 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function assertLockWorked($forTime = 2, $notLongerThan = null)
|
|
|
|
{
|
|
|
|
if ($notLongerThan === null) {
|
|
|
|
$notLongerThan = $forTime + 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->gearman->runTasks();
|
|
|
|
|
|
|
|
$this->assertTrue($this->maxRunTime > $forTime,
|
|
|
|
"Because of locking this tests should have run at least " . $forTime . " seconds, ".
|
|
|
|
"but only did for " . $this->maxRunTime . " seconds.");
|
|
|
|
$this->assertTrue($this->maxRunTime < $notLongerThan,
|
|
|
|
"The longest task should not run longer than " . $notLongerThan . " seconds, ".
|
|
|
|
"but did for " . $this->maxRunTime . " seconds."
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function asyncFindWithLock($entityName, $entityId, $lockMode)
|
|
|
|
{
|
2010-05-15 09:58:39 +02:00
|
|
|
$this->startJob('findWithLock', array(
|
2010-05-02 13:04:25 +02:00
|
|
|
'entityName' => $entityName,
|
|
|
|
'entityId' => $entityId,
|
|
|
|
'lockMode' => $lockMode,
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function asyncDqlWithLock($dql, $params, $lockMode)
|
|
|
|
{
|
2010-05-15 09:58:39 +02:00
|
|
|
$this->startJob('dqlWithLock', array(
|
2010-05-02 13:04:25 +02:00
|
|
|
'dql' => $dql,
|
|
|
|
'dqlParams' => $params,
|
|
|
|
'lockMode' => $lockMode,
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function asyncLock($entityName, $entityId, $lockMode)
|
|
|
|
{
|
2010-05-15 09:58:39 +02:00
|
|
|
$this->startJob('lock', array(
|
2010-05-02 13:04:25 +02:00
|
|
|
'entityName' => $entityName,
|
|
|
|
'entityId' => $entityId,
|
|
|
|
'lockMode' => $lockMode,
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
2010-05-15 09:58:39 +02:00
|
|
|
protected function startJob($fn, $fixture)
|
2010-05-02 13:04:25 +02:00
|
|
|
{
|
|
|
|
$this->gearman->addTask($fn, serialize(array(
|
|
|
|
'conn' => $this->_em->getConnection()->getParams(),
|
|
|
|
'fixture' => $fixture
|
|
|
|
)));
|
|
|
|
|
|
|
|
$this->assertEquals(GEARMAN_SUCCESS, $this->gearman->returnCode());
|
|
|
|
}
|
|
|
|
}
|