*
  • Concrete implementations need to initialize Repository::$queryColumns
  • *
  • The datasource passed to a repository must implement the Selectable interface
  • *
  • The datasource must yield an instance of Queryable when its select() method is called
  • * */ abstract class Repository implements Selectable { /** * The format to use when converting values of type date_time */ const DATETIME_FORMAT = 'd/m/Y g:i A'; /** * The name of this repository * * @var string */ protected $name; /** * The datasource being used * * @var Selectable */ protected $ds; /** * The base table name this repository is responsible for * * This will be automatically set to the first key of $queryColumns if not explicitly set. * * @var string */ protected $baseTable; /** * The query columns being provided * * This must be initialized by concrete repository implementations, in the following format *
    
         *  array(
         *      'baseTable' => array(
         *          'column1',
         *          'alias1' => 'column2',
         *          'alias2' => 'column3'
         *      )
         *  )
         * 
    
         *
         * @var array
         */
        protected $queryColumns;
    
        /**
         * The columns (or aliases) which are not permitted to be queried. (by design)
         *
         * @var array   An array of strings
         */
        protected $filterColumns;
    
        /**
         * The default sort rules to be applied on a query
         *
         * This may be initialized by concrete repository implementations, in the following format
         * 
    
         *  array(
         *      'alias_or_column_name' => array(
         *          'order'     => 'asc'
         *      ),
         *      'alias_or_column_name' => array(
         *          'columns'   => array(
         *              'once_more_the_alias_or_column_name_as_in_the_parent_key',
         *              'an_additional_alias_or_column_name_with_a_specific_direction asc'
         *          ),
         *          'order'     => 'desc'
         *      ),
         *      'alias_or_column_name' => array(
         *          'columns'   => array('a_different_alias_or_column_name_designated_to_act_as_the_only_sort_column')
         *          // Ascendant sort by default
         *      )
         *  )
         * 
    
         * Note that it's mandatory to supply the alias name in case there is one.
         *
         * @var array
         */
        protected $sortRules;
    
        /**
         * The value conversion rules to apply on a query or statement
         *
         * This may be initialized by concrete repository implementations and describes for which aliases or column
         * names what type of conversion is available. For entries, where the key is the alias/column and the value
         * is the type identifier, the repository attempts to find a conversion method for the alias/column first and,
         * if none is found, then for the type. If an entry only provides a value, which is the alias/column, the
         * repository only attempts to find a conversion method for the alias/column. The name of a conversion method
         * is expected to be declared using lowerCamelCase. (e.g. user_name will be translated to persistUserName and
         * groupname will be translated to retrieveGroupname)
         *
         * @var array
         */
        protected $conversionRules;
    
        /**
         * An array to map table names to aliases
         *
         * @var array
         */
        protected $aliasTableMap;
    
        /**
         * A flattened array to map query columns to aliases
         *
         * @var array
         */
        protected $aliasColumnMap;
    
        /**
         * Create a new repository object
         *
         * @param   Selectable  $ds     The datasource to use
         */
        public function __construct(Selectable $ds)
        {
            $this->ds = $ds;
            $this->aliasTableMap = array();
            $this->aliasColumnMap = array();
    
            $this->init();
        }
    
        /**
         * Initialize this repository
         *
         * Supposed to be overwritten by concrete repository implementations.
         */
        protected function init()
        {
    
        }
    
        /**
         * Set this repository's name
         *
         * @param   string  $name
         *
         * @return  $this
         */
        public function setName($name)
        {
            $this->name = $name;
            return $this;
        }
    
        /**
         * Return this repository's name
         *
         * In case no name has been explicitly set yet, the class name is returned.
         *
         * @return  string
         */
        public function getName()
        {
            return $this->name ?: __CLASS__;
        }
    
        /**
         * Return the datasource being used
         *
         * @return  Selectable
         */
        public function getDataSource()
        {
            return $this->ds;
        }
    
        /**
         * Return the base table name this repository is responsible for
         *
         * @return  string
         *
         * @throws  ProgrammingError    In case no base table name has been set and
         *                               $this->queryColumns does not provide one either
         */
        public function getBaseTable()
        {
            if ($this->baseTable === null) {
                $queryColumns = $this->getQueryColumns();
                reset($queryColumns);
                $this->baseTable = key($queryColumns);
                if (is_int($this->baseTable) || !is_array($queryColumns[$this->baseTable])) {
                    throw new ProgrammingError('"%s" is not a valid base table', $this->baseTable);
                }
            }
    
            return $this->baseTable;
        }
    
        /**
         * Return the query columns being provided
         *
         * Calls $this->initializeQueryColumns() in case $this->queryColumns is null.
         *
         * @return  array
         */
        public function getQueryColumns()
        {
            if ($this->queryColumns === null) {
                $this->queryColumns = $this->initializeQueryColumns();
            }
    
            return $this->queryColumns;
        }
    
        /**
         * Overwrite this in your repository implementation in case you need to initialize the query columns lazily
         *
         * @return  array
         */
        protected function initializeQueryColumns()
        {
            return array();
        }
    
        /**
         * Return the columns (or aliases) which are not permitted to be queried
         *
         * Calls $this->initializeFilterColumns() in case $this->filterColumns is null.
         *
         * @return  array
         */
        public function getFilterColumns()
        {
            if ($this->filterColumns === null) {
                $this->filterColumns = $this->initializeFilterColumns();
            }
    
            return $this->filterColumns;
        }
    
        /**
         * Overwrite this in your repository implementation in case you need to initialize the filter columns lazily
         *
         * @return  array
         */
        protected function initializeFilterColumns()
        {
            return array();
        }
    
        /**
         * Return the default sort rules to be applied on a query
         *
         * Calls $this->initializeSortRules() in case $this->sortRules is null.
         *
         * @return  array
         */
        public function getSortRules()
        {
            if ($this->sortRules === null) {
                $this->sortRules = $this->initializeSortRules();
            }
    
            return $this->sortRules;
        }
    
        /**
         * Overwrite this in your repository implementation in case you need to initialize the sort rules lazily
         *
         * @return  array
         */
        protected function initializeSortRules()
        {
            return array();
        }
    
        /**
         * Return the value conversion rules to apply on a query
         *
         * Calls $this->initializeConversionRules() in case $this->conversionRules is null.
         *
         * @return  array
         */
        public function getConversionRules()
        {
            if ($this->conversionRules === null) {
                $this->conversionRules = $this->initializeConversionRules();
            }
    
            return $this->conversionRules;
        }
    
        /**
         * Overwrite this in your repository implementation in case you need to initialize the conversion rules lazily
         *
         * @return  array
         */
        protected function initializeConversionRules()
        {
            return array();
        }
    
        /**
         * Return an array to map table names to aliases
         *
         * @return  array
         */
        protected function getAliasTableMap()
        {
            if (empty($this->aliasTableMap)) {
                $this->initializeAliasMaps();
            }
    
            return $this->aliasTableMap;
        }
    
        /**
         * Return a flattened array to map query columns to aliases
         *
         * @return  array
         */
        protected function getAliasColumnMap()
        {
            if (empty($this->aliasColumnMap)) {
                $this->initializeAliasMaps();
            }
    
            return $this->aliasColumnMap;
        }
    
        /**
         * Initialize $this->aliasTableMap and $this->aliasColumnMap
         *
         * @throws  ProgrammingError    In case $this->queryColumns does not provide any column information
         */
        protected function initializeAliasMaps()
        {
            $queryColumns = $this->getQueryColumns();
            if (empty($queryColumns)) {
                throw new ProgrammingError('Repositories are required to initialize $this->queryColumns first');
            }
    
            foreach ($queryColumns as $table => $columns) {
                foreach ($columns as $alias => $column) {
                    if (! is_string($alias)) {
                        $key = $column;
                    } else {
                        $key = $alias;
                        $column = preg_replace('~\n\s*~', ' ', $column);
                    }
    
                    if (array_key_exists($key, $this->aliasTableMap)) {
                        if ($this->aliasTableMap[$key] !== null) {
                            $existingTable = $this->aliasTableMap[$key];
                            $existingColumn = $this->aliasColumnMap[$key];
                            $this->aliasTableMap[$existingTable . '.' . $key] = $existingTable;
                            $this->aliasColumnMap[$existingTable . '.' . $key] = $existingColumn;
                            $this->aliasTableMap[$key] = null;
                            $this->aliasColumnMap[$key] = null;
                        }
    
                        $this->aliasTableMap[$table . '.' . $key] = $table;
                        $this->aliasColumnMap[$table . '.' . $key] = $column;
                    } else {
                        $this->aliasTableMap[$key] = $table;
                        $this->aliasColumnMap[$key] = $column;
                    }
                }
            }
        }
    
        /**
         * Return a new query for the given columns
         *
         * @param   array   $columns    The desired columns, if null all columns will be queried
         *
         * @return  RepositoryQuery
         */
        public function select(array $columns = null)
        {
            $query = new RepositoryQuery($this);
            $query->from($this->getBaseTable(), $columns);
            return $query;
        }
    
        /**
         * Return whether this repository is capable of converting values for the given table
         *
         * @param   string  $table
         *
         * @return  bool
         */
        public function providesValueConversion($table)
        {
            $conversionRules = $this->getConversionRules();
            return !empty($conversionRules) && isset($conversionRules[$table]);
        }
    
        /**
         * Convert a value supposed to be transmitted to the data source
         *
         * @param   string  $table      The table where to persist the value
         * @param   string  $name       The alias or column name
         * @param   mixed   $value      The value to convert
         *
         * @return  mixed               If conversion was possible, the converted value, otherwise the unchanged value
         */
        public function persistColumn($table, $name, $value)
        {
            $converter = $this->getConverter($table, $name, 'persist');
            if ($converter !== null) {
                $value = $this->$converter($value);
            }
    
            return $value;
        }
    
        /**
         * Convert a value which was fetched from the data source
         *
         * @param   string  $table      The table the value has been fetched from
         * @param   string  $name       The alias or column name
         * @param   mixed   $value      The value to convert
         *
         * @return  mixed               If conversion was possible, the converted value, otherwise the unchanged value
         */
        public function retrieveColumn($table, $name, $value)
        {
            $converter = $this->getConverter($table, $name, 'retrieve');
            if ($converter !== null) {
                $value = $this->$converter($value);
            }
    
            return $value;
        }
    
        /**
         * Return the name of the conversion method for the given alias or column name and context
         *
         * @param   string  $table      The datasource's table
         * @param   string  $name       The alias or column name for which to return a conversion method
         * @param   string  $context    The context of the conversion: persist or retrieve
         *
         * @return  string
         *
         * @throws  ProgrammingError    In case a conversion rule is found but not any conversion method
         */
        protected function getConverter($table, $name, $context)
        {
            $conversionRules = $this->getConversionRules();
            if (! isset($conversionRules[$table])) {
                return;
            }
    
            $tableRules = $conversionRules[$table];
    
            // Check for a conversion method for the alias/column first
            if (array_key_exists($name, $tableRules) || in_array($name, $tableRules)) {
                $methodName = $context . join('', array_map('ucfirst', explode('_', $name)));
                if (method_exists($this, $methodName)) {
                    return $methodName;
                }
            }
    
            // The conversion method for the type is just a fallback, but it is required to exist if defined
            if (isset($tableRules[$name])) {
                $identifier = join('', array_map('ucfirst', explode('_', $tableRules[$name])));
                if (! method_exists($this, $context . $identifier)) {
                    // Do not throw an error in case at least one conversion method exists
                    if (! method_exists($this, ($context === 'persist' ? 'retrieve' : 'persist') . $identifier)) {
                        throw new ProgrammingError(
                            'Cannot find any conversion method for type "%s"'
                            . '. Add a proper conversion method or remove the type definition',
                            $tableRules[$name]
                        );
                    }
    
                    Logger::debug(
                        'Conversion method "%s" for type definition "%s" does not exist in repository "%s".',
                        $context . $identifier,
                        $tableRules[$name],
                        $this->getName()
                    );
                } else {
                    return $context . $identifier;
                }
            }
        }
    
        /**
         * Convert a timestamp or DateTime object to a string formatted using static::DATETIME_FORMAT
         *
         * @param   mixed   $value
         *
         * @return  string
         */
        protected function persistDateTime($value)
        {
            if (is_numeric($value)) {
                $value = date(static::DATETIME_FORMAT, $value);
            } elseif ($value instanceof DateTime) {
                $value = date(static::DATETIME_FORMAT, $value->getTimestamp()); // Using date here, to ignore any timezone
            } elseif ($value !== null) {
                throw new ProgrammingError(
                    'Cannot persist value "%s" as type date_time. It\'s not a timestamp or DateTime object',
                    $value
                );
            }
    
            return $value;
        }
    
        /**
         * Convert a string formatted using static::DATETIME_FORMAT to a unix timestamp
         *
         * @param   string  $value
         *
         * @return  int
         */
        protected function retrieveDateTime($value)
        {
            if (is_numeric($value)) {
                $value = (int) $value;
            } elseif (is_string($value)) {
                $dateTime = DateTime::createFromFormat(static::DATETIME_FORMAT, $value);
                if ($dateTime === false) {
                    Logger::debug(
                        'Unable to parse string "%s" as type date_time with format "%s" in repository "%s"',
                        $value,
                        static::DATETIME_FORMAT,
                        $this->getName()
                    );
                    $value = null;
                } else {
                    $value = $dateTime->getTimestamp();
                }
            } elseif ($value !== null) {
                throw new ProgrammingError(
                    'Cannot retrieve value "%s" as type date_time. It\'s not a integer or (numeric) string',
                    $value
                );
            }
    
            return $value;
        }
    
        /**
         * Convert the given array to an comma separated string
         *
         * @param   array|string    $value
         *
         * @return  string
         */
        protected function persistCommaSeparatedString($value)
        {
            if (is_array($value)) {
                $value = join(',', array_map('trim', $value));
            } elseif ($value !== null && !is_string($value)) {
                throw new ProgrammingError('Cannot persist value "%s" as comma separated string', $value);
            }
    
            return $value;
        }
    
        /**
         * Convert the given comma separated string to an array
         *
         * @param   string  $value
         *
         * @return  array
         */
        protected function retrieveCommaSeparatedString($value)
        {
            if ($value && is_string($value)) {
                $value = String::trimSplit($value);
            } elseif ($value !== null) {
                throw new ProgrammingError('Cannot retrieve value "%s" as array. It\'s not a string', $value);
            }
    
            return $value;
        }
    
        /**
         * Parse the given value based on the ASN.1 standard (GeneralizedTime) and return its timestamp representation
         *
         * @param   string|null     $value
         *
         * @return  int
         */
        protected function retrieveGeneralizedTime($value)
        {
            if ($value === null) {
                return $value;
            }
    
            if (
                ($dateTime = DateTime::createFromFormat('YmdHis.uO', $value)) !== false
                || ($dateTime = DateTime::createFromFormat('YmdHis.uZ', $value)) !== false
                || ($dateTime = DateTime::createFromFormat('YmdHis.u', $value)) !== false
                || ($dateTime = DateTime::createFromFormat('YmdHis', $value)) !== false
                || ($dateTime = DateTime::createFromFormat('YmdHi', $value)) !== false
                || ($dateTime = DateTime::createFromFormat('YmdH', $value)) !== false
            ) {
                return $dateTime->getTimeStamp();
            } else {
                Logger::debug(sprintf(
                    'Failed to parse "%s" based on the ASN.1 standard (GeneralizedTime) in repository "%s".',
                    $value,
                    $this->getName()
                ));
            }
        }
    
        /**
         * Validate that the requested table exists
         *
         * @param   string              $table      The table to validate
         * @param   RepositoryQuery     $query      An optional query to pass as context
         *                                          (unused by the base implementation)
         *
         * @return  string                          The table's name, may differ from the given one
         *
         * @throws  ProgrammingError                In case the given table does not exist
         */
        public function requireTable($table, RepositoryQuery $query = null)
        {
            $queryColumns = $this->getQueryColumns();
            if (! isset($queryColumns[$table])) {
                throw new ProgrammingError('Table "%s" not found', $table);
            }
    
            return $table;
        }
    
        /**
         * Recurse the given filter, require each column for the given table and convert all values
         *
         * @param   string              $table      The table being filtered
         * @param   Filter              $filter     The filter to recurse
         * @param   RepositoryQuery     $query      An optional query to pass as context
         *                                          (Directly passed through to $this->requireFilterColumn)
         * @param   bool                $clone      Whether to clone $filter first
         *
         * @return  Filter                          The udpated filter
         */
        public function requireFilter($table, Filter $filter, RepositoryQuery $query = null, $clone = true)
        {
            if ($clone) {
                $filter = clone $filter;
            }
    
            if ($filter->isExpression()) {
                $column = $filter->getColumn();
                $filter->setColumn($this->requireFilterColumn($table, $column, $query));
                $filter->setExpression($this->persistColumn($table, $column, $filter->getExpression()));
            } elseif ($filter->isChain()) {
                foreach ($filter->filters() as $chainOrExpression) {
                    $this->requireFilter($table, $chainOrExpression, $query, false);
                }
            }
    
            return $filter;
        }
    
        /**
         * Return this repository's query columns of the given table mapped to their respective aliases
         *
         * @param   string  $table
         *
         * @return  array
         *
         * @throws  ProgrammingError    In case $table does not exist
         */
        public function requireAllQueryColumns($table)
        {
            $queryColumns = $this->getQueryColumns();
            if (! array_key_exists($table, $queryColumns)) {
                throw new ProgrammingError('Table name "%s" not found', $table);
            }
    
            $filterColumns = $this->getFilterColumns();
            $columns = array();
            foreach ($queryColumns[$table] as $alias => $column) {
                if (! in_array(is_string($alias) ? $alias : $column, $filterColumns)) {
                    $columns[$alias] = $column;
                }
            }
    
            return $columns;
        }
    
        /**
         * Return the query column name for the given alias or null in case the alias does not exist
         *
         * @param   string  $table
         * @param   string  $alias
         *
         * @return  string|null
         */
        public function resolveQueryColumnAlias($table, $alias)
        {
            $aliasColumnMap = $this->getAliasColumnMap();
            if (isset($aliasColumnMap[$alias])) {
                return $aliasColumnMap[$alias];
            }
    
            $prefixedAlias = $table . '.' . $alias;
            if (isset($aliasColumnMap[$prefixedAlias])) {
                return $aliasColumnMap[$prefixedAlias];
            }
        }
    
        /**
         * Return whether the given alias or query column name is available in the given table
         *
         * @param   string  $table
         * @param   string  $alias
         *
         * @return  bool
         */
        public function validateQueryColumnAssociation($table, $alias)
        {
            $aliasTableMap = $this->getAliasTableMap();
            if (isset($aliasTableMap[$alias])) {
                return $aliasTableMap[$alias] === $table;
            }
    
            $prefixedAlias = $table . '.' . $alias;
            return isset($aliasTableMap[$prefixedAlias]);
        }
    
        /**
         * Return whether the given column name or alias is a valid query column
         *
         * @param   string  $table  The table where to look for the column or alias
         * @param   string  $name   The column name or alias to check
         *
         * @return  bool
         */
        public function hasQueryColumn($table, $name)
        {
            if (in_array($name, $this->getFilterColumns())) {
                return false;
            }
    
            return $this->resolveQueryColumnAlias($table, $name) !== null
                && $this->validateQueryColumnAssociation($table, $name);
        }
    
        /**
         * Validate that the given column is a valid query target and return it or the actual name if it's an alias
         *
         * @param   string              $table  The table where to look for the column or alias
         * @param   string              $name   The name or alias of the column to validate
         * @param   RepositoryQuery     $query  An optional query to pass as context (unused by the base implementation)
         *
         * @return  string                      The given column's name
         *
         * @throws  QueryException              In case the given column is not a valid query column
         */
        public function requireQueryColumn($table, $name, RepositoryQuery $query = null)
        {
            if (in_array($name, $this->getFilterColumns())) {
                throw new QueryException(t('Filter column "%s" cannot be queried'), $name);
            }
    
            if (($column = $this->resolveQueryColumnAlias($table, $name)) === null) {
                throw new QueryException(t('Query column "%s" not found'), $name);
            }
    
            if (! $this->validateQueryColumnAssociation($table, $name)) {
                throw new QueryException(t('Query column "%s" not found in table "%s"'), $name, $table);
            }
    
            return $column;
        }
    
        /**
         * Return whether the given column name or alias is a valid filter column
         *
         * @param   string  $table  The table where to look for the column or alias
         * @param   string  $name   The column name or alias to check
         *
         * @return  bool
         */
        public function hasFilterColumn($table, $name)
        {
            return $this->resolveQueryColumnAlias($table, $name) !== null
                && $this->validateQueryColumnAssociation($table, $name);
        }
    
        /**
         * Validate that the given column is a valid filter target and return it or the actual name if it's an alias
         *
         * @param   string              $table  The table where to look for the column or alias
         * @param   string              $name   The name or alias of the column to validate
         * @param   RepositoryQuery     $query  An optional query to pass as context (unused by the base implementation)
         *
         * @return  string                      The given column's name
         *
         * @throws  QueryException              In case the given column is not a valid filter column
         */
        public function requireFilterColumn($table, $name, RepositoryQuery $query = null)
        {
            if (($column = $this->resolveQueryColumnAlias($table, $name)) === null) {
                throw new QueryException(t('Filter column "%s" not found'), $name);
            }
    
            if (! $this->validateQueryColumnAssociation($table, $name)) {
                throw new QueryException(t('Filter column "%s" not found in table "%s"'), $name, $table);
            }
    
            return $column;
        }
    
        /**
         * Return whether the given column name or alias of the given table is a valid statement column
         *
         * @param   string  $table  The table where to look for the column or alias
         * @param   string  $name   The column name or alias to check
         *
         * @return  bool
         */
        public function hasStatementColumn($table, $name)
        {
            return $this->hasQueryColumn($table, $name);
        }
    
        /**
         * Validate that the given column is a valid statement column and return it or the actual name if it's an alias
         *
         * @param   string  $table      The table for which to require the column
         * @param   string  $name       The name or alias of the column to validate
         *
         * @return  string              The given column's name
         *
         * @throws  StatementException  In case the given column is not a statement column
         */
        public function requireStatementColumn($table, $name)
        {
            if (in_array($name, $this->filterColumns)) {
                throw new StatementException('Filter column "%s" cannot be referenced in a statement', $name);
            }
    
            if (($column = $this->resolveQueryColumnAlias($table, $name)) === null) {
                throw new StatementException('Statement column "%s" not found', $name);
            }
    
            if (! $this->validateQueryColumnAssociation($table, $name)) {
                throw new StatementException('Statement column "%s" not found in table "%s"', $name, $table);
            }
    
            return $column;
        }
    
        /**
         * Resolve the given aliases or column names of the given table supposed to be persisted and convert their values
         *
         * @param   string  $table
         * @param   array   $data
         *
         * @return  array
         */
        public function requireStatementColumns($table, array $data)
        {
            $resolved = array();
            foreach ($data as $alias => $value) {
                $resolved[$this->requireStatementColumn($table, $alias)] = $this->persistColumn($table, $alias, $value);
            }
    
            return $resolved;
        }
    }