Various updates.
This commit is contained in:
parent
029a9293f0
commit
27178cfd09
|
@ -188,12 +188,12 @@
|
|||
"source": {
|
||||
"type": "git",
|
||||
"url": "https://github.com/vatesfr/rekodi.git",
|
||||
"reference": "7a8d126055acfe4e3dbbdf01d50e3143952b8b77"
|
||||
"reference": "b95faecb9b5b0f9e541a0cf8aac680cad834a4be"
|
||||
},
|
||||
"dist": {
|
||||
"type": "zip",
|
||||
"url": "https://api.github.com/repos/vatesfr/rekodi/zipball/7a8d126055acfe4e3dbbdf01d50e3143952b8b77",
|
||||
"reference": "7a8d126055acfe4e3dbbdf01d50e3143952b8b77",
|
||||
"url": "https://api.github.com/repos/vatesfr/rekodi/zipball/b95faecb9b5b0f9e541a0cf8aac680cad834a4be",
|
||||
"reference": "b95faecb9b5b0f9e541a0cf8aac680cad834a4be",
|
||||
"shasum": ""
|
||||
},
|
||||
"require": {
|
||||
|
@ -222,7 +222,7 @@
|
|||
"keywords": [
|
||||
"event"
|
||||
],
|
||||
"time": "2013-03-03 13:25:27"
|
||||
"time": "2013-03-22 12:24:15"
|
||||
}
|
||||
],
|
||||
"packages-dev": [
|
||||
|
|
|
@ -25,7 +25,8 @@
|
|||
return array(
|
||||
|
||||
'database' => array(
|
||||
'json' => '#{root_dir}/database.json',
|
||||
'type' => 'json',
|
||||
'file' => '#{root_dir}/database.json',
|
||||
),
|
||||
|
||||
'listen' => array(
|
||||
|
|
|
@ -22,31 +22,13 @@
|
|||
* @package Xen Orchestra Server
|
||||
*/
|
||||
|
||||
use Bean\User;
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
final class Application extends Base
|
||||
{
|
||||
/**
|
||||
*
|
||||
*/
|
||||
const NONE = 0;
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
const READ = 1;
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
const WRITE = 2;
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
const ADMIN = 3;
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
|
@ -75,30 +57,30 @@ final class Application extends Base
|
|||
return array(0, 'already authenticated');
|
||||
}
|
||||
|
||||
$users = $this->_di->get('users');
|
||||
|
||||
// Checks the user exists.
|
||||
if (!isset($this->_usersByName[$name]))
|
||||
$user = $users->getBy('name', $name, false);
|
||||
if (!$user)
|
||||
{
|
||||
return array(1, 'invalid credential');
|
||||
}
|
||||
|
||||
$uid = $this->_usersByName[$name];
|
||||
$hash = &$this->_users[$uid]['password'];
|
||||
|
||||
// Checks the password matches.
|
||||
if (!password_verify($password, $hash))
|
||||
if (!password_verify($password, $user->password))
|
||||
{
|
||||
return array(1, 'invalid credential');
|
||||
}
|
||||
|
||||
// Checks whether the hash needs to be updated.
|
||||
if (password_needs_rehash($hash, PASSWORD_DEFAULT))
|
||||
if (password_needs_rehash($user->password, PASSWORD_DEFAULT))
|
||||
{
|
||||
$hash = password_hash($password, PASSWORD_DEFAULT);
|
||||
$this->_saveDatabase();
|
||||
$user->password = password_hash($password, PASSWORD_DEFAULT);
|
||||
$users->save($user);
|
||||
}
|
||||
|
||||
// Marks the client as authenticated.
|
||||
$c->uid = $uid;
|
||||
$c->uid = $user->id;
|
||||
|
||||
// Returns success.
|
||||
$c->respond($id, true);
|
||||
|
@ -122,23 +104,24 @@ final class Application extends Base
|
|||
return array(0, 'already authenticated');
|
||||
}
|
||||
|
||||
$tokens = $this->_di->get('tokens');
|
||||
|
||||
// Checks the token exists.
|
||||
if (!isset($this->_tokens[$token]))
|
||||
$token = $tokens->get($token, false);
|
||||
if (!$token)
|
||||
{
|
||||
return array(1, 'invalid token');
|
||||
}
|
||||
|
||||
$record = $this->_tokens[$token];
|
||||
|
||||
// Checks the token is valid.
|
||||
if ($record['expiration'] < time())
|
||||
if ($token->expiration < time())
|
||||
{
|
||||
unset($this->_tokens[$token]);
|
||||
$tokens->delete($token->id);
|
||||
return array(1, 'invalid token');
|
||||
}
|
||||
|
||||
// Marks the client as authenticated.
|
||||
$c->uid = $record['uid'];
|
||||
$c->uid = $token->user_id;
|
||||
|
||||
// Returns success.
|
||||
$c->respond($id, true);
|
||||
|
@ -154,10 +137,12 @@ final class Application extends Base
|
|||
return array(0, 'not authenticated');
|
||||
}
|
||||
|
||||
$user = $this->_di->get('users')->get($c->uid);
|
||||
|
||||
$c->respond($id, array(
|
||||
'id' => (string) $c->uid,
|
||||
'name' => $this->_users[$c->uid]['name'],
|
||||
'permission' => $this->_users[$c->uid]['permission'],
|
||||
'id' => $user->id,
|
||||
'name' => $user->name,
|
||||
'permission' => $user->permission,
|
||||
));
|
||||
}
|
||||
|
||||
|
@ -172,6 +157,8 @@ final class Application extends Base
|
|||
return array(0, 'not authenticated');
|
||||
}
|
||||
|
||||
$tokens = $this->_di->get('tokens');
|
||||
|
||||
// Generates the token and makes sure it is unique.
|
||||
do
|
||||
{
|
||||
|
@ -188,14 +175,14 @@ final class Application extends Base
|
|||
{
|
||||
$token = uniqid('', true);
|
||||
}
|
||||
} while (isset($this->_tokens[$token]));
|
||||
} while ($tokens->get($token, false));
|
||||
|
||||
// Registers it.
|
||||
$this->_tokens[$token] = array(
|
||||
$tokens->create(array(
|
||||
'id' => $token,
|
||||
'expiration' => time() + 604800, // One week
|
||||
'uid' => $c->uid,
|
||||
);
|
||||
$this->_saveDatabase();
|
||||
'user_id' => $c->uid,
|
||||
));
|
||||
|
||||
// Returns it.
|
||||
$c->respond($id, $token);
|
||||
|
@ -206,15 +193,23 @@ final class Application extends Base
|
|||
*/
|
||||
function api_session_destroyToken($id, array $params, Client $c)
|
||||
{
|
||||
// Checks parameters.
|
||||
if (!isset($params[0]))
|
||||
{
|
||||
return -32602; // Invalid params.
|
||||
}
|
||||
$token = $params[0];
|
||||
|
||||
$tokens = $this->_di->get('tokens');
|
||||
|
||||
// Checks the token exists.
|
||||
if (!isset($this->_tokens[$token]))
|
||||
if (!$tokens->get($token, false))
|
||||
{
|
||||
return array(0, 'invalid token');
|
||||
}
|
||||
|
||||
// Deletes it.
|
||||
unset($this->_tokens[$token]);
|
||||
$this->_saveDatabase();
|
||||
$tokens->delete($token);
|
||||
|
||||
// Returns success.
|
||||
$c->respond($id, true);
|
||||
|
@ -234,21 +229,19 @@ final class Application extends Base
|
|||
|
||||
// Checks credentials.
|
||||
if (!$c->isAuthenticated()
|
||||
|| !$this->_checkPermission($c->uid, self::ADMIN))
|
||||
|| !$this->_checkPermission($c->uid, User::ADMIN))
|
||||
{
|
||||
return array(0, 'not authorized');
|
||||
}
|
||||
|
||||
// Checks the provided user name.
|
||||
if (!is_string($name)
|
||||
|| !preg_match('/^[a-z0-9]+(?:[-_.][a-z0-9]+)*$/', $name))
|
||||
if (!User::check('name', $name))
|
||||
{
|
||||
return array(1, 'invalid user name');
|
||||
}
|
||||
|
||||
// Checks the provided password.
|
||||
if (!is_string($password)
|
||||
|| !preg_match('/^.{8,}$/', $password))
|
||||
if (!User::check('password', $password))
|
||||
{
|
||||
return array(2, 'invalid password');
|
||||
}
|
||||
|
@ -256,36 +249,34 @@ final class Application extends Base
|
|||
// Checks provided permission.
|
||||
if (isset($params[2]))
|
||||
{
|
||||
$permission = self::_permissionFromString($params[2]);
|
||||
if ($permission === false)
|
||||
$permission = $params[2];
|
||||
if (!User::check('permission', $permission))
|
||||
{
|
||||
return array(3, 'invalid permission');
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
$permission = self::NONE;
|
||||
$permission = User::NONE;
|
||||
}
|
||||
|
||||
$users = $this->_di->get('users');
|
||||
|
||||
// Checks if the user name is already used.
|
||||
if (isset($this->_usersByName[$name]))
|
||||
if ($users->getBy('name', $name, false))
|
||||
{
|
||||
return array(4, 'user name already taken');
|
||||
}
|
||||
|
||||
// Creates the user.
|
||||
$this->_users[] = array(
|
||||
$user = $users->create(array(
|
||||
'name' => $name,
|
||||
'password' => password_hash($password, PASSWORD_DEFAULT),
|
||||
'permission' => $permission,
|
||||
);
|
||||
end($this->_users);
|
||||
$uid = (string) key($this->_users);
|
||||
$this->_usersByName[$name] = $uid;
|
||||
$this->_saveDatabase();
|
||||
));
|
||||
|
||||
// Returns the identifier.
|
||||
$c->respond($id, $uid);
|
||||
$c->respond($id, $user->id);
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -302,22 +293,22 @@ final class Application extends Base
|
|||
|
||||
// Checks credentials.
|
||||
if (!$c->isAuthenticated()
|
||||
|| !$this->_checkPermission($c->uid, self::ADMIN))
|
||||
|| !$this->_checkPermission($c->uid, User::ADMIN))
|
||||
{
|
||||
return array(0, 'not authorized');
|
||||
}
|
||||
|
||||
$users = $this->_di->get('users');
|
||||
|
||||
// Checks user exists and is not the current user.
|
||||
if (!isset($this->_users[$uid])
|
||||
|| ($uid === $c->uid))
|
||||
if (($uid === $c->uid)
|
||||
|| !$users->get($uid, false))
|
||||
{
|
||||
return array(1, 'invalid user');
|
||||
}
|
||||
|
||||
// Deletes the user.
|
||||
$name = $this->_users[$uid]['name'];
|
||||
unset($this->_users[$uid], $this->_usersByName[$name]);
|
||||
$this->_saveDatabase();
|
||||
$users->delete($uid);
|
||||
|
||||
// Returns success.
|
||||
$c->respond($id, true);
|
||||
|
@ -341,23 +332,24 @@ final class Application extends Base
|
|||
return array(0, 'not authenticated');
|
||||
}
|
||||
|
||||
$hash = &$this->_users[$c->uid]['password'];
|
||||
$users = $this->_di->get('users');
|
||||
$user = $users->get($c->uid);
|
||||
|
||||
// Checks the old password matches.
|
||||
if (!password_verify($old, $hash))
|
||||
if (!password_verify($old, $user->password))
|
||||
{
|
||||
return array(1, 'invalid credential');
|
||||
}
|
||||
|
||||
// Checks the new password is valid.
|
||||
if (!is_string($new)
|
||||
|| !preg_match('/^.{8,}$/', $new))
|
||||
if (($new === $old)
|
||||
|| !User::check('password', $new))
|
||||
{
|
||||
return array(2, 'invalid password');
|
||||
}
|
||||
|
||||
$hash = password_hash($new, PASSWORD_DEFAULT);
|
||||
$this->_saveDatabase();
|
||||
$user->password = password_hash($new, PASSWORD_DEFAULT);
|
||||
$users->save($user);
|
||||
|
||||
// Returns success.
|
||||
$c->respond($id, true);
|
||||
|
@ -370,23 +362,56 @@ final class Application extends Base
|
|||
{
|
||||
// Checks credentials.
|
||||
if (!$c->isAuthenticated()
|
||||
|| !$this->_checkPermission($c->uid, self::ADMIN))
|
||||
|| !$this->_checkPermission($c->uid, User::ADMIN))
|
||||
{
|
||||
return array(0, 'not authorized');
|
||||
}
|
||||
|
||||
$users = array();
|
||||
foreach ($this->_users as $uid => $user)
|
||||
{
|
||||
$users[] = array(
|
||||
'id' => $uid,
|
||||
'name' => $user['name'],
|
||||
'permission' => self::_permissionToString($user['permission']),
|
||||
);
|
||||
}
|
||||
$users = $this->_di->get('users')->getArray(
|
||||
null,
|
||||
array('id', 'name', 'permission')
|
||||
);
|
||||
|
||||
$c->respond($id, $users);
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
function api_user_set($id, array $params, Client $c)
|
||||
{
|
||||
// Checks parameter.
|
||||
if (!isset($params[0], $params[1]))
|
||||
{
|
||||
return -32602; // Invalid params.
|
||||
}
|
||||
list($id, $properties) = $params;
|
||||
|
||||
if (!$c->isAuthenticated()
|
||||
|| !$this->_checkPermission($c->uid, User::ADMIN))
|
||||
{
|
||||
return array(0, 'not authorized');
|
||||
}
|
||||
|
||||
$users = $this->_di->get('users');
|
||||
$user = $users->get($id);
|
||||
|
||||
foreach ($properties as $field => $value)
|
||||
{
|
||||
switch ($field)
|
||||
{
|
||||
case 'name':
|
||||
case 'password':
|
||||
case 'permission':
|
||||
default:
|
||||
return array(2, 'invalid property');
|
||||
}
|
||||
}
|
||||
$users->save($user);
|
||||
|
||||
$c->respond($id, true);
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
|
@ -394,7 +419,7 @@ final class Application extends Base
|
|||
{
|
||||
// @todo Handles parameter.
|
||||
|
||||
$c->respond($id, $this->_xenVms);
|
||||
$c->respond($id, $this->_di->get('vms')->getArray());
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -515,29 +540,27 @@ final class Application extends Base
|
|||
*/
|
||||
function updateXenVms(array $vms)
|
||||
{
|
||||
foreach ($vms as $ref => $vm)
|
||||
$manager = $this->_di->get('vms');
|
||||
|
||||
foreach ($vms as $id => $properties)
|
||||
{
|
||||
if ($vm['is_a_template'])
|
||||
$properties['id'] = $id;
|
||||
|
||||
$vm = $manager->get($id, false);
|
||||
if (!$vm)
|
||||
{
|
||||
$_ = 'template';
|
||||
}
|
||||
elseif ($vm['is_a_snapshot'])
|
||||
{
|
||||
$_ = 'snapshot';
|
||||
}
|
||||
elseif ($vm['is_control_domain'])
|
||||
{
|
||||
$_ = 'control_domain';
|
||||
$manager->create($properties);
|
||||
echo "new VM: $id\n";
|
||||
}
|
||||
else
|
||||
{
|
||||
$_ = 'normal';
|
||||
$vm->set($properties, true);
|
||||
$keys = array_keys($vm->getDirty());
|
||||
sort($keys);
|
||||
$dirty = implode(', ', $keys);
|
||||
$manager->save($vm);
|
||||
echo "updated VM: $id ($dirty)\n";
|
||||
}
|
||||
|
||||
$this->_update(
|
||||
$this->_xenVms[$_][$ref],
|
||||
$vm
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -546,10 +569,6 @@ final class Application extends Base
|
|||
*/
|
||||
function run()
|
||||
{
|
||||
$this->_loadDatabase();
|
||||
|
||||
//--------------------------------------
|
||||
|
||||
$config = $this->_di->get('config');
|
||||
$loop = $this->_di->get('loop');
|
||||
|
||||
|
@ -628,37 +647,6 @@ final class Application extends Base
|
|||
*/
|
||||
private $_di;
|
||||
|
||||
/**
|
||||
* Mapping from user identifier to record.
|
||||
*
|
||||
* Each record contains:
|
||||
* - “name” (string): the user name used for sign in;
|
||||
* - “password” (string): the user password hashed for sign in.
|
||||
*
|
||||
* @var array
|
||||
*/
|
||||
private $_users = array();
|
||||
|
||||
/**
|
||||
* Mapping from user name to identifier.
|
||||
*
|
||||
* @var array
|
||||
*/
|
||||
private $_usersByName = array();
|
||||
|
||||
/**
|
||||
* Tokens that may be used to authenticate clients.
|
||||
*
|
||||
* Each token record is an array containing:
|
||||
* - “expiration” (integer): timestamp of when this token will be
|
||||
* considered invalid;
|
||||
* - “uid” (string): the identifier of the user authenticated with
|
||||
* this token.
|
||||
*
|
||||
* @var array
|
||||
*/
|
||||
private $_tokens = array();
|
||||
|
||||
/**
|
||||
* @var array
|
||||
*/
|
||||
|
@ -669,11 +657,6 @@ final class Application extends Base
|
|||
*/
|
||||
private $_xenHosts = array();
|
||||
|
||||
/**
|
||||
* @var array
|
||||
*/
|
||||
private $_xenVms = array();
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
|
@ -686,40 +669,6 @@ final class Application extends Base
|
|||
return gettype($val);
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
private static function _permissionFromString($string)
|
||||
{
|
||||
$permissions = array(
|
||||
'none' => self::NONE,
|
||||
'read' => self::READ,
|
||||
'write' => self::WRITE,
|
||||
'admin' => self::ADMIN
|
||||
);
|
||||
|
||||
return isset($permissions[$string])
|
||||
? $permissions[$string]
|
||||
: false;
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
private static function _permissionToString($permission)
|
||||
{
|
||||
$permissions = array(
|
||||
self::NONE => 'none',
|
||||
self::READ => 'read',
|
||||
self::WRITE => 'write',
|
||||
self::ADMIN => 'admin',
|
||||
);
|
||||
|
||||
return isset($permissions[$permission])
|
||||
? $permissions[$permission]
|
||||
: false;
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
|
@ -770,90 +719,13 @@ final class Application extends Base
|
|||
$old = $new;
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
private function _saveDatabase()
|
||||
{
|
||||
$data = json_encode(array(
|
||||
'users' => $this->_users,
|
||||
'usersByName' => $this->_usersByName,
|
||||
'tokens' => $this->_tokens,
|
||||
));
|
||||
|
||||
$bytes = @file_put_contents(
|
||||
$this->_di->get('config')['database.json'],
|
||||
$data
|
||||
);
|
||||
if ($bytes === false)
|
||||
{
|
||||
trigger_error(
|
||||
'could not write the database',
|
||||
E_USER_ERROR
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
private function _loadDatabase()
|
||||
{
|
||||
$file = $this->_di->get('config')['database.json'];
|
||||
if (!file_exists($file))
|
||||
{
|
||||
trigger_error(
|
||||
'no such database, using default values (admin:admin)',
|
||||
E_USER_WARNING
|
||||
);
|
||||
|
||||
// @todo Factorizes this code with api_user_create().
|
||||
|
||||
$this->_users = array(
|
||||
1 => array(
|
||||
'name' => 'admin',
|
||||
'password' => '$2y$10$VzBQqiwnhG5zc2.MQmmW4ORcPW6FE7SLhPr1VBV2ubn5zJoesnmli',
|
||||
'permission' => self::ADMIN,
|
||||
),
|
||||
);
|
||||
$this->_usersByName = array(
|
||||
'admin' => '1',
|
||||
);
|
||||
|
||||
return;
|
||||
}
|
||||
|
||||
$data = @file_get_contents(
|
||||
$this->_di->get('config')['database.json']
|
||||
);
|
||||
if (($data === false)
|
||||
|| (($data = json_decode($data, true)) === null))
|
||||
{
|
||||
trigger_error(
|
||||
'could not read the database',
|
||||
E_USER_ERROR
|
||||
);
|
||||
}
|
||||
|
||||
foreach (array('users', 'usersByName', 'tokens') as $entry)
|
||||
{
|
||||
if (!isset($data[$entry]))
|
||||
{
|
||||
trigger_error(
|
||||
"missing entry from the database: $entry",
|
||||
E_USER_ERROR
|
||||
);
|
||||
}
|
||||
|
||||
$this->{'_'.$entry} = $data[$entry];
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
private function _checkPermission($uid, $permission, $object = null)
|
||||
{
|
||||
return ($this->_users[$uid]['permission'] >= $permission);
|
||||
$user = $this->_di->get('users')->get($uid);
|
||||
|
||||
return ($user->permission >= $permission);
|
||||
}
|
||||
}
|
||||
|
|
|
@ -0,0 +1,47 @@
|
|||
<?php
|
||||
/**
|
||||
* This file is a part of Xen Orchestra Server.
|
||||
*
|
||||
* Xen Orchestra Server is free software: you can redistribute it
|
||||
* and/or modify it under the terms of the GNU General Public License
|
||||
* as published by the Free Software Foundation, either version 3 of
|
||||
* the License, or (at your option) any later version.
|
||||
*
|
||||
* Xen Orchestra Server is distributed in the hope that it will be
|
||||
* useful, but WITHOUT ANY WARRANTY; without even the implied warranty
|
||||
* of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||
* General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with Xen Orchestra Server. If not, see
|
||||
* <http://www.gnu.org/licenses/>.
|
||||
*
|
||||
* @author Julien Fontanet <julien.fontanet@vates.fr>
|
||||
* @license http://www.gnu.org/licenses/gpl-3.0-standalone.html GPLv3
|
||||
*
|
||||
* @package Xen Orchestra Server
|
||||
*/
|
||||
|
||||
namespace Bean;
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
final class Token extends \Rekodi\Bean
|
||||
{
|
||||
/**
|
||||
* This function is not necessary but allow us to dynamically
|
||||
* initialize our bean.
|
||||
*/
|
||||
static function init()
|
||||
{
|
||||
self::$_fields = array_flip(array(
|
||||
'id',
|
||||
'expiration',
|
||||
'user_id',
|
||||
));
|
||||
}
|
||||
|
||||
protected static $_fields;
|
||||
}
|
||||
Token::init();
|
|
@ -29,6 +29,60 @@ namespace Bean;
|
|||
*/
|
||||
final class User extends \Rekodi\Bean
|
||||
{
|
||||
/**
|
||||
*
|
||||
*/
|
||||
const NONE = 0;
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
const READ = 1;
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
const WRITE = 2;
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
const ADMIN = 3;
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
static function permissionFromString($string)
|
||||
{
|
||||
$permissions = array(
|
||||
'none' => self::NONE,
|
||||
'read' => self::READ,
|
||||
'write' => self::WRITE,
|
||||
'admin' => self::ADMIN
|
||||
);
|
||||
|
||||
return isset($permissions[$string])
|
||||
? $permissions[$string]
|
||||
: false;
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
static function permissionToString($permission)
|
||||
{
|
||||
$permissions = array(
|
||||
self::NONE => 'none',
|
||||
self::READ => 'read',
|
||||
self::WRITE => 'write',
|
||||
self::ADMIN => 'admin',
|
||||
);
|
||||
|
||||
return isset($permissions[$permission])
|
||||
? $permissions[$permission]
|
||||
: false;
|
||||
}
|
||||
|
||||
/**
|
||||
* This function is not necessary but allow us to dynamically
|
||||
* initialize our bean.
|
||||
|
@ -39,9 +93,37 @@ final class User extends \Rekodi\Bean
|
|||
'id',
|
||||
'name',
|
||||
'password',
|
||||
'permission',
|
||||
));
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
static function check($field, &$value)
|
||||
{
|
||||
switch ($field)
|
||||
{
|
||||
case 'id':
|
||||
return true;
|
||||
case 'name':
|
||||
return (
|
||||
is_string($value)
|
||||
&& preg_match('/^[a-z0-9]+(?:[-_.][a-z0-9]+)*$/', $value)
|
||||
);
|
||||
case 'password':
|
||||
return (
|
||||
is_string($value)
|
||||
&& preg_match('/^.{8,}$/', $value)
|
||||
);
|
||||
case 'permission':
|
||||
$value = self::permissionFromString($value);
|
||||
return (false !== $value);
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
protected static $_fields;
|
||||
}
|
||||
User::init();
|
||||
|
|
|
@ -0,0 +1,173 @@
|
|||
<?php
|
||||
/**
|
||||
* This file is a part of Xen Orchestra Server.
|
||||
*
|
||||
* Xen Orchestra Server is free software: you can redistribute it
|
||||
* and/or modify it under the terms of the GNU General Public License
|
||||
* as published by the Free Software Foundation, either version 3 of
|
||||
* the License, or (at your option) any later version.
|
||||
*
|
||||
* Xen Orchestra Server is distributed in the hope that it will be
|
||||
* useful, but WITHOUT ANY WARRANTY; without even the implied warranty
|
||||
* of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||
* General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with Xen Orchestra Server. If not, see
|
||||
* <http://www.gnu.org/licenses/>.
|
||||
*
|
||||
* @author Julien Fontanet <julien.fontanet@vates.fr>
|
||||
* @license http://www.gnu.org/licenses/gpl-3.0-standalone.html GPLv3
|
||||
*
|
||||
* @package Xen Orchestra Server
|
||||
*/
|
||||
|
||||
namespace Bean;
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
final class VM extends \Rekodi\Bean
|
||||
{
|
||||
/**
|
||||
* This function is not necessary but allow us to dynamically
|
||||
* initialize our bean.
|
||||
*/
|
||||
static function init()
|
||||
{
|
||||
self::$_fields = array_flip(array(
|
||||
// Identifiers.
|
||||
'id',
|
||||
'name_label',
|
||||
'resident_on', // The host this VM is currently resident on.
|
||||
'domarch', // Domain architecture if available, null otherwise.
|
||||
'domid', // Domain ID.
|
||||
|
||||
// Description.
|
||||
'name_description',
|
||||
'is_a_template',
|
||||
'is_a_snapshot',
|
||||
'is_control_domain',
|
||||
'tags',
|
||||
|
||||
// Technical characteristics.
|
||||
'attached_PCIs',
|
||||
'platform',
|
||||
'VBDs', // Virtual Block Devices.
|
||||
'VCPUs_at_startup',
|
||||
'VCPUs_max',
|
||||
'VCPUs_params',
|
||||
'VGPUs',
|
||||
'VIFs', // Virtual Network Interface (string[]). @todo
|
||||
//'VTPMs', // Virtual Trust Platform Modules ???
|
||||
|
||||
// Event-related actions.
|
||||
'actions_after_crash',
|
||||
'actions_after_reboot',
|
||||
'actions_after_shutdown',
|
||||
|
||||
// Current state.
|
||||
'guest_metrics',
|
||||
'memory_dynamic_max',
|
||||
'memory_dynamic_min',
|
||||
'memory_overhead',
|
||||
'memory_static_max',
|
||||
'memory_static_min',
|
||||
'memory_target',
|
||||
'metrics',
|
||||
'power_state',
|
||||
|
||||
// Snapshot-related info.
|
||||
'shutdown_delay',
|
||||
'snapshot_info',
|
||||
'snapshot_metadata',
|
||||
'snapshot_of',
|
||||
'snapshot_time',
|
||||
'snapshots',
|
||||
'transportable_snapshot_id',
|
||||
|
||||
// Operations.
|
||||
'allowed_operations',
|
||||
'current_operations',
|
||||
|
||||
// Various.
|
||||
'consoles',
|
||||
|
||||
// 'affinity',
|
||||
// 'appliance',
|
||||
// 'bios_strings',
|
||||
// 'blobs',
|
||||
// 'blocked_operations',
|
||||
// 'children', // ???
|
||||
// 'crash_dumps',
|
||||
// 'ha_always_run',
|
||||
// 'ha_restart_priority',
|
||||
// 'HVM_boot_params',
|
||||
// 'HVM_boot_policy',
|
||||
// 'HVM_shadow_multiplier',
|
||||
// 'is_snapshot_from_vmpp',
|
||||
// 'last_boot_CPU_flags',
|
||||
// 'last_booted_record',
|
||||
// 'order',
|
||||
// 'other_config',
|
||||
// 'parent', // ???
|
||||
// 'PCI_bus',
|
||||
// 'protection_policy',
|
||||
// 'PV_args',
|
||||
// 'PV_bootloader',
|
||||
// 'PV_bootloader_args',
|
||||
// 'PV_kernel',
|
||||
// 'PV_legacy_args',
|
||||
// 'PV_ramdisk',
|
||||
// 'recommendations',
|
||||
// 'start_delay',
|
||||
// 'suspend_SR',
|
||||
// 'suspend_VDI',
|
||||
// 'user_version',
|
||||
// 'version',
|
||||
// 'xenstore_data',
|
||||
));
|
||||
}
|
||||
|
||||
function __get($name)
|
||||
{
|
||||
static $json_encoded;
|
||||
if (!$json_encoded)
|
||||
{
|
||||
$json_encoded = array_flip(array(
|
||||
'VBDs',
|
||||
'VCPUs_params',
|
||||
'VGPUs',
|
||||
'VIFs',
|
||||
'allowed_operations',
|
||||
'consoles',
|
||||
'crash_dumps',
|
||||
'current_operations',
|
||||
'platform',
|
||||
'snapshot_info',
|
||||
'tags',
|
||||
));
|
||||
}
|
||||
|
||||
$value = parent::__get($name);
|
||||
if (isset($json_encoded[$name]))
|
||||
{
|
||||
return json_decode($value, true);
|
||||
}
|
||||
return $value;
|
||||
}
|
||||
|
||||
function __set($name, $value)
|
||||
{
|
||||
if (is_array($value)
|
||||
|| is_object($value))
|
||||
{
|
||||
$value = json_encode($value);
|
||||
}
|
||||
|
||||
return parent::__set($name, $value);
|
||||
}
|
||||
|
||||
protected static $_fields;
|
||||
}
|
||||
VM::init();
|
|
@ -30,7 +30,7 @@ final class Client extends Base
|
|||
/**
|
||||
*
|
||||
*/
|
||||
public $uid = 0;
|
||||
public $uid = null;
|
||||
|
||||
/**
|
||||
* @param Loop $loop
|
||||
|
@ -54,7 +54,7 @@ final class Client extends Base
|
|||
*/
|
||||
function isAuthenticated()
|
||||
{
|
||||
return ($this->uid !== 0);
|
||||
return (null !== $this->uid);
|
||||
}
|
||||
|
||||
/**
|
||||
|
|
|
@ -89,7 +89,7 @@ final class Config extends Base implements
|
|||
$entries = $entries->_entries;
|
||||
}
|
||||
|
||||
$this->_entries = array_merge_recursive($this->_entries, $entries);
|
||||
self::_merge($this->_entries, $entries);
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -195,6 +195,26 @@ final class Config extends Base implements
|
|||
|
||||
//--------------------------------------
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
static private function _merge(array &$ar1, array $ar2)
|
||||
{
|
||||
foreach ($ar2 as $key => $val)
|
||||
{
|
||||
if (is_array($val)
|
||||
&& isset($ar1[$key])
|
||||
&& is_array($ar1[$key]))
|
||||
{
|
||||
self::_merge($ar1[$key], $val);
|
||||
}
|
||||
else
|
||||
{
|
||||
$ar1[$key] = $val;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @var array
|
||||
*/
|
||||
|
|
91
lib/DI.php
91
lib/DI.php
|
@ -71,6 +71,71 @@ final class DI extends Base
|
|||
return new Application($this);
|
||||
}
|
||||
|
||||
private function _init_database()
|
||||
{
|
||||
$config = $this->get('config');
|
||||
|
||||
$type = $config['database.type'];
|
||||
if ('json' !== $type)
|
||||
{
|
||||
trigger_error(
|
||||
'unsupported database type ('.$type.')',
|
||||
E_USER_ERROR
|
||||
);
|
||||
}
|
||||
|
||||
$file = $config['database.file'];
|
||||
if (file_exists($file))
|
||||
{
|
||||
$data = @file_get_contents($file);
|
||||
if ((false === $data)
|
||||
|| (null === ($data = json_decode($data, true))))
|
||||
{
|
||||
trigger_error(
|
||||
'could not read the database',
|
||||
E_USER_ERROR
|
||||
);
|
||||
}
|
||||
|
||||
return \Rekodi\Manager\Memory::createFromState($data);
|
||||
}
|
||||
|
||||
$manager = new \Rekodi\Manager\Memory;
|
||||
|
||||
// Create tables.
|
||||
$manager->createTable('tokens', function ($table) {
|
||||
$table
|
||||
->string('id')->unique()
|
||||
->integer('expiration')
|
||||
->string('user_id')
|
||||
;
|
||||
});
|
||||
$manager->createTable('users', function ($table) {
|
||||
$table
|
||||
->integer('id')->autoIncremented()
|
||||
->string('name')->unique()
|
||||
->string('password')
|
||||
->integer('permission')
|
||||
;
|
||||
});
|
||||
|
||||
// Insert initial data.
|
||||
$manager->create('users', array(
|
||||
array(
|
||||
'name' => 'admin',
|
||||
'password' => '$2y$10$VzBQqiwnhG5zc2.MQmmW4ORcPW6FE7SLhPr1VBV2ubn5zJoesnmli',
|
||||
'permission' => \Bean\User::ADMIN,
|
||||
),
|
||||
));
|
||||
|
||||
trigger_error(
|
||||
'no existing database, creating default user (admin:admin)',
|
||||
E_USER_WARNING
|
||||
);
|
||||
|
||||
return $manager;
|
||||
}
|
||||
|
||||
private function _init_errorLogger()
|
||||
{
|
||||
return new ErrorLogger($this->get('logger'));
|
||||
|
@ -109,4 +174,30 @@ final class DI extends Base
|
|||
{
|
||||
return new Loop;
|
||||
}
|
||||
|
||||
private function _init_tokens()
|
||||
{
|
||||
return new \Manager\Tokens(
|
||||
$this->get('database')
|
||||
);
|
||||
}
|
||||
|
||||
private function _init_users()
|
||||
{
|
||||
return new \Manager\Users(
|
||||
$this->get('database')
|
||||
);
|
||||
}
|
||||
|
||||
private function _init_vms()
|
||||
{
|
||||
$database = new \Rekodi\Manager\Memory;
|
||||
$database->createTable('vms', function ($table) {
|
||||
$table
|
||||
->string('id')->unique()
|
||||
;
|
||||
});
|
||||
|
||||
return new \Manager\VMs($database);
|
||||
}
|
||||
}
|
||||
|
|
12
lib/Loop.php
12
lib/Loop.php
|
@ -119,8 +119,10 @@ final class Loop extends Base
|
|||
|
||||
foreach ($read as $handle)
|
||||
{
|
||||
$callback = $this->_readCallbacks[(int) $handle];
|
||||
$result = call_user_func($callback, $handle, $user_data);
|
||||
$result = call_user_func(
|
||||
$this->_readCallbacks[(int) $handle],
|
||||
$handle, $user_data
|
||||
);
|
||||
|
||||
if (!is_resource($handle))
|
||||
{
|
||||
|
@ -133,8 +135,10 @@ final class Loop extends Base
|
|||
}
|
||||
foreach ($write as $handle)
|
||||
{
|
||||
$callback = $this->_writeCallbacks[(int) $handle];
|
||||
$result = call_user_func($callback, $handle, $user_data);
|
||||
$result = call_user_func(
|
||||
$this->_writeCallbacks[(int) $handle],
|
||||
$handle, $user_data
|
||||
);
|
||||
|
||||
if (!is_resource($handle))
|
||||
{
|
||||
|
|
|
@ -0,0 +1,186 @@
|
|||
<?php
|
||||
/**
|
||||
* This file is a part of Xen Orchestra Server.
|
||||
*
|
||||
* Xen Orchestra Server is free software: you can redistribute it
|
||||
* and/or modify it under the terms of the GNU General Public License
|
||||
* as published by the Free Software Foundation, either version 3 of
|
||||
* the License, or (at your option) any later version.
|
||||
*
|
||||
* Xen Orchestra Server is distributed in the hope that it will be
|
||||
* useful, but WITHOUT ANY WARRANTY; without even the implied warranty
|
||||
* of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||
* General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with Xen Orchestra Server. If not, see
|
||||
* <http://www.gnu.org/licenses/>.
|
||||
*
|
||||
* @author Julien Fontanet <julien.fontanet@vates.fr>
|
||||
* @license http://www.gnu.org/licenses/gpl-3.0-standalone.html GPLv3
|
||||
*
|
||||
* @package Xen Orchestra Server
|
||||
*/
|
||||
|
||||
namespace Manager;
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
abstract class ManagerAbstract
|
||||
{
|
||||
/**
|
||||
*
|
||||
*/
|
||||
protected function __construct(\Rekodi\Manager $manager, $table, $bean)
|
||||
{
|
||||
$this->_manager = $manager;
|
||||
$this->_table = $table;
|
||||
$this->_bean = $bean;
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
function __destruct()
|
||||
{}
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
function create(array $properties)
|
||||
{
|
||||
$class = $this->_bean;
|
||||
$bean = new $class($properties, true);
|
||||
|
||||
$new_props = $this->_manager->create(
|
||||
$this->_table,
|
||||
array($bean->getProperties())
|
||||
);
|
||||
|
||||
if (count($new_props) !== 1)
|
||||
{
|
||||
trigger_error(
|
||||
'unexpected number of created '.$this->_table.' ('.$n.')',
|
||||
E_USER_ERROR
|
||||
);
|
||||
}
|
||||
|
||||
$bean->set($new_props[0]);
|
||||
$bean->markAsClean();
|
||||
return $bean;
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
function delete($id)
|
||||
{
|
||||
$n = $this->_manager->delete($this->_table, array('id' => $id));
|
||||
|
||||
if (1 !== $n)
|
||||
{
|
||||
trigger_error(
|
||||
'unexpected number of deleted '.$this->_table.' ('.$n.')',
|
||||
E_USER_ERROR
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
function getBy($field, $value, $default = 'fatal error')
|
||||
{
|
||||
$beans = $this->_manager->get(
|
||||
$this->_table,
|
||||
array($field => $value)
|
||||
);
|
||||
|
||||
if ($beans)
|
||||
{
|
||||
$class = $this->_bean;
|
||||
return new $class($beans[0], true);
|
||||
}
|
||||
|
||||
if (func_num_args() >= 3)
|
||||
{
|
||||
return $default;
|
||||
}
|
||||
|
||||
trigger_error(
|
||||
'no such '.$this->_table.' ('.$field.' = '.$value.')',
|
||||
E_USER_ERROR
|
||||
);
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
* @param string $id
|
||||
* @param mixed $default
|
||||
*
|
||||
* @return Bean
|
||||
*/
|
||||
function get($id, $default = 'fatal error')
|
||||
{
|
||||
if (func_num_args() === 1)
|
||||
{
|
||||
return $this->getBy('id', $id);
|
||||
}
|
||||
return $this->getBy('id', $id, $default);
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
function getArray($filter = null, $fields = null)
|
||||
{
|
||||
return $this->_manager->get(
|
||||
$this->_table,
|
||||
$filter,
|
||||
$fields
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
function save(\Rekodi\Bean $bean)
|
||||
{
|
||||
$dirty = $bean->getDirty();
|
||||
if (!$dirty)
|
||||
{
|
||||
// Nothing has changed.
|
||||
return;
|
||||
}
|
||||
|
||||
$n = $this->_manager->update(
|
||||
$this->_table,
|
||||
array('id' => $bean->id),
|
||||
$bean->getDirty()
|
||||
);
|
||||
|
||||
if (1 !== $n)
|
||||
{
|
||||
trigger_error(
|
||||
'unexpected number of updated '.$this->_table.' ('.$n.')',
|
||||
E_USER_ERROR
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @var \Rekodi\Manager
|
||||
*/
|
||||
private $_manager;
|
||||
|
||||
/**
|
||||
* @var string
|
||||
*/
|
||||
private $_table;
|
||||
|
||||
/**
|
||||
* @var string
|
||||
*/
|
||||
private $_bean;
|
||||
}
|
|
@ -0,0 +1,39 @@
|
|||
<?php
|
||||
/**
|
||||
* This file is a part of Xen Orchestra Server.
|
||||
*
|
||||
* Xen Orchestra Server is free software: you can redistribute it
|
||||
* and/or modify it under the terms of the GNU General Public License
|
||||
* as published by the Free Software Foundation, either version 3 of
|
||||
* the License, or (at your option) any later version.
|
||||
*
|
||||
* Xen Orchestra Server is distributed in the hope that it will be
|
||||
* useful, but WITHOUT ANY WARRANTY; without even the implied warranty
|
||||
* of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||
* General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with Xen Orchestra Server. If not, see
|
||||
* <http://www.gnu.org/licenses/>.
|
||||
*
|
||||
* @author Julien Fontanet <julien.fontanet@vates.fr>
|
||||
* @license http://www.gnu.org/licenses/gpl-3.0-standalone.html GPLv3
|
||||
*
|
||||
* @package Xen Orchestra Server
|
||||
*/
|
||||
|
||||
namespace Manager;
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
final class Tokens extends ManagerAbstract
|
||||
{
|
||||
/**
|
||||
*
|
||||
*/
|
||||
function __construct(\Rekodi\Manager $manager)
|
||||
{
|
||||
parent::__construct($manager, 'tokens', '\Bean\Token');
|
||||
}
|
||||
}
|
|
@ -0,0 +1,39 @@
|
|||
<?php
|
||||
/**
|
||||
* This file is a part of Xen Orchestra Server.
|
||||
*
|
||||
* Xen Orchestra Server is free software: you can redistribute it
|
||||
* and/or modify it under the terms of the GNU General Public License
|
||||
* as published by the Free Software Foundation, either version 3 of
|
||||
* the License, or (at your option) any later version.
|
||||
*
|
||||
* Xen Orchestra Server is distributed in the hope that it will be
|
||||
* useful, but WITHOUT ANY WARRANTY; without even the implied warranty
|
||||
* of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||
* General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with Xen Orchestra Server. If not, see
|
||||
* <http://www.gnu.org/licenses/>.
|
||||
*
|
||||
* @author Julien Fontanet <julien.fontanet@vates.fr>
|
||||
* @license http://www.gnu.org/licenses/gpl-3.0-standalone.html GPLv3
|
||||
*
|
||||
* @package Xen Orchestra Server
|
||||
*/
|
||||
|
||||
namespace Manager;
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
final class Users extends ManagerAbstract
|
||||
{
|
||||
/**
|
||||
*
|
||||
*/
|
||||
function __construct(\Rekodi\Manager $manager)
|
||||
{
|
||||
parent::__construct($manager, 'users', '\Bean\User');
|
||||
}
|
||||
}
|
|
@ -0,0 +1,39 @@
|
|||
<?php
|
||||
/**
|
||||
* This file is a part of Xen Orchestra Server.
|
||||
*
|
||||
* Xen Orchestra Server is free software: you can redistribute it
|
||||
* and/or modify it under the terms of the GNU General Public License
|
||||
* as published by the Free Software Foundation, either version 3 of
|
||||
* the License, or (at your option) any later version.
|
||||
*
|
||||
* Xen Orchestra Server is distributed in the hope that it will be
|
||||
* useful, but WITHOUT ANY WARRANTY; without even the implied warranty
|
||||
* of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||
* General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with Xen Orchestra Server. If not, see
|
||||
* <http://www.gnu.org/licenses/>.
|
||||
*
|
||||
* @author Julien Fontanet <julien.fontanet@vates.fr>
|
||||
* @license http://www.gnu.org/licenses/gpl-3.0-standalone.html GPLv3
|
||||
*
|
||||
* @package Xen Orchestra Server
|
||||
*/
|
||||
|
||||
namespace Manager;
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
final class VMs extends ManagerAbstract
|
||||
{
|
||||
/**
|
||||
*
|
||||
*/
|
||||
function __construct(\Rekodi\Manager $manager)
|
||||
{
|
||||
parent::__construct($manager, 'vms', '\Bean\VM');
|
||||
}
|
||||
}
|
|
@ -1,113 +0,0 @@
|
|||
<?php
|
||||
/**
|
||||
* This file is a part of Xen Orchestra Server.
|
||||
*
|
||||
* Xen Orchestra Server is free software: you can redistribute it
|
||||
* and/or modify it under the terms of the GNU General Public License
|
||||
* as published by the Free Software Foundation, either version 3 of
|
||||
* the License, or (at your option) any later version.
|
||||
*
|
||||
* Xen Orchestra Server is distributed in the hope that it will be
|
||||
* useful, but WITHOUT ANY WARRANTY; without even the implied warranty
|
||||
* of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||
* General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with Xen Orchestra Server. If not, see
|
||||
* <http://www.gnu.org/licenses/>.
|
||||
*
|
||||
* @author Julien Fontanet <julien.fontanet@vates.fr>
|
||||
* @license http://www.gnu.org/licenses/gpl-3.0-standalone.html GPLv3
|
||||
*
|
||||
* @package Xen Orchestra Server
|
||||
*/
|
||||
|
||||
namespace Model;
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
final class Users
|
||||
{
|
||||
/**
|
||||
*
|
||||
*/
|
||||
function __construct(\Rekodi\Manager $manager)
|
||||
{
|
||||
$this->_manager = $manager;
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
function delete($id)
|
||||
{
|
||||
$n = $this->_manager->delete(array('id' => $id));
|
||||
|
||||
if ($n !== 1)
|
||||
{
|
||||
trigger_error(
|
||||
'unexpected number of deleted users ('.$n.')',
|
||||
E_USER_ERROR
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @param string $id
|
||||
* @param mixed $default
|
||||
*
|
||||
* @return User
|
||||
*/
|
||||
function get($id, $default = 'fatal error')
|
||||
{
|
||||
$users = $this->_manager->get(array('id' => $id));
|
||||
|
||||
if ($users)
|
||||
{
|
||||
return $users[0];
|
||||
}
|
||||
|
||||
if (func_num_args() >= 2)
|
||||
{
|
||||
return $default;
|
||||
}
|
||||
|
||||
trigger_error(
|
||||
'no such user (id = '.$id.')',
|
||||
E_USER_ERROR
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
function save(User $user)
|
||||
{
|
||||
if (!isset($user->id))
|
||||
{
|
||||
|
||||
// @todo Fills the user with its generated id.
|
||||
|
||||
return;
|
||||
}
|
||||
|
||||
$n = $this->_manager->update(
|
||||
$user->getOriginals(),
|
||||
$user->getDirty()
|
||||
);
|
||||
|
||||
if ($n !== 1)
|
||||
{
|
||||
trigger_error(
|
||||
'unexpected number of updated users ('.$n.')',
|
||||
E_USER_ERROR
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @var \Rekodi\Manager
|
||||
*/
|
||||
private $_manager;
|
||||
}
|
|
@ -91,8 +91,7 @@ final class XCP extends Base
|
|||
$request = array_shift($this->_queue);
|
||||
if ($request[0] !== null)
|
||||
{
|
||||
$callback = $request[0];
|
||||
$result = call_user_func($callback, $response, $this);
|
||||
$result = call_user_func($request[0], $response, $this);
|
||||
if ($result === true)
|
||||
{
|
||||
$this->_queue[] = $request;
|
||||
|
|
Loading…
Reference in New Issue