<?php class TDomSqlMapBuilder { const DEFAULT_CONFIG_FILE = 'sqlmap.xml'; private $_document; private $_sqlMapper; private $_configFile; private $_properties; private $_deserialize; private $_useNamespaces = false; private $_FlushOnExecuteStatements=array(); public function __construct($cachedir='./cache') { $this->_properties = new TMap; $this->_deserialize = new TConfigDeserialize($this->_properties); } public function configure($resource=null) { if($resource instanceof SimpleXMLElement) return $this->build($resource); if(!is_string($resource)) $resource = self::DEFAULT_CONFIG_FILE; $this->_configFile = $resource; if(!is_file($resource)) throw new TSqlMapConfigurationException( 'sqlmap_unable_to_find_config', $resource); return $this->build($this->getConfigAsXmlDocument($resource)); } protected function getConfigAsXmlDocument($file) { return simplexml_load_file($file); } public function build(SimpleXMLElement $document) { $this->_document = $document; $this->initializeSQLMap($document); return $this->_sqlMapper; } protected function initializeSQLMap($document) { $this->_sqlMapper = new TSqlMapper(new TTypeHandlerFactory); if(isset($document->properties)) { $this->loadGlobalProperties($document->properties); } if(isset($document->settings) && isset($document->settings->setting)) $this->configureSettings($document->settings); foreach($document->xpath('//typeHandler') as $handler) $this->loadTypeHandler($handler, $this->_configFile); //load database provider if(isset($document->provider) && isset($document->provider->datasource)) { $this->loadProvider($document->provider, $document->provider->datasource, $document, $this->_configFile); } else { throw new TSqlMapConfigurationException( 'sqlmap_unable_to_find_db_config', $this->_configFile); } foreach($document->xpath('//sqlMap') as $sqlmap) $this->loadSqlMappingFiles($sqlmap); $this->resolveResultMapping(); if($this->_sqlMapper->getIsCacheModelsEnabled()) $this->attachCacheModel(); } protected function configureSettings($node) { foreach($node->setting as $setting) { if(isset($setting['useStatementNamespaces'])) { $this->_useNamespaces = TPropertyValue::ensureBoolean( (string)$setting['useStatementNamespaces']); } if(isset($setting['cacheModelsEnabled'])) { $this->_sqlMapper->setCacheModelsEnabled( TPropertyValue::ensureBoolean( (string)$setting['cacheModelsEnabled'])); } } } /** * Attach CacheModel to statement and register trigger statements for * cache models */ protected function attachCacheModel() { foreach($this->_sqlMapper->getStatements() as $mappedStatement) { if(strlen($model = $mappedStatement->getStatement()->getCacheModel()) > 0) { $cache = $this->_sqlMapper->getCache($model); //var_dump($model); $mappedStatement->getStatement()->setCache($cache); } } foreach($this->_FlushOnExecuteStatements as $cacheID => $statementIDs) { if(count($statementIDs) > 0) { foreach($statementIDs as $statementID) { $cacheModel = $this->_sqlMapper->getCache($cacheID); $statement = $this->_sqlMapper->getMappedStatement($statementID); $cacheModel->registerTriggerStatement($statement); } } } } protected function loadGlobalProperties($node) { if(isset($node['resource'])) $this->loadPropertyResource($node); foreach($node->children() as $child) { if(isset($child['resource'])) $this->loadPropertyResource($child); $this->_properties[(string)$child['key']] = (string)$child['value']; } } protected function loadPropertyResource($node) { $resource = $this->getResourceFromPath((string)$node['resource']); $properties = $this->getConfigAsXmlDocument($resource); $this->loadGlobalProperties($properties); } protected function loadProvider($providerNode, $node, $document, $file) { //$id = (string)$node['id']; $class = (string)$providerNode['class']; if(strlen($class) > 0) { if(class_exists($class,false)) { $provider = new $class; $this->_deserialize->loadConfiguration($provider, $node,$file); $this->_sqlMapper->setDataProvider($provider); } else { throw new TSqlMapConfigurationException( 'sqlmap_unable_find_provider_class_def', $file, $class); } } else { throw new TSqlMapConfigurationException( 'sqlmap_unable_find_provider_class', $file); } //var_dump($node); } protected function loadTypeHandler($node, $file) { if(!is_null($node['type']) && !is_null($node['callback'])) { $type = (string)$node['type']; $dbType = (string)$node['dbType']; $class = (string)$node['callback']; if(class_exists('Prado', false)) { $handler = Prado::createComponent($class); } else { if(class_exists($class,false)) $handler = new $class; else throw new TSqlMapConfigurationException( 'sqlmap_type_handler_class_undef', $file, $class); } $factory = $this->_sqlMapper->getTypeHandlerFactory(); $factory->register($type, $handler, $dbType); } else { throw new TSqlMapConfigurationException( 'sqlmap_type_handler_callback_undef', $file); } } protected function loadSqlMappingFiles($node) { $resource = $this->getResourceFromPath((string)$node['resource']); $sqlmap = $this->getConfigAsXmlDocument($resource); $this->configureSqlMap($sqlmap,$resource); } protected function getResourceFromPath($resource) { $basedir = dirname($this->_configFile); $file = realpath($basedir.'/'.$resource); if(!is_string($file) || !is_file($file)) $file = realpath($resource); if(is_string($file) && is_file($file)) return $file; else throw new TSqlMapConfigurationException( 'sqlmap_unable_to_find_resource', $resource); } protected function configureSqlMap($document,$file) { // if(isset($document->typeAlias)) // foreach($document->typeAlias as $node) // TTypeAliasDeSerializer::Deserialize($node, $this->_sqlMapper); foreach($document->xpath('//resultMap') as $node) $this->loadResultMap($node,$document,$file); foreach($document->xpath('//parameterMap') as $node) $this->loadParameterMap($node, $document, $file); foreach($document->xpath('//statement') as $node) $this->loadStatementTag($node, $document,$file); foreach($document->xpath('//select') as $node) $this->loadSelectTag($node, $document, $file); foreach($document->xpath('//insert') as $node) $this->loadInsertTag($node, $document, $file); foreach($document->xpath('//update') as $node) $this->loadUpdateTag($node, $document, $file); foreach($document->xpath('//delete') as $node) $this->loadDeleteTag($node, $document, $file); /* if(isset($document->procedure)) foreach($document->procedure as $node) $this->loadProcedureTag($node); */ if($this->_sqlMapper->getIsCacheModelsEnabled()) { if(isset($document->cacheModel)) foreach($document->cacheModel as $node) $this->loadCacheModel($node, $document, $file); } } protected function loadCacheModel($node, $document, $file) { $cacheModel = $this->_deserialize->cacheModel($node, $this->_sqlMapper, $file); if(isset($node->flushOnExecute)) { foreach($node->flushOnExecute as $flush) { $id = $cacheModel->getID(); if(!isset($this->_FlushOnExecuteStatements[$id])) $this->_FlushOnExecuteStatements[$id] = array(); $this->_FlushOnExecuteStatements[$id][] = (string)$flush['statement']; } } //var_dump($cacheModel); $cacheModel->initialize($this->_sqlMapper); $this->_sqlMapper->addCache($cacheModel); } protected function loadUpdateTag($node, $document, $file) { $update = $this->_deserialize->insert($node, $this->_sqlMapper, $file); if(!is_null($update->getGenerate())) { var_dump('generate update'); } else { $this->processSqlStatement($update, $document, $node, $file); } $mappedStatement = new TUpdateMappedStatement($this->_sqlMapper, $update); $this->_sqlMapper->addMappedStatement($mappedStatement); } protected function loadDeleteTag($node, $document, $file) { $delete = $this->_deserialize->delete($node, $this->_sqlMapper, $file); if(!is_null($delete->getGenerate())) { var_dump('generate delete'); } else { $this->processSqlStatement($delete, $document, $node, $file); } $mappedStatement = new TDeleteMappedStatement($this->_sqlMapper, $delete); $this->_sqlMapper->addMappedStatement($mappedStatement); } protected function loadParameterMap($node, $document, $file) { $id = (string)$node['id']; if($this->_sqlMapper->getParameterMaps()->contains($id)) return; $parameterMap = $this->_deserialize->parameterMap($node, $this->_sqlMapper, $file); $extendMap = $parameterMap->getExtends(); if(strlen($extendMap) > 0) { if($this->_sqlMapper->getParameterMaps()->contains($extendMap) == false) { $nodes = $document->xpath("//parameterMap[@id='{$extendMap}']"); if(isset($nodes[0])) $this->loadParameterMap($nodes[0],$document,$file); else throw new TSqlMapConfigurationException( 'sqlmap_unable_to_find_parent_parameter_map', $extendMap, $file); } $superMap = $this->_sqlMapper->getParameterMap($extendMap); $index = 0; foreach($superMap->getPropertyNames() as $propertyName) { $parameterMap->insertParameterProperty($index++, $superMap->getProperty($propertyName)); } } $this->_sqlMapper->addParameterMap($parameterMap); } protected function loadInsertTag($node, $document, $file) { $insert = $this->_deserialize->insert($node, $this->_sqlMapper, $file); if(!is_null($insert->getGenerate())) { var_dump("generate insert"); } else { $this->processSqlStatement($insert, $document, $node, $file); } $mappedStatement = new TInsertMappedStatement($this->_sqlMapper, $insert); $this->_sqlMapper->addMappedStatement($mappedStatement); if(!is_null($insert->getSelectKey())) { $selectKey = $insert->getSelectKey(); $selectKey->setID($insert->getID()); $selectKey->initialize($this->_sqlMapper); $selectKey->setID($insert->getID().'.SelectKey'); $this->processSqlStatement($selectKey, $document, $node->selectKey, $file); $mappedStatement = new TMappedStatement($this->_sqlMapper, $selectKey); $this->_sqlMapper->addMappedStatement($mappedStatement); } } protected function processSqlStatement($statement, $document, $node, $file) { $commandText = (string)$node; if(strlen($extend = $statement->getExtends()) > 0) { $superNodes = $document->xpath("//*[@id='{$extend}']"); if(isset($superNodes[0])) $commandText = (string)$superNodes[0] . $commandText; else throw new TSqlMapConfigurationException( 'sqlmap_unable_to_find_parent_sql', $extend, $file); } //$sql = new TStaticSql(); //$sql->buildPreparedStatement($statement, (string)$node); $commandText = $this->_deserialize->replaceProperties($commandText); $this->applyInlineParameterMap($statement, $commandText, $node, $file); //$statement->setSql($sql); } protected function applyInlineParameterMap($statement, $sqlStatement, $node, $file) { $scope['statement'] = $statement->getID(); $scope['file'] = $file; if($statement->parameterMap() == null) { // Build a Parametermap with the inline parameters. // if they exist. Then delete inline infos from sqltext. $parameterParser = new TInlineParameterMapParser; $sqlText = $parameterParser->parse( $this->_sqlMapper, $statement, $sqlStatement, $scope); if(count($sqlText['parameters']) > 0) { $map = new TParameterMap(); $map->setID($statement->getID().'-InLineParameterMap'); $statement->setInlineParameterMap($map); foreach($sqlText['parameters'] as $property) $map->addParameterProperty($property); } $sqlStatement = $sqlText['sql']; } $simpleDynamic = new TSimpleDynamicParser; $dynamics = $simpleDynamic->parse($this->_sqlMapper, $statement, $sqlStatement, $scope); if(count($dynamics['parameters']) > 0) { $sql = new TSimpleDynamicSql($dynamics['parameters']); $sqlStatement = $dynamics['sql']; } else $sql = new TStaticSql(); $sql->buildPreparedStatement($statement, $sqlStatement); $statement->setSql($sql); } protected function resolveResultMapping() { $maps = $this->_sqlMapper->getResultMaps(); foreach($maps as $entry) { foreach($entry->getColumns() as $item) { $resultMap = $item->getResultMapping(); if(strlen($resultMap) > 0) { if($maps->contains($resultMap)) $item->setNestedResultMap($maps[$resultMap]); else throw new TSqlMapConfigurationException( 'sqlmap_unable_to_find_result_mapping', $resultMap, $this->_configFile, $entry->getID()); } } if(!is_null($entry->getDiscriminator())) { $entry->getDiscriminator()->initialize($this->_sqlMapper); } } } protected function loadSelectTag($node, $document, $file) { $select = $this->_deserialize->select($node, $this->_sqlMapper, $file); if(!is_null($select->getGenerate())) { var_dump("generate select"); } else { $this->processSqlStatement($select, $document, $node, $file); /*$sql = new TStaticSql(); $sql->buildPreparedStatement($select, (string)$node); $select->setSql($sql);*/ } $mappedStatement = new TMappedStatement($this->_sqlMapper, $select); if($this->_sqlMapper->getIsCacheModelsEnabled() && strlen($select->getCacheModel()) > 0) { $mappedStatement = new TCachingStatement($mappedStatement); } $this->_sqlMapper->addMappedStatement($mappedStatement); } protected function loadResultMap($node,$document,$file) { $resultMap = $this->_deserialize->resultMap($node, $this->_sqlMapper,$file); $extendMap = $resultMap->getExtends(); if(strlen($extendMap) > 0) { if(!$this->_sqlMapper->getResultMaps()->contains($extendMap)) { $nodes = $document->xpath("//resultMap[@id='{$extendMap}']"); if(isset($nodes[0])) $this->loadResultMap($nodes[0],$document,$file); else throw new TSqlMapConfigurationException( 'sqlmap_unable_to_find_parent_result_map', $extendMap, $file); } $superMap = $this->_sqlMapper->getResultMap($extendMap); $resultMap->getColumns()->mergeWith($superMap->getColumns()); } if(!$this->_sqlMapper->getResultMaps()->contains($resultMap->getID())) $this->_sqlMapper->addResultMap($resultMap); } protected function loadStatementTag($node, $document, $file) { $statement = $this->_deserialize->statement($node, $this->_sqlMapper, $file); /*$sql = new TStaticSql(); $sql->buildPreparedStatement($statement, (string)$node); $statement->setSql($sql);*/ $this->processSqlStatement($statement, $document, $node, $file); $mappedStatement = new TMappedStatement($this->_sqlMapper, $statement); $this->_sqlMapper->addMappedStatement($mappedStatement); } } ?>