. */ Doctrine::autoload('Doctrine_Hydrate'); /** * Doctrine_RawSql * * @package Doctrine * @license http://www.opensource.org/licenses/lgpl-license.php LGPL * @category Object Relational Mapping * @link www.phpdoctrine.com * @since 1.0 * @version $Revision$ * @author Konsta Vesterinen */ class Doctrine_RawSql extends Doctrine_Hydrate { /** * @var array $fields */ private $fields; /** * __call * method overloader * * @param string $name * @param array $args * @return Doctrine_RawSql */ public function __call($name, $args) { if ( ! isset($this->parts[$name])) { throw new Doctrine_RawSql_Exception("Unknown overload method $name. Availible overload methods are ".implode(" ",array_keys($this->parts))); } if ($name == 'select') { preg_match_all('/{([^}{]*)}/U', $args[0], $m); $this->fields = $m[1]; $this->parts["select"] = array(); } else { $this->parts[$name][] = $args[0]; } return $this; } /** * get */ public function get($name) { if ( ! isset($this->parts[$name])) { throw new Doctrine_RawSql_Exception('Unknown query part ' . $name); } return $this->parts[$name]; } /** * parseQuery * * @param string $query * @return Doctrine_RawSql */ public function parseQuery($query) { preg_match_all('/{([^}{]*)}/U', $query, $m); $this->fields = $m[1]; $this->clear(); $e = Doctrine_Query::sqlExplode($query,' '); foreach ($e as $k => $part) { $low = strtolower($part); switch (strtolower($part)) { case "select": case "from": case "where": case "limit": case "offset": case "having": $p = $low; if ( ! isset($parts[$low])) { $parts[$low] = array(); } break; case "order": case "group": $i = ($k + 1); if (isset($e[$i]) && strtolower($e[$i]) === "by") { $p = $low; $p .= "by"; $parts[$low."by"] = array(); } else { $parts[$p][] = $part; } break; case "by": continue; default: if ( ! isset($parts[$p][0])) { $parts[$p][0] = $part; } else { $parts[$p][0] .= ' '.$part; } }; }; $this->parts = $parts; $this->parts["select"] = array(); return $this; } /** * getQuery * * * @return string */ public function getQuery() { foreach ($this->fields as $field) { $e = explode('.', $field); if ( ! isset($e[1])) { throw new Doctrine_RawSql_Exception('All selected fields in Sql query must be in format tableAlias.fieldName'); } if ( ! isset($this->tables[$e[0]])) { try { $this->addComponent($e[0], ucwords($e[0])); } catch(Doctrine_Exception $exception) { throw new Doctrine_RawSql_Exception('The associated component for table alias ' . $e[0] . ' couldn\'t be found.'); } } if ($e[1] == '*') { foreach ($this->tables[$e[0]]->getColumnNames() as $name) { $field = $e[0].'.'.$name; $this->parts['select'][$field] = $field.' AS '.$e[0].'__'.$name; } } else { $field = $e[0].'.'.$e[1]; $this->parts['select'][$field] = $field.' AS '.$e[0].'__'.$e[1]; } } // force-add all primary key fields foreach ($this->tableAliases as $alias) { foreach ($this->tables[$alias]->getPrimaryKeys() as $key) { $field = $alias . '.' . $key; if ( ! isset($this->parts["select"][$field])) { $this->parts["select"][$field] = $field." AS ".$alias."__".$key; } } } $q = 'SELECT ' . implode(', ', $this->parts['select']); $string = $this->applyInheritance(); if ( ! empty($string)) { $this->parts['where'][] = $string; } $copy = $this->parts; unset($copy['select']); $q .= ( ! empty($this->parts['from']))? ' FROM ' . implode(' ', $this->parts['from']) : ''; $q .= ( ! empty($this->parts['where']))? ' WHERE ' . implode(' AND ', $this->parts['where']) : ''; $q .= ( ! empty($this->parts['groupby']))? ' GROUP BY ' . implode(', ', $this->parts['groupby']) : ''; $q .= ( ! empty($this->parts['having']))? ' HAVING ' . implode(' ', $this->parts['having']) : ''; $q .= ( ! empty($this->parts['orderby']))? ' ORDER BY ' . implode(' ', $this->parts['orderby']) : ''; $q .= ( ! empty($this->parts['limit']))? ' LIMIT ' . implode(' ', $this->parts['limit']) : ''; $q .= ( ! empty($this->parts['offset']))? ' OFFSET ' . implode(' ', $this->parts['offset']) : ''; if ( ! empty($string)) { array_pop($this->parts['where']); } return $q; } /** * getFields * * @return array */ public function getFields() { return $this->fields; } /** * addComponent * * @param string $tableAlias * @param string $componentName * @return Doctrine_RawSql */ public function addComponent($tableAlias, $componentName) { $e = explode('.', $componentName); $currPath = ''; $table = null; foreach ($e as $k => $component) { $currPath .= '.' . $component; if ($k == 0) $currPath = substr($currPath,1); if (isset($this->tableAliases[$currPath])) { $alias = $this->tableAliases[$currPath]; } else { $alias = $tableAlias; } if ($table) { $tableName = $table->getAliasName($component); } $table = $this->conn->getTable($component); $this->tables[$alias] = $table; $this->fetchModes[$alias] = Doctrine::FETCH_IMMEDIATE; $this->tableAliases[$currPath] = $alias; if ($k !== 0) { $this->joins[$alias] = $prevAlias; } $prevAlias = $alias; $prevPath = $currPath; } return $this; } }