acquire(); $this->assertTrue($ok); $lock->release(); } public function test_consecutive_locking() { $path = __DIR__ . '/../tmp/precious'; $lock = new phpbb_lock_flock($path); $ok = $lock->acquire(); $this->assertTrue($ok); $lock->release(); $ok = $lock->acquire(); $this->assertTrue($ok); $lock->release(); $ok = $lock->acquire(); $this->assertTrue($ok); $lock->release(); } /* This hangs the process. public function test_concurrent_locking_fail() { $path = __DIR__ . '/../tmp/precious'; $lock1 = new phpbb_lock_flock($path); $ok = $lock1->acquire(); $this->assertTrue($ok); $lock2 = new phpbb_lock_flock($path); $ok = $lock2->acquire(); $this->assertFalse($ok); $lock->release(); $ok = $lock2->acquire(); $this->assertTrue($ok); } */ public function test_concurrent_locking() { if (!function_exists('pcntl_fork')) { $this->markTestSkipped('pcntl extension and pcntl_fork are required for this test'); } $path = __DIR__ . '/../tmp/precious'; $pid = pcntl_fork(); if ($pid) { // parent // wait 0.5 s, acquire the lock, note how long it took sleep(1); $lock = new phpbb_lock_flock($path); $start = time(); $ok = $lock->acquire(); $delta = time() - $start; $this->assertTrue($ok); $this->assertGreaterThan(0.5, $delta, 'First lock acquired too soon'); $lock->release(); // acquire again, this should be instantaneous $start = time(); $ok = $lock->acquire(); $delta = time() - $start; $this->assertTrue($ok); $this->assertLessThan(0.1, $delta, 'Second lock not acquired instantaneously'); // reap the child $status = null; pcntl_waitpid($pid, $status); } else { // child // immediately acquire the lock and sleep for 2 s $lock = new phpbb_lock_flock($path); $ok = $lock->acquire(); $this->assertTrue($ok); sleep(2); $lock->release(); // and go away silently pcntl_exec('/usr/bin/env', array('true')); } } }