Skip to content

Commit a9cd22e

Browse files
committed
Add README.md
1 parent 4c613ed commit a9cd22e

File tree

1 file changed

+142
-0
lines changed

1 file changed

+142
-0
lines changed

README.md

+142
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,142 @@
1+
# php-lock
2+
3+
[![Build Status](https://travis-ci.org/texthtml/php-lock.svg?branch=master)](https://travis-ci.org/texthtml/php-lock)
4+
[![Latest Stable Version](https://poser.pugx.org/texthtml/php-lock/v/stable.svg)](https://packagist.org/packages/texthtml/php-lock)
5+
[![License](https://poser.pugx.org/texthtml/php-lock/license.svg)](http://www.gnu.org/licenses/agpl-3.0.html)
6+
[![Total Downloads](https://poser.pugx.org/texthtml/php-lock/downloads.svg)](https://packagist.org/packages/texthtml/php-lock)
7+
[![Scrutinizer Code Quality](https://scrutinizer-ci.com/g/texthtml/php-lock/badges/quality-score.png?b=master)](https://scrutinizer-ci.com/g/texthtml/php-lock/?branch=master)
8+
9+
[php-lock](https://packagist.org/packages/texthtml/php-lock) is a library that makes locking on resources easy. It can be used to avoid access to a file during write operations or to prevent crontabs to overlap. And it's designed to integrate well with Dependancy Injection (eg Symfony Container or Pimple).
10+
11+
## Installation
12+
13+
Through Composer :
14+
15+
```json
16+
{
17+
"require": {
18+
"texthtml/php-lock": "~1.0"
19+
}
20+
}
21+
```
22+
23+
## Usage
24+
25+
You can create an object that represent a lock on a file. You can then try to acquire that lock by calling `$lock->acquire()`. If the lock fail it will throw an `Exception` (useful for CLI tools built with [Symfony Console Components documentation](http://symfony.com/doc/current/components/console/introduction.html)). If the lock is acquired the program can continue.
26+
27+
### Locking a file exclusively
28+
29+
```php
30+
use TH\Lock\FileLock;
31+
32+
$lock = new FileLock('/path/to/file');
33+
34+
$lock->acquire();
35+
36+
// other processes that try to acquire a lock on the file will fail
37+
38+
// edit /path/to/file
39+
40+
$lock->release();
41+
42+
// other processes can now acquire a lock on the file
43+
```
44+
45+
### Sharing a lock on a file
46+
47+
```php
48+
49+
use TH\Lock\FileLock;
50+
51+
$lock = new FileLock('/path/to/file');
52+
53+
$lock->acquire(FileLock::SHARED);
54+
55+
// other processes that try to acquire an exclusive lock on the file will fail,
56+
// processes that try to acquire an shared lock on the file will succeed
57+
58+
// read /path/to/file
59+
60+
$lock->acquire();
61+
62+
// other processes can now acquire an exclusive lock on the file if no other shared lock remains.
63+
```
64+
65+
### Upgrading or downgrading a lock
66+
67+
`$lock->acquire()` can be called multiple times to either upgrade a shared lock to an exclusive lock or downgrade an exclusive lock to a shared one. You can use it when you're done editing the file.
68+
69+
### Auto release
70+
71+
`$lock->release()` is called automatically when the lock is destroyed so you d'ont need to manually release it at the end of a script or if it got out of scope.
72+
73+
```php
74+
use TH\Lock\FileLock;
75+
76+
function batch() {
77+
$lock = new FileLock('/fome/file');
78+
$lock->acquire();
79+
80+
// lot of editing on file
81+
}
82+
83+
batch();
84+
85+
// the lock will be released here even if $lock->release() is not called in batch()
86+
```
87+
88+
### Using lock for crontabs
89+
90+
When you don't want some crontabs to overlap you can make a lock on the same file in each crontab. The `TH\Lock\LockFactory` can ease the process and provide more helpful message in case of overlap.
91+
92+
```
93+
$lock = $factory->create('protected resource', 'process 1');
94+
95+
$lock->acquire();
96+
97+
// process 1 does stuff
98+
```
99+
100+
```
101+
$lock = $factory->create('protected resource', 'process 2');
102+
103+
$lock->acquire();
104+
105+
// process 2 does stuff
106+
```
107+
108+
When process 1 is running and we start process 2, an Exception will be thrown: "Could not acquire exclusive lock on protected resource" and if the factory was configured with a `\Psr\Log\LoggerInterface`, messages explaining what happend would be logged:
109+
110+
process 1: exclusive lock acquired on protected resource
111+
process 2: could not acquire exclusive lock on protected resource
112+
process 2: lock released on protected resource
113+
114+
The only `LockFactory` available at the moment is the `TH\Lock\FileFactory`. This factory autmatically create lock files for your resources in the specified folder.
115+
116+
```php
117+
118+
use TH\Lock\FileFactory;
119+
120+
$factory = new FileFactory('/path/to/lock_dir/');
121+
$factory->setLogger($psrLogger);
122+
123+
$factory->create(...);
124+
```
125+
126+
## API
127+
128+
There are three methods you can use on a `FileLock`:
129+
130+
* `\TH\Lock\FileLock::acquire($exclusive = FileLock::EXCLUSIVE, $blocking = FileLock::NON_BLOCKING)` used to acquire a lock on the file
131+
* `\TH\Lock\FileLock::release()` used to release a lock on the file
132+
* `\TH\Lock\FileLock::setLogger(\Psr\Log\LoggerInterface $logger)` used to configure this lock logger (configured by the factory if the `LockFile` come from the factory)
133+
134+
And two on `FileFactory':
135+
136+
* `\TH\Lock\FileFactory::create($resource, $owner = null)` used to create a `FileLock` for $resource
137+
* `\TH\Lock\FileFactory::setLogger(\Psr\Log\LoggerInterface $logger)` used to configure lock build by this factory
138+
`
139+
140+
## Notes
141+
142+
Only lock file are currently supported. It means distributed processes can't be lock this way. It should be easy to implements the `TH\Lock\Lock` and `TH\Lock\LockFactory` interface to use a distributed lock mechanism (maybe with [Redis](http://redis.io/topics/distlock)).

0 commit comments

Comments
 (0)