| Recommend this page to a friend! | 
|  Download | 
| Info | Documentation |  Files |  Install with Composer |  Download | Reputation | Support forum | Blog | Links | 
| Last Updated | Ratings | Unique User Downloads | Download Rankings | |||||
| 2025-05-08 (8 hours ago)  |     68% | Total: 850 This week: 1 | All time:  4,103 This week: 42  | |||||
| Version | License | PHP version | Categories | |||
| tad-php 0.4.13 | MIT/X Consortium ... | 5.4 | Networking, PHP 5, Web services, Hard... | 
| Description | Author | |||
| This package can communicate with ZK time and attendance devices. Innovation Award 
 | 
A simple PHP class to interacts with ZK Time & Attendance Devices.
TAD: A class that implements an interface to interacts with ZK Time & Attendance devices.
Documentation found about ZK SOAP api is very limited or poor, however TAD class implements most SOAP functions supported by ZK devices. Specifically TAD class exposes the following 35 methods:
get_date, get_att_log, get_user_info, get_all_user_info, get_user_template, get_combination, get_option, get_free_sizes, get_platform, get_fingerprint_algorithm, get_serial_number, get_oem_vendor, get_mac_address, get_device_name, get_manufacture_time, get_antipassback_mode, get_workcode, get_ext_format_mode, get_encrypted_mode, get_pin2_width, get_ssr_mode, get_firmware_version, set_date, set_user_info, set_user_template, delete_user, delete_template, delete_data, delete_user_password, delete_admin, enable, disable, refresh_db, restart, and poweroff.
All methods above are implemented by 2 classes: Providers\TADSoap and Providers\TADZKLib.
There are some SOAP functions supported by ZK devices that it's suppossed, according to the official docs (which incidentally it's very limited and so poor!!!) must show an expected behaviour, but when they are invoked don't work like is expected, so they become useless (e.g. Restart SOAP call). For these situations, TAD class implement them by Providers\TADZKLib class ([PHP_ZKLib] - http://dnaextrim.github.io/php_zklib/). This class takes a different approach to "talk to the device": it uses UDP protocol at device standard port 4370.
PHP_ZKLib class it's been fully integrated, after a refactoring process taking out all duplicated code (DRY).
For practical purposes, you don't have to be worried about when to use TAD class or PHP_ZKLib class because you only have to get a TAD instance (as shown below) and call any of its methods available. The class decides about when runs the method invoked using TAD class or PHP_ZKLib class.
After download TAD-PHP, you have 2 ways to get your enviroment configured to use the classes:
Composer is the PHP's package manager and is the recommended way to get packages for your projects. It's also able to build automatically *autoloaders* if you wrote down your code using PSR-0 and/or PSR-4 standards, avoiding you headaches about everything related to loading classes.
TADPHP is built follows PSR-4 standard and comes with a specific file named composer.json that allows Composer to generate a file named autoload.php (beside others files of course). This files generated is the only one you need to include in your project to get all classes required by TADPHP loaded in memory:
Install Composer:
    curl -s https://getcomposer.org/installer | php
Get inside TADPHP root folder and generate the autoload.php file:
    php composer.phar dump-autoload
    ```
    The command above will generate a folder called vendor. Inside of it, you'll see the autoload.php
<?php require 'vendor/autoload.php'; ...
```
Even if Composer it's the preferred method to generate the files needed to get all classes loaded, maybe you want to do the task by hand:
Require/Include all classes required by TAD-PHP using the relative TAD-PHP path
    <?php
require 'tad/lib/TADFactory.php';
    require 'tad/lib/TAD.php';
