mirror of
https://git.code.sf.net/p/seeddms/code
synced 2025-10-24 01:41:01 +00:00
1284 lines
36 KiB
PHP
1284 lines
36 KiB
PHP
<?php
|
||
/**
|
||
* Implementation of the attribute object in the document management system
|
||
*
|
||
* @category DMS
|
||
* @package SeedDMS_Core
|
||
* @license GPL 2
|
||
* @version @version@
|
||
* @author Uwe Steinmann <uwe@steinmann.cx>
|
||
* @copyright Copyright (C) 2012 Uwe Steinmann
|
||
* @version Release: @package_version@
|
||
*/
|
||
|
||
/**
|
||
* Class to represent an attribute in the document management system
|
||
*
|
||
* Attributes are key/value pairs which can be attachted to documents,
|
||
* folders and document content. The number of attributes is unlimited.
|
||
* Each attribute has a value and is related to an attribute definition,
|
||
* which holds the name and other information about the attribute.
|
||
*
|
||
* @see SeedDMS_Core_AttributeDefinition
|
||
*
|
||
* @category DMS
|
||
* @package SeedDMS_Core
|
||
* @author Uwe Steinmann <uwe@steinmann.cx>
|
||
* @copyright Copyright (C) 2012-2013 Uwe Steinmann
|
||
* @version Release: @package_version@
|
||
*/
|
||
class SeedDMS_Core_Attribute { /* {{{ */
|
||
/**
|
||
* @var integer id of attribute
|
||
*
|
||
* @access protected
|
||
*/
|
||
protected $_id;
|
||
|
||
/**
|
||
* @var SeedDMS_Core_Folder|SeedDMS_Core_Document|SeedDMS_Core_DocumentContent SeedDMS_Core_Object folder, document or document content
|
||
* this attribute belongs to
|
||
*
|
||
* @access protected
|
||
*/
|
||
protected $_obj;
|
||
|
||
/**
|
||
* @var SeedDMS_Core_AttributeDefinition definition of this attribute
|
||
*
|
||
* @access protected
|
||
*/
|
||
protected $_attrdef;
|
||
|
||
/**
|
||
* @var mixed value of this attribute
|
||
*
|
||
* @access protected
|
||
*/
|
||
protected $_value;
|
||
|
||
/**
|
||
* @var integer validation error
|
||
*
|
||
* @access protected
|
||
*/
|
||
protected $_validation_error;
|
||
|
||
/**
|
||
* @var SeedDMS_Core_DMS reference to the dms instance this attribute belongs to
|
||
*
|
||
* @access protected
|
||
*/
|
||
protected $_dms;
|
||
|
||
/**
|
||
* SeedDMS_Core_Attribute constructor.
|
||
* @param $id
|
||
* @param $obj
|
||
* @param $attrdef
|
||
* @param $value
|
||
*/
|
||
function __construct($id, $obj, $attrdef, $value) { /* {{{ */
|
||
$this->_id = $id;
|
||
$this->_obj = $obj;
|
||
$this->_attrdef = $attrdef;
|
||
$this->_value = $value;
|
||
$this->_validation_error = 0;
|
||
$this->_dms = null;
|
||
} /* }}} */
|
||
|
||
/**
|
||
* Set reference to dms
|
||
*
|
||
* @param SeedDMS_Core_DMS $dms
|
||
*/
|
||
function setDMS($dms) { /* {{{ */
|
||
$this->_dms = $dms;
|
||
} /* }}} */
|
||
|
||
/**
|
||
* Get internal id of attribute
|
||
*
|
||
* @return integer id
|
||
*/
|
||
function getID() { return $this->_id; }
|
||
|
||
/**
|
||
* Return attribute value as stored in database
|
||
*
|
||
* This function will return the value of multi value attributes
|
||
* including the separator char.
|
||
*
|
||
* @return string the attribute value as it is stored in the database.
|
||
*/
|
||
function getValue() { return $this->_value; }
|
||
|
||
/**
|
||
* Return attribute values as an array
|
||
*
|
||
* This function returns the attribute value as an array. The array
|
||
* has one element for non multi value attributes and n elements for
|
||
* multi value attributes.
|
||
*
|
||
* @return array the attribute values
|
||
*/
|
||
function getValueAsArray() { /* {{{ */
|
||
if($this->_attrdef->getMultipleValues()) {
|
||
/* If the value doesn't start with the separator used in the value set,
|
||
* then assume that the value was not saved with a leading separator.
|
||
* This can happen, if the value was previously a single value from
|
||
* the value set and later turned into a multi value attribute.
|
||
*/
|
||
$sep = substr($this->_value, 0, 1);
|
||
if(!($vsep = $this->_attrdef->getValueSetSeparator()))
|
||
$vsep = $sep;
|
||
if($sep == $vsep)
|
||
return(explode($sep, substr($this->_value, 1)));
|
||
else
|
||
return(array($this->_value));
|
||
} else {
|
||
return array($this->_value);
|
||
}
|
||
} /* }}} */
|
||
|
||
/**
|
||
* Set a value of an attribute
|
||
*
|
||
* The attribute is completely deleted if the value is an empty string
|
||
* or empty array. An array of values is only allowed if the attribute may
|
||
* have multiple values. If an array is passed and the attribute may
|
||
* have only a single value, then the first element of the array will
|
||
* be taken.
|
||
*
|
||
* @param string $values value as string or array to be set
|
||
* @return boolean true if operation was successfull, otherwise false
|
||
*/
|
||
function setValue($values) { /* {{{*/
|
||
$db = $this->_dms->getDB();
|
||
|
||
if($this->_attrdef->getMultipleValues()) {
|
||
$valuesetstr = $this->_attrdef->getValueSet();
|
||
/* Multiple values without a value set is not allowed */
|
||
/* No need to have valueset anymore. If none is given, the values are
|
||
* expected to be separated by ','
|
||
if(!$valuesetstr)
|
||
return false;
|
||
*/
|
||
$valueset = $this->_attrdef->getValueSetAsArray();
|
||
|
||
if(is_array($values)) {
|
||
if($values) {
|
||
$vsep = $this->getValueSetSeparator();
|
||
if($valueset) {
|
||
$error = false;
|
||
foreach($values as $v) {
|
||
if(!in_array($v, $valueset)) { $error = true; break; }
|
||
}
|
||
if($error)
|
||
return false;
|
||
$valuesetstr = $this->_attrdef->getValueSet();
|
||
$value = $vsep.implode($vsep, $values);
|
||
} else {
|
||
$value = $vsep.implode($vsep, $values);
|
||
}
|
||
} else {
|
||
$value = '';
|
||
}
|
||
} else {
|
||
if($values) {
|
||
if($valuesetstr) {
|
||
if($valuesetstr[0] != $values[0])
|
||
$values = explode($valuesetstr[0], $values);
|
||
else
|
||
$values = explode($valuesetstr[0], substr($values, 1));
|
||
} else {
|
||
$values = explode(',', substr($values, 1));
|
||
}
|
||
|
||
if($valueset) {
|
||
$error = false;
|
||
foreach($values as $v) {
|
||
if(!in_array($v, $valueset)) { $error = true; break; }
|
||
}
|
||
if($error)
|
||
return false;
|
||
$value = $valuesetstr[0].implode($valuesetstr[0], $values);
|
||
} else {
|
||
$value = ','.implode(',', $values);
|
||
}
|
||
} else {
|
||
$value = $values;
|
||
}
|
||
}
|
||
} else {
|
||
if(is_array($values)) {
|
||
if($values)
|
||
$value = $values[0];
|
||
else
|
||
$value = '';
|
||
} else {
|
||
$value = $values;
|
||
}
|
||
}
|
||
|
||
switch(get_class($this->_obj)) {
|
||
case $this->_dms->getClassname('document'):
|
||
if(trim($value) === '')
|
||
$queryStr = "DELETE FROM `tblDocumentAttributes` WHERE `document` = " . $this->_obj->getID() . " AND `attrdef` = " . $this->_attrdef->getId();
|
||
else
|
||
$queryStr = "UPDATE `tblDocumentAttributes` SET `value` = ".$db->qstr($value)." WHERE `document` = " . $this->_obj->getID() . " AND `attrdef` = " . $this->_attrdef->getId();
|
||
break;
|
||
case $this->_dms->getClassname('documentcontent'):
|
||
if(trim($value) === '')
|
||
$queryStr = "DELETE FROM `tblDocumentContentAttributes` WHERE `content` = " . $this->_obj->getID() . " AND `attrdef` = " . $this->_attrdef->getId();
|
||
else
|
||
$queryStr = "UPDATE `tblDocumentContentAttributes` SET `value` = ".$db->qstr($value)." WHERE `content` = " . $this->_obj->getID() . " AND `attrdef` = " . $this->_attrdef->getId();
|
||
break;
|
||
case $this->_dms->getClassname('folder'):
|
||
if(trim($value) === '')
|
||
$queryStr = "DELETE FROM `tblFolderAttributes WHERE` `folder` = " . $this->_obj->getID() . " AND `attrdef` = " . $this->_attrdef->getId();
|
||
else
|
||
$queryStr = "UPDATE `tblFolderAttributes` SET `value` = ".$db->qstr($value)." WHERE `folder` = " . $this->_obj->getID() . " AND `attrdef` = " . $this->_attrdef->getId();
|
||
break;
|
||
default:
|
||
return false;
|
||
}
|
||
if (!$db->getResult($queryStr))
|
||
return false;
|
||
|
||
$this->_value = $value;
|
||
|
||
return true;
|
||
} /* }}} */
|
||
|
||
/**
|
||
* Validate attribute value
|
||
*
|
||
* This function checks if the attribute values fits the attribute
|
||
* definition.
|
||
* If the validation fails the validation error will be set which
|
||
* can be requested by SeedDMS_Core_Attribute::getValidationError()
|
||
*
|
||
* @return boolean true if validation succeeds, otherwise false
|
||
*/
|
||
function validate() { /* {{{ */
|
||
/** @var SeedDMS_Core_AttributeDefinition $attrdef */
|
||
$attrdef = $this->_attrdef;
|
||
$result = $attrdef->validate($this->_value);
|
||
$this->_validation_error = $attrdef->getValidationError();
|
||
return $result;
|
||
} /* }}} */
|
||
|
||
/**
|
||
* Get validation error from last validation
|
||
*
|
||
* @return integer error code
|
||
*/
|
||
function getValidationError() { return $this->_validation_error; }
|
||
|
||
/**
|
||
* Set validation error
|
||
*
|
||
* @param integer error code
|
||
*/
|
||
function setValidationError($error) { $this->_validation_error = $error; }
|
||
|
||
/**
|
||
* Get definition of attribute
|
||
*
|
||
* @return object attribute definition
|
||
*/
|
||
function getAttributeDefinition() { return $this->_attrdef; }
|
||
|
||
} /* }}} */
|
||
|
||
/**
|
||
* Class to represent an attribute definition in the document management system
|
||
*
|
||
* Attribute definitions specify the name, type, object type, minimum and
|
||
* maximum values and a value set. The object type determines the object
|
||
* an attribute may be attached to. If the object type is set to object_all
|
||
* the attribute can be used for documents, document content and folders.
|
||
*
|
||
* The type of an attribute specifies the skalar data type.
|
||
*
|
||
* Attributes for which multiple values are allowed must have the
|
||
* multiple flag set to true and specify a value set. A value set
|
||
* is a string consisting of n separated values. The separator is the
|
||
* first char of the value set. A possible value could be '|REV-A|REV-B'
|
||
* If multiple values are allowed, then minvalues and maxvalues may
|
||
* restrict the allowed number of values.
|
||
*
|
||
* @see SeedDMS_Core_Attribute
|
||
*
|
||
* @category DMS
|
||
* @package SeedDMS_Core
|
||
* @author Markus Westphal, Malcolm Cowe, Uwe Steinmann <uwe@steinmann.cx>
|
||
* @copyright Copyright (C) 2012 Uwe Steinmann
|
||
* @version Release: @package_version@
|
||
*/
|
||
class SeedDMS_Core_AttributeDefinition { /* {{{ */
|
||
/**
|
||
* @var integer id of attribute definition
|
||
*
|
||
* @access protected
|
||
*/
|
||
protected $_id;
|
||
|
||
/**
|
||
* @var string name of attribute definition
|
||
*
|
||
* @access protected
|
||
*/
|
||
protected $_name;
|
||
|
||
/**
|
||
* @var string object type of attribute definition. This can be one of
|
||
* type_int, type_float, type_string, type_boolean, type_url, or type_email.
|
||
*
|
||
* @access protected
|
||
*/
|
||
protected $_type;
|
||
|
||
/**
|
||
* @var string type of attribute definition. This can be one of objtype_all,
|
||
* objtype_folder, objtype_document, or objtype_documentcontent.
|
||
*
|
||
* @access protected
|
||
*/
|
||
protected $_objtype;
|
||
|
||
/**
|
||
* @var boolean whether an attribute can have multiple values
|
||
*
|
||
* @access protected
|
||
*/
|
||
protected $_multiple;
|
||
|
||
/**
|
||
* @var integer minimum values of an attribute
|
||
*
|
||
* @access protected
|
||
*/
|
||
protected $_minvalues;
|
||
|
||
/**
|
||
* @var integer maximum values of an attribute
|
||
*
|
||
* @access protected
|
||
*/
|
||
protected $_maxvalues;
|
||
|
||
/**
|
||
* @var string list of possible values of an attribute
|
||
*
|
||
* @access protected
|
||
*/
|
||
protected $_valueset;
|
||
|
||
/**
|
||
* @var string regular expression the value must match
|
||
*
|
||
* @access protected
|
||
*/
|
||
protected $_regex;
|
||
|
||
/**
|
||
* @var integer validation error
|
||
*
|
||
* @access protected
|
||
*/
|
||
protected $_validation_error;
|
||
|
||
/**
|
||
* @var SeedDMS_Core_DMS reference to the dms instance this attribute definition belongs to
|
||
*
|
||
* @access protected
|
||
*/
|
||
protected $_dms;
|
||
|
||
/**
|
||
* @var string
|
||
*/
|
||
protected $_separator;
|
||
|
||
/*
|
||
* Possible skalar data types of an attribute
|
||
*/
|
||
const type_int = '1';
|
||
const type_float = '2';
|
||
const type_string = '3';
|
||
const type_boolean = '4';
|
||
const type_url = '5';
|
||
const type_email = '6';
|
||
const type_date = '7';
|
||
|
||
/*
|
||
* Addtional data types of an attribute representing objects in seeddms
|
||
*/
|
||
const type_folder = '101';
|
||
const type_document = '102';
|
||
//const type_documentcontent = '103';
|
||
const type_user = '104';
|
||
const type_group = '105';
|
||
|
||
/*
|
||
* The object type for which a attribute may be used
|
||
*/
|
||
const objtype_all = '0';
|
||
const objtype_folder = '1';
|
||
const objtype_document = '2';
|
||
const objtype_documentcontent = '3';
|
||
|
||
/**
|
||
* Constructor
|
||
*
|
||
* @param integer $id internal id of attribute definition
|
||
* @param string $name name of attribute
|
||
* @param integer $objtype type of object for which this attribute definition
|
||
* may be used.
|
||
* @param integer $type skalar type of attribute
|
||
* @param boolean $multiple set to true if multiple values are allowed
|
||
* @param integer $minvalues minimum number of values
|
||
* @param integer $maxvalues maximum number of values
|
||
* @param string $valueset separated list of allowed values, the first char
|
||
* is taken as the separator
|
||
* @param $regex
|
||
*/
|
||
function __construct($id, $name, $objtype, $type, $multiple, $minvalues, $maxvalues, $valueset, $regex) { /* {{{ */
|
||
$this->_id = $id;
|
||
$this->_name = $name;
|
||
$this->_type = $type;
|
||
$this->_objtype = $objtype;
|
||
$this->_multiple = $multiple;
|
||
$this->_minvalues = $minvalues;
|
||
$this->_maxvalues = $maxvalues;
|
||
$this->_valueset = $valueset;
|
||
$this->_separator = '';
|
||
$this->_regex = $regex;
|
||
$this->_dms = null;
|
||
$this->_validation_error = 0;
|
||
} /* }}} */
|
||
|
||
/**
|
||
* Set reference to dms
|
||
*
|
||
* @param SeedDMS_Core_DMS $dms
|
||
*/
|
||
function setDMS($dms) { /* {{{ */
|
||
$this->_dms = $dms;
|
||
} /* }}} */
|
||
|
||
/**
|
||
* Get internal id of attribute definition
|
||
*
|
||
* @return integer id
|
||
*/
|
||
function getID() { return $this->_id; }
|
||
|
||
/**
|
||
* Get name of attribute definition
|
||
*
|
||
* @return string name
|
||
*/
|
||
function getName() { return $this->_name; }
|
||
|
||
function setName($name) { /* {{{ */
|
||
$db = $this->_dms->getDB();
|
||
|
||
$queryStr = "UPDATE `tblAttributeDefinitions` SET `name` =".$db->qstr($name)." WHERE `id` = " . $this->_id;
|
||
$res = $db->getResult($queryStr);
|
||
if (!$res)
|
||
return false;
|
||
|
||
$this->_name = $name;
|
||
return true;
|
||
} /* }}} */
|
||
|
||
/**
|
||
* Get object type of attribute definition
|
||
*
|
||
* This can be one of objtype_all,
|
||
* objtype_folder, objtype_document, or objtype_documentcontent.
|
||
*
|
||
* @return integer type
|
||
*/
|
||
function getObjType() { return $this->_objtype; }
|
||
|
||
/**
|
||
* Set object type of attribute definition
|
||
*
|
||
* This can be one of objtype_all,
|
||
* objtype_folder, objtype_document, or objtype_documentcontent.
|
||
*
|
||
* @param integer $objtype type
|
||
* @return bool
|
||
*/
|
||
function setObjType($objtype) { /* {{{ */
|
||
$db = $this->_dms->getDB();
|
||
|
||
$queryStr = "UPDATE `tblAttributeDefinitions` SET `objtype` =".intval($objtype)." WHERE `id` = " . $this->_id;
|
||
$res = $db->getResult($queryStr);
|
||
if (!$res)
|
||
return false;
|
||
|
||
$this->_objtype = $objtype;
|
||
return true;
|
||
} /* }}} */
|
||
|
||
/**
|
||
* Get type of attribute definition
|
||
*
|
||
* This can be one of type_int, type_float, type_string, type_boolean,
|
||
* type_url, type_email.
|
||
*
|
||
* @return integer type
|
||
*/
|
||
function getType() { return $this->_type; }
|
||
|
||
/**
|
||
* Set type of attribute definition
|
||
*
|
||
* This can be one of type_int, type_float, type_string, type_boolean,
|
||
* type_url, type_email.
|
||
*
|
||
* @param integer $type type
|
||
* @return bool
|
||
*/
|
||
function setType($type) { /* {{{ */
|
||
$db = $this->_dms->getDB();
|
||
|
||
$queryStr = "UPDATE `tblAttributeDefinitions` SET `type` =".intval($type)." WHERE `id` = " . $this->_id;
|
||
$res = $db->getResult($queryStr);
|
||
if (!$res)
|
||
return false;
|
||
|
||
$this->_type = $type;
|
||
return true;
|
||
} /* }}} */
|
||
|
||
/**
|
||
* Check if attribute definition allows multi values for attribute
|
||
*
|
||
* @return boolean true if attribute may have multiple values
|
||
*/
|
||
function getMultipleValues() { return $this->_multiple; }
|
||
|
||
/**
|
||
* Set if attribute definition allows multi values for attribute
|
||
*
|
||
* @param boolean $mv true if attribute may have multiple values, otherwise
|
||
* false
|
||
* @return bool
|
||
*/
|
||
function setMultipleValues($mv) { /* {{{ */
|
||
$db = $this->_dms->getDB();
|
||
|
||
$queryStr = "UPDATE `tblAttributeDefinitions` SET `multiple` =".intval($mv)." WHERE `id` = " . $this->_id;
|
||
$res = $db->getResult($queryStr);
|
||
if (!$res)
|
||
return false;
|
||
|
||
$this->_multiple = $mv;
|
||
return true;
|
||
} /* }}} */
|
||
|
||
/**
|
||
* Return minimum number of values for attributes
|
||
*
|
||
* Attributes with multiple values may be limited to a range
|
||
* of values. This functions returns the minimum number of values.
|
||
*
|
||
* @return integer minimum number of values
|
||
*/
|
||
function getMinValues() { return $this->_minvalues; }
|
||
|
||
function setMinValues($minvalues) { /* {{{ */
|
||
$db = $this->_dms->getDB();
|
||
|
||
$queryStr = "UPDATE `tblAttributeDefinitions` SET `minvalues` =".intval($minvalues)." WHERE `id` = " . $this->_id;
|
||
$res = $db->getResult($queryStr);
|
||
if (!$res)
|
||
return false;
|
||
|
||
$this->_minvalues = $minvalues;
|
||
return true;
|
||
} /* }}} */
|
||
|
||
/**
|
||
* Return maximum number of values for attributes
|
||
*
|
||
* Attributes with multiple values may be limited to a range
|
||
* of values. This functions returns the maximum number of values.
|
||
*
|
||
* @return integer maximum number of values
|
||
*/
|
||
function getMaxValues() { return $this->_maxvalues; }
|
||
|
||
function setMaxValues($maxvalues) { /* {{{ */
|
||
$db = $this->_dms->getDB();
|
||
|
||
$queryStr = "UPDATE `tblAttributeDefinitions` SET `maxvalues` =".intval($maxvalues)." WHERE `id` = " . $this->_id;
|
||
$res = $db->getResult($queryStr);
|
||
if (!$res)
|
||
return false;
|
||
|
||
$this->_maxvalues = $maxvalues;
|
||
return true;
|
||
} /* }}} */
|
||
|
||
/**
|
||
* Get the value set as saved in the database
|
||
*
|
||
* This is a string containing the list of valueѕ separated by a
|
||
* delimiter which also precedes the whole string, e.g. '|Yes|No'
|
||
*
|
||
* Use {@link SeedDMS_Core_AttributeDefinition::getValueSetAsArray()}
|
||
* for a list of values returned as an array.
|
||
*
|
||
* @return string value set
|
||
*/
|
||
function getValueSet() { /* {{{ */
|
||
return $this->_valueset;
|
||
} /* }}} */
|
||
|
||
/**
|
||
* Get the separator used for the value set
|
||
*
|
||
* This is the first char of the value set string.
|
||
*
|
||
* @return string separator or an empty string if a value set is not set
|
||
*/
|
||
function getValueSetSeparator() { /* {{{ */
|
||
if(strlen($this->_valueset) > 1) {
|
||
return $this->_valueset[0];
|
||
} elseif($this->_multiple) {
|
||
if($this->_type == SeedDMS_Core_AttributeDefinition::type_user || $this->_type == SeedDMS_Core_AttributeDefinition::type_group)
|
||
return ',';
|
||
else
|
||
return '';
|
||
} else {
|
||
return '';
|
||
}
|
||
} /* }}} */
|
||
|
||
/**
|
||
* Get the whole value set as an array
|
||
*
|
||
* @return array values of value set or false if the value set has
|
||
* less than 2 chars
|
||
*/
|
||
function getValueSetAsArray() { /* {{{ */
|
||
if(strlen($this->_valueset) > 1)
|
||
return array_map('trim', explode($this->_valueset[0], substr($this->_valueset, 1)));
|
||
else
|
||
return array();
|
||
} /* }}} */
|
||
|
||
/**
|
||
* Get the n'th value of a value set
|
||
*
|
||
* @param $ind
|
||
* @return string n'th value of value set or false if the index is
|
||
* out of range or the value set has less than 2 chars
|
||
* @internal param int $index
|
||
*/
|
||
function getValueSetValue($ind) { /* {{{ */
|
||
if(strlen($this->_valueset) > 1) {
|
||
$tmp = explode($this->_valueset[0], substr($this->_valueset, 1));
|
||
if(isset($tmp[$ind]))
|
||
return trim($tmp[$ind]);
|
||
else
|
||
return false;
|
||
} else
|
||
return false;
|
||
} /* }}} */
|
||
|
||
/**
|
||
* Set the value set
|
||
*
|
||
* A value set is a list of values allowed for an attribute. The values
|
||
* are separated by a char which must also be the first char of the
|
||
* value set string.
|
||
*
|
||
* @param string $valueset
|
||
* @return boolean true if value set could be set, otherwise false
|
||
*/
|
||
function setValueSet($valueset) { /* {{{ */
|
||
/*
|
||
$tmp = array();
|
||
foreach($valueset as $value) {
|
||
$tmp[] = str_replace('"', '""', $value);
|
||
}
|
||
$valuesetstr = implode(",", $tmp);
|
||
*/
|
||
if(trim($valueset)) {
|
||
$valuesetarr = array_map('trim', explode($valueset[0], substr($valueset, 1)));
|
||
$valuesetstr = $valueset[0].implode($valueset[0], $valuesetarr);
|
||
} else {
|
||
$valuesetstr = '';
|
||
}
|
||
|
||
$db = $this->_dms->getDB();
|
||
|
||
$queryStr = "UPDATE `tblAttributeDefinitions` SET `valueset` =".$db->qstr($valuesetstr)." WHERE `id` = " . $this->_id;
|
||
$res = $db->getResult($queryStr);
|
||
if (!$res)
|
||
return false;
|
||
|
||
$this->_valueset = $valueset;
|
||
$this->_separator = substr($valueset, 0, 1);
|
||
return true;
|
||
} /* }}} */
|
||
|
||
/**
|
||
* Get the regular expression as saved in the database
|
||
*
|
||
* @return string regular expression
|
||
*/
|
||
function getRegex() { /* {{{ */
|
||
return $this->_regex;
|
||
} /* }}} */
|
||
|
||
/**
|
||
* Set the regular expression
|
||
*
|
||
* A value of the attribute must match this regular expression.
|
||
*
|
||
* @param string $regex
|
||
* @return boolean true if regex could be set, otherwise false
|
||
*/
|
||
function setRegex($regex) { /* {{{ */
|
||
$db = $this->_dms->getDB();
|
||
|
||
$queryStr = "UPDATE `tblAttributeDefinitions` SET `regex` =".$db->qstr($regex)." WHERE `id` = " . $this->_id;
|
||
$res = $db->getResult($queryStr);
|
||
if (!$res)
|
||
return false;
|
||
|
||
$this->_regex = $regex;
|
||
return true;
|
||
} /* }}} */
|
||
|
||
/**
|
||
* Check if the attribute definition is used
|
||
*
|
||
* Checks all documents, folders and document content whether at least
|
||
* one of them referenceѕ this attribute definition
|
||
*
|
||
* @return boolean true if attribute definition is used, otherwise false
|
||
*/
|
||
function isUsed() { /* {{{ */
|
||
$db = $this->_dms->getDB();
|
||
|
||
$queryStr = "SELECT * FROM `tblDocumentAttributes` WHERE `attrdef`=".$this->_id;
|
||
$resArr = $db->getResultArray($queryStr);
|
||
if (is_array($resArr) && count($resArr) == 0) {
|
||
$queryStr = "SELECT * FROM `tblFolderAttributes` WHERE `attrdef`=".$this->_id;
|
||
$resArr = $db->getResultArray($queryStr);
|
||
if (is_array($resArr) && count($resArr) == 0) {
|
||
$queryStr = "SELECT * FROM `tblDocumentContentAttributes` WHERE `attrdef`=".$this->_id;
|
||
$resArr = $db->getResultArray($queryStr);
|
||
if (is_array($resArr) && count($resArr) == 0) {
|
||
|
||
return false;
|
||
}
|
||
}
|
||
}
|
||
return true;
|
||
} /* }}} */
|
||
|
||
/**
|
||
* Parse a given value according to attribute definition
|
||
*
|
||
* The return value is always an array, even if the attribute is single
|
||
* value attribute.
|
||
*
|
||
* @param $value
|
||
* @return array|bool
|
||
*/
|
||
function parseValue($value) { /* {{{ */
|
||
if($this->getMultipleValues()) {
|
||
/* If the value doesn't start with the separator used in the value set,
|
||
* then assume that the value was not saved with a leading separator.
|
||
* This can happen, if the value was previously a single value from
|
||
* the value set and later turned into a multi value attribute.
|
||
*/
|
||
$sep = substr($value, 0, 1);
|
||
$vsep = $this->getValueSetSeparator();
|
||
if($sep == $vsep)
|
||
$values = explode($sep, substr($value, 1));
|
||
else
|
||
$values = array($value);
|
||
} else {
|
||
$values = array($value);
|
||
}
|
||
|
||
switch((string) $this->getType()) {
|
||
case self::type_document:
|
||
foreach($values as $value) {
|
||
if($u = $this->_dms->getDocument((int) $value))
|
||
$tmp[] = $u->getName();
|
||
else
|
||
$tmp[] = '???';
|
||
}
|
||
$values = $tmp;
|
||
break;
|
||
case self::type_folder:
|
||
foreach($values as $value) {
|
||
if($u = $this->_dms->getFolder((int) $value))
|
||
$tmp[] = $u->getName();
|
||
else
|
||
$tmp[] = '???';
|
||
}
|
||
$values = $tmp;
|
||
break;
|
||
case self::type_user:
|
||
foreach($values as $value) {
|
||
if($u = $this->_dms->getUser((int) $value))
|
||
$tmp[] = $u->getLogin();
|
||
else
|
||
$tmp[] = '???';
|
||
}
|
||
$values = $tmp;
|
||
break;
|
||
case self::type_group:
|
||
foreach($values as $value) {
|
||
if($u = $this->_dms->getGroup((int) $value))
|
||
$tmp[] = $u->getName();
|
||
else
|
||
$tmp[] = '???';
|
||
}
|
||
$values = $tmp;
|
||
break;
|
||
}
|
||
return $values;
|
||
} /* }}} */
|
||
|
||
/**
|
||
* Return a list of documents, folders, document contents where this
|
||
* attribute definition is used
|
||
*
|
||
* @param integer $limit return not more the n objects of each type
|
||
* @return array|bool
|
||
*/
|
||
function getStatistics($limit=0) { /* {{{ */
|
||
$db = $this->_dms->getDB();
|
||
|
||
$result = array('docs'=>array(), 'folders'=>array(), 'contents'=>array());
|
||
if($this->_objtype == SeedDMS_Core_AttributeDefinition::objtype_all ||
|
||
$this->_objtype == SeedDMS_Core_AttributeDefinition::objtype_document) {
|
||
$queryStr = "SELECT * FROM `tblDocumentAttributes` WHERE `attrdef`=".$this->_id;
|
||
if($limit)
|
||
$queryStr .= " limit ".(int) $limit;
|
||
$resArr = $db->getResultArray($queryStr);
|
||
if($resArr) {
|
||
foreach($resArr as $rec) {
|
||
if($doc = $this->_dms->getDocument($rec['document'])) {
|
||
$result['docs'][] = $doc;
|
||
}
|
||
}
|
||
}
|
||
$queryStr = "SELECT count(*) c, `value` FROM `tblDocumentAttributes` WHERE `attrdef`=".$this->_id." GROUP BY `value` ORDER BY c DESC";
|
||
$resArr = $db->getResultArray($queryStr);
|
||
if($resArr) {
|
||
foreach($resArr as $row) {
|
||
$tmpattr = new SeedDMS_Core_Attribute(0, null, $this, $row['value']);
|
||
foreach($tmpattr->getValueAsArray() as $value) {
|
||
if(isset($possiblevalues[md5($value)])) {
|
||
$possiblevalues[md5($value)]['c'] += $row['c'];
|
||
} else {
|
||
$possiblevalues[md5($value)] = array('value'=>$value, 'c'=>$row['c']);
|
||
}
|
||
}
|
||
}
|
||
$result['frequencies']['document'] = $possiblevalues;
|
||
}
|
||
}
|
||
|
||
if($this->_objtype == SeedDMS_Core_AttributeDefinition::objtype_all ||
|
||
$this->_objtype == SeedDMS_Core_AttributeDefinition::objtype_folder) {
|
||
$queryStr = "SELECT * FROM `tblFolderAttributes` WHERE `attrdef`=".$this->_id;
|
||
if($limit)
|
||
$queryStr .= " limit ".(int) $limit;
|
||
$resArr = $db->getResultArray($queryStr);
|
||
if($resArr) {
|
||
foreach($resArr as $rec) {
|
||
if($folder = $this->_dms->getFolder($rec['folder'])) {
|
||
$result['folders'][] = $folder;
|
||
}
|
||
}
|
||
}
|
||
$valueset = $this->getValueSetAsArray();
|
||
$possiblevalues = array();
|
||
foreach($valueset as $value) {
|
||
$possiblevalues[md5($value)] = array('value'=>$value, 'c'=>0);
|
||
}
|
||
$queryStr = "SELECT count(*) c, `value` FROM `tblFolderAttributes` WHERE `attrdef`=".$this->_id." GROUP BY `value` ORDER BY c DESC";
|
||
$resArr = $db->getResultArray($queryStr);
|
||
if($resArr) {
|
||
foreach($resArr as $row) {
|
||
$tmpattr = new SeedDMS_Core_Attribute(0, null, $this, $row['value']);
|
||
foreach($tmpattr->getValueAsArray() as $value) {
|
||
if(isset($possiblevalues[md5($value)])) {
|
||
$possiblevalues[md5($value)]['c'] += $row['c'];
|
||
} else {
|
||
$possiblevalues[md5($value)] = array('value'=>$value, 'c'=>$row['c']);
|
||
}
|
||
}
|
||
}
|
||
$result['frequencies']['folder'] = $possiblevalues;
|
||
}
|
||
}
|
||
|
||
if($this->_objtype == SeedDMS_Core_AttributeDefinition::objtype_all ||
|
||
$this->_objtype == SeedDMS_Core_AttributeDefinition::objtype_documentcontent) {
|
||
$queryStr = "SELECT * FROM `tblDocumentContentAttributes` WHERE `attrdef`=".$this->_id;
|
||
if($limit)
|
||
$queryStr .= " limit ".(int) $limit;
|
||
$resArr = $db->getResultArray($queryStr);
|
||
if($resArr) {
|
||
foreach($resArr as $rec) {
|
||
if($content = $this->_dms->getDocumentContent($rec['content'])) {
|
||
$result['contents'][] = $content;
|
||
}
|
||
}
|
||
}
|
||
$valueset = $this->getValueSetAsArray();
|
||
$possiblevalues = array();
|
||
foreach($valueset as $value) {
|
||
$possiblevalues[md5($value)] = array('value'=>$value, 'c'=>0);
|
||
}
|
||
$queryStr = "SELECT count(*) c, `value` FROM `tblDocumentContentAttributes` WHERE `attrdef`=".$this->_id." GROUP BY `value` ORDER BY c DESC";
|
||
$resArr = $db->getResultArray($queryStr);
|
||
if($resArr) {
|
||
foreach($resArr as $row) {
|
||
$tmpattr = new SeedDMS_Core_Attribute(0, null, $this, $row['value']);
|
||
foreach($tmpattr->getValueAsArray() as $value) {
|
||
if(isset($possiblevalues[md5($value)])) {
|
||
$possiblevalues[md5($value)]['c'] += $row['c'];
|
||
} else {
|
||
$possiblevalues[md5($value)] = array('value'=>$value, 'c'=>$row['c']);
|
||
}
|
||
}
|
||
}
|
||
$result['frequencies']['content'] = $possiblevalues;
|
||
}
|
||
}
|
||
|
||
return $result;
|
||
} /* }}} */
|
||
|
||
/**
|
||
* Remove the attribute definition
|
||
* Removal is only executed when the definition is not used anymore.
|
||
*
|
||
* @return boolean true on success or false in case of an error
|
||
*/
|
||
function remove() { /* {{{ */
|
||
$db = $this->_dms->getDB();
|
||
|
||
if($this->isUsed())
|
||
return false;
|
||
|
||
// Delete user itself
|
||
$queryStr = "DELETE FROM `tblAttributeDefinitions` WHERE `id` = " . $this->_id;
|
||
if (!$db->getResult($queryStr)) return false;
|
||
|
||
return true;
|
||
} /* }}} */
|
||
|
||
/**
|
||
* Get all documents and folders by a given attribute value
|
||
*
|
||
* @param string $attrvalue value of attribute
|
||
* @param integer $limit limit number of documents/folders
|
||
* @return array array containing list of documents and folders
|
||
*/
|
||
public function getObjects($attrvalue, $limit='') { /* {{{ */
|
||
$db = $this->_dms->getDB();
|
||
|
||
$result = array('docs'=>array(), 'folders'=>array(), 'contents'=>array());
|
||
if($this->_objtype == SeedDMS_Core_AttributeDefinition::objtype_all ||
|
||
$this->_objtype == SeedDMS_Core_AttributeDefinition::objtype_document) {
|
||
$queryStr = "SELECT * FROM `tblDocumentAttributes` WHERE `attrdef`=".$this->_id." AND ";
|
||
if($this->getMultipleValues()) {
|
||
$sep = $this->getValueSetSeparator();
|
||
$queryStr .= "(`value` like ".$db->qstr($sep.$attrvalue.'%')." OR `value` like ".$db->qstr('%'.$sep.$attrvalue.$sep.'%')." OR `value` like ".$db->qstr('%'.$sep.$attrvalue).")";
|
||
} else {
|
||
$queryStr .= "`value`=".$db->qstr($attrvalue);
|
||
}
|
||
if($limit)
|
||
$queryStr .= " limit ".(int) $limit;
|
||
$resArr = $db->getResultArray($queryStr);
|
||
if($resArr) {
|
||
foreach($resArr as $rec) {
|
||
if($doc = $this->_dms->getDocument($rec['document'])) {
|
||
$result['docs'][] = $doc;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
if($this->_objtype == SeedDMS_Core_AttributeDefinition::objtype_all ||
|
||
$this->_objtype == SeedDMS_Core_AttributeDefinition::objtype_folder) {
|
||
$queryStr = "SELECT * FROM `tblFolderAttributes` WHERE `attrdef`=".$this->_id." AND ";
|
||
if($this->getMultipleValues()) {
|
||
$sep = $this->getValueSetSeparator();
|
||
$queryStr .= "(`value` like ".$db->qstr($sep.$attrvalue.'%')." OR `value` like ".$db->qstr('%'.$sep.$attrvalue.$sep.'%')." OR `value` like ".$db->qstr('%'.$sep.$attrvalue).")";
|
||
} else {
|
||
$queryStr .= "`value`=".$db->qstr($attrvalue);
|
||
}
|
||
if($limit)
|
||
$queryStr .= " limit ".(int) $limit;
|
||
$resArr = $db->getResultArray($queryStr);
|
||
if($resArr) {
|
||
foreach($resArr as $rec) {
|
||
if($folder = $this->_dms->getFolder($rec['folder'])) {
|
||
$result['folders'][] = $folder;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
return $result;
|
||
} /* }}} */
|
||
|
||
/**
|
||
* Remove a given attribute value from all documents, versions and folders
|
||
*
|
||
* @param string $attrvalue value of attribute
|
||
* @return array array containing list of documents and folders
|
||
*/
|
||
public function removeValue($attrvalue) { /* {{{ */
|
||
$db = $this->_dms->getDB();
|
||
|
||
foreach(array('document', 'documentcontent', 'folder') as $type) {
|
||
if($type == 'document') {
|
||
$tablename = "tblDocumentAttributes";
|
||
$objtype = SeedDMS_Core_AttributeDefinition::objtype_document;
|
||
} elseif($type == 'documentcontent') {
|
||
$tablename = "tblDocumentContentAttributes";
|
||
$objtype = SeedDMS_Core_AttributeDefinition::objtype_documentcontent;
|
||
} elseif($type == 'folder') {
|
||
$tablename = "tblFolderAttributes";
|
||
$objtype = SeedDMS_Core_AttributeDefinition::objtype_folder;
|
||
}
|
||
if($this->_objtype == SeedDMS_Core_AttributeDefinition::objtype_all || $objtype) {
|
||
$queryStr = "SELECT * FROM `".$tablename."` WHERE `attrdef`=".$this->_id." AND ";
|
||
if($this->getMultipleValues()) {
|
||
$sep = $this->getValueSetSeparator();
|
||
$queryStr .= "(`value` like ".$db->qstr($sep.$attrvalue.'%')." OR `value` like ".$db->qstr('%'.$sep.$attrvalue.$sep.'%')." OR `value` like ".$db->qstr('%'.$sep.$attrvalue).")";
|
||
} else {
|
||
$queryStr .= "`value`=".$db->qstr($attrvalue);
|
||
}
|
||
|
||
$resArr = $db->getResultArray($queryStr);
|
||
if($resArr) {
|
||
$db->startTransaction();
|
||
foreach($resArr as $rec) {
|
||
if($rec['value'] == $attrvalue) {
|
||
$queryStr = "DELETE FROM `".$tablename."` WHERE `id`=".$rec['id'];
|
||
} else {
|
||
if($this->getMultipleValues()) {
|
||
$sep = substr($rec['value'], 0, 1);
|
||
$vsep = $this->getValueSetSeparator();
|
||
if($sep == $vsep)
|
||
$values = explode($sep, substr($rec['value'], 1));
|
||
else
|
||
$values = array($rec['value']);
|
||
if (($key = array_search($attrvalue, $values)) !== false) {
|
||
unset($values[$key]);
|
||
}
|
||
if($values) {
|
||
$queryStr = "UPDATE `".$tablename."` SET `value`=".$db->qstr($sep.implode($sep, $values))." WHERE `id`=".$rec['id'];
|
||
} else {
|
||
$queryStr = "DELETE FROM `".$tablename."` WHERE `id`=".$rec['id'];
|
||
}
|
||
} else {
|
||
}
|
||
}
|
||
if (!$db->getResult($queryStr)) {
|
||
$db->rollbackTransaction();
|
||
return false;
|
||
}
|
||
}
|
||
$db->commitTransaction();
|
||
}
|
||
}
|
||
}
|
||
return true;
|
||
} /* }}} */
|
||
|
||
/**
|
||
* Validate value against attribute definition
|
||
*
|
||
* This function checks if the given value fits the attribute
|
||
* definition.
|
||
* If the validation fails the validation error will be set which
|
||
* can be requested by SeedDMS_Core_Attribute::getValidationError()
|
||
*
|
||
* @param string|array $attrvalue attribute value
|
||
* @return boolean true if validation succeds, otherwise false
|
||
*/
|
||
function validate($attrvalue) { /* {{{ */
|
||
/* Check if 'onAttributeValidate' callback is set */
|
||
if(isset($this->_dms->callbacks['onAttributeValidate'])) {
|
||
foreach($this->_dms->callbacks['onAttributeValidate'] as $callback) {
|
||
$ret = call_user_func($callback[0], $callback[1], $this);
|
||
if(is_bool($ret))
|
||
return $ret;
|
||
}
|
||
}
|
||
|
||
/* Turn $attrvalue into an array of values. Checks if $attrvalue starts
|
||
* with a separator char as set in the value set and uses it to explode
|
||
* the $attrvalue. If the separator doesn't match or this attribute
|
||
* definition doesn't have a value set, then just create a one element
|
||
* array. if $attrvalue is empty, then create an empty array.
|
||
*/
|
||
if($this->getMultipleValues()) {
|
||
if(is_string($attrvalue)) {
|
||
$sep = $attrvalue[0];
|
||
$vsep = $this->getValueSetSeparator();
|
||
if($sep == $vsep)
|
||
$values = explode($attrvalue[0], substr($attrvalue, 1));
|
||
else
|
||
$values = array($attrvalue);
|
||
} else
|
||
$values = $attrvalue;
|
||
} elseif($attrvalue) {
|
||
$values = array($attrvalue);
|
||
} else {
|
||
$values = array();
|
||
}
|
||
|
||
/* Check if attribute value has at least the minimum number of values */
|
||
$this->_validation_error = 0;
|
||
if($this->getMinValues() > count($values)) {
|
||
$this->_validation_error = 1;
|
||
return false;
|
||
}
|
||
/* Check if attribute value has not more than maximum number of values */
|
||
if($this->getMaxValues() && $this->getMaxValues() < count($values)) {
|
||
$this->_validation_error = 2;
|
||
return false;
|
||
}
|
||
|
||
$success = true;
|
||
switch((string) $this->getType()) {
|
||
case self::type_boolean:
|
||
foreach($values as $value) {
|
||
$success &= preg_match('/^[01]$/', $value) ? true : false;
|
||
}
|
||
if(!$success)
|
||
$this->_validation_error = 8;
|
||
break;
|
||
case self::type_int:
|
||
foreach($values as $value) {
|
||
$success &= preg_match('/^[0-9]*$/', $value) ? true : false;
|
||
}
|
||
if(!$success)
|
||
$this->_validation_error = 6;
|
||
break;
|
||
case self::type_date:
|
||
foreach($values as $value) {
|
||
$success &= preg_match('/^[12][0-9]{3}-[01][0-9]-[0-9]{2}$/', $value) ? true : false;
|
||
}
|
||
if(!$success)
|
||
$this->_validation_error = 9;
|
||
break;
|
||
case self::type_float:
|
||
foreach($values as $value) {
|
||
$success &= is_numeric($value);
|
||
}
|
||
if(!$success)
|
||
$this->_validation_error = 7;
|
||
break;
|
||
case self::type_string:
|
||
if(trim($this->getRegex()) != '') {
|
||
foreach($values as $value) {
|
||
$success &= preg_match($this->getRegex(), $value) ? true : false;
|
||
}
|
||
}
|
||
if(!$success)
|
||
$this->_validation_error = 3;
|
||
break;
|
||
case self::type_email:
|
||
foreach($values as $value) {
|
||
$success &= preg_match('/^[a-z0-9._-]+@+[a-z0-9._-]+\.+[a-z]{2,4}$/i', $value);
|
||
}
|
||
if(!$success)
|
||
$this->_validation_error = 5;
|
||
break;
|
||
case self::type_url:
|
||
foreach($values as $value) {
|
||
$success &= preg_match('/^http(s)?:\/\/[a-z0-9-]+(.[a-z0-9-]+)*(:[0-9]+)?(\/.*)?$/i', $value);
|
||
}
|
||
if(!$success)
|
||
$this->_validation_error = 4;
|
||
break;
|
||
case self::type_document:
|
||
foreach($values as $value) {
|
||
$success = true;
|
||
if(!$this->_dms->getDocument((int) $value))
|
||
$success = false;
|
||
}
|
||
if(!$success)
|
||
$this->_validation_error = 10;
|
||
break;
|
||
case self::type_folder:
|
||
foreach($values as $value) {
|
||
$success = true;
|
||
if(!$this->_dms->getFolder((int) $value))
|
||
$success = false;
|
||
}
|
||
if(!$success)
|
||
$this->_validation_error = 11;
|
||
break;
|
||
case self::type_user:
|
||
foreach($values as $value) {
|
||
$success = true;
|
||
if(!$this->_dms->getUser((int) $value))
|
||
$success = false;
|
||
}
|
||
if(!$success)
|
||
$this->_validation_error = 12;
|
||
break;
|
||
case self::type_group:
|
||
foreach($values as $value) {
|
||
$success = true;
|
||
if(!$this->_dms->getGroup((int) $value))
|
||
$success = false;
|
||
}
|
||
if(!$success)
|
||
$this->_validation_error = 13;
|
||
break;
|
||
}
|
||
|
||
if(!$success)
|
||
return $success;
|
||
|
||
/* Check if value is in value set */
|
||
if($valueset = $this->getValueSetAsArray()) {
|
||
foreach($values as $value) {
|
||
if(!in_array($value, $valueset)) {
|
||
$success = false;
|
||
$this->_validation_error = 10;
|
||
}
|
||
}
|
||
}
|
||
|
||
return $success;
|
||
|
||
} /* }}} */
|
||
|
||
/**
|
||
* Get validation error from last validation
|
||
*
|
||
* @return integer error code
|
||
*/
|
||
function getValidationError() { return $this->_validation_error; }
|
||
|
||
} /* }}} */
|