636 lines
15 KiB
PHP
636 lines
15 KiB
PHP
|
<?php
|
||
|
/**
|
||
|
* Classes and functions for export data from LDAP
|
||
|
*
|
||
|
* These classes provide differnet export formats.
|
||
|
*
|
||
|
* @author The phpLDAPadmin development team
|
||
|
* @package phpLDAPadmin
|
||
|
* @see export.php and export_form.php
|
||
|
*/
|
||
|
|
||
|
/**
|
||
|
* Exporter Class
|
||
|
*
|
||
|
* This class serves as a top level exporter class, which will return
|
||
|
* the correct Export class.
|
||
|
*
|
||
|
* @package phpLDAPadmin
|
||
|
* @subpackage Export
|
||
|
*/
|
||
|
class Exporter {
|
||
|
# Server ID that the export is linked to
|
||
|
private $server_id;
|
||
|
# Exporter Type
|
||
|
private $template_id;
|
||
|
private $template;
|
||
|
|
||
|
public function __construct($server_id,$template_id) {
|
||
|
$this->server_id = $server_id;
|
||
|
$this->template_id = $template_id;
|
||
|
|
||
|
$this->accept();
|
||
|
}
|
||
|
|
||
|
static function types() {
|
||
|
$type = array();
|
||
|
|
||
|
$details = ExportCSV::getType();
|
||
|
$type[$details['type']] = $details;
|
||
|
$details = ExportDSML::getType();
|
||
|
$type[$details['type']] = $details;
|
||
|
$details = ExportLDIF::getType();
|
||
|
$type[$details['type']] = $details;
|
||
|
$details = ExportVCARD::getType();
|
||
|
$type[$details['type']] = $details;
|
||
|
|
||
|
return $type;
|
||
|
}
|
||
|
|
||
|
private function accept() {
|
||
|
switch($this->template_id) {
|
||
|
case 'CSV':
|
||
|
$this->template = new ExportCSV();
|
||
|
break;
|
||
|
|
||
|
case 'DSML':
|
||
|
$this->template = new ExportDSML();
|
||
|
break;
|
||
|
|
||
|
case 'LDIF':
|
||
|
$this->template = new ExportLDIF();
|
||
|
break;
|
||
|
|
||
|
case 'VCARD':
|
||
|
$this->template = new ExportVCARD();
|
||
|
break;
|
||
|
|
||
|
default:
|
||
|
die();
|
||
|
}
|
||
|
|
||
|
$this->template->accept();
|
||
|
}
|
||
|
|
||
|
public function getTemplate() {
|
||
|
return $this->template;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Export Class
|
||
|
*
|
||
|
* This abstract classes provides all the common methods and variables for the
|
||
|
* custom export classes.
|
||
|
*
|
||
|
* @package phpLDAPadmin
|
||
|
* @subpackage Export
|
||
|
*/
|
||
|
abstract class Export {
|
||
|
# Line Break
|
||
|
protected $br;
|
||
|
# Compress the output
|
||
|
protected $compress;
|
||
|
# Export Results
|
||
|
protected $results;
|
||
|
protected $resultsdata;
|
||
|
protected $items = 0;
|
||
|
|
||
|
/**
|
||
|
* Return this LDAP Server object
|
||
|
*
|
||
|
* @return object DataStore Server
|
||
|
*/
|
||
|
protected function getServer() {
|
||
|
return $_SESSION[APPCONFIG]->getServer($this->getServerID());
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Return the LDAP server ID
|
||
|
*
|
||
|
* @return int Server ID
|
||
|
*/
|
||
|
protected function getServerID() {
|
||
|
return get_request('server_id','REQUEST');
|
||
|
}
|
||
|
|
||
|
public function accept() {
|
||
|
$server = $this->getServer();
|
||
|
|
||
|
# Get the data to be exported
|
||
|
$query = array();
|
||
|
$base = get_request('dn','REQUEST');
|
||
|
$query['baseok'] = true;
|
||
|
$query['filter'] = get_request('filter','REQUEST',false,'objectclass=*');
|
||
|
$query['scope'] = get_request('scope','REQUEST',false,'base');
|
||
|
$query['deref'] = $_SESSION[APPCONFIG]->getValue('deref','export');
|
||
|
$query['size_limit'] = 0;
|
||
|
$attrs = get_request('attributes','REQUEST');
|
||
|
|
||
|
$attrs = preg_replace('/\s+/','',$attrs);
|
||
|
if ($attrs)
|
||
|
$query['attrs'] = explode(',',$attrs);
|
||
|
else
|
||
|
$query['attrs'] = array('*');
|
||
|
|
||
|
if (get_request('sys_attr')) {
|
||
|
if (! in_array('*',$query['attrs']))
|
||
|
array_push($query['attrs'],'*');
|
||
|
array_push($query['attrs'],'+');
|
||
|
}
|
||
|
|
||
|
if (! $base)
|
||
|
$bases = $server->getBaseDN();
|
||
|
else
|
||
|
$bases = array($base);
|
||
|
|
||
|
foreach ($bases as $base) {
|
||
|
$query['base'] = $base;
|
||
|
|
||
|
$time_start = utime();
|
||
|
$this->results[$base] = $server->query($query,null);
|
||
|
$time_end = utime();
|
||
|
|
||
|
usort($this->results[$base],'pla_compare_dns');
|
||
|
$this->resultsdata[$base]['time'] = round($time_end-$time_start,2);
|
||
|
|
||
|
# If no result, there is a something wrong
|
||
|
if (! $this->results[$base] && $server->getErrorNum(null))
|
||
|
system_message(array(
|
||
|
'title'=>_('Encountered an error while performing search.'),
|
||
|
'body'=>ldap_error_msg($server->getErrorMessage(null),$server->getErrorNum(null)),
|
||
|
'type'=>'error'));
|
||
|
|
||
|
$this->items += count($this->results[$base]);
|
||
|
}
|
||
|
|
||
|
$this->resultsdata['scope'] = $query['scope'];
|
||
|
$this->resultsdata['filter'] = $query['filter'];
|
||
|
$this->resultsdata['attrs'] = $query['attrs'];
|
||
|
|
||
|
# Other settings
|
||
|
switch (get_request('format','POST',false,'unix')) {
|
||
|
case 'win':
|
||
|
$this->br = "\r\n";
|
||
|
break;
|
||
|
|
||
|
case 'mac':
|
||
|
$this->br = "\r";
|
||
|
break;
|
||
|
|
||
|
case 'unix':
|
||
|
default:
|
||
|
$this->br = "\n";
|
||
|
}
|
||
|
|
||
|
if (get_request('compress','REQUEST') == 'on')
|
||
|
$this->compress = true;
|
||
|
}
|
||
|
|
||
|
public function isCompressed() {
|
||
|
return $this->compress;
|
||
|
}
|
||
|
|
||
|
protected function getHeader() {
|
||
|
$server = $this->getServer();
|
||
|
$type = $this->getType();
|
||
|
|
||
|
$output = '';
|
||
|
|
||
|
$output .= sprintf('# %s: %s%s',_('Search scope'),$this->resultsdata['scope'],$this->br);
|
||
|
$output .= sprintf('# %s: %s%s',_('Search filter'),$this->resultsdata['filter'],$this->br);
|
||
|
$output .= sprintf('# %s: %s%s',_('Total entries'),$this->items,$this->br);
|
||
|
$output .= sprintf('#%s',$this->br);
|
||
|
$output .= sprintf('# Generated by %s (%s) on %s%s',app_name(),get_href('web'),date('F j, Y g:i a'),$this->br);
|
||
|
$output .= sprintf('# Version: %s%s',app_version(),$this->br);
|
||
|
|
||
|
$output .= $this->br;
|
||
|
|
||
|
return $output;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Helper method to check if the attribute value should be base 64 encoded.
|
||
|
*
|
||
|
* @param The string to check.
|
||
|
* @return boolean true if the string is safe ascii, false otherwise.
|
||
|
*/
|
||
|
protected function isSafeAscii($str) {
|
||
|
for ($i=0;$i<strlen($str);$i++)
|
||
|
if (ord($str{$i}) < 32 || ord($str{$i}) > 127)
|
||
|
return false;
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Export entries to CSV
|
||
|
*
|
||
|
* @package phpLDAPadmin
|
||
|
* @subpackage Export
|
||
|
*/
|
||
|
class ExportCSV extends Export {
|
||
|
private $separator = ',';
|
||
|
private $qualifier = '"';
|
||
|
private $multivalue_separator = ' | ';
|
||
|
private $escapeCode = '"';
|
||
|
|
||
|
static public function getType() {
|
||
|
return array('type'=>'CSV','description' => 'CSV (Spreadsheet)','extension'=>'csv');
|
||
|
}
|
||
|
|
||
|
function export() {
|
||
|
$server = $this->getServer();
|
||
|
|
||
|
/* Go thru and find all the attribute names first. This is needed, because, otherwise we have
|
||
|
* no idea as to which search attributes were actually populated with data */
|
||
|
$headers = array('dn');
|
||
|
$entries = array();
|
||
|
foreach ($this->results as $base => $results) {
|
||
|
foreach ($results as $dndetails) {
|
||
|
array_push($entries,$dndetails);
|
||
|
|
||
|
unset($dndetails['dn']);
|
||
|
foreach (array_keys($dndetails) as $key)
|
||
|
if (! in_array($key,$headers))
|
||
|
array_push($headers,$key);
|
||
|
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$output = '';
|
||
|
$num_headers = count($headers);
|
||
|
|
||
|
# Print out the headers
|
||
|
for ($i=0; $i<$num_headers; $i++) {
|
||
|
$output .= sprintf('%s%s%s',$this->qualifier,$headers[$i],$this->qualifier);
|
||
|
|
||
|
if ($i < $num_headers-1)
|
||
|
$output .= $this->separator;
|
||
|
}
|
||
|
|
||
|
# Drop out our DN header.
|
||
|
array_shift($headers);
|
||
|
$num_headers--;
|
||
|
|
||
|
$output .= $this->br;
|
||
|
|
||
|
# Loop on every entry
|
||
|
foreach ($entries as $index => $entry) {
|
||
|
$dn = $entry['dn'];
|
||
|
unset($entry['dn']);
|
||
|
$output .= sprintf('%s%s%s%s',$this->qualifier,$this->LdapEscape($dn),$this->qualifier,$this->separator);
|
||
|
|
||
|
# Print the attributes
|
||
|
for ($j=0; $j<$num_headers; $j++) {
|
||
|
$attr = $headers[$j];
|
||
|
$output .= $this->qualifier;
|
||
|
|
||
|
if (array_key_exists($attr,$entry)) {
|
||
|
$binary_attribute = $server->isAttrBinary($attr) ? 1 : 0;
|
||
|
|
||
|
if (! is_array($entry[$attr]))
|
||
|
$attr_values = array($entry[$attr]);
|
||
|
else
|
||
|
$attr_values = $entry[$attr];
|
||
|
|
||
|
$num_attr_values = count($attr_values);
|
||
|
|
||
|
for ($i=0; $i<$num_attr_values; $i++) {
|
||
|
if ($binary_attribute)
|
||
|
$output .= base64_encode($attr_values[$i]);
|
||
|
else
|
||
|
$output .= $this->LdapEscape($attr_values[$i]);
|
||
|
|
||
|
if ($i < $num_attr_values-1)
|
||
|
$output .= $this->multivalue_separator;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$output .= $this->qualifier;
|
||
|
|
||
|
if ($j < $num_headers-1)
|
||
|
$output .= $this->separator;
|
||
|
}
|
||
|
|
||
|
$output .= $this->br;
|
||
|
}
|
||
|
|
||
|
if ($this->compress)
|
||
|
return gzencode($output);
|
||
|
else
|
||
|
return $output;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Function to escape data, where the qualifier happens to also
|
||
|
* be in the data.
|
||
|
*/
|
||
|
private function LdapEscape ($var) {
|
||
|
return str_replace($this->qualifier,$this->escapeCode.$this->qualifier,$var);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Export entries to DSML v.1
|
||
|
*
|
||
|
* @package phpLDAPadmin
|
||
|
* @subpackage Export
|
||
|
*/
|
||
|
class ExportDSML extends Export {
|
||
|
static public function getType() {
|
||
|
return array('type'=>'DSML','description' => _('DSML V.1 Export'),'extension'=>'xml');
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Export entries to DSML format
|
||
|
*/
|
||
|
function export() {
|
||
|
$server = $this->getServer();
|
||
|
|
||
|
# Not very elegant, but do the job for the moment as we have just 4 level
|
||
|
$indent = array();
|
||
|
$indent['dir'] = ' ';
|
||
|
$indent['ent'] = ' ';
|
||
|
$indent['att'] = ' ';
|
||
|
$indent['val'] = ' ';
|
||
|
|
||
|
# Print declaration
|
||
|
$output = sprintf('<?xml version="1.0"?>%s',$this->br);
|
||
|
|
||
|
# Print root element
|
||
|
$output .= sprintf('<dsml>%s',$this->br);
|
||
|
|
||
|
# Print info related to this export
|
||
|
$output .= sprintf('<!--%s',$this->br);
|
||
|
$output .= $this->getHeader();
|
||
|
$output .= sprintf('-->%s',$this->br);
|
||
|
$output .= $this->br;
|
||
|
|
||
|
$output .= sprintf('%s<directory-entries>%s',$indent['dir'],$this->br);
|
||
|
|
||
|
# Sift through the entries.
|
||
|
$counter = 0;
|
||
|
foreach ($this->results as $base => $results) {
|
||
|
foreach ($results as $dndetails) {
|
||
|
$counter++;
|
||
|
|
||
|
$dn = $dndetails['dn'];
|
||
|
unset($dndetails['dn']);
|
||
|
ksort($dndetails);
|
||
|
|
||
|
# Display DN
|
||
|
$output .= sprintf('%s<entry dn="%s">%s',$indent['ent'],htmlspecialchars($dn),$this->br);
|
||
|
|
||
|
# Display the objectClass attributes first
|
||
|
if (isset($dndetails['objectClass'])) {
|
||
|
if (! is_array($dndetails['objectClass']))
|
||
|
$dndetails['objectClass'] = array($dndetails['objectClass']);
|
||
|
|
||
|
$output .= sprintf('%s<objectClass>%s',$indent['att'],$this->br);
|
||
|
|
||
|
foreach ($dndetails['objectClass'] as $ocValue)
|
||
|
$output .= sprintf('%s<oc-value>%s</oc-value>%s',$indent['val'],$ocValue,$this->br);
|
||
|
|
||
|
$output .= sprintf('%s</objectClass>%s',$indent['att'],$this->br);
|
||
|
unset($dndetails['objectClass']);
|
||
|
}
|
||
|
|
||
|
# Display the attributes
|
||
|
foreach ($dndetails as $key => $attr) {
|
||
|
if (! is_array($attr))
|
||
|
$attr = array($attr);
|
||
|
|
||
|
$output .= sprintf('%s<attr name="%s">%s',$indent['att'],$key,$this->br);
|
||
|
|
||
|
# If the attribute is binary, set the flag $binary_mode to true
|
||
|
$binary_mode = $server->isAttrBinary($key) ? 1 : 0;
|
||
|
|
||
|
foreach ($attr as $value)
|
||
|
$output .= sprintf('%s<value>%s</value>%s',
|
||
|
$indent['val'],($binary_mode ? base64_encode($value) : htmlspecialchars($value)),$this->br);
|
||
|
|
||
|
$output .= sprintf('%s</attr>%s',$indent['att'],$this->br);
|
||
|
}
|
||
|
|
||
|
$output .= sprintf('%s</entry>%s',$indent['ent'],$this->br);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$output .= sprintf('%s</directory-entries>%s',$indent['dir'],$this->br);
|
||
|
$output .= sprintf('</dsml>%s',$this->br);
|
||
|
|
||
|
if ($this->compress)
|
||
|
return gzencode($output);
|
||
|
else
|
||
|
return $output;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Export from LDAP using an LDIF format
|
||
|
*
|
||
|
* @package phpLDAPadmin
|
||
|
* @subpackage Export
|
||
|
*/
|
||
|
class ExportLDIF extends Export {
|
||
|
# The maximum length of the ldif line
|
||
|
private $line_length = 76;
|
||
|
|
||
|
static public function getType() {
|
||
|
return array('type'=>'LDIF','description' => _('LDIF Export'),'extension'=>'ldif');
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Export entries to LDIF format
|
||
|
*/
|
||
|
public function export() {
|
||
|
if (! $this->results) {
|
||
|
echo _('Nothing to export');
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
$server = $this->getServer();
|
||
|
|
||
|
$output = $this->getHeader();
|
||
|
|
||
|
# Add our version.
|
||
|
$output .= 'version: 1';
|
||
|
$output .= $this->br;
|
||
|
$output .= $this->br;
|
||
|
|
||
|
# Sift through the entries.
|
||
|
$counter = 0;
|
||
|
foreach ($this->results as $base => $results) {
|
||
|
foreach ($results as $dndetails) {
|
||
|
$counter++;
|
||
|
|
||
|
$dn = $dndetails['dn'];
|
||
|
unset($dndetails['dn']);
|
||
|
ksort($dndetails);
|
||
|
|
||
|
$title_string = sprintf('# %s %s: %s%s',_('Entry'),$counter,$dn,$this->br);
|
||
|
|
||
|
if (strlen($title_string) > $this->line_length-3)
|
||
|
$title_string = substr($title_string,0,$this->line_length-3).'...'.$this->br;
|
||
|
|
||
|
$output .= $title_string;
|
||
|
|
||
|
# Display dn
|
||
|
if ($this->isSafeAscii($dn))
|
||
|
$output .= $this->multiLineDisplay(sprintf('dn: %s',$dn));
|
||
|
else
|
||
|
$output .= $this->multiLineDisplay(sprintf('dn:: %s',base64_encode($dn)));
|
||
|
|
||
|
# display the attributes
|
||
|
foreach ($dndetails as $key => $attr) {
|
||
|
if (! is_array($attr))
|
||
|
$attr = array($attr);
|
||
|
|
||
|
foreach ($attr as $value)
|
||
|
if (! $this->isSafeAscii($value) || $server->isAttrBinary($key))
|
||
|
$output .= $this->multiLineDisplay(sprintf('%s:: %s',$key,base64_encode($value)));
|
||
|
else
|
||
|
$output .= $this->multiLineDisplay(sprintf('%s: %s',$key,$value));
|
||
|
}
|
||
|
|
||
|
$output .= $this->br;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ($this->compress)
|
||
|
return gzencode($output);
|
||
|
else
|
||
|
return $output;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Helper method to wrap ldif lines
|
||
|
*
|
||
|
* @param The line to be wrapped if needed.
|
||
|
*/
|
||
|
private function multiLineDisplay($str) {
|
||
|
$length_string = strlen($str);
|
||
|
$length_max = $this->line_length;
|
||
|
|
||
|
$output = '';
|
||
|
while ($length_string > $length_max) {
|
||
|
$output .= substr($str,0,$length_max).$this->br.' ';
|
||
|
$str = substr($str,$length_max,$length_string);
|
||
|
$length_string = strlen($str);
|
||
|
|
||
|
/* Need to do minus one to align on the right
|
||
|
* the first line with the possible following lines
|
||
|
* as these will have an extra space. */
|
||
|
$length_max = $this->line_length-1;
|
||
|
}
|
||
|
|
||
|
$output .= $str.$this->br;
|
||
|
|
||
|
return $output;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Export entries to VCARD v2.1
|
||
|
*
|
||
|
* @package phpLDAPadmin
|
||
|
* @subpackage Export
|
||
|
*/
|
||
|
class ExportVCARD extends Export {
|
||
|
static public function getType() {
|
||
|
return array('type'=>'VCARD','description' => _('VCARD 2.1 Export'),'extension'=>'vcf');
|
||
|
}
|
||
|
|
||
|
# Mappping one to one attribute
|
||
|
private $mapping = array(
|
||
|
'cn' => 'FN',
|
||
|
'title' => 'TITLE',
|
||
|
'homephone' => 'TEL;HOME',
|
||
|
'mobile' => 'TEL;CELL',
|
||
|
'mail' => 'EMAIL;Internet',
|
||
|
'labeleduri' =>'URL',
|
||
|
'o' => 'ORG',
|
||
|
'audio' => 'SOUND',
|
||
|
'facsmiletelephoneNumber' =>'TEL;WORK;HOME;VOICE;FAX',
|
||
|
'jpegphoto' => 'PHOTO;ENCODING=BASE64',
|
||
|
'businesscategory' => 'ROLE',
|
||
|
'description' => 'NOTE'
|
||
|
);
|
||
|
|
||
|
private $deliveryAddress = array(
|
||
|
'postofficebox',
|
||
|
'street',
|
||
|
'l',
|
||
|
'st',
|
||
|
'postalcode',
|
||
|
'c');
|
||
|
|
||
|
/**
|
||
|
* Export entries to VCARD format
|
||
|
*/
|
||
|
function export() {
|
||
|
$server = $this->getServer();
|
||
|
|
||
|
# Sift through the entries.
|
||
|
foreach ($this->results as $base => $results) {
|
||
|
foreach ($results as $dndetails) {
|
||
|
$dndetails = array_change_key_case($dndetails);
|
||
|
|
||
|
# Check the attributes needed for the delivery address field
|
||
|
$addr = 'ADR:';
|
||
|
foreach ($this->deliveryAddress as $attr) {
|
||
|
if (isset($dndetails[$attr])) {
|
||
|
$addr .= $dndetails[$attr];
|
||
|
unset($dndetails[$attr]);
|
||
|
}
|
||
|
$addr .= ';';
|
||
|
}
|
||
|
|
||
|
$output = sprintf('BEGIN:VCARD%s',$this->br);
|
||
|
|
||
|
# Loop for the attributes
|
||
|
foreach ($dndetails as $key => $attr) {
|
||
|
if (! is_array($attr))
|
||
|
$attr = array($attr);
|
||
|
|
||
|
# If an attribute of the ldap entry exist in the mapping array for vcard
|
||
|
if (isset($this->mapping[$key])) {
|
||
|
|
||
|
# Case of organisation. Need to append the possible ou attribute
|
||
|
if ($key == 'o') {
|
||
|
$output .= sprintf('%s:%s',$this->mapping[$key],$attr[0]);
|
||
|
|
||
|
if (isset($entry['ou']))
|
||
|
foreach ($entry['ou'] as $ou_value)
|
||
|
$output .= sprintf(';%s',$ou_value);
|
||
|
|
||
|
# The attribute is binary. (to do : need to fold the line)
|
||
|
} elseif (in_array($key,array('audio','jpegphoto'))) {
|
||
|
$output .= $this->mapping[$key].':'.$this->br;
|
||
|
$output .= ' '.base64_encode($attr[0]);
|
||
|
|
||
|
} else {
|
||
|
$output .= $this->mapping[$key].':'.$attr[0];
|
||
|
}
|
||
|
|
||
|
$output .= $this->br;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$output .= sprintf('UID:%s%s',isset($dndetails['entryUUID']) ? $dndetails['entryUUID'] : $dndetails['dn'],$this->br);
|
||
|
$output .= sprintf('VERSION:2.1%s',$this->br);
|
||
|
$output .= sprintf('%s%s',$addr,$this->br);
|
||
|
$output .= sprintf('END:VCARD%s',$this->br);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ($this->compress)
|
||
|
return gzencode($output);
|
||
|
else
|
||
|
return $output;
|
||
|
}
|
||
|
}
|
||
|
?>
|