require 'tad/lib/TADResponse.php';
require 'tad/lib/Providers/TADSoap.php';
require 'tad/lib/Providers/TADZKLib.php';
require 'tad/lib/Exceptions/ConnectionError.php';
require 'tad/lib/Exceptions/FilterArgumentError.php';
require 'tad/lib/Exceptions/UnrecognizedArgument.php';
require 'tad/lib/Exceptions/UnrecognizedCommand.php';
    ```
All TAD-PHP classes are under the namespace named TADPHP. So, to use any class you need to use the Fully qualified class name. For example, to get a new instance of TADFactory class you need to use:
<?php
...
$tad_factory = new TADPHP\TADFactory();
...
However, as your project grows up using fully qualified class names becomes annoying, so it's better to use PHP USE sentence:
<?php
...
use TADPHP\TADFactory;
use TADPHP\TAD;
...
$comands = TAD::commands_available();
$tad = (new TADFactory(['ip'=>'192.168.100.156', 'com_key'=>0]))->get_instance();
...
First, instantiate a TADFactory object, then use it to create a TAD object.
<?php
...
$tad_factory = new TADFactory(['ip'=>'192.168.0.1']);
$tad = $tad_factory->get_instance();
...
Or you can get a TAD object in one single step (valid only in PHP 5.4+):
<?php
  $tad = (new TADFactory(['ip'=>'192.168.0.1']))->get_instance();
You can customize TAD object traits passing an options array:
<?php
  $options = [
    'ip' => '192.168.0.1',   // '169.254.0.1' by default (totally useless!!!).
    'internal_id' => 100,    // 1 by default.
    'com_key' => 123,        // 0 by default.
    'description' => 'TAD1', // 'N/A' by default.
    'soap_port' => 8080,     // 80 by default,
    'udp_port' => 20000      // 4370 by default.
    'encoding' => 'utf-8'    // iso8859-1 by default.
  ];
  
  $tad_factory = new TADFactory($options);
  $tad = $tad_factory->get_instance();  
SOAP API is implemented by TADSoap class. All methods that use UDP Protocol are implemented by PHP_ZKLib class. Even though you have 2 classes, you do not have to be worried about which method is been calling using SOAP api or through PHP_ZKLib. You've got a single interface.
Some methods need that you set up some parameters prior you can call them. TAD class uses associative arrays as way to pass params to the methods. Using associative arrays is a "more verbose way" that helps you to remember which params you have to pass.
Valid params supported by TAD class are:
com_key, pin, time, template, name, password, group, privilege, card, pin2, tz1, tz2, tz3, finger_id, option_name, date, size, valid, value
As you can see, params names are so intuitive and easy to remember.
// Get a full list of commands supported by TADPHP\TAD class.
$commands_list = TAD::commands_available();
// Get a list of commands implemented via TADPHP\TADSoap.
$soap_commands =  = TAD::soap_commands_available();
// Get a list of commands implemented via TAD\PHP\TADSoap.
$zklib_commands = TAD::zklib_commands_available();
// Getting current time and date
$dt = $tad->get_date();
// Setting device's date to '2014-01-01' (time will be set to now!)
$response = $tad->set_date(['date'=>'2014-01-01']);
// Setting device's time to '12:30:15' (date will be set to today!)
$response = $tad->set_date(['time'=>'12:30:15']);
// Setting device's date & time
$response = $tad->set_date(['date'=>'2014-01-01', 'time'=>'12:30:15']);
// Setting device's date & time to now.
$response = $tad->set_date();
You can retrieve attendance logs for all user or just for one:
// Getting attendance logs from all users.
$logs = $tad->get_att_log();
// Getting attendance logs from one user.
$logs = $tad->get_att_log(['pin'=>123]);
You can get all information about a single user or all users. The information you can get include:
// Getting info from a specific user.
$user_info = $tad->get_user_info(['pin'=>123]);
// Getting info from all users.
$all_user_info = $tad->get_all_user_info();
TAD class allows you to register new users in the device or even you can update (change) information about an user already registered. However to achieve this, TAD class needs to delete the user (of course this applies when you are updating user's information) and then creates the user. Maybe this is not the best way to do that, but if TAD just calls the method to create a user, it will be created as many times as you call it.
If you look into PHP_ZKLib code, you'll see a method to create / update users. However, when you call that method, it generates a PIN code (not PIN2 code) in a way that if that code already exists in the device, it refuses to create the user. This is a method that should be modified to make it working properly but the way how PIN code is created is unknown.
In the meantime, TAD class uses delete and create SOAP calls. Of course, to make things easy for you, you have to call just 1 method.
// We are creating a superadmin user named 'Foo Bar' with a PIN = 123 and password = 4321.
$r = $tad->set_user_info([
    'pin' => 123,
    'name'=> 'Foo Bar',
    'privilege'=> 14,
    'password' => 4321
]);
The device uses an algorithm to encode fingerprints called "BioBridge" and it has 2 flavors: VX 9.0 and the new one VX 10.0. According the documentation, VX 10.0 generates shorter encoded fingerprints and it's faster when the device has to make searchings for a fingerprint match process. However, TAD class exposes a method to upload fingerprints but it works only when device is configured to use the old BioBridge VX 9.0 algorithm. When device uses VX 10.0 algorithm, the machine freezes!!!. When asked to ZK Software forum, the answer got was: "It has to work with any biobridge version. Check your code!". Any help about this, would be appreciated.
/ Setting a user template (fingerprint).
 * 
 * You can upload until 10 templates per user. You have to use 'finger_id' param to indicate
 * which fingerprint you are uploading to.
 * 
 * Till now, this method only works with VX 9.0 BioBridge algorithm :-(. Any help
 * arround this issue will be appreciated. 
 */
 
// The folowing string represents a fingerprint encoded using BioBridge algorithm VX 9.0
$template1_vx9 = "ocosgoulTUEdNKVRwRQ0I27BDTEkdMEONK9KQQunMVSBK6VPLEENk9MwgQ+DP3PBC1FTXEEG4ihpQQQ3vFQBO4K+WwERYilHAQ8ztktBEBbKQ0ELDtJrwQ7dqCiBCz+/IgEGKrBjQQhEO0zBFQNDQYEKFbhrQQdLF1wBDxclfUELMNFXwQRvvmHBCslKUAEZfU1OQRzmIU5BXRW0eoEKPMltgQnQGUyBJQSfRIEUSzIdAQ45l3gBByHUTMEJ5yVhQQmi0UZBFHvYPUEGeKxTAQ6rFGNBCIYURoEOZS9VwR+1M4RoE5m0DRUTF8DHd6HdqxHAxWmj393M28DDX2FkanKi/t7LGsDCWqGarmt1BaL/25nAwVaiipu/cgcQGKG6mcDBU6KYmr5wChQcobmJIsDBUKKJmZ1uExyi+ZaYwMFMgU2CQCSinYdnJsDBR4Ghl3Q4owa3dnfAwUamdlZlR5p2Zi7AwUSndERlfOpWZlfAwUOiQzVkLDhDopRUVTLAwT2iQ0ZjIzVMolNFRcDBN6I0ZlQebVaiEjRVwMEyolVVUxVxXKEBRUTAwS+iZVYyD3JhoQJFTMDBLKJlVUIKcWShBVVTwMIkoWVkFQhyaaEVZ1rAwh6hVlUPAW+iNGd3wMIToWdlBnWiRWZ3aMDDCqRmZjRpZmrAxASjd2Vnh2/gAA==";
$template1_data = [
  'pin' => 123,
  'finger_id' => 0, // First fingerprint has 0 as index.
  'size' => 514,    // Be careful, this is not string length of $template1_vx9 var.
  'valid' => 1,
  'template' => $template1_vx9
];
$tad->set_user_template( $template1_data );
When you have to delete user's fingerprints, you delete all of them. You cannot delete fingerprint one by one.
// Delete all fingerprints (template) for user with PIN = 123.
$tad->delete_template(['pin'=>123]);
// Delete password for user with PIN = 123.
$tad->delete_user_password(['pin'=>123]);
/// Delete user with PIN = 123.
$tad->delete_user(['pin'=>123]);
From the device point of view, users that have a privilege level not equal to 0, are considered admin users, so this method enables you wipe them out with one single step.
$tad->delete_admin();
You can clear information from device according a code you specify.
Code meaning:
1: clear all device data, 2: clear all users templates and 3: clear all attendance logs.
// Delete all attendance logs stored.
$tad->delete_data(['value'=>3]);
You can get some valuable statistics about your device including:
* Space available for templates. * Space available for attendance logs. * Total storage capacity for attendance logs. * Total storage capacity for user templates. * Total users stored * Total user passwords stored. * Total attendance logs stored. * Total templates stored.
// Get some device statistics.
$fs = $tad->get_free_sizes();
Sometimes you need to lock device's screen and keypad to prevent users can use it. You can disable the device and when you want to get it back working, just enable it!
// Disabling device.
$tad->disable();
...
// Enabling device.
$tad->enable();
// Restart the device!!!
$tad->restart();
...
// Shut down the device!!!
$tad->poweroff();
Every command executed via TAD class will return an object that is an instance of TADResponse class. This object contains all information about the device's response received. This way you can get full flexibility to manipulate the responses that you got from the device: you can transform it in XML, JSON or even you can get an array. Also you can set some criterias to make a filtering process on the response.
TADResponse class offers 14 methods:
get_response, set_response, get_encoding, set_encoding, get_header, set_header, get_response_body, to_xml, to_json, to_array, count, is_empty_response, filter_xml and filter_by
When you call any TAD method, all responses (including those empty ones) are returned as an TADResponse object, so you can invoke any of methods mentioned above:
$r= $tad->get_date();
// Get response in XML format.
$xml_date = $r->get_response();
// Or you can specify the format.
$json_date = $r->get_response(['format'=>'json']);
// Or you want to get just response's body (without XML header).
$raw_response = $r->get_response_body(); // This always will be in XML format!!!
Sometimes you would like to build a TADResponse object by hand:
$response = '<Response><Name>Foo</Name><Address>Bar</Address></Response>';
$encoding = 'iso8859-1';
$tr = new TADResponse($response, $encoding);
...
// Maybe later you want to change the response you set before
$new_response = '<CrazyResponse><Joke>foo bar taz</Joke></CrazyResponse>';
$tr->set_response($new_response);
...
$r = $tad->get_date();
// Get current response's encoding.
$encoding = $r->get_encoding();
// Perhaps you want to change response's encoding.
$r->set_encoding('utf-8');
Instead of getting a full XML response, you can get just the header's response and you can change it even:
$r = $tad->get_date();
$header = $r->get_header();
// Method above returns '<?xml version="1.0" encoding="iso8859-1" standalone="no"?>'
$new_header = '<?xml version="1.1" encoding="utf-8" standalone="yes"?>';
// Lets set a new response's header.
$r->set_header($new_header);
As you seen above, you can get device's responses in different formats using get_response() method and specifying the format you want. However, you can use the following methods too:
// Get attendance logs for all users.
$att_logs = $tad->get_att_logs(); // $att_logs is an TADResponse object.
// Get response in XML format.
$xml_att_logs = $att_logs->to_xml();
// Get response in JSON format.
$json_att_logs = $att_logs->to_json();
// Get an array from response.
$array_att_logs = $att_logs->to_array().
// Lets get an XML response in one single step.
$xml = $tad->get_att_logs()->to_xml();
When you are interested just in how many items has the response, just count them:
$att_logs = $tad->get_att_logs();
// Get just the number of logs you retrived.
$logs_number = $att_logs->count();
Sometimes some queries to the device returns an empty answer. Because the original response from the device is in XML format, to know if you got any relevant data, you should have to parse the responses. That's not very handy:
$r = $tad->get_att_logs(['pin'=>123]); // This employee does not have any logs!!!
if ($r->is_empty_response()) {
    echo 'The employee does not have logs recorded';
}
...
As you saw above, all device's responses are handled by TADResponse class. You get the raw XML but you always get the whole set. What if you you'd like to do some kind of processing on reponses? Now, you can process the whole XML response by applying filters. This way, you can get just XML responses that really needs.
// Get attendance logs for all users;
$att_logs = $tad->get_att_logs();
// Now, you want filter the resulset to get att logs between '2014-01-10' and '2014-03-20'.
$filtered_att_logs = $att_logs->filter_by_date(
    ['start' => '2014-01-10','end' => '2014-03-20']
);
// Maybe you need to be more specific: get logs of users with privilege of Enroller
// that generated a 'check-out' event after '2014-08-01'.
$filtered_att_logs = $att_logs->filter_by_privilege_and_status_and_date(
    2, // 2 = Enroller role.
    1, // 1 = Check-out.
    ['start'] => '2014-08-01'
);
// You can do specific string searching too!!!
$users = $tad->get_all_user_info();
// Searches for all employees with the string 'Foo' as part of its name.
$foo_employees = $users->filter_by_name(['like'=>'Foo']);
Notes:
TADPHP is not perfect!!!. As mentioned at the beggining, it's been developed after hours, and hours, and hours of googling and it's been tested using just Fingertec Q2i Time & attendance device (that it's I have in my work), so it's possible that you can find errors when you use it with others devices or even you can find better ways to do the things. For that reason, there are some things to do:
Jorge Cobis - <http://twitter.com/cobisja>.
By the way, I'm from Bolivarian Republic of Venezuela :-D
Feel free to contribute!!!. Welcome aboard!!!
*0.4.2 (Saturday, 17th January 2015)
0.4.1 (Friday, 16th January 2015)
0.4.0 (Sunday, 11th January 2015)
0.3.2 (Saturday, 3rd January 2015)
0.3.1 (Monday, 29th December 2014)
0.3.0 (Saturday, 27th December 2014)
0.2.0 (Wednesday, 24th December 2014)
0.1.0 (Monday, 22nd December 2014)
Copyright (c) 2014 Jorge Cobis (<jcobis@gmail.com>)
MIT License
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|  Files (19) | 
| File | Role | Description | ||
|---|---|---|---|---|
|  lib (3 files, 2 directories) | ||||
|  test (2 directories) | ||||
|    composer.json | Data | composer rules | ||
|    LICENSE | Lic. | TAD License | ||
|    phpunit.xml | Data | phpunit conf | ||
|    README.md | Doc. | Readme | ||
|  Files (19) | / | lib | 
| File | Role | Description | ||
|---|---|---|---|---|
|  Exceptions (4 files) | ||||
|  Providers (2 files) | ||||
|  TAD.php | Class | Main class | ||
|  TADFactory.php | Class | Class to get a TAD instance | ||
|  TADResponse.php | Class | Class to handle device's responses | ||
|  Files (19) | / | lib | / | Exceptions | 
| File | Role | Description | 
|---|---|---|
|  ConnectionError.php | Class | Connection Error exception | 
|  FilterArgumentError.php | Class | Filter Argument exception | 
|  UnrecognizedArgument.php | Class | Unrecognized argument exception | 
|  UnrecognizedCommand.php | Class | Unrecognized command exception | 
|  Files (19) | / | lib | / | Providers | 
| File | Role | Description | 
|---|---|---|
|  TADSoap.php | Class | Class to handle device's comm via SOAP | 
|  TADZKLib.php | Class | Class to handle device's comm via UDP | 
|  Files (19) | / | test | / | lib | 
| File | Role | Description | ||
|---|---|---|---|---|
|  Providers (2 files) | ||||
|    RealTADTest.php | Test | part of test suite | ||
|    TADResponseTest.php | Test | part of test suite | ||
|    TADTest.php | Test | part of test suite | ||
|  Files (19) | / | test | / | lib | / | Providers | 
| File | Role | Description | 
|---|---|---|
|    TADSoapTest.php | Test | part of test suite | 
|    TADZKLibTest.php | Test | part of test suite | 
| The PHP Classes site has supported package installation using the Composer tool since 2013, as you may verify by reading this instructions page. | 
|  Install with Composer | 
| Version Control | Unique User Downloads | Download Rankings | |||||||||||||||
| 100% | 
 | 
 | 
| User Ratings | User Comments (2) | |||||||||||||||||||||||||||||||||||||
| 
 | 
 | |||||||||||||||||||||||||||||||||||||
| Applications that use this package | 
 If you know an application of this package, send a message to the author to add a link here.
 If you know an application of this package, send a message to the author to add a link here.