fix for stable Univention 4.4-2 REST API

This commit is contained in:
Ralf Becker 2019-09-30 12:37:29 +02:00
parent 6f799d7cb7
commit 5dcf1e842f
4 changed files with 102 additions and 53 deletions

View File

@ -557,10 +557,19 @@ class Ldap
{ {
$to_write['gidnumber'] = abs($data['account_id']); $to_write['gidnumber'] = abs($data['account_id']);
$to_write['cn'] = $data['account_lid']; $to_write['cn'] = $data['account_lid'];
if (!empty($data['account_description']) || $old) // do not overwrite exitsting description, if non is given
if (isset($data['account_description']))
{ {
$to_write['description'] = !empty($data['account_description']) ? $data['account_description'] : array(); $to_write['description'] = !empty($data['account_description']) ? $data['account_description'] : array();
} }
// to kope with various dependencies / requirements of objectclasses, simply write everything again
foreach($old as $name => $value)
{
if (!isset($to_write[$name]) && !in_array($name, ['dn', 'objectclass']))
{
$to_write[$name] = $value;
}
}
return $to_write; return $to_write;
} }

View File

@ -34,6 +34,27 @@ use EGroupware\Api;
*/ */
class Univention extends Ldap class Univention extends Ldap
{ {
/**
* Name of mail attribute
*/
const MAIL_ATTR = 'mailprimaryaddress';
/**
* Constructor
*
* @param Api\Accounts $frontend reference to the frontend class, to be able to call it's methods if needed
*/
function __construct(Api\Accounts $frontend)
{
parent::__construct($frontend);
// remove not supported groupOfNames to skip parent::save() first tries with it, before trying without
if (($groupOfNameKey = array_search('groupofnames', $this->requiredObjectClasses['group'])))
{
unset($this->requiredObjectClasses['group'][$groupOfNameKey]);
}
}
/** /**
* Saves / adds the data of one account * Saves / adds the data of one account
* *
@ -82,9 +103,9 @@ class Univention extends Ldap
$data['account_dn'] = $udm->createUser($data); $data['account_dn'] = $udm->createUser($data);
$data['account_id'] = $this->name2id($data['account_lid'], 'account_lid', 'u'); $data['account_id'] = $this->name2id($data['account_lid'], 'account_lid', 'u');
} }
// create new groups with given account_id via directory-manager too, to be able to set the RID // create new groups incl. Samba objectclass and SID
elseif($data['account_type'] === 'g' && !empty($data['account_id']) && elseif($data['account_type'] === 'g' && (empty($data['account_id']) ||
$data['account_id'] >= Ads::MIN_ACCOUNT_ID && !$this->id2name($data['account_id'])) $data['account_id'] >= Ads::MIN_ACCOUNT_ID && !$this->id2name($data['account_id'])))
{ {
$data['account_dn'] = $udm->createGroup($data); $data['account_dn'] = $udm->createGroup($data);
$data['account_id'] = $this->name2id($data['account_lid'], 'account_lid', 'g'); $data['account_id'] = $this->name2id($data['account_lid'], 'account_lid', 'g');

View File

@ -52,7 +52,7 @@ class Udm
/** /**
* Log webservice-calls to error_log * Log webservice-calls to error_log
*/ */
const DEBUG = true; const DEBUG = false;
/** /**
* Constructor * Constructor
@ -96,13 +96,16 @@ class Udm
{ {
$curl = curl_init(); $curl = curl_init();
// fix error: Request argument "policies" is not a "dict" (PHP encodes empty arrays as array, not object) // fix error like: Request argument "policies" is not a "dict" (PHP encodes empty arrays as array, not object)
if (array_key_exists('policies', $_payload) && empty($_payload['policies'])) if (array_key_exists('policies', $_payload) && empty($_payload['policies']))
{ {
$_payload['policies'] = new \stdClass(); // force "policies": {} $_payload['policies'] = new \stdClass(); // force "policies": {}
} }
if (is_array($_payload['properties']) && array_key_exists('umcProperty', $_payload['properties']) && empty($_payload['properties']['umcProperty']))
{
$_payload['properties']['umcProperty'] = new \stdClass(); // force "umcProperty": {}
}
$headers = [];
$curlOpts = [ $curlOpts = [
CURLOPT_URL => 'https://'.$this->host.($_path[0] !== '/' ? self::PREFIX : '').$_path, CURLOPT_URL => 'https://'.$this->host.($_path[0] !== '/' ? self::PREFIX : '').$_path,
CURLOPT_USERPWD => $this->user.':'.$this->config['ldap_root_pw'], CURLOPT_USERPWD => $this->user.':'.$this->config['ldap_root_pw'],
@ -115,29 +118,7 @@ class Udm
//CURLOPT_FOLLOWLOCATION => 1, //CURLOPT_FOLLOWLOCATION => 1,
CURLOPT_TIMEOUT => 30, // setting a timeout of 30 seconds, as recommended by Univention CURLOPT_TIMEOUT => 30, // setting a timeout of 30 seconds, as recommended by Univention
CURLOPT_VERBOSE => 1, CURLOPT_VERBOSE => 1,
CURLOPT_HEADERFUNCTION => CURLOPT_HEADER => 1,
function($curl, $header) use (&$headers)
{
$len = strlen($header);
$header = explode(':', $header, 2);
if (count($header) < 2)
{
$headers[] = $header[0]; // http status
return $len;
}
$name = strtolower(trim($header[0]));
if (!array_key_exists($name, $headers))
{
$headers[$name] = trim($header[1]);
}
else
{
$headers[$name] = [$headers[$name]];
$headers[$name][] = trim($header[1]);
}
unset($curl); // not used, but required by function signature
return $len;
},
]; ];
if (isset($if_match)) if (isset($if_match))
{ {
@ -162,27 +143,32 @@ class Udm
curl_setopt_array($curl, $curlOpts); curl_setopt_array($curl, $curlOpts);
$response = curl_exec($curl); $response = curl_exec($curl);
$header_size = curl_getinfo($curl, CURLINFO_HEADER_SIZE);
$headers = self::getHeaders(substr($response, 0, $header_size));
$body = substr($response, $header_size);
$path = urldecode($_path); // for nicer error-messages $path = urldecode($_path); // for nicer error-messages
if (!$response || !($json = json_decode($response, true)) && json_last_error()) if ($response === false || $body !== '' && !($json = json_decode($body, true)) && json_last_error())
{ {
$info = curl_getinfo($curl); $info = curl_getinfo($curl);
curl_close($curl); curl_close($curl);
if ($retry > 0) if ($retry > 0)
{ {
error_log(__METHOD__."($path, $_method, ...) failed, retrying in 100ms, returned $response, headers=".json_encode($headers, JSON_UNESCAPED_SLASHES|JSON_UNESCAPED_UNICODE).", curl_getinfo()=".json_encode($info, JSON_UNESCAPED_SLASHES|JSON_UNESCAPED_UNICODE)); error_log(__METHOD__."($path, $_method, ...) failed, retrying in 100ms, returned $body, headers=".json_encode($headers, JSON_UNESCAPED_SLASHES|JSON_UNESCAPED_UNICODE).", curl_getinfo()=".json_encode($info, JSON_UNESCAPED_SLASHES|JSON_UNESCAPED_UNICODE));
usleep(100000); usleep(100000);
return $this->call($_path, $_method, $_payload, $headers, $if_match, $return_dn, --$retry); return $this->call($_path, $_method, $_payload, $headers, $if_match, $return_dn, --$retry);
} }
error_log(__METHOD__."($path, $_method, ...) returned $response, headers=".json_encode($headers, JSON_UNESCAPED_SLASHES|JSON_UNESCAPED_UNICODE).", curl_getinfo()=".json_encode($info, JSON_UNESCAPED_SLASHES|JSON_UNESCAPED_UNICODE)); error_log(__METHOD__."($path, $_method, ...) returned $body, headers=".json_encode($headers, JSON_UNESCAPED_SLASHES|JSON_UNESCAPED_UNICODE).", curl_getinfo()=".json_encode($info, JSON_UNESCAPED_SLASHES|JSON_UNESCAPED_UNICODE));
error_log(__METHOD__."($path, $_method, ".json_encode($_payload, JSON_UNESCAPED_SLASHES|JSON_UNESCAPED_UNICODE).")"); error_log(__METHOD__."($path, $_method, ".json_encode($_payload, JSON_UNESCAPED_SLASHES|JSON_UNESCAPED_UNICODE).")");
throw new UdmCantConnect("Error contacting Univention UDM REST Api ($_path)".($response ? ': '.json_last_error() : '')); throw new UdmCantConnect("Error contacting Univention UDM REST Api ($path)".($response !== false ? ': '.json_last_error() : ''));
} }
curl_close($curl); curl_close($curl);
if (!empty($json['error'])) // error in json or non 20x http status
if (!empty($json['error']) || !preg_match('|^HTTP/[0-9.]+ 20|', $headers[0]))
{ {
error_log(__METHOD__."($path, $_method, ...) returned $response, headers=".json_encode($headers, JSON_UNESCAPED_SLASHES|JSON_UNESCAPED_UNICODE)); error_log(__METHOD__."($path, $_method, ...) returned $response, headers=".json_encode($headers, JSON_UNESCAPED_SLASHES|JSON_UNESCAPED_UNICODE));
error_log(__METHOD__."($path, $_method, ".json_encode($_payload, JSON_UNESCAPED_SLASHES|JSON_UNESCAPED_UNICODE).")"); error_log(__METHOD__."($path, $_method, ".json_encode($_payload, JSON_UNESCAPED_SLASHES|JSON_UNESCAPED_UNICODE).")");
throw new UdmError("UDM REST Api (".urldecode($_path)."): ".(empty($json['error']['message']) ? $response : $json['error']['message']), $json['error']['code']); throw new UdmError("UDM REST Api ($path): ".(empty($json['error']['message']) ? $headers[0] : $json['error']['message']), $json['error']['code']);
} }
if (self::DEBUG) if (self::DEBUG)
{ {
@ -195,13 +181,54 @@ class Udm
$matches = null; $matches = null;
if (!isset($headers['location']) || !preg_match('|/([^/]+)$|', $headers['location'], $matches)) if (!isset($headers['location']) || !preg_match('|/([^/]+)$|', $headers['location'], $matches))
{ {
throw new UdmMissingLocation("UDM REST Api ($_path) did not return Location header!"); throw new UdmMissingLocation("UDM REST Api ($path) did not return Location header!");
} }
return urldecode($matches[1]); return urldecode($matches[1]);
} }
return $json; return $json;
} }
/**
* Convert header string in array of headers
*
* A "HTTP/1.1 100 Continue" is NOT returned!
*
* @param string $head
* @return array with name => value pairs, 0: http-status, value can be an array for multiple headers with same name
*/
protected static function getHeaders($head)
{
$headers = [];
foreach(explode("\r\n", $head) as $header)
{
if (empty($header)) continue;
$parts = explode(':', $header, 2);
if (count($parts) < 2)
{
$headers[0] = $header; // http-status
}
else
{
$name = strtolower($parts[0]);
if (!isset($headers[$name]))
{
$headers[$name] = trim($parts[1]);
}
else
{
if (!is_array($headers[$name]))
{
$headers[$name] = [$headers[$name]];
}
$headers[$name][] = trim($parts[1]);
}
}
}
if (self::DEBUG) error_log(__METHOD__."(\$head) returning ".json_encode($headers));
return $headers;
}
/** /**
* Create a user * Create a user
* *
@ -212,7 +239,7 @@ class Udm
public function createUser(array $data) public function createUser(array $data)
{ {
// set default values // set default values
$payload = $this->user2udm($data, $this->call('users/user/add')['entry']); $payload = $this->user2udm($data, $this->call('users/user/add'));
$payload['superordinate'] = null; $payload['superordinate'] = null;
$payload['position'] = $this->config['ldap_context']; $payload['position'] = $this->config['ldap_context'];
@ -236,16 +263,7 @@ class Udm
$payload = $this->user2udm($data, $this->call('users/user/'.urlencode($dn), 'GET', [], $get_headers)); $payload = $this->user2udm($data, $this->call('users/user/'.urlencode($dn), 'GET', [], $get_headers));
$headers = []; $headers = [];
$ret = $this->call('users/user/'.urlencode($dn), 'PUT', $payload, $headers, $get_headers['etag'], true); return $this->call('users/user/'.urlencode($dn), 'PUT', $payload, $headers, $get_headers['etag'], true);
// you can not set the password and force a password change for next login in the same call
// the forced password change will be lost --> call again without password to force the change on next login
if (!empty($data['account_passwd']) && !empty($data['mustchangepassword']))
{
unset($data['account_passwd']);
$ret = $this->updateUser($ret, $data);
}
return $ret;
} }
/** /**
@ -315,7 +333,7 @@ class Udm
public function createGroup(array $data) public function createGroup(array $data)
{ {
// set default values // set default values
$payload = $this->group2udm($data, $this->call('groups/group/add')['entry']); $payload = $this->group2udm($data, $this->call('groups/group/add'));
$payload['superordinate'] = null; $payload['superordinate'] = null;
$payload['position'] = empty($this->config['ldap_group_context']) ? $this->config['ldap_context'] : $this->config['ldap_group_context']; $payload['position'] = empty($this->config['ldap_group_context']) ? $this->config['ldap_context'] : $this->config['ldap_group_context'];
@ -336,7 +354,7 @@ class Udm
{ {
// set existing values // set existing values
$get_headers = []; $get_headers = [];
$payload = $this->user2udm($data, $this->call('groups/group/'.urlencode($dn), 'GET', [], $get_headers)); $payload = $this->group2udm($data, $this->call('groups/group/'.urlencode($dn), 'GET', [], $get_headers));
$headers = []; $headers = [];
return $this->call('groups/group/'.urlencode($dn), 'PUT', $payload, $headers, $get_headers['etag'], true); return $this->call('groups/group/'.urlencode($dn), 'PUT', $payload, $headers, $get_headers['etag'], true);
@ -353,7 +371,7 @@ class Udm
{ {
foreach([ foreach([
'account_lid' => 'name', 'account_lid' => 'name',
'account_id' => ['gidNumber', 'sambaRID'], 'account_id' => 'gidNumber',
] as $egw => $names) ] as $egw => $names)
{ {
if (!empty($data[$egw])) if (!empty($data[$egw]))
@ -364,7 +382,8 @@ class Udm
{ {
throw new \Exception ("No '$name' in properties: ".json_encode($payload['properties'])); throw new \Exception ("No '$name' in properties: ".json_encode($payload['properties']));
} }
$payload['properties'][$name] = $data[$egw]; // our account_id is negative for groups!
$payload['properties'][$name] = $egw === 'account_id' ? abs($data[$egw]) : $data[$egw];
} }
} }
} }

View File

@ -199,7 +199,7 @@ class Ldap implements Backend
isset($allValues[0]['sambapwdlastset']) && (string)$allValues[0]['sambapwdlastset'][0] === '0' || isset($allValues[0]['sambapwdlastset']) && (string)$allValues[0]['sambapwdlastset'][0] === '0' ||
isset($allValues[0]['krb5passwordend']) && Api\DateTime::user2server($allValues[0]['krb5passwordend'][0]) < time()) isset($allValues[0]['krb5passwordend']) && Api\DateTime::user2server($allValues[0]['krb5passwordend'][0]) < time())
{ {
error_log(__METHOD__."('$_username') shadowlastchange={$allValues[0]['shadowlastchange']}, sambapwdlastset={$allValues[0]['sambapwdlastset'][0]}, krb5passwordend={$allValues[0]['krb5passwordend'][0]} --> return 0"); if ($this->debug) error_log(__METHOD__."('$_username') shadowlastchange={$allValues[0]['shadowlastchange'][0]}, sambapwdlastset={$allValues[0]['sambapwdlastset'][0]}, krb5passwordend={$allValues[0]['krb5passwordend'][0]} --> return 0");
return 0; return 0;
} }
if (!isset($allValues[0]['shadowlastchange'])) if (!isset($allValues[0]['shadowlastchange']))