941 lines
34 KiB
PHP
941 lines
34 KiB
PHP
<?php
|
|
use \LAM\PDF\PDFTable;
|
|
use \LAM\PDF\PDFTableCell;
|
|
use \LAM\PDF\PDFTableRow;
|
|
/*
|
|
|
|
This code is part of LDAP Account Manager (http://www.ldap-account-manager.org/)
|
|
Copyright (C) 2003 - 2006 Tilo Lutz
|
|
2007 - 2019 Roland Gruber
|
|
|
|
This program 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 2 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program 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 this program; if not, write to the Free Software
|
|
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*/
|
|
|
|
|
|
/**
|
|
* Manages quotas for users and groups.
|
|
*
|
|
* @package modules
|
|
*
|
|
* @author Tilo Lutz
|
|
* @author Roland Gruber
|
|
* @author Michael Duergner
|
|
*/
|
|
|
|
/**
|
|
* Manages quotas for users and groups.
|
|
*
|
|
* @package modules
|
|
*/
|
|
class quota extends baseModule {
|
|
|
|
/** delimiter for lamdaemon commands */
|
|
private static $SPLIT_DELIMITER = "###x##y##x###";
|
|
/** prefix for lamdaemon results */
|
|
private static $QUOTA_PREFIX = 'QUOTA_ENTRY ';
|
|
|
|
/** this function fills the error message array with messages
|
|
**/
|
|
function load_Messages() {
|
|
// error messages for input checks
|
|
$this->messages['softblock'][0] = array('ERROR', _('Block soft quota'), _('Block soft quota contains invalid characters. Only natural numbers are allowed.'));
|
|
$this->messages['softblock'][1] = array('ERROR', _('Account %s:') . ' %s', _('Block soft quota contains invalid characters. Only natural numbers are allowed.'));
|
|
$this->messages['hardblock'][0] = array('ERROR', _('Block hard quota'), _('Block hard quota contains invalid characters. Only natural numbers are allowed.'));
|
|
$this->messages['hardblock'][1] = array('ERROR', _('Account %s:') . ' %s', _('Block hard quota contains invalid characters. Only natural numbers are allowed.'));
|
|
$this->messages['softinode'][0] = array('ERROR', _('Inode soft quota'), _('Inode soft quota contains invalid characters. Only natural numbers are allowed.'));
|
|
$this->messages['softinode'][1] = array('ERROR', _('Account %s:') . ' %s', _('Inode soft quota contains invalid characters. Only natural numbers are allowed.'));
|
|
$this->messages['hardinode'][0] = array('ERROR', _('Inode hard quota'), _('Inode hard quota contains invalid characters. Only natural numbers are allowed.'));
|
|
$this->messages['hardinode'][1] = array('ERROR', _('Account %s:') . ' %s', _('Inode hard quota contains invalid characters. Only natural numbers are allowed.'));
|
|
$this->messages['block_cmp'][0] = array('ERROR', _('Block quota'), _('Block soft quota must be smaller than block hard quota.'));
|
|
$this->messages['block_cmp'][1] = array('ERROR', _('Account %s:') . ' %s', _('Block soft quota must be smaller than block hard quota.'));
|
|
$this->messages['inode_cmp'][0] = array('ERROR', _('Inode quota'), _('Inode soft quota must be smaller than inode hard quota.'));
|
|
$this->messages['inode_cmp'][1] = array('ERROR', _('Account %s:') . ' %s', _('Inode soft quota must be smaller than inode hard quota.'));
|
|
$this->messages['upload'][0] = array('ERROR', _('Account %s:') . ' %s', _('Quota has wrong format!'));
|
|
}
|
|
|
|
/**
|
|
* Returns true if this module can manage accounts of the current type, otherwise false.
|
|
*
|
|
* @return boolean true if module fits
|
|
*/
|
|
public function can_manage() {
|
|
return in_array($this->get_scope(), array('user', 'group'));
|
|
}
|
|
|
|
/**
|
|
* Returns meta data that is interpreted by parent class
|
|
*
|
|
* @return array array with meta data
|
|
*
|
|
* @see baseModule::get_metaData()
|
|
*/
|
|
function get_metaData() {
|
|
$return = array();
|
|
// icon
|
|
$return['icon'] = 'hard-driveBig.png';
|
|
// alias name
|
|
$return["alias"] = _('Quota');
|
|
if ($this->get_scope() == 'group') {
|
|
// module dependencies
|
|
$return['dependencies'] = array('depends' => array(array('posixGroup', 'rfc2307bisPosixGroup', 'windowsPosixGroup')), 'conflicts' => array());
|
|
}
|
|
if ($this->get_scope() == 'user') {
|
|
// module dependencies
|
|
$return['dependencies'] = array('depends' => array('posixAccount'), 'conflicts' => array());
|
|
}
|
|
// managed attributes
|
|
$return['attributes'] = array('uid', 'cn');
|
|
// available PDF fields
|
|
$return['PDF_fields'] = array(
|
|
'quotas' => _('Quota')
|
|
);
|
|
// help entries
|
|
$numbersHelp = _('Symbols K, M, G and T can be appended to numeric value to express kibibytes, mebibytes, gibibytes and tebibytes for blocks.')
|
|
. _(' Symbols k, m, g and t can be appended to numeric value to express multiples of 10^3, 10^6, 10^9 and 10^12 inodes.');
|
|
$return['help'] = array(
|
|
"Mountpoint" => array(
|
|
"Headline" => _("Mountpoint"),
|
|
"Text" => _("Mountpoint of device with enabled quotas.")
|
|
),
|
|
"UsedBlocks" => array(
|
|
"Headline" => _("Used blocks"),
|
|
"Text" => _("Used blocks. 1000 blocks are usually 1MB")
|
|
),
|
|
"SoftBlockLimit" => array(
|
|
"Headline" => _("Soft block limit"),
|
|
"Text" => _("Soft block limit.") . '<br>' . $numbersHelp, "SeeAlso" => array(
|
|
'link' => 'http://www.ibiblio.org/pub/Linux/docs/HOWTO/other-formats/html_single/Quota.html#ss4.4',
|
|
'text' => 'Quota How-To')
|
|
),
|
|
"HardBlockLimit" => array(
|
|
"Headline" => _("Hard block limit"),
|
|
"Text" => _("Hard block limit") . '<br>' . $numbersHelp, "SeeAlso" => array(
|
|
'link' => 'http://www.ibiblio.org/pub/Linux/docs/HOWTO/other-formats/html_single/Quota.html#ss4.5',
|
|
'text' => 'Quota How-To')
|
|
),
|
|
"GraceBlockPeriod" => array(
|
|
"Headline" => _("Grace block period"),
|
|
"Text" => _("Grace block period. Most filesystems use a fixed maximum value of 7 days."), "SeeAlso" => array(
|
|
'link' => 'http://www.ibiblio.org/pub/Linux/docs/HOWTO/other-formats/html_single/Quota.html#ss4.6',
|
|
'text' => 'Quota How-To')
|
|
),
|
|
"UsedInodes" => array(
|
|
"Headline" => _("Used inodes"),
|
|
"Text" => _("Used inodes (files)").'.'
|
|
),
|
|
"SoftInodeLimit" => array(
|
|
"Headline" => _("Soft inode limit"),
|
|
"Text" => _("Soft inode (files) limit.") . '<br>' . $numbersHelp, "SeeAlso" => array(
|
|
'link' => 'http://www.ibiblio.org/pub/Linux/docs/HOWTO/other-formats/html_single/Quota.html#ss4.4',
|
|
'text' => 'Quota How-To')
|
|
),
|
|
"HardInodeLimit" => array(
|
|
"Headline" => _("Hard inode limit"),
|
|
"Text" => _("Hard inode (files) limit") . '<br>' . $numbersHelp, "SeeAlso" => array(
|
|
'link' => 'http://www.ibiblio.org/pub/Linux/docs/HOWTO/other-formats/html_single/Quota.html#ss4.5',
|
|
'text' => 'Quota How-To')
|
|
),
|
|
"GraceInodePeriod" => array(
|
|
"Headline" => _("Grace inode period"),
|
|
"Text" => _("Grace inode (files) period. Most filesystems use a fixed maximum value of 7 days."), "SeeAlso" => array(
|
|
'link' => 'http://www.ibiblio.org/pub/Linux/docs/HOWTO/other-formats/html_single/Quota.html#ss4.6',
|
|
'text' => 'Quota How-To')
|
|
),
|
|
"upload" => array(
|
|
"Headline" => _("Quota"),
|
|
"Text" => _("Please enter the quota settings for this mount point. The syntax is: {soft block limit},{hard block limit},{soft inode limit},{hard inode limit}")
|
|
)
|
|
);
|
|
return $return;
|
|
}
|
|
|
|
/** Saves the quota settings */
|
|
private $quota;
|
|
|
|
/**
|
|
* Initializes the quota values.
|
|
*/
|
|
function initQuotas() {
|
|
if (isset($this->quota)) {
|
|
return;
|
|
}
|
|
$userName = '+';
|
|
if (($this->getAccountContainer() != null) && !$this->getAccountContainer()->isNewAccount) {
|
|
if ($this->get_scope() == 'user') {
|
|
if (!isset($this->attributes['uid'][0])) {
|
|
return;
|
|
}
|
|
$userName = $this->attributes['uid'][0];
|
|
}
|
|
else if ($this->get_scope() == 'group') {
|
|
if (!isset($this->attributes['cn'][0])) {
|
|
return;
|
|
}
|
|
$userName = $this->attributes['cn'][0];
|
|
}
|
|
}
|
|
// get list of lamdaemon servers
|
|
$lamdaemonServers = $_SESSION['config']->getConfiguredScriptServers();
|
|
foreach ($lamdaemonServers as $lamdaemonServer) {
|
|
$server = $lamdaemonServer->getServer();
|
|
// get quotas
|
|
$remote = new \LAM\REMOTE\Remote();
|
|
$remote->connect($lamdaemonServer);
|
|
$quotas = $remote->execute(implode(quota::$SPLIT_DELIMITER, array($userName, "quota", "get", $this->get_scope())));
|
|
$remote->disconnect();
|
|
if (empty($quotas)) {
|
|
continue;
|
|
}
|
|
$allQuotas = explode(":", $quotas);
|
|
array_pop($allQuotas); // remove empty element at the end
|
|
for ($i = 0; $i < sizeof($allQuotas); $i++) {
|
|
if (strpos($allQuotas[$i], quota::$QUOTA_PREFIX) !== 0) {
|
|
continue;
|
|
}
|
|
$allQuotas[$i] = substr($allQuotas[$i], strlen(quota::$QUOTA_PREFIX));
|
|
$singleQuota = explode(",", $allQuotas[$i]);
|
|
$this->quota[$server][$i] = $singleQuota;
|
|
if ($this->quota[$server][$i][4] < time()) {
|
|
$this->quota[$server][$i][4] = '';
|
|
}
|
|
else {
|
|
$this->quota[$server][$i][4] = strval(intval(($this->quota[$server][$i][4] - time())/3600)) .' '. _('hours');
|
|
}
|
|
if ($this->quota[$server][$i][8] < time()) {
|
|
$this->quota[$server][$i][8] = '';
|
|
}
|
|
else {
|
|
$this->quota[$server][$i][8] = strval(intval(($this->quota[$server][$i][8] - time())/3600)) .' '. _('hours');
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Gets the cn from the Unix group module.
|
|
*
|
|
* @return String cn attribute
|
|
*/
|
|
private function getCn() {
|
|
$modules = array('posixGroup', 'windowsPosixGroup', 'rfc2307bisPosixGroup', 'groupOfNames', 'groupOfUniqueNames', 'windowsGroup');
|
|
for ($i = 0; $i < sizeof($modules); $i++) {
|
|
if ($this->getAccountContainer()->getAccountModule($modules[$i]) != null) {
|
|
$attrs = $this->getAccountContainer()->getAccountModule($modules[$i])->getAttributes();
|
|
if (isset($attrs['cn'][0])) {
|
|
return $attrs['cn'][0];
|
|
}
|
|
}
|
|
}
|
|
return null;
|
|
}
|
|
|
|
/**
|
|
* This function is used to check if this module page can be displayed.
|
|
* It returns false if a module depends on data from other modules which was not yet entered.
|
|
*
|
|
* @return boolean true, if page can be displayed
|
|
*/
|
|
function module_ready() {
|
|
if ($this->get_scope()=='user') {
|
|
$attrs = $this->getAccountContainer()->getAccountModule('posixAccount')->getAttributes();
|
|
if ($attrs['uid'][0]=='') {
|
|
return false;
|
|
}
|
|
}
|
|
return ($this->get_scope() != 'group') || !empty($this->getCn());
|
|
}
|
|
|
|
/**
|
|
* Quotas are set in postmodify.
|
|
*
|
|
* @see baseModule::postModifyActions()
|
|
*
|
|
* @param boolean $newAccount is new account
|
|
* @param array $attributes LDAP attributes of this entry
|
|
* @return array array which contains status messages. Each entry is an array containing the status message parameters.
|
|
*/
|
|
public function postModifyActions($newAccount, $attributes) {
|
|
$messages = array();
|
|
if (!isset($this->quota) || !is_array($this->quota)) {
|
|
return $messages;
|
|
}
|
|
// determine if this is a user or group account
|
|
if ($this->get_scope()=='user') {
|
|
$attrs = $this->getAccountContainer()->getAccountModule('posixAccount')->getAttributes();
|
|
$id = $attrs['uid'][0];
|
|
}
|
|
if ($this->get_scope()=='group') {
|
|
$id = $this->getCn();
|
|
}
|
|
// get list of lamdaemon servers
|
|
$servers = array_keys($this->quota);
|
|
for ($q = 0; $q < sizeof($servers); $q++) {
|
|
$server = $servers[$q];
|
|
$i=0;
|
|
$quotastring = "";
|
|
for ($i = 0; $i < sizeof($this->quota[$server]); $i++) {
|
|
$quotastring .= $this->quota[$server][$i][0];
|
|
$quotastring .= ',' . $this->getQuotaNumber($this->quota[$server][$i][2]);
|
|
$quotastring .= ',' . $this->getQuotaNumber($this->quota[$server][$i][3]);
|
|
$quotastring .= ',' . $this->getQuotaNumber($this->quota[$server][$i][6]);
|
|
$quotastring .= ',' . $this->getQuotaNumber($this->quota[$server][$i][7]) . ':';
|
|
}
|
|
$remote = new \LAM\REMOTE\Remote();
|
|
$remoteServer = $_SESSION['config']->getScriptServerByName($server);
|
|
$remote->connect($remoteServer);
|
|
$remote->execute(implode(quota::$SPLIT_DELIMITER, array($id, "quota", "set", $this->get_scope(), "$quotastring\n")));
|
|
$remote->disconnect();
|
|
}
|
|
return $messages;
|
|
}
|
|
|
|
/**
|
|
* Returns the quota value as plain number.
|
|
*
|
|
* @param string $quotaInput quota value (might contain e.g. "K" for KB)
|
|
* @return int quota number
|
|
*/
|
|
private function getQuotaNumber($quotaInput) {
|
|
if (empty($quotaInput) || (strlen($quotaInput) < 2)) {
|
|
return $quotaInput;
|
|
}
|
|
if (substr($quotaInput, -1, 1) === 'K') {
|
|
return substr($quotaInput, 0, -1);
|
|
}
|
|
if (substr($quotaInput, -1, 1) === 'M') {
|
|
return 1024 * substr($quotaInput, 0, -1);
|
|
}
|
|
if (substr($quotaInput, -1, 1) === 'G') {
|
|
return 1024 * 1024 * substr($quotaInput, 0, -1);
|
|
}
|
|
if (substr($quotaInput, -1, 1) === 'T') {
|
|
return 1024 * 1024 * 1024 * 1024 * substr($quotaInput, 0, -1);
|
|
}
|
|
if (substr($quotaInput, -1, 1) === 'k') {
|
|
return 1000 * substr($quotaInput, 0, -1);
|
|
}
|
|
if (substr($quotaInput, -1, 1) === 'm') {
|
|
return 1000 * 1000 * substr($quotaInput, 0, -1);
|
|
}
|
|
if (substr($quotaInput, -1, 1) === 'g') {
|
|
return 1000 * 1000 * 1000 * substr($quotaInput, 0, -1);
|
|
}
|
|
if (substr($quotaInput, -1, 1) === 't') {
|
|
return 1000 * 1000 * 1000 * 1000 * substr($quotaInput, 0, -1);
|
|
}
|
|
return $quotaInput;
|
|
}
|
|
|
|
/**
|
|
* Allows the module to run commands before the LDAP entry is deleted.
|
|
*
|
|
* @return array Array which contains status messages. Each entry is an array containing the status message parameters.
|
|
*/
|
|
function preDeleteActions() {
|
|
try {
|
|
$this->initQuotas();
|
|
}
|
|
catch (LAMException $e) {
|
|
return array(array('ERROR', $e->getTitle(), $e->getMessage()));
|
|
}
|
|
if (!isset($this->quota) || !is_array($this->quota)) {
|
|
return array();
|
|
}
|
|
// determine if this is a user or group account
|
|
if ($this->get_scope()=='user') {
|
|
$attrs = $this->getAccountContainer()->getAccountModule('posixAccount')->getAttributes();
|
|
$id = $attrs['uid'][0];
|
|
}
|
|
if ($this->get_scope()=='group') {
|
|
$id = $this->getCn();
|
|
}
|
|
// get list of lamdaemon servers
|
|
$servers = array_keys($this->quota);
|
|
for ($q = 0; $q < sizeof($servers); $q++) {
|
|
$server = $servers[$q];
|
|
$i=0;
|
|
$quotastring = "";
|
|
while (isset($this->quota[$server][$i][0])) {
|
|
$quotastring = $quotastring . $this->quota[$server][$i][0] . ',0,0,0,0:';
|
|
$i++;
|
|
}
|
|
$remote = new \LAM\REMOTE\Remote();
|
|
$remoteServer = $_SESSION['config']->getScriptServerByName($server);
|
|
$remote->connect($remoteServer);
|
|
$remote->execute(implode(quota::$SPLIT_DELIMITER, array($id, "quota", "set", $this->get_scope(), "$quotastring\n")));
|
|
$remote->disconnect();
|
|
}
|
|
return array();
|
|
}
|
|
|
|
/**
|
|
* Processes user input of the primary module page.
|
|
* It checks if all input values are correct and updates the associated LDAP attributes.
|
|
*
|
|
* @return array list of info/error messages
|
|
*/
|
|
function process_attributes() {
|
|
if (!isset($this->quota) || !is_array($this->quota)) {
|
|
return array();
|
|
}
|
|
$errors = array();
|
|
// get list of lamdaemon servers
|
|
$servers = array_keys($this->quota);
|
|
for ($q = 0; $q < sizeof($servers); $q++) {
|
|
$server = $servers[$q];
|
|
$id = $this->replaceSpecialChars($server);
|
|
$i=0;
|
|
// loop for every mointpoint with quotas
|
|
while (isset($this->quota[$server][$i][0])) {
|
|
$this->quota[$server][$i][2] = $_POST[$i . '_2_' . $id];
|
|
$this->quota[$server][$i][3] = $_POST[$i . '_3_' . $id];
|
|
$this->quota[$server][$i][6] = $_POST[$i . '_6_' . $id];
|
|
$this->quota[$server][$i][7] = $_POST[$i . '_7_' . $id];
|
|
// Check if values are OK and set automatic values. if not error-variable will be set
|
|
if (!get_preg($this->quota[$server][$i][2], 'quotaNumber')) {
|
|
$errors[] = $this->messages['softblock'][0];
|
|
}
|
|
if (!get_preg($this->quota[$server][$i][3], 'quotaNumber')) {
|
|
$errors[] = $this->messages['hardblock'][0];
|
|
}
|
|
if (!get_preg($this->quota[$server][$i][6], 'quotaNumber')) {
|
|
$errors[] = $this->messages['softinode'][0];
|
|
}
|
|
if (!get_preg($this->quota[$server][$i][7], 'quotaNumber')) {
|
|
$errors[] = $this->messages['hardinode'][0];
|
|
}
|
|
if ($this->getQuotaNumber($this->quota[$server][$i][2]) > $this->getQuotaNumber($this->quota[$server][$i][3])) {
|
|
$errors[] = $this->messages['block_cmp'][0];
|
|
}
|
|
if ($this->getQuotaNumber($this->quota[$server][$i][6]) > $this->getQuotaNumber($this->quota[$server][$i][7])) {
|
|
$errors[] = $this->messages['inode_cmp'][0];
|
|
}
|
|
$i++;
|
|
}
|
|
}
|
|
return $errors;
|
|
}
|
|
|
|
/**
|
|
* Returns the HTML meta data for the main account page.
|
|
*
|
|
* @return htmlElement HTML meta data
|
|
*/
|
|
function display_html_attributes() {
|
|
$return = new htmlResponsiveRow();
|
|
$return->setCSSClasses(array('maxrow'));
|
|
try {
|
|
$this->initQuotas();
|
|
}
|
|
catch (LAMException $e) {
|
|
$return->add(new htmlStatusMessage('ERROR', $e->getTitle(), $e->getMessage()), 12);
|
|
return $return;
|
|
}
|
|
if (!is_array($this->quota)) {
|
|
return $return;
|
|
}
|
|
// get list of lamdaemon servers
|
|
$serverDescriptions = array();
|
|
$lamdaemonServers = $_SESSION['config']->getConfiguredScriptServers();
|
|
for ($s = 0; $s < sizeof($lamdaemonServers); $s++) {
|
|
$lamdaemonServer = $lamdaemonServers[$s];
|
|
$serverDescriptions[$lamdaemonServer->getServer()] = $lamdaemonServer->getLabel();
|
|
}
|
|
$servers = array_keys($this->quota);
|
|
for ($q = 0; $q < sizeof($servers); $q++) {
|
|
$server = $servers[$q];
|
|
$id = $this->replaceSpecialChars($server);
|
|
$title = $server;
|
|
if ($serverDescriptions[$server] != $server) {
|
|
$title = $serverDescriptions[$server] . " (" . $server . ")";
|
|
}
|
|
$return->add(new htmlSubTitle($title), 12);
|
|
|
|
$titles = array(
|
|
_('Mountpoint'), _('Used blocks'), _('Soft block limit'), _('Hard block limit'), _('Grace block period'),
|
|
_('Used inodes'), _('Soft inode limit'), _('Hard inode limit'), _('Grace inode period')
|
|
);
|
|
$data = array(
|
|
array(
|
|
new htmlHelpLink('Mountpoint'), new htmlHelpLink('UsedBlocks'), new htmlHelpLink('SoftBlockLimit'),
|
|
new htmlHelpLink('HardBlockLimit'), new htmlHelpLink('GraceBlockPeriod'), new htmlHelpLink('UsedInodes'),
|
|
new htmlHelpLink('SoftInodeLimit'), new htmlHelpLink('HardInodeLimit'), new htmlHelpLink('GraceInodePeriod')
|
|
)
|
|
);
|
|
|
|
$i=0;
|
|
// loop for every mointpoint with enabled quotas
|
|
while (isset($this->quota[$server][$i][0])) {
|
|
$dataRow = array();
|
|
$dataRow[] = new htmlOutputText($this->quota[$server][$i][0]);
|
|
$dataRow[] = new htmlOutputText($this->quota[$server][$i][1]);
|
|
$sbLimitInput = new htmlInputField($i . '_2_' . $id, $this->quota[$server][$i][2]);
|
|
$sbLimitInput->setFieldMaxLength(20);
|
|
$dataRow[] = $sbLimitInput;
|
|
$hbLimit = new htmlInputField($i . '_3_' . $id, $this->quota[$server][$i][3]);
|
|
$hbLimit->setFieldMaxLength(20);
|
|
$dataRow[] = $hbLimit;
|
|
$dataRow[] = new htmlOutputText($this->quota[$server][$i][4]);
|
|
$dataRow[] = new htmlOutputText($this->quota[$server][$i][5]);
|
|
$siLimit = new htmlInputField($i . '_6_' . $id, $this->quota[$server][$i][6]);
|
|
$siLimit->setFieldMaxLength(20);
|
|
$dataRow[] = $siLimit;
|
|
$hiLimit = new htmlInputField($i . '_7_' . $id, $this->quota[$server][$i][7]);
|
|
$hiLimit->setFieldMaxLength(20);
|
|
$dataRow[] = $hiLimit;
|
|
$dataRow[] = new htmlOutputText($this->quota[$server][$i][8]);
|
|
$data[] = $dataRow;
|
|
$i++;
|
|
}
|
|
$table = new htmlResponsiveTable($titles, $data);
|
|
$table->setWidths(array('20%', '5%', '10%', '10%', '5%', '5%', '10%', '10%', '5%'));
|
|
$return->add($table, 12);
|
|
}
|
|
return $return;
|
|
}
|
|
|
|
/**
|
|
* Replaces special characters in HTML name values.
|
|
*
|
|
* @param string $input input string
|
|
* @return string output string
|
|
*/
|
|
function replaceSpecialChars($input) {
|
|
return str_replace(".", "_", $input);
|
|
}
|
|
|
|
/**
|
|
* {@inheritDoc}
|
|
*/
|
|
function get_profileOptions($typeId) {
|
|
$return = new htmlResponsiveRow();
|
|
$optionsAvailable = false;
|
|
// get list of lamdaemon servers
|
|
$lamdaemonServers = $_SESSION['config']->getConfiguredScriptServers();
|
|
try {
|
|
foreach ($lamdaemonServers as $lamdaemonServer) {
|
|
$server = $lamdaemonServer->getServer();
|
|
$id = $this->replaceSpecialChars($server);
|
|
$description = $lamdaemonServer->getLabel();
|
|
if ($description != $server) {
|
|
$description = $description . ' (' . $server . ')';
|
|
}
|
|
// Get quotas
|
|
$remote = new \LAM\REMOTE\Remote();
|
|
$remote->connect($lamdaemonServer);
|
|
$quotas = $remote->execute(implode(quota::$SPLIT_DELIMITER, array("+", "quota", "get", $this->get_scope())));
|
|
$remote->disconnect();
|
|
if (empty($quotas)) {
|
|
continue;
|
|
}
|
|
$dirs = explode(":", $quotas);
|
|
array_pop($dirs); // remove empty element at the end
|
|
for ($i = 0; $i < sizeof($dirs); $i++) {
|
|
if (strpos($dirs[$i], quota::$QUOTA_PREFIX) !== 0) {
|
|
unset($dirs[$i]);
|
|
$dirs = array_values($dirs);
|
|
$i--;
|
|
continue;
|
|
}
|
|
$dirs[$i] = substr($dirs[$i], strlen(quota::$QUOTA_PREFIX));
|
|
$dirs[$i] = explode(",", $dirs[$i]);
|
|
$dirs[$i] = $dirs[$i][0];
|
|
}
|
|
$dirs = array_values($dirs);
|
|
if (sizeof($dirs) < 1) {
|
|
continue; // stop if no quota directories were found
|
|
}
|
|
$optionsAvailable = true;
|
|
$return->add(new htmlSubTitle($description), 12);
|
|
for ($i = 0; $i < sizeof($dirs); $i++) {
|
|
$return->add(new htmlOutputText($dirs[$i]), 12);
|
|
$sbLimit = new htmlResponsiveInputField(_('Soft block limit'), "quota_softblock_" . $id . "_" . $dirs[$i], null, 'SoftBlockLimit');
|
|
$sbLimit->setFieldMaxLength(20);
|
|
$return->add($sbLimit, 12);
|
|
$hbLimit = new htmlResponsiveInputField(_('Hard block limit'), "quota_hardblock_" . $id . "_" . $dirs[$i], null, 'HardBlockLimit');
|
|
$hbLimit->setFieldMaxLength(20);
|
|
$return->add($hbLimit, 12);
|
|
$siLimit = new htmlResponsiveInputField(_('Soft inode limit'), "quota_softinode_" . $id . "_" . $dirs[$i], null, 'SoftInodeLimit');
|
|
$siLimit->setFieldMaxLength(20);
|
|
$return->add($siLimit, 12);
|
|
$hiLimit = new htmlResponsiveInputField(_('Hard inode limit'), "quota_hardinode_" . $id . "_" . $dirs[$i], null, 'HardInodeLimit');
|
|
$hiLimit->setFieldMaxLength(20);
|
|
$return->add($hiLimit, 12);
|
|
}
|
|
}
|
|
}
|
|
catch (LAMException $e) {
|
|
$return->add(new htmlStatusMessage('WARN', $e->getTitle()), 12);
|
|
return $return;
|
|
}
|
|
if (!$optionsAvailable) {
|
|
return null;
|
|
}
|
|
return $return;
|
|
}
|
|
|
|
/**
|
|
* {@inheritDoc}
|
|
*/
|
|
function check_profileOptions($options, $typeId) {
|
|
$return = array();
|
|
// get list of lamdaemon servers
|
|
$lamdaemonServers = $_SESSION['config']->getConfiguredScriptServers();
|
|
foreach ($lamdaemonServers as $lamdaemonServer) {
|
|
$server = $lamdaemonServer->getServer();
|
|
$id = $this->replaceSpecialChars($server);
|
|
// Get quotas
|
|
$remote = new \LAM\REMOTE\Remote();
|
|
$remote->connect($lamdaemonServer);
|
|
$quotas = $remote->execute(implode(quota::$SPLIT_DELIMITER, array("+", "quota", "get", $this->get_scope())));
|
|
$remote->disconnect();
|
|
$dirs = explode(":", $quotas);
|
|
array_pop($dirs); // remove empty element at the end
|
|
for ($i = 0; $i < sizeof($dirs); $i++) {
|
|
if (strpos($dirs[$i], quota::$QUOTA_PREFIX) !== 0) {
|
|
unset($dirs[$i]);
|
|
$i--;
|
|
continue;
|
|
}
|
|
$dirs[$i] = substr($dirs[$i], strlen(quota::$QUOTA_PREFIX));
|
|
$dirs[$i] = explode(",", $dirs[$i]);
|
|
$dirs[$i] = $dirs[$i][0];
|
|
}
|
|
$dirs = array_values($dirs);
|
|
for ($i = 0; $i < sizeof($dirs); $i++) {
|
|
if (!get_preg($options["quota_softblock_" . $id . "_" . $dirs[$i]][0], 'digit')) {
|
|
$return[] = $this->messages['softblock'][0];
|
|
}
|
|
if (!get_preg($options["quota_hardblock_" . $id . "_" . $dirs[$i]][0], 'digit')) {
|
|
$return[] = $this->messages['hardblock'][0];
|
|
}
|
|
if (!get_preg($options["quota_softinode_" . $id . "_" . $dirs[$i]][0], 'digit')) {
|
|
$return[] = $this->messages['softinode'][0];
|
|
}
|
|
if (!get_preg($options["quota_hardinode_" . $id . "_" . $dirs[$i]][0], 'digit')) {
|
|
$return[] = $this->messages['hardinode'][0];
|
|
}
|
|
if ($this->getQuotaNumber($options["quota_softblock_" . $id . "_" . $dirs[$i]][0]) > $this->getQuotaNumber($options["quota_hardblock_" . $id . "_" . $dirs[$i]][0])) {
|
|
$return[] = $this->messages['block_cmp'][0];
|
|
}
|
|
if ($this->getQuotaNumber($options["quota_softinode_" . $id . "_" . $dirs[$i]][0]) > $this->getQuotaNumber($options["quota_hardinode_" . $id . "_" . $dirs[$i]][0])) {
|
|
$return[] = $this->messages['inode_cmp'][0];
|
|
}
|
|
$i++;
|
|
}
|
|
}
|
|
return $return;
|
|
}
|
|
|
|
/**
|
|
* Loads the values of an account profile into internal variables.
|
|
*
|
|
* @param array $profile hash array with profile values (identifier => value)
|
|
*/
|
|
function load_profile($profile) {
|
|
try {
|
|
$this->initQuotas();
|
|
}
|
|
catch (LAMException $e) {
|
|
logNewMessage(LOG_ERR, $e->getTitle(), $e->getMessage());
|
|
}
|
|
if (!isset($this->quota) || !is_array($this->quota)) {
|
|
return;
|
|
}
|
|
$servers = array_keys($this->quota);
|
|
for ($s = 0; $s < sizeof($servers); $s++) {
|
|
$server = $servers[$s];
|
|
$id = $this->replaceSpecialChars($server);
|
|
for ($i = 0; $i < sizeof($this->quota[$server]); $i++) {
|
|
$dir = $this->quota[$server][$i][0];
|
|
if (isset($profile["quota_softblock_" . $id . "_" . $dir])) {
|
|
$this->quota[$server][$i][2] = $profile["quota_softblock_" . $id . "_" . $dir][0];
|
|
}
|
|
if (isset($profile["quota_hardblock_" . $id . "_" . $dir])) {
|
|
$this->quota[$server][$i][3] = $profile["quota_hardblock_" . $id . "_" . $dir][0];
|
|
}
|
|
if (isset($profile["quota_softinode_" . $id . "_" . $dir])) {
|
|
$this->quota[$server][$i][6] = $profile["quota_softinode_" . $id . "_" . $dir][0];
|
|
}
|
|
if (isset($profile["quota_hardinode_" . $id . "_" . $dir])) {
|
|
$this->quota[$server][$i][7] = $profile["quota_hardinode_" . $id . "_" . $dir][0];
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* {@inheritDoc}
|
|
* @see baseModule::get_pdfEntries()
|
|
*/
|
|
function get_pdfEntries($pdfKeys, $typeId) {
|
|
try {
|
|
$this->initQuotas();
|
|
}
|
|
catch (LAMException $e) {
|
|
logNewMessage(LOG_ERR, $e->getTitle(), $e->getMessage());
|
|
return array();
|
|
}
|
|
if (!isset($this->quota) || !is_array($this->quota)) {
|
|
return array();
|
|
}
|
|
if (sizeof($this->quota) > 0) {
|
|
$pdfTable = new PDFTable();
|
|
// get list of lamdaemon servers
|
|
$lamdaemonServers = $_SESSION['config']->getConfiguredScriptServers();
|
|
foreach ($lamdaemonServers as $lamdaemonServer) {
|
|
$server = $lamdaemonServer->getServer();
|
|
$description = $lamdaemonServer->getLabel();
|
|
if ($description != $server) {
|
|
$description = $description . " (" . $server . ")";
|
|
}
|
|
if (!isset($this->quota[$server]) || (sizeof($this->quota[$server]) < 1)) {
|
|
continue;
|
|
}
|
|
$pdfRow = new PDFTableRow();
|
|
$pdfRow->cells[] = new PDFTableCell($description, null, null, true);
|
|
$pdfTable->rows[] = $pdfRow;
|
|
$pdfRow = new PDFTableRow();
|
|
$pdfRow->cells[] = new PDFTableCell(_('Mountpoint'), '28%', null, true);
|
|
$pdfRow->cells[] = new PDFTableCell(_('Soft block'), '18%', null, true);
|
|
$pdfRow->cells[] = new PDFTableCell(_('Hard block'), '18%', null, true);
|
|
$pdfRow->cells[] = new PDFTableCell(_('Soft inode'), '18%', null, true);
|
|
$pdfRow->cells[] = new PDFTableCell(_('Hard inode'), '18%', null, true);
|
|
$pdfTable->rows[] = $pdfRow;
|
|
for ($i = 0; $i < sizeof($this->quota[$server]); $i++) {
|
|
$pdfRow = new PDFTableRow();
|
|
$pdfRow->cells[] = new PDFTableCell($this->quota[$server][$i][0], '28%');
|
|
$pdfRow->cells[] = new PDFTableCell($this->quota[$server][$i][2], '18%');
|
|
$pdfRow->cells[] = new PDFTableCell($this->quota[$server][$i][3], '18%');
|
|
$pdfRow->cells[] = new PDFTableCell($this->quota[$server][$i][6], '18%');
|
|
$pdfRow->cells[] = new PDFTableCell($this->quota[$server][$i][7], '18%');
|
|
$pdfTable->rows[] = $pdfRow;
|
|
}
|
|
$pdfRow = new PDFTableRow();
|
|
$pdfRow->cells[] = new PDFTableCell(' ');
|
|
$pdfTable->rows[] = $pdfRow;
|
|
}
|
|
$return = array();
|
|
$this->addPDFTable($return, 'quotas', $pdfTable);
|
|
return $return;
|
|
}
|
|
else {
|
|
return array();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* {@inheritDoc}
|
|
* @see baseModule::get_uploadColumns()
|
|
*/
|
|
function get_uploadColumns($selectedModules, &$type) {
|
|
try {
|
|
$this->initQuotas();
|
|
}
|
|
catch (LAMException $e) {
|
|
logNewMessage(LOG_ERR, $e->getTitle(), $e->getMessage());
|
|
return array();
|
|
}
|
|
if (!isset($this->quota) || !is_array($this->quota)) {
|
|
return array();
|
|
}
|
|
$return = array();
|
|
if (sizeof($this->quota) > 0) {
|
|
// get list of lamdaemon servers
|
|
$lamdaemonServers = $_SESSION['config']->getConfiguredScriptServers();
|
|
foreach ($lamdaemonServers as $lamdaemonServer) {
|
|
$server = $lamdaemonServer->getServer();
|
|
// Get quotas
|
|
$remote = new \LAM\REMOTE\Remote();
|
|
$remote->connect($lamdaemonServer);
|
|
$quotas = $remote->execute(implode(quota::$SPLIT_DELIMITER, array("+", "quota", "get", $this->get_scope())));
|
|
$remote->disconnect();
|
|
$dirs = explode(":", $quotas);
|
|
array_pop($dirs); // remove empty element at the end
|
|
for ($i = 0; $i < sizeof($dirs); $i++) {
|
|
if (strpos($dirs[$i], quota::$QUOTA_PREFIX) !== 0) {
|
|
unset($dirs[$i]);
|
|
$i--;
|
|
continue;
|
|
}
|
|
$dirs[$i] = substr($dirs[$i], strlen(quota::$QUOTA_PREFIX));
|
|
$dirs[$i] = explode(",", $dirs[$i]);
|
|
$dirs[$i] = $dirs[$i][0];
|
|
}
|
|
$dirs = array_values($dirs);
|
|
for ($i = 0; $i < sizeof($dirs); $i++) {
|
|
$return[] = array(
|
|
'name' => 'quota_' . $server . ':' . $dirs[$i],
|
|
'description' => sprintf(_('Quota for %s on %s'), $dirs[$i], $server),
|
|
'help' => 'upload',
|
|
'example' => '2000,2500,3000,3500');
|
|
}
|
|
}
|
|
}
|
|
return $return;
|
|
}
|
|
|
|
/**
|
|
* {@inheritDoc}
|
|
* @see baseModule::doUploadPostActions()
|
|
*/
|
|
function doUploadPostActions(&$data, $ids, $failed, &$temp, &$accounts, $selectedModules, $type) {
|
|
$errors = array();
|
|
// first call, get list of user names and quota values
|
|
if (!isset($temp['counter'])) {
|
|
$temp['counter'] = 0;
|
|
// create list of quota columns
|
|
$temp['quotas'] = array();
|
|
$columns = array_keys($ids);
|
|
for ($i = 0; $i < sizeof($columns); $i++) {
|
|
if (strpos($columns[$i], 'quota_') === 0) {
|
|
$temp['quotas'][] = substr($columns[$i], 6);
|
|
}
|
|
}
|
|
// select user/group name depending on current scope
|
|
$temp['accounts'] = array();
|
|
$col = 'invalid';
|
|
if ($this->get_scope() == 'user') {
|
|
$col = $ids['posixAccount_userName'];
|
|
}
|
|
elseif ($this->get_scope() == 'group') {
|
|
$col = $ids['posixGroup_cn'];
|
|
}
|
|
// create list of account names and their quota values
|
|
for ($i = 0; $i < sizeof($data); $i++) {
|
|
if (in_array($i, $failed)) {
|
|
continue; // ignore failed accounts
|
|
}
|
|
$name = $data[$i][$col];
|
|
for ($m = 0; $m < sizeof($temp['quotas']); $m++) {
|
|
if ($data[$i][$ids['quota_' . $temp['quotas'][$m]]] != '') {
|
|
$parts = explode(',', $data[$i][$ids['quota_' . $temp['quotas'][$m]]]);
|
|
// check syntax
|
|
if (sizeof($parts) != 4) {
|
|
$errMsg = $this->messages['upload'][0];
|
|
array_push($errMsg, array($i, 'quota_' . $temp['quotas'][$m]));
|
|
$errors[] = $errMsg;
|
|
continue;
|
|
}
|
|
if (!get_preg($parts[0], 'quotaNumber')) {
|
|
$errMsg = $this->messages['softblock'][1];
|
|
array_push($errMsg, array($i, 'quota_' . $temp['quotas'][$m]));
|
|
$errors[] = $errMsg;
|
|
continue;
|
|
}
|
|
if (!get_preg($parts[1], 'quotaNumber')) {
|
|
$errMsg = $this->messages['hardblock'][1];
|
|
array_push($errMsg, array($i, 'quota_' . $temp['quotas'][$m]));
|
|
$errors[] = $errMsg;
|
|
continue;
|
|
}
|
|
if (!get_preg($parts[2], 'quotaNumber')) {
|
|
$errMsg = $this->messages['softinode'][1];
|
|
array_push($errMsg, array($i, 'quota_' . $temp['quotas'][$m]));
|
|
$errors[] = $errMsg;
|
|
continue;
|
|
}
|
|
if (!get_preg($parts[3], 'quotaNumber')) {
|
|
$errMsg = $this->messages['hardinode'][1];
|
|
array_push($errMsg, array($i, 'quota_' . $temp['quotas'][$m]));
|
|
$errors[] = $errMsg;
|
|
continue;
|
|
}
|
|
if ($this->getQuotaNumber($parts[0]) > $this->getQuotaNumber($parts[1])) {
|
|
$errMsg = $this->messages['block_cmp'][1];
|
|
array_push($errMsg, array($i, 'quota_' . $temp['quotas'][$m]));
|
|
$errors[] = $errMsg;
|
|
continue;
|
|
}
|
|
if ($this->getQuotaNumber($parts[2]) > $this->getQuotaNumber($parts[3])) {
|
|
$errMsg = $this->messages['inode_cmp'][1];
|
|
array_push($errMsg, array($i, 'quota_' . $temp['quotas'][$m]));
|
|
$errors[] = $errMsg;
|
|
continue;
|
|
}
|
|
$parts[0] = $this->getQuotaNumber($parts[0]);
|
|
$parts[1] = $this->getQuotaNumber($parts[1]);
|
|
$parts[2] = $this->getQuotaNumber($parts[2]);
|
|
$parts[3] = $this->getQuotaNumber($parts[3]);
|
|
// save quota settings
|
|
$temp['accounts'][$name][$temp['quotas'][$m]] = $parts;
|
|
}
|
|
}
|
|
}
|
|
return array('status' => 'inProgress', 'progress' => 5, 'errors' => $errors);
|
|
}
|
|
// quotas are ready to set
|
|
elseif ($temp['counter'] < sizeof($temp['accounts'])) {
|
|
$names = array_keys($temp['accounts']);
|
|
$name = $names[$temp['counter']];
|
|
$mountPoints = array_keys($temp['accounts'][$name]);
|
|
// set quota
|
|
for ($m = 0; $m < sizeof($mountPoints); $m++) {
|
|
$mpParts = explode(":", $mountPoints[$m]);
|
|
$server = $mpParts[0];
|
|
$dir = $mpParts[1];
|
|
$quotaString = implode(quota::$SPLIT_DELIMITER, array($name, "quota", "set", $this->get_scope(), $dir . ',' .
|
|
implode(',', $temp['accounts'][$name][$mountPoints[$m]]) . "\n"));
|
|
$remote = new \LAM\REMOTE\Remote();
|
|
$remoteServer = $_SESSION['config']->getScriptServerByName($server);
|
|
$remote->connect($remoteServer);
|
|
$result = $remote->execute($quotaString);
|
|
$remote->disconnect();
|
|
if (!empty($result)) {
|
|
$parts = explode(",", $result);
|
|
if ($parts[0] == 'ERROR') {
|
|
$errors[] = array('ERROR', $parts[1], $parts[2]);
|
|
}
|
|
}
|
|
}
|
|
// set counters to next account/mount point
|
|
$temp['counter']++;
|
|
return array(
|
|
'status' => 'inProgress',
|
|
'progress' => 5 + (95 * ($temp['counter'] / sizeof($temp['accounts']))),
|
|
'errors' => $errors);
|
|
}
|
|
return array('status' => 'finished');
|
|
}
|
|
|
|
/**
|
|
* Returns a list of modifications which have to be made to the LDAP account.
|
|
*
|
|
* Calling this method requires the existence of an enclosing {@link accountContainer}.<br>
|
|
* <br>
|
|
*
|
|
* <br>This function returns an array with 3 entries:
|
|
* <br>array( DN1 ('add' => array($attr), 'remove' => array($attr), 'modify' => array($attr)), DN2 .... )
|
|
* <br>DN is the DN to change. It is possible to change several DNs (e.g. create a new user and add him
|
|
* to some groups via attribute memberUid)<br>
|
|
* <br><b>"add"</b> are attributes which have to be added to the LDAP entry
|
|
* <br><b>"remove"</b> are attributes which have to be removed from the LDAP entry
|
|
* <br><b>"modify"</b> are attributes which have to be modified in the LDAP entry
|
|
* <br><b>"notchanged"</b> are attributes which stay unchanged
|
|
* <br><b>"info"</b> values with informational value (e.g. to be used later by pre/postModify actions)
|
|
* <br>
|
|
* <br>This builds the required comands from $this-attributes and $this->orig.
|
|
*
|
|
* @return array list of modifications
|
|
*/
|
|
public function save_attributes() {
|
|
// no LDAP changes
|
|
return $this->getAccountContainer()->save_module_attributes(array(), array());
|
|
}
|
|
|
|
}
|
|
|
|
?>
|