@@ -14,138 +14,138 @@ |
||
14 | 14 | */ |
15 | 15 | class TDBMSchemaAnalyzer |
16 | 16 | { |
17 | - private $connection; |
|
18 | - |
|
19 | - /** |
|
20 | - * @var Schema |
|
21 | - */ |
|
22 | - private $schema; |
|
23 | - |
|
24 | - /** |
|
25 | - * @var string |
|
26 | - */ |
|
27 | - private $cachePrefix; |
|
28 | - |
|
29 | - /** |
|
30 | - * @var Cache |
|
31 | - */ |
|
32 | - private $cache; |
|
33 | - |
|
34 | - /** |
|
35 | - * @var SchemaAnalyzer |
|
36 | - */ |
|
37 | - private $schemaAnalyzer; |
|
38 | - |
|
39 | - /** |
|
40 | - * @param Connection $connection The DBAL DB connection to use |
|
41 | - * @param Cache $cache A cache service to be used |
|
42 | - * @param SchemaAnalyzer $schemaAnalyzer The schema analyzer that will be used to find shortest paths... |
|
43 | - * Will be automatically created if not passed |
|
44 | - */ |
|
45 | - public function __construct(Connection $connection, Cache $cache, SchemaAnalyzer $schemaAnalyzer) |
|
46 | - { |
|
47 | - $this->connection = $connection; |
|
48 | - $this->cache = $cache; |
|
49 | - $this->schemaAnalyzer = $schemaAnalyzer; |
|
50 | - } |
|
51 | - |
|
52 | - /** |
|
53 | - * Returns a unique ID for the current connection. Useful for namespacing cache entries in the current connection. |
|
54 | - * |
|
55 | - * @return string |
|
56 | - */ |
|
57 | - public function getCachePrefix() |
|
58 | - { |
|
59 | - if ($this->cachePrefix === null) { |
|
60 | - $this->cachePrefix = hash('md4', $this->connection->getHost().'-'.$this->connection->getPort().'-'.$this->connection->getDatabase().'-'.$this->connection->getDriver()->getName()); |
|
61 | - } |
|
62 | - |
|
63 | - return $this->cachePrefix; |
|
64 | - } |
|
65 | - |
|
66 | - /** |
|
67 | - * Returns the (cached) schema. |
|
68 | - * |
|
69 | - * @return Schema |
|
70 | - */ |
|
71 | - public function getSchema() |
|
72 | - { |
|
73 | - if ($this->schema === null) { |
|
74 | - $cacheKey = $this->getCachePrefix().'_schema'; |
|
75 | - if ($this->cache->contains($cacheKey)) { |
|
76 | - $this->schema = $this->cache->fetch($cacheKey); |
|
77 | - } else { |
|
78 | - $this->schema = $this->connection->getSchemaManager()->createSchema(); |
|
79 | - $this->cache->save($cacheKey, $this->schema); |
|
80 | - } |
|
81 | - } |
|
82 | - |
|
83 | - return $this->schema; |
|
84 | - } |
|
85 | - |
|
86 | - /** |
|
87 | - * Returns the list of pivot tables linked to table $tableName. |
|
88 | - * |
|
89 | - * @param string $tableName |
|
90 | - * |
|
91 | - * @return array|string[] |
|
92 | - */ |
|
93 | - public function getPivotTableLinkedToTable($tableName) |
|
94 | - { |
|
95 | - $cacheKey = $this->getCachePrefix().'_pivottables_link_'.$tableName; |
|
96 | - if ($this->cache->contains($cacheKey)) { |
|
97 | - return $this->cache->fetch($cacheKey); |
|
98 | - } |
|
99 | - |
|
100 | - $pivotTables = []; |
|
101 | - |
|
102 | - $junctionTables = $this->schemaAnalyzer->detectJunctionTables(true); |
|
103 | - foreach ($junctionTables as $table) { |
|
104 | - $fks = $table->getForeignKeys(); |
|
105 | - foreach ($fks as $fk) { |
|
106 | - if ($fk->getForeignTableName() == $tableName) { |
|
107 | - $pivotTables[] = $table->getName(); |
|
108 | - break; |
|
109 | - } |
|
110 | - } |
|
111 | - } |
|
112 | - |
|
113 | - $this->cache->save($cacheKey, $pivotTables); |
|
114 | - |
|
115 | - return $pivotTables; |
|
116 | - } |
|
117 | - |
|
118 | - /** |
|
119 | - * Returns the list of foreign keys pointing to the table represented by this bean, excluding foreign keys |
|
120 | - * from junction tables and from inheritance. |
|
121 | - * |
|
122 | - * @return ForeignKeyConstraint[] |
|
123 | - */ |
|
124 | - public function getIncomingForeignKeys($tableName) |
|
125 | - { |
|
126 | - $junctionTables = $this->schemaAnalyzer->detectJunctionTables(true); |
|
127 | - $junctionTableNames = array_map(function (Table $table) { |
|
128 | - return $table->getName(); |
|
129 | - }, $junctionTables); |
|
130 | - $childrenRelationships = $this->schemaAnalyzer->getChildrenRelationships($tableName); |
|
131 | - |
|
132 | - $fks = []; |
|
133 | - foreach ($this->getSchema()->getTables() as $table) { |
|
134 | - foreach ($table->getForeignKeys() as $fk) { |
|
135 | - if ($fk->getForeignTableName() === $tableName) { |
|
136 | - if (in_array($fk->getLocalTableName(), $junctionTableNames)) { |
|
137 | - continue; |
|
138 | - } |
|
139 | - foreach ($childrenRelationships as $childFk) { |
|
140 | - if ($fk->getLocalTableName() === $childFk->getLocalTableName() && $fk->getLocalColumns() === $childFk->getLocalColumns()) { |
|
141 | - continue 2; |
|
142 | - } |
|
143 | - } |
|
144 | - $fks[] = $fk; |
|
145 | - } |
|
146 | - } |
|
147 | - } |
|
148 | - |
|
149 | - return $fks; |
|
150 | - } |
|
17 | + private $connection; |
|
18 | + |
|
19 | + /** |
|
20 | + * @var Schema |
|
21 | + */ |
|
22 | + private $schema; |
|
23 | + |
|
24 | + /** |
|
25 | + * @var string |
|
26 | + */ |
|
27 | + private $cachePrefix; |
|
28 | + |
|
29 | + /** |
|
30 | + * @var Cache |
|
31 | + */ |
|
32 | + private $cache; |
|
33 | + |
|
34 | + /** |
|
35 | + * @var SchemaAnalyzer |
|
36 | + */ |
|
37 | + private $schemaAnalyzer; |
|
38 | + |
|
39 | + /** |
|
40 | + * @param Connection $connection The DBAL DB connection to use |
|
41 | + * @param Cache $cache A cache service to be used |
|
42 | + * @param SchemaAnalyzer $schemaAnalyzer The schema analyzer that will be used to find shortest paths... |
|
43 | + * Will be automatically created if not passed |
|
44 | + */ |
|
45 | + public function __construct(Connection $connection, Cache $cache, SchemaAnalyzer $schemaAnalyzer) |
|
46 | + { |
|
47 | + $this->connection = $connection; |
|
48 | + $this->cache = $cache; |
|
49 | + $this->schemaAnalyzer = $schemaAnalyzer; |
|
50 | + } |
|
51 | + |
|
52 | + /** |
|
53 | + * Returns a unique ID for the current connection. Useful for namespacing cache entries in the current connection. |
|
54 | + * |
|
55 | + * @return string |
|
56 | + */ |
|
57 | + public function getCachePrefix() |
|
58 | + { |
|
59 | + if ($this->cachePrefix === null) { |
|
60 | + $this->cachePrefix = hash('md4', $this->connection->getHost().'-'.$this->connection->getPort().'-'.$this->connection->getDatabase().'-'.$this->connection->getDriver()->getName()); |
|
61 | + } |
|
62 | + |
|
63 | + return $this->cachePrefix; |
|
64 | + } |
|
65 | + |
|
66 | + /** |
|
67 | + * Returns the (cached) schema. |
|
68 | + * |
|
69 | + * @return Schema |
|
70 | + */ |
|
71 | + public function getSchema() |
|
72 | + { |
|
73 | + if ($this->schema === null) { |
|
74 | + $cacheKey = $this->getCachePrefix().'_schema'; |
|
75 | + if ($this->cache->contains($cacheKey)) { |
|
76 | + $this->schema = $this->cache->fetch($cacheKey); |
|
77 | + } else { |
|
78 | + $this->schema = $this->connection->getSchemaManager()->createSchema(); |
|
79 | + $this->cache->save($cacheKey, $this->schema); |
|
80 | + } |
|
81 | + } |
|
82 | + |
|
83 | + return $this->schema; |
|
84 | + } |
|
85 | + |
|
86 | + /** |
|
87 | + * Returns the list of pivot tables linked to table $tableName. |
|
88 | + * |
|
89 | + * @param string $tableName |
|
90 | + * |
|
91 | + * @return array|string[] |
|
92 | + */ |
|
93 | + public function getPivotTableLinkedToTable($tableName) |
|
94 | + { |
|
95 | + $cacheKey = $this->getCachePrefix().'_pivottables_link_'.$tableName; |
|
96 | + if ($this->cache->contains($cacheKey)) { |
|
97 | + return $this->cache->fetch($cacheKey); |
|
98 | + } |
|
99 | + |
|
100 | + $pivotTables = []; |
|
101 | + |
|
102 | + $junctionTables = $this->schemaAnalyzer->detectJunctionTables(true); |
|
103 | + foreach ($junctionTables as $table) { |
|
104 | + $fks = $table->getForeignKeys(); |
|
105 | + foreach ($fks as $fk) { |
|
106 | + if ($fk->getForeignTableName() == $tableName) { |
|
107 | + $pivotTables[] = $table->getName(); |
|
108 | + break; |
|
109 | + } |
|
110 | + } |
|
111 | + } |
|
112 | + |
|
113 | + $this->cache->save($cacheKey, $pivotTables); |
|
114 | + |
|
115 | + return $pivotTables; |
|
116 | + } |
|
117 | + |
|
118 | + /** |
|
119 | + * Returns the list of foreign keys pointing to the table represented by this bean, excluding foreign keys |
|
120 | + * from junction tables and from inheritance. |
|
121 | + * |
|
122 | + * @return ForeignKeyConstraint[] |
|
123 | + */ |
|
124 | + public function getIncomingForeignKeys($tableName) |
|
125 | + { |
|
126 | + $junctionTables = $this->schemaAnalyzer->detectJunctionTables(true); |
|
127 | + $junctionTableNames = array_map(function (Table $table) { |
|
128 | + return $table->getName(); |
|
129 | + }, $junctionTables); |
|
130 | + $childrenRelationships = $this->schemaAnalyzer->getChildrenRelationships($tableName); |
|
131 | + |
|
132 | + $fks = []; |
|
133 | + foreach ($this->getSchema()->getTables() as $table) { |
|
134 | + foreach ($table->getForeignKeys() as $fk) { |
|
135 | + if ($fk->getForeignTableName() === $tableName) { |
|
136 | + if (in_array($fk->getLocalTableName(), $junctionTableNames)) { |
|
137 | + continue; |
|
138 | + } |
|
139 | + foreach ($childrenRelationships as $childFk) { |
|
140 | + if ($fk->getLocalTableName() === $childFk->getLocalTableName() && $fk->getLocalColumns() === $childFk->getLocalColumns()) { |
|
141 | + continue 2; |
|
142 | + } |
|
143 | + } |
|
144 | + $fks[] = $fk; |
|
145 | + } |
|
146 | + } |
|
147 | + } |
|
148 | + |
|
149 | + return $fks; |
|
150 | + } |
|
151 | 151 | } |
@@ -27,100 +27,100 @@ |
||
27 | 27 | */ |
28 | 28 | class InnerResultArray extends InnerResultIterator |
29 | 29 | { |
30 | - /** |
|
31 | - * The list of results already fetched. |
|
32 | - * |
|
33 | - * @var AbstractTDBMObject[] |
|
34 | - */ |
|
35 | - private $results = []; |
|
30 | + /** |
|
31 | + * The list of results already fetched. |
|
32 | + * |
|
33 | + * @var AbstractTDBMObject[] |
|
34 | + */ |
|
35 | + private $results = []; |
|
36 | 36 | |
37 | - /** |
|
38 | - * Whether a offset exists. |
|
39 | - * |
|
40 | - * @link http://php.net/manual/en/arrayaccess.offsetexists.php |
|
41 | - * |
|
42 | - * @param mixed $offset <p> |
|
43 | - * An offset to check for. |
|
44 | - * </p> |
|
45 | - * |
|
46 | - * @return bool true on success or false on failure. |
|
47 | - * </p> |
|
48 | - * <p> |
|
49 | - * The return value will be casted to boolean if non-boolean was returned |
|
50 | - * |
|
51 | - * @since 5.0.0 |
|
52 | - */ |
|
53 | - public function offsetExists($offset) |
|
54 | - { |
|
55 | - try { |
|
56 | - $this->toIndex($offset); |
|
57 | - } catch (TDBMInvalidOffsetException $e) { |
|
58 | - return false; |
|
59 | - } |
|
37 | + /** |
|
38 | + * Whether a offset exists. |
|
39 | + * |
|
40 | + * @link http://php.net/manual/en/arrayaccess.offsetexists.php |
|
41 | + * |
|
42 | + * @param mixed $offset <p> |
|
43 | + * An offset to check for. |
|
44 | + * </p> |
|
45 | + * |
|
46 | + * @return bool true on success or false on failure. |
|
47 | + * </p> |
|
48 | + * <p> |
|
49 | + * The return value will be casted to boolean if non-boolean was returned |
|
50 | + * |
|
51 | + * @since 5.0.0 |
|
52 | + */ |
|
53 | + public function offsetExists($offset) |
|
54 | + { |
|
55 | + try { |
|
56 | + $this->toIndex($offset); |
|
57 | + } catch (TDBMInvalidOffsetException $e) { |
|
58 | + return false; |
|
59 | + } |
|
60 | 60 | |
61 | - return true; |
|
62 | - } |
|
61 | + return true; |
|
62 | + } |
|
63 | 63 | |
64 | - /** |
|
65 | - * Offset to retrieve. |
|
66 | - * |
|
67 | - * @link http://php.net/manual/en/arrayaccess.offsetget.php |
|
68 | - * |
|
69 | - * @param mixed $offset <p> |
|
70 | - * The offset to retrieve. |
|
71 | - * </p> |
|
72 | - * |
|
73 | - * @return mixed Can return all value types |
|
74 | - * |
|
75 | - * @since 5.0.0 |
|
76 | - */ |
|
77 | - public function offsetGet($offset) |
|
78 | - { |
|
79 | - $this->toIndex($offset); |
|
64 | + /** |
|
65 | + * Offset to retrieve. |
|
66 | + * |
|
67 | + * @link http://php.net/manual/en/arrayaccess.offsetget.php |
|
68 | + * |
|
69 | + * @param mixed $offset <p> |
|
70 | + * The offset to retrieve. |
|
71 | + * </p> |
|
72 | + * |
|
73 | + * @return mixed Can return all value types |
|
74 | + * |
|
75 | + * @since 5.0.0 |
|
76 | + */ |
|
77 | + public function offsetGet($offset) |
|
78 | + { |
|
79 | + $this->toIndex($offset); |
|
80 | 80 | |
81 | - return $this->results[$offset]; |
|
82 | - } |
|
81 | + return $this->results[$offset]; |
|
82 | + } |
|
83 | 83 | |
84 | - private function toIndex($offset) |
|
85 | - { |
|
86 | - if ($offset < 0 || filter_var($offset, FILTER_VALIDATE_INT) === false) { |
|
87 | - throw new TDBMInvalidOffsetException('Trying to access result set using offset "'.$offset.'". An offset must be a positive integer.'); |
|
88 | - } |
|
89 | - if ($this->statement === null) { |
|
90 | - $this->executeQuery(); |
|
91 | - } |
|
92 | - while (!isset($this->results[$offset])) { |
|
93 | - $this->next(); |
|
94 | - if ($this->current === null) { |
|
95 | - throw new TDBMInvalidOffsetException('Offset "'.$offset.'" does not exist in result set.'); |
|
96 | - } |
|
97 | - } |
|
98 | - } |
|
84 | + private function toIndex($offset) |
|
85 | + { |
|
86 | + if ($offset < 0 || filter_var($offset, FILTER_VALIDATE_INT) === false) { |
|
87 | + throw new TDBMInvalidOffsetException('Trying to access result set using offset "'.$offset.'". An offset must be a positive integer.'); |
|
88 | + } |
|
89 | + if ($this->statement === null) { |
|
90 | + $this->executeQuery(); |
|
91 | + } |
|
92 | + while (!isset($this->results[$offset])) { |
|
93 | + $this->next(); |
|
94 | + if ($this->current === null) { |
|
95 | + throw new TDBMInvalidOffsetException('Offset "'.$offset.'" does not exist in result set.'); |
|
96 | + } |
|
97 | + } |
|
98 | + } |
|
99 | 99 | |
100 | - public function next() |
|
101 | - { |
|
102 | - // Let's overload the next() method to store the result. |
|
103 | - if (isset($this->results[$this->key + 1])) { |
|
104 | - ++$this->key; |
|
105 | - $this->current = $this->results[$this->key]; |
|
106 | - } else { |
|
107 | - parent::next(); |
|
108 | - if ($this->current !== null) { |
|
109 | - $this->results[$this->key] = $this->current; |
|
110 | - } |
|
111 | - } |
|
112 | - } |
|
100 | + public function next() |
|
101 | + { |
|
102 | + // Let's overload the next() method to store the result. |
|
103 | + if (isset($this->results[$this->key + 1])) { |
|
104 | + ++$this->key; |
|
105 | + $this->current = $this->results[$this->key]; |
|
106 | + } else { |
|
107 | + parent::next(); |
|
108 | + if ($this->current !== null) { |
|
109 | + $this->results[$this->key] = $this->current; |
|
110 | + } |
|
111 | + } |
|
112 | + } |
|
113 | 113 | |
114 | - /** |
|
115 | - * Overloads the rewind implementation. |
|
116 | - * Do not reexecute the query. |
|
117 | - */ |
|
118 | - public function rewind() |
|
119 | - { |
|
120 | - if (!$this->fetchStarted) { |
|
121 | - $this->executeQuery(); |
|
122 | - } |
|
123 | - $this->key = -1; |
|
124 | - $this->next(); |
|
125 | - } |
|
114 | + /** |
|
115 | + * Overloads the rewind implementation. |
|
116 | + * Do not reexecute the query. |
|
117 | + */ |
|
118 | + public function rewind() |
|
119 | + { |
|
120 | + if (!$this->fetchStarted) { |
|
121 | + $this->executeQuery(); |
|
122 | + } |
|
123 | + $this->key = -1; |
|
124 | + $this->next(); |
|
125 | + } |
|
126 | 126 | } |
@@ -33,42 +33,42 @@ |
||
33 | 33 | */ |
34 | 34 | class TDBMObject extends AbstractTDBMObject |
35 | 35 | { |
36 | - public function getProperty($var, $tableName = null) |
|
37 | - { |
|
38 | - return $this->get($var, $tableName); |
|
39 | - } |
|
36 | + public function getProperty($var, $tableName = null) |
|
37 | + { |
|
38 | + return $this->get($var, $tableName); |
|
39 | + } |
|
40 | 40 | |
41 | - public function setProperty($var, $value, $tableName = null) |
|
42 | - { |
|
43 | - $this->set($var, $value, $tableName); |
|
44 | - } |
|
41 | + public function setProperty($var, $value, $tableName = null) |
|
42 | + { |
|
43 | + $this->set($var, $value, $tableName); |
|
44 | + } |
|
45 | 45 | |
46 | - /** |
|
47 | - * Specify data which should be serialized to JSON. |
|
48 | - * |
|
49 | - * @link http://php.net/manual/en/jsonserializable.jsonserialize.php |
|
50 | - * |
|
51 | - * @return mixed data which can be serialized by <b>json_encode</b>, |
|
52 | - * which is a value of any type other than a resource |
|
53 | - * |
|
54 | - * @since 5.4.0 |
|
55 | - */ |
|
56 | - public function jsonSerialize() |
|
57 | - { |
|
58 | - throw new TDBMException('Json serialization is only implemented for generated beans.'); |
|
59 | - } |
|
46 | + /** |
|
47 | + * Specify data which should be serialized to JSON. |
|
48 | + * |
|
49 | + * @link http://php.net/manual/en/jsonserializable.jsonserialize.php |
|
50 | + * |
|
51 | + * @return mixed data which can be serialized by <b>json_encode</b>, |
|
52 | + * which is a value of any type other than a resource |
|
53 | + * |
|
54 | + * @since 5.4.0 |
|
55 | + */ |
|
56 | + public function jsonSerialize() |
|
57 | + { |
|
58 | + throw new TDBMException('Json serialization is only implemented for generated beans.'); |
|
59 | + } |
|
60 | 60 | |
61 | - /** |
|
62 | - * Returns an array of used tables by this bean (from parent to child relationship). |
|
63 | - * |
|
64 | - * @return string[] |
|
65 | - */ |
|
66 | - protected function getUsedTables() |
|
67 | - { |
|
68 | - $tableNames = array_keys($this->dbRows); |
|
69 | - $tableNames = $this->tdbmService->_getLinkBetweenInheritedTables($tableNames); |
|
70 | - $tableNames = array_reverse($tableNames); |
|
61 | + /** |
|
62 | + * Returns an array of used tables by this bean (from parent to child relationship). |
|
63 | + * |
|
64 | + * @return string[] |
|
65 | + */ |
|
66 | + protected function getUsedTables() |
|
67 | + { |
|
68 | + $tableNames = array_keys($this->dbRows); |
|
69 | + $tableNames = $this->tdbmService->_getLinkBetweenInheritedTables($tableNames); |
|
70 | + $tableNames = array_reverse($tableNames); |
|
71 | 71 | |
72 | - return $tableNames; |
|
73 | - } |
|
72 | + return $tableNames; |
|
73 | + } |
|
74 | 74 | } |
@@ -17,202 +17,202 @@ discard block |
||
17 | 17 | */ |
18 | 18 | class TDBMDaoGenerator |
19 | 19 | { |
20 | - /** |
|
21 | - * @var SchemaAnalyzer |
|
22 | - */ |
|
23 | - private $schemaAnalyzer; |
|
24 | - |
|
25 | - /** |
|
26 | - * @var Schema |
|
27 | - */ |
|
28 | - private $schema; |
|
29 | - |
|
30 | - /** |
|
31 | - * The root directory of the project. |
|
32 | - * |
|
33 | - * @var string |
|
34 | - */ |
|
35 | - private $rootPath; |
|
36 | - |
|
37 | - /** |
|
38 | - * Name of composer file. |
|
39 | - * |
|
40 | - * @var string |
|
41 | - */ |
|
42 | - private $composerFile; |
|
43 | - |
|
44 | - /** |
|
45 | - * @var TDBMSchemaAnalyzer |
|
46 | - */ |
|
47 | - private $tdbmSchemaAnalyzer; |
|
48 | - |
|
49 | - /** |
|
50 | - * Constructor. |
|
51 | - * |
|
52 | - * @param SchemaAnalyzer $schemaAnalyzer |
|
53 | - * @param Schema $schema |
|
54 | - * @param TDBMSchemaAnalyzer $tdbmSchemaAnalyzer |
|
55 | - */ |
|
56 | - public function __construct(SchemaAnalyzer $schemaAnalyzer, Schema $schema, TDBMSchemaAnalyzer $tdbmSchemaAnalyzer) |
|
57 | - { |
|
58 | - $this->schemaAnalyzer = $schemaAnalyzer; |
|
59 | - $this->schema = $schema; |
|
60 | - $this->tdbmSchemaAnalyzer = $tdbmSchemaAnalyzer; |
|
61 | - $this->rootPath = __DIR__.'/../../../../../../../../'; |
|
62 | - $this->composerFile = 'composer.json'; |
|
63 | - } |
|
64 | - |
|
65 | - /** |
|
66 | - * Generates all the daos and beans. |
|
67 | - * |
|
68 | - * @param string $daoFactoryClassName The classe name of the DAO factory |
|
69 | - * @param string $daonamespace The namespace for the DAOs, without trailing \ |
|
70 | - * @param string $beannamespace The Namespace for the beans, without trailing \ |
|
71 | - * @param bool $storeInUtc If the generated daos should store the date in UTC timezone instead of user's timezone |
|
72 | - * |
|
73 | - * @return \string[] the list of tables |
|
74 | - * |
|
75 | - * @throws TDBMException |
|
76 | - */ |
|
77 | - public function generateAllDaosAndBeans($daoFactoryClassName, $daonamespace, $beannamespace, $storeInUtc) |
|
78 | - { |
|
79 | - $classNameMapper = ClassNameMapper::createFromComposerFile($this->rootPath.$this->composerFile); |
|
80 | - // TODO: check that no class name ends with "Base". Otherwise, there will be name clash. |
|
81 | - |
|
82 | - $tableList = $this->schema->getTables(); |
|
83 | - |
|
84 | - // Remove all beans and daos from junction tables |
|
85 | - $junctionTables = $this->schemaAnalyzer->detectJunctionTables(true); |
|
86 | - $junctionTableNames = array_map(function (Table $table) { |
|
87 | - return $table->getName(); |
|
88 | - }, $junctionTables); |
|
89 | - |
|
90 | - $tableList = array_filter($tableList, function (Table $table) use ($junctionTableNames) { |
|
91 | - return !in_array($table->getName(), $junctionTableNames); |
|
92 | - }); |
|
93 | - |
|
94 | - foreach ($tableList as $table) { |
|
95 | - $this->generateDaoAndBean($table, $daonamespace, $beannamespace, $classNameMapper, $storeInUtc); |
|
96 | - } |
|
97 | - |
|
98 | - $this->generateFactory($tableList, $daoFactoryClassName, $daonamespace, $classNameMapper); |
|
99 | - |
|
100 | - // Ok, let's return the list of all tables. |
|
101 | - // These will be used by the calling script to create Mouf instances. |
|
102 | - |
|
103 | - return array_map(function (Table $table) { |
|
104 | - return $table->getName(); |
|
105 | - }, $tableList); |
|
106 | - } |
|
107 | - |
|
108 | - /** |
|
109 | - * Generates in one method call the daos and the beans for one table. |
|
110 | - * |
|
111 | - * @param Table $table |
|
112 | - * @param string $daonamespace |
|
113 | - * @param string $beannamespace |
|
114 | - * @param ClassNameMapper $classNameMapper |
|
115 | - * @param bool $storeInUtc |
|
116 | - * |
|
117 | - * @throws TDBMException |
|
118 | - */ |
|
119 | - public function generateDaoAndBean(Table $table, $daonamespace, $beannamespace, ClassNameMapper $classNameMapper, $storeInUtc) |
|
120 | - { |
|
121 | - $tableName = $table->getName(); |
|
122 | - $daoName = $this->getDaoNameFromTableName($tableName); |
|
123 | - $beanName = $this->getBeanNameFromTableName($tableName); |
|
124 | - $baseBeanName = $this->getBaseBeanNameFromTableName($tableName); |
|
125 | - $baseDaoName = $this->getBaseDaoNameFromTableName($tableName); |
|
126 | - |
|
127 | - $beanDescriptor = new BeanDescriptor($table, $this->schemaAnalyzer, $this->schema, $this->tdbmSchemaAnalyzer); |
|
128 | - $this->generateBean($beanDescriptor, $beanName, $baseBeanName, $table, $beannamespace, $classNameMapper, $storeInUtc); |
|
129 | - $this->generateDao($beanDescriptor, $daoName, $baseDaoName, $beanName, $table, $daonamespace, $beannamespace, $classNameMapper); |
|
130 | - } |
|
131 | - |
|
132 | - /** |
|
133 | - * Returns the name of the bean class from the table name. |
|
134 | - * |
|
135 | - * @param $tableName |
|
136 | - * |
|
137 | - * @return string |
|
138 | - */ |
|
139 | - public static function getBeanNameFromTableName($tableName) |
|
140 | - { |
|
141 | - return self::toSingular(self::toCamelCase($tableName)).'Bean'; |
|
142 | - } |
|
143 | - |
|
144 | - /** |
|
145 | - * Returns the name of the DAO class from the table name. |
|
146 | - * |
|
147 | - * @param $tableName |
|
148 | - * |
|
149 | - * @return string |
|
150 | - */ |
|
151 | - public static function getDaoNameFromTableName($tableName) |
|
152 | - { |
|
153 | - return self::toSingular(self::toCamelCase($tableName)).'Dao'; |
|
154 | - } |
|
155 | - |
|
156 | - /** |
|
157 | - * Returns the name of the base bean class from the table name. |
|
158 | - * |
|
159 | - * @param $tableName |
|
160 | - * |
|
161 | - * @return string |
|
162 | - */ |
|
163 | - public static function getBaseBeanNameFromTableName($tableName) |
|
164 | - { |
|
165 | - return self::toSingular(self::toCamelCase($tableName)).'BaseBean'; |
|
166 | - } |
|
167 | - |
|
168 | - /** |
|
169 | - * Returns the name of the base DAO class from the table name. |
|
170 | - * |
|
171 | - * @param $tableName |
|
172 | - * |
|
173 | - * @return string |
|
174 | - */ |
|
175 | - public static function getBaseDaoNameFromTableName($tableName) |
|
176 | - { |
|
177 | - return self::toSingular(self::toCamelCase($tableName)).'BaseDao'; |
|
178 | - } |
|
179 | - |
|
180 | - /** |
|
181 | - * Writes the PHP bean file with all getters and setters from the table passed in parameter. |
|
182 | - * |
|
183 | - * @param BeanDescriptor $beanDescriptor |
|
184 | - * @param string $className The name of the class |
|
185 | - * @param string $baseClassName The name of the base class which will be extended (name only, no directory) |
|
186 | - * @param Table $table The table |
|
187 | - * @param string $beannamespace The namespace of the bean |
|
188 | - * @param ClassNameMapper $classNameMapper |
|
189 | - * |
|
190 | - * @throws TDBMException |
|
191 | - */ |
|
192 | - public function generateBean(BeanDescriptor $beanDescriptor, $className, $baseClassName, Table $table, $beannamespace, ClassNameMapper $classNameMapper, $storeInUtc) |
|
193 | - { |
|
194 | - $str = $beanDescriptor->generatePhpCode($beannamespace); |
|
195 | - |
|
196 | - $possibleBaseFileNames = $classNameMapper->getPossibleFileNames($beannamespace.'\\Generated\\'.$baseClassName); |
|
197 | - if (empty($possibleBaseFileNames)) { |
|
198 | - throw new TDBMException('Sorry, autoload namespace issue. The class "'.$beannamespace.'\\'.$baseClassName.'" is not autoloadable.'); |
|
199 | - } |
|
200 | - $possibleBaseFileName = $this->rootPath.$possibleBaseFileNames[0]; |
|
201 | - |
|
202 | - $this->ensureDirectoryExist($possibleBaseFileName); |
|
203 | - file_put_contents($possibleBaseFileName, $str); |
|
204 | - @chmod($possibleBaseFileName, 0664); |
|
205 | - |
|
206 | - $possibleFileNames = $classNameMapper->getPossibleFileNames($beannamespace.'\\'.$className); |
|
207 | - if (empty($possibleFileNames)) { |
|
208 | - // @codeCoverageIgnoreStart |
|
209 | - throw new TDBMException('Sorry, autoload namespace issue. The class "'.$beannamespace.'\\'.$className.'" is not autoloadable.'); |
|
210 | - // @codeCoverageIgnoreEnd |
|
211 | - } |
|
212 | - $possibleFileName = $this->rootPath.$possibleFileNames[0]; |
|
213 | - if (!file_exists($possibleFileName)) { |
|
214 | - $tableName = $table->getName(); |
|
215 | - $str = "<?php |
|
20 | + /** |
|
21 | + * @var SchemaAnalyzer |
|
22 | + */ |
|
23 | + private $schemaAnalyzer; |
|
24 | + |
|
25 | + /** |
|
26 | + * @var Schema |
|
27 | + */ |
|
28 | + private $schema; |
|
29 | + |
|
30 | + /** |
|
31 | + * The root directory of the project. |
|
32 | + * |
|
33 | + * @var string |
|
34 | + */ |
|
35 | + private $rootPath; |
|
36 | + |
|
37 | + /** |
|
38 | + * Name of composer file. |
|
39 | + * |
|
40 | + * @var string |
|
41 | + */ |
|
42 | + private $composerFile; |
|
43 | + |
|
44 | + /** |
|
45 | + * @var TDBMSchemaAnalyzer |
|
46 | + */ |
|
47 | + private $tdbmSchemaAnalyzer; |
|
48 | + |
|
49 | + /** |
|
50 | + * Constructor. |
|
51 | + * |
|
52 | + * @param SchemaAnalyzer $schemaAnalyzer |
|
53 | + * @param Schema $schema |
|
54 | + * @param TDBMSchemaAnalyzer $tdbmSchemaAnalyzer |
|
55 | + */ |
|
56 | + public function __construct(SchemaAnalyzer $schemaAnalyzer, Schema $schema, TDBMSchemaAnalyzer $tdbmSchemaAnalyzer) |
|
57 | + { |
|
58 | + $this->schemaAnalyzer = $schemaAnalyzer; |
|
59 | + $this->schema = $schema; |
|
60 | + $this->tdbmSchemaAnalyzer = $tdbmSchemaAnalyzer; |
|
61 | + $this->rootPath = __DIR__.'/../../../../../../../../'; |
|
62 | + $this->composerFile = 'composer.json'; |
|
63 | + } |
|
64 | + |
|
65 | + /** |
|
66 | + * Generates all the daos and beans. |
|
67 | + * |
|
68 | + * @param string $daoFactoryClassName The classe name of the DAO factory |
|
69 | + * @param string $daonamespace The namespace for the DAOs, without trailing \ |
|
70 | + * @param string $beannamespace The Namespace for the beans, without trailing \ |
|
71 | + * @param bool $storeInUtc If the generated daos should store the date in UTC timezone instead of user's timezone |
|
72 | + * |
|
73 | + * @return \string[] the list of tables |
|
74 | + * |
|
75 | + * @throws TDBMException |
|
76 | + */ |
|
77 | + public function generateAllDaosAndBeans($daoFactoryClassName, $daonamespace, $beannamespace, $storeInUtc) |
|
78 | + { |
|
79 | + $classNameMapper = ClassNameMapper::createFromComposerFile($this->rootPath.$this->composerFile); |
|
80 | + // TODO: check that no class name ends with "Base". Otherwise, there will be name clash. |
|
81 | + |
|
82 | + $tableList = $this->schema->getTables(); |
|
83 | + |
|
84 | + // Remove all beans and daos from junction tables |
|
85 | + $junctionTables = $this->schemaAnalyzer->detectJunctionTables(true); |
|
86 | + $junctionTableNames = array_map(function (Table $table) { |
|
87 | + return $table->getName(); |
|
88 | + }, $junctionTables); |
|
89 | + |
|
90 | + $tableList = array_filter($tableList, function (Table $table) use ($junctionTableNames) { |
|
91 | + return !in_array($table->getName(), $junctionTableNames); |
|
92 | + }); |
|
93 | + |
|
94 | + foreach ($tableList as $table) { |
|
95 | + $this->generateDaoAndBean($table, $daonamespace, $beannamespace, $classNameMapper, $storeInUtc); |
|
96 | + } |
|
97 | + |
|
98 | + $this->generateFactory($tableList, $daoFactoryClassName, $daonamespace, $classNameMapper); |
|
99 | + |
|
100 | + // Ok, let's return the list of all tables. |
|
101 | + // These will be used by the calling script to create Mouf instances. |
|
102 | + |
|
103 | + return array_map(function (Table $table) { |
|
104 | + return $table->getName(); |
|
105 | + }, $tableList); |
|
106 | + } |
|
107 | + |
|
108 | + /** |
|
109 | + * Generates in one method call the daos and the beans for one table. |
|
110 | + * |
|
111 | + * @param Table $table |
|
112 | + * @param string $daonamespace |
|
113 | + * @param string $beannamespace |
|
114 | + * @param ClassNameMapper $classNameMapper |
|
115 | + * @param bool $storeInUtc |
|
116 | + * |
|
117 | + * @throws TDBMException |
|
118 | + */ |
|
119 | + public function generateDaoAndBean(Table $table, $daonamespace, $beannamespace, ClassNameMapper $classNameMapper, $storeInUtc) |
|
120 | + { |
|
121 | + $tableName = $table->getName(); |
|
122 | + $daoName = $this->getDaoNameFromTableName($tableName); |
|
123 | + $beanName = $this->getBeanNameFromTableName($tableName); |
|
124 | + $baseBeanName = $this->getBaseBeanNameFromTableName($tableName); |
|
125 | + $baseDaoName = $this->getBaseDaoNameFromTableName($tableName); |
|
126 | + |
|
127 | + $beanDescriptor = new BeanDescriptor($table, $this->schemaAnalyzer, $this->schema, $this->tdbmSchemaAnalyzer); |
|
128 | + $this->generateBean($beanDescriptor, $beanName, $baseBeanName, $table, $beannamespace, $classNameMapper, $storeInUtc); |
|
129 | + $this->generateDao($beanDescriptor, $daoName, $baseDaoName, $beanName, $table, $daonamespace, $beannamespace, $classNameMapper); |
|
130 | + } |
|
131 | + |
|
132 | + /** |
|
133 | + * Returns the name of the bean class from the table name. |
|
134 | + * |
|
135 | + * @param $tableName |
|
136 | + * |
|
137 | + * @return string |
|
138 | + */ |
|
139 | + public static function getBeanNameFromTableName($tableName) |
|
140 | + { |
|
141 | + return self::toSingular(self::toCamelCase($tableName)).'Bean'; |
|
142 | + } |
|
143 | + |
|
144 | + /** |
|
145 | + * Returns the name of the DAO class from the table name. |
|
146 | + * |
|
147 | + * @param $tableName |
|
148 | + * |
|
149 | + * @return string |
|
150 | + */ |
|
151 | + public static function getDaoNameFromTableName($tableName) |
|
152 | + { |
|
153 | + return self::toSingular(self::toCamelCase($tableName)).'Dao'; |
|
154 | + } |
|
155 | + |
|
156 | + /** |
|
157 | + * Returns the name of the base bean class from the table name. |
|
158 | + * |
|
159 | + * @param $tableName |
|
160 | + * |
|
161 | + * @return string |
|
162 | + */ |
|
163 | + public static function getBaseBeanNameFromTableName($tableName) |
|
164 | + { |
|
165 | + return self::toSingular(self::toCamelCase($tableName)).'BaseBean'; |
|
166 | + } |
|
167 | + |
|
168 | + /** |
|
169 | + * Returns the name of the base DAO class from the table name. |
|
170 | + * |
|
171 | + * @param $tableName |
|
172 | + * |
|
173 | + * @return string |
|
174 | + */ |
|
175 | + public static function getBaseDaoNameFromTableName($tableName) |
|
176 | + { |
|
177 | + return self::toSingular(self::toCamelCase($tableName)).'BaseDao'; |
|
178 | + } |
|
179 | + |
|
180 | + /** |
|
181 | + * Writes the PHP bean file with all getters and setters from the table passed in parameter. |
|
182 | + * |
|
183 | + * @param BeanDescriptor $beanDescriptor |
|
184 | + * @param string $className The name of the class |
|
185 | + * @param string $baseClassName The name of the base class which will be extended (name only, no directory) |
|
186 | + * @param Table $table The table |
|
187 | + * @param string $beannamespace The namespace of the bean |
|
188 | + * @param ClassNameMapper $classNameMapper |
|
189 | + * |
|
190 | + * @throws TDBMException |
|
191 | + */ |
|
192 | + public function generateBean(BeanDescriptor $beanDescriptor, $className, $baseClassName, Table $table, $beannamespace, ClassNameMapper $classNameMapper, $storeInUtc) |
|
193 | + { |
|
194 | + $str = $beanDescriptor->generatePhpCode($beannamespace); |
|
195 | + |
|
196 | + $possibleBaseFileNames = $classNameMapper->getPossibleFileNames($beannamespace.'\\Generated\\'.$baseClassName); |
|
197 | + if (empty($possibleBaseFileNames)) { |
|
198 | + throw new TDBMException('Sorry, autoload namespace issue. The class "'.$beannamespace.'\\'.$baseClassName.'" is not autoloadable.'); |
|
199 | + } |
|
200 | + $possibleBaseFileName = $this->rootPath.$possibleBaseFileNames[0]; |
|
201 | + |
|
202 | + $this->ensureDirectoryExist($possibleBaseFileName); |
|
203 | + file_put_contents($possibleBaseFileName, $str); |
|
204 | + @chmod($possibleBaseFileName, 0664); |
|
205 | + |
|
206 | + $possibleFileNames = $classNameMapper->getPossibleFileNames($beannamespace.'\\'.$className); |
|
207 | + if (empty($possibleFileNames)) { |
|
208 | + // @codeCoverageIgnoreStart |
|
209 | + throw new TDBMException('Sorry, autoload namespace issue. The class "'.$beannamespace.'\\'.$className.'" is not autoloadable.'); |
|
210 | + // @codeCoverageIgnoreEnd |
|
211 | + } |
|
212 | + $possibleFileName = $this->rootPath.$possibleFileNames[0]; |
|
213 | + if (!file_exists($possibleFileName)) { |
|
214 | + $tableName = $table->getName(); |
|
215 | + $str = "<?php |
|
216 | 216 | /* |
217 | 217 | * This file has been automatically generated by TDBM. |
218 | 218 | * You can edit this file as it will not be overwritten. |
@@ -229,76 +229,76 @@ discard block |
||
229 | 229 | { |
230 | 230 | |
231 | 231 | }"; |
232 | - $this->ensureDirectoryExist($possibleFileName); |
|
233 | - file_put_contents($possibleFileName, $str); |
|
234 | - @chmod($possibleFileName, 0664); |
|
235 | - } |
|
236 | - } |
|
237 | - |
|
238 | - /** |
|
239 | - * Tries to find a @defaultSort annotation in one of the columns. |
|
240 | - * |
|
241 | - * @param Table $table |
|
242 | - * |
|
243 | - * @return array First item: column name, Second item: column order (asc/desc) |
|
244 | - */ |
|
245 | - private function getDefaultSortColumnFromAnnotation(Table $table) |
|
246 | - { |
|
247 | - $defaultSort = null; |
|
248 | - $defaultSortDirection = null; |
|
249 | - foreach ($table->getColumns() as $column) { |
|
250 | - $comments = $column->getComment(); |
|
251 | - $matches = []; |
|
252 | - if (preg_match('/@defaultSort(\((desc|asc)\))*/', $comments, $matches) != 0) { |
|
253 | - $defaultSort = $column->getName(); |
|
254 | - if (count($matches) === 3) { |
|
255 | - $defaultSortDirection = $matches[2]; |
|
256 | - } else { |
|
257 | - $defaultSortDirection = 'ASC'; |
|
258 | - } |
|
259 | - } |
|
260 | - } |
|
261 | - |
|
262 | - return [$defaultSort, $defaultSortDirection]; |
|
263 | - } |
|
264 | - |
|
265 | - /** |
|
266 | - * Writes the PHP bean DAO with simple functions to create/get/save objects. |
|
267 | - * |
|
268 | - * @param BeanDescriptor $beanDescriptor |
|
269 | - * @param string $className The name of the class |
|
270 | - * @param string $baseClassName |
|
271 | - * @param string $beanClassName |
|
272 | - * @param Table $table |
|
273 | - * @param string $daonamespace |
|
274 | - * @param string $beannamespace |
|
275 | - * @param ClassNameMapper $classNameMapper |
|
276 | - * |
|
277 | - * @throws TDBMException |
|
278 | - */ |
|
279 | - public function generateDao(BeanDescriptor $beanDescriptor, $className, $baseClassName, $beanClassName, Table $table, $daonamespace, $beannamespace, ClassNameMapper $classNameMapper) |
|
280 | - { |
|
281 | - $tableName = $table->getName(); |
|
282 | - $primaryKeyColumns = $table->getPrimaryKeyColumns(); |
|
283 | - |
|
284 | - list($defaultSort, $defaultSortDirection) = $this->getDefaultSortColumnFromAnnotation($table); |
|
285 | - |
|
286 | - // FIXME: lowercase tables with _ in the name should work! |
|
287 | - $tableCamel = self::toSingular(self::toCamelCase($tableName)); |
|
288 | - |
|
289 | - $beanClassWithoutNameSpace = $beanClassName; |
|
290 | - $beanClassName = $beannamespace.'\\'.$beanClassName; |
|
291 | - |
|
292 | - list($usedBeans, $findByDaoCode) = $beanDescriptor->generateFindByDaoCode($beannamespace, $beanClassWithoutNameSpace); |
|
293 | - |
|
294 | - $usedBeans[] = $beanClassName; |
|
295 | - // Let's suppress duplicates in used beans (if any) |
|
296 | - $usedBeans = array_flip(array_flip($usedBeans)); |
|
297 | - $useStatements = array_map(function ($usedBean) { |
|
298 | - return "use $usedBean;\n"; |
|
299 | - }, $usedBeans); |
|
300 | - |
|
301 | - $str = "<?php |
|
232 | + $this->ensureDirectoryExist($possibleFileName); |
|
233 | + file_put_contents($possibleFileName, $str); |
|
234 | + @chmod($possibleFileName, 0664); |
|
235 | + } |
|
236 | + } |
|
237 | + |
|
238 | + /** |
|
239 | + * Tries to find a @defaultSort annotation in one of the columns. |
|
240 | + * |
|
241 | + * @param Table $table |
|
242 | + * |
|
243 | + * @return array First item: column name, Second item: column order (asc/desc) |
|
244 | + */ |
|
245 | + private function getDefaultSortColumnFromAnnotation(Table $table) |
|
246 | + { |
|
247 | + $defaultSort = null; |
|
248 | + $defaultSortDirection = null; |
|
249 | + foreach ($table->getColumns() as $column) { |
|
250 | + $comments = $column->getComment(); |
|
251 | + $matches = []; |
|
252 | + if (preg_match('/@defaultSort(\((desc|asc)\))*/', $comments, $matches) != 0) { |
|
253 | + $defaultSort = $column->getName(); |
|
254 | + if (count($matches) === 3) { |
|
255 | + $defaultSortDirection = $matches[2]; |
|
256 | + } else { |
|
257 | + $defaultSortDirection = 'ASC'; |
|
258 | + } |
|
259 | + } |
|
260 | + } |
|
261 | + |
|
262 | + return [$defaultSort, $defaultSortDirection]; |
|
263 | + } |
|
264 | + |
|
265 | + /** |
|
266 | + * Writes the PHP bean DAO with simple functions to create/get/save objects. |
|
267 | + * |
|
268 | + * @param BeanDescriptor $beanDescriptor |
|
269 | + * @param string $className The name of the class |
|
270 | + * @param string $baseClassName |
|
271 | + * @param string $beanClassName |
|
272 | + * @param Table $table |
|
273 | + * @param string $daonamespace |
|
274 | + * @param string $beannamespace |
|
275 | + * @param ClassNameMapper $classNameMapper |
|
276 | + * |
|
277 | + * @throws TDBMException |
|
278 | + */ |
|
279 | + public function generateDao(BeanDescriptor $beanDescriptor, $className, $baseClassName, $beanClassName, Table $table, $daonamespace, $beannamespace, ClassNameMapper $classNameMapper) |
|
280 | + { |
|
281 | + $tableName = $table->getName(); |
|
282 | + $primaryKeyColumns = $table->getPrimaryKeyColumns(); |
|
283 | + |
|
284 | + list($defaultSort, $defaultSortDirection) = $this->getDefaultSortColumnFromAnnotation($table); |
|
285 | + |
|
286 | + // FIXME: lowercase tables with _ in the name should work! |
|
287 | + $tableCamel = self::toSingular(self::toCamelCase($tableName)); |
|
288 | + |
|
289 | + $beanClassWithoutNameSpace = $beanClassName; |
|
290 | + $beanClassName = $beannamespace.'\\'.$beanClassName; |
|
291 | + |
|
292 | + list($usedBeans, $findByDaoCode) = $beanDescriptor->generateFindByDaoCode($beannamespace, $beanClassWithoutNameSpace); |
|
293 | + |
|
294 | + $usedBeans[] = $beanClassName; |
|
295 | + // Let's suppress duplicates in used beans (if any) |
|
296 | + $usedBeans = array_flip(array_flip($usedBeans)); |
|
297 | + $useStatements = array_map(function ($usedBean) { |
|
298 | + return "use $usedBean;\n"; |
|
299 | + }, $usedBeans); |
|
300 | + |
|
301 | + $str = "<?php |
|
302 | 302 | |
303 | 303 | /* |
304 | 304 | * This file has been automatically generated by TDBM. |
@@ -375,10 +375,10 @@ discard block |
||
375 | 375 | } |
376 | 376 | "; |
377 | 377 | |
378 | - if (count($primaryKeyColumns) === 1) { |
|
379 | - $primaryKeyColumn = $primaryKeyColumns[0]; |
|
380 | - $primaryKeyPhpType = self::dbalTypeToPhpType($table->getColumn($primaryKeyColumn)->getType()); |
|
381 | - $str .= " |
|
378 | + if (count($primaryKeyColumns) === 1) { |
|
379 | + $primaryKeyColumn = $primaryKeyColumns[0]; |
|
380 | + $primaryKeyPhpType = self::dbalTypeToPhpType($table->getColumn($primaryKeyColumn)->getType()); |
|
381 | + $str .= " |
|
382 | 382 | /** |
383 | 383 | * Get $beanClassWithoutNameSpace specified by its ID (its primary key) |
384 | 384 | * If the primary key does not exist, an exception is thrown. |
@@ -393,8 +393,8 @@ discard block |
||
393 | 393 | return \$this->tdbmService->findObjectByPk('$tableName', ['$primaryKeyColumn' => \$id], [], \$lazyLoading); |
394 | 394 | } |
395 | 395 | "; |
396 | - } |
|
397 | - $str .= " |
|
396 | + } |
|
397 | + $str .= " |
|
398 | 398 | /** |
399 | 399 | * Deletes the $beanClassWithoutNameSpace passed in parameter. |
400 | 400 | * |
@@ -493,33 +493,33 @@ discard block |
||
493 | 493 | } |
494 | 494 | "; |
495 | 495 | |
496 | - $str .= $findByDaoCode; |
|
497 | - $str .= '} |
|
496 | + $str .= $findByDaoCode; |
|
497 | + $str .= '} |
|
498 | 498 | '; |
499 | 499 | |
500 | - $possibleBaseFileNames = $classNameMapper->getPossibleFileNames($daonamespace.'\\Generated\\'.$baseClassName); |
|
501 | - if (empty($possibleBaseFileNames)) { |
|
502 | - // @codeCoverageIgnoreStart |
|
503 | - throw new TDBMException('Sorry, autoload namespace issue. The class "'.$baseClassName.'" is not autoloadable.'); |
|
504 | - // @codeCoverageIgnoreEnd |
|
505 | - } |
|
506 | - $possibleBaseFileName = $this->rootPath.$possibleBaseFileNames[0]; |
|
507 | - |
|
508 | - $this->ensureDirectoryExist($possibleBaseFileName); |
|
509 | - file_put_contents($possibleBaseFileName, $str); |
|
510 | - @chmod($possibleBaseFileName, 0664); |
|
511 | - |
|
512 | - $possibleFileNames = $classNameMapper->getPossibleFileNames($daonamespace.'\\'.$className); |
|
513 | - if (empty($possibleFileNames)) { |
|
514 | - // @codeCoverageIgnoreStart |
|
515 | - throw new TDBMException('Sorry, autoload namespace issue. The class "'.$className.'" is not autoloadable.'); |
|
516 | - // @codeCoverageIgnoreEnd |
|
517 | - } |
|
518 | - $possibleFileName = $this->rootPath.$possibleFileNames[0]; |
|
519 | - |
|
520 | - // Now, let's generate the "editable" class |
|
521 | - if (!file_exists($possibleFileName)) { |
|
522 | - $str = "<?php |
|
500 | + $possibleBaseFileNames = $classNameMapper->getPossibleFileNames($daonamespace.'\\Generated\\'.$baseClassName); |
|
501 | + if (empty($possibleBaseFileNames)) { |
|
502 | + // @codeCoverageIgnoreStart |
|
503 | + throw new TDBMException('Sorry, autoload namespace issue. The class "'.$baseClassName.'" is not autoloadable.'); |
|
504 | + // @codeCoverageIgnoreEnd |
|
505 | + } |
|
506 | + $possibleBaseFileName = $this->rootPath.$possibleBaseFileNames[0]; |
|
507 | + |
|
508 | + $this->ensureDirectoryExist($possibleBaseFileName); |
|
509 | + file_put_contents($possibleBaseFileName, $str); |
|
510 | + @chmod($possibleBaseFileName, 0664); |
|
511 | + |
|
512 | + $possibleFileNames = $classNameMapper->getPossibleFileNames($daonamespace.'\\'.$className); |
|
513 | + if (empty($possibleFileNames)) { |
|
514 | + // @codeCoverageIgnoreStart |
|
515 | + throw new TDBMException('Sorry, autoload namespace issue. The class "'.$className.'" is not autoloadable.'); |
|
516 | + // @codeCoverageIgnoreEnd |
|
517 | + } |
|
518 | + $possibleFileName = $this->rootPath.$possibleFileNames[0]; |
|
519 | + |
|
520 | + // Now, let's generate the "editable" class |
|
521 | + if (!file_exists($possibleFileName)) { |
|
522 | + $str = "<?php |
|
523 | 523 | |
524 | 524 | /* |
525 | 525 | * This file has been automatically generated by TDBM. |
@@ -538,22 +538,22 @@ discard block |
||
538 | 538 | |
539 | 539 | } |
540 | 540 | "; |
541 | - $this->ensureDirectoryExist($possibleFileName); |
|
542 | - file_put_contents($possibleFileName, $str); |
|
543 | - @chmod($possibleFileName, 0664); |
|
544 | - } |
|
545 | - } |
|
546 | - |
|
547 | - /** |
|
548 | - * Generates the factory bean. |
|
549 | - * |
|
550 | - * @param Table[] $tableList |
|
551 | - */ |
|
552 | - private function generateFactory(array $tableList, $daoFactoryClassName, $daoNamespace, ClassNameMapper $classNameMapper) |
|
553 | - { |
|
554 | - // For each table, let's write a property. |
|
555 | - |
|
556 | - $str = "<?php |
|
541 | + $this->ensureDirectoryExist($possibleFileName); |
|
542 | + file_put_contents($possibleFileName, $str); |
|
543 | + @chmod($possibleFileName, 0664); |
|
544 | + } |
|
545 | + } |
|
546 | + |
|
547 | + /** |
|
548 | + * Generates the factory bean. |
|
549 | + * |
|
550 | + * @param Table[] $tableList |
|
551 | + */ |
|
552 | + private function generateFactory(array $tableList, $daoFactoryClassName, $daoNamespace, ClassNameMapper $classNameMapper) |
|
553 | + { |
|
554 | + // For each table, let's write a property. |
|
555 | + |
|
556 | + $str = "<?php |
|
557 | 557 | |
558 | 558 | /* |
559 | 559 | * This file has been automatically generated by TDBM. |
@@ -562,13 +562,13 @@ discard block |
||
562 | 562 | |
563 | 563 | namespace {$daoNamespace}\\Generated; |
564 | 564 | "; |
565 | - foreach ($tableList as $table) { |
|
566 | - $tableName = $table->getName(); |
|
567 | - $daoClassName = $this->getDaoNameFromTableName($tableName); |
|
568 | - $str .= "use {$daoNamespace}\\".$daoClassName.";\n"; |
|
569 | - } |
|
565 | + foreach ($tableList as $table) { |
|
566 | + $tableName = $table->getName(); |
|
567 | + $daoClassName = $this->getDaoNameFromTableName($tableName); |
|
568 | + $str .= "use {$daoNamespace}\\".$daoClassName.";\n"; |
|
569 | + } |
|
570 | 570 | |
571 | - $str .= " |
|
571 | + $str .= " |
|
572 | 572 | /** |
573 | 573 | * The $daoFactoryClassName provides an easy access to all DAOs generated by TDBM. |
574 | 574 | * |
@@ -577,12 +577,12 @@ discard block |
||
577 | 577 | { |
578 | 578 | "; |
579 | 579 | |
580 | - foreach ($tableList as $table) { |
|
581 | - $tableName = $table->getName(); |
|
582 | - $daoClassName = $this->getDaoNameFromTableName($tableName); |
|
583 | - $daoInstanceName = self::toVariableName($daoClassName); |
|
580 | + foreach ($tableList as $table) { |
|
581 | + $tableName = $table->getName(); |
|
582 | + $daoClassName = $this->getDaoNameFromTableName($tableName); |
|
583 | + $daoInstanceName = self::toVariableName($daoClassName); |
|
584 | 584 | |
585 | - $str .= ' /** |
|
585 | + $str .= ' /** |
|
586 | 586 | * @var '.$daoClassName.' |
587 | 587 | */ |
588 | 588 | private $'.$daoInstanceName.'; |
@@ -607,158 +607,158 @@ discard block |
||
607 | 607 | } |
608 | 608 | |
609 | 609 | '; |
610 | - } |
|
610 | + } |
|
611 | 611 | |
612 | - $str .= ' |
|
612 | + $str .= ' |
|
613 | 613 | } |
614 | 614 | '; |
615 | 615 | |
616 | - $possibleFileNames = $classNameMapper->getPossibleFileNames($daoNamespace.'\\Generated\\'.$daoFactoryClassName); |
|
617 | - if (empty($possibleFileNames)) { |
|
618 | - throw new TDBMException('Sorry, autoload namespace issue. The class "'.$daoNamespace.'\\'.$daoFactoryClassName.'" is not autoloadable.'); |
|
619 | - } |
|
620 | - $possibleFileName = $this->rootPath.$possibleFileNames[0]; |
|
621 | - |
|
622 | - $this->ensureDirectoryExist($possibleFileName); |
|
623 | - file_put_contents($possibleFileName, $str); |
|
624 | - @chmod($possibleFileName, 0664); |
|
625 | - } |
|
626 | - |
|
627 | - /** |
|
628 | - * Transforms a string to camelCase (except the first letter will be uppercase too). |
|
629 | - * Underscores and spaces are removed and the first letter after the underscore is uppercased. |
|
630 | - * |
|
631 | - * @param $str string |
|
632 | - * |
|
633 | - * @return string |
|
634 | - */ |
|
635 | - public static function toCamelCase($str) |
|
636 | - { |
|
637 | - $str = strtoupper(substr($str, 0, 1)).substr($str, 1); |
|
638 | - while (true) { |
|
639 | - if (strpos($str, '_') === false && strpos($str, ' ') === false) { |
|
640 | - break; |
|
641 | - } |
|
642 | - |
|
643 | - $pos = strpos($str, '_'); |
|
644 | - if ($pos === false) { |
|
645 | - $pos = strpos($str, ' '); |
|
646 | - } |
|
647 | - $before = substr($str, 0, $pos); |
|
648 | - $after = substr($str, $pos + 1); |
|
649 | - $str = $before.strtoupper(substr($after, 0, 1)).substr($after, 1); |
|
650 | - } |
|
651 | - |
|
652 | - return $str; |
|
653 | - } |
|
654 | - |
|
655 | - /** |
|
656 | - * Tries to put string to the singular form (if it is plural). |
|
657 | - * We assume the table names are in english. |
|
658 | - * |
|
659 | - * @param $str string |
|
660 | - * |
|
661 | - * @return string |
|
662 | - */ |
|
663 | - public static function toSingular($str) |
|
664 | - { |
|
665 | - return Inflector::singularize($str); |
|
666 | - } |
|
667 | - |
|
668 | - /** |
|
669 | - * Put the first letter of the string in lower case. |
|
670 | - * Very useful to transform a class name into a variable name. |
|
671 | - * |
|
672 | - * @param $str string |
|
673 | - * |
|
674 | - * @return string |
|
675 | - */ |
|
676 | - public static function toVariableName($str) |
|
677 | - { |
|
678 | - return strtolower(substr($str, 0, 1)).substr($str, 1); |
|
679 | - } |
|
680 | - |
|
681 | - /** |
|
682 | - * Ensures the file passed in parameter can be written in its directory. |
|
683 | - * |
|
684 | - * @param string $fileName |
|
685 | - * |
|
686 | - * @throws TDBMException |
|
687 | - */ |
|
688 | - private function ensureDirectoryExist($fileName) |
|
689 | - { |
|
690 | - $dirName = dirname($fileName); |
|
691 | - if (!file_exists($dirName)) { |
|
692 | - $old = umask(0); |
|
693 | - $result = mkdir($dirName, 0775, true); |
|
694 | - umask($old); |
|
695 | - if ($result === false) { |
|
696 | - throw new TDBMException("Unable to create directory: '".$dirName."'."); |
|
697 | - } |
|
698 | - } |
|
699 | - } |
|
700 | - |
|
701 | - /** |
|
702 | - * Absolute path to composer json file. |
|
703 | - * |
|
704 | - * @param string $composerFile |
|
705 | - */ |
|
706 | - public function setComposerFile($composerFile) |
|
707 | - { |
|
708 | - $this->rootPath = dirname($composerFile).'/'; |
|
709 | - $this->composerFile = basename($composerFile); |
|
710 | - } |
|
711 | - |
|
712 | - /** |
|
713 | - * Transforms a DBAL type into a PHP type (for PHPDoc purpose). |
|
714 | - * |
|
715 | - * @param Type $type The DBAL type |
|
716 | - * |
|
717 | - * @return string The PHP type |
|
718 | - */ |
|
719 | - public static function dbalTypeToPhpType(Type $type) |
|
720 | - { |
|
721 | - $map = [ |
|
722 | - Type::TARRAY => 'array', |
|
723 | - Type::SIMPLE_ARRAY => 'array', |
|
724 | - Type::JSON_ARRAY => 'array', |
|
725 | - Type::BIGINT => 'string', |
|
726 | - Type::BOOLEAN => 'bool', |
|
727 | - Type::DATETIME => '\DateTimeInterface', |
|
728 | - Type::DATETIMETZ => '\DateTimeInterface', |
|
729 | - Type::DATE => '\DateTimeInterface', |
|
730 | - Type::TIME => '\DateTimeInterface', |
|
731 | - Type::DECIMAL => 'float', |
|
732 | - Type::INTEGER => 'int', |
|
733 | - Type::OBJECT => 'string', |
|
734 | - Type::SMALLINT => 'int', |
|
735 | - Type::STRING => 'string', |
|
736 | - Type::TEXT => 'string', |
|
737 | - Type::BINARY => 'string', |
|
738 | - Type::BLOB => 'string', |
|
739 | - Type::FLOAT => 'float', |
|
740 | - Type::GUID => 'string', |
|
741 | - ]; |
|
742 | - |
|
743 | - return isset($map[$type->getName()]) ? $map[$type->getName()] : $type->getName(); |
|
744 | - } |
|
745 | - |
|
746 | - /** |
|
747 | - * @param string $beanNamespace |
|
748 | - * |
|
749 | - * @return \string[] Returns a map mapping table name to beans name |
|
750 | - */ |
|
751 | - public function buildTableToBeanMap($beanNamespace) |
|
752 | - { |
|
753 | - $tableToBeanMap = []; |
|
754 | - |
|
755 | - $tables = $this->schema->getTables(); |
|
756 | - |
|
757 | - foreach ($tables as $table) { |
|
758 | - $tableName = $table->getName(); |
|
759 | - $tableToBeanMap[$tableName] = $beanNamespace.'\\'.self::getBeanNameFromTableName($tableName); |
|
760 | - } |
|
761 | - |
|
762 | - return $tableToBeanMap; |
|
763 | - } |
|
616 | + $possibleFileNames = $classNameMapper->getPossibleFileNames($daoNamespace.'\\Generated\\'.$daoFactoryClassName); |
|
617 | + if (empty($possibleFileNames)) { |
|
618 | + throw new TDBMException('Sorry, autoload namespace issue. The class "'.$daoNamespace.'\\'.$daoFactoryClassName.'" is not autoloadable.'); |
|
619 | + } |
|
620 | + $possibleFileName = $this->rootPath.$possibleFileNames[0]; |
|
621 | + |
|
622 | + $this->ensureDirectoryExist($possibleFileName); |
|
623 | + file_put_contents($possibleFileName, $str); |
|
624 | + @chmod($possibleFileName, 0664); |
|
625 | + } |
|
626 | + |
|
627 | + /** |
|
628 | + * Transforms a string to camelCase (except the first letter will be uppercase too). |
|
629 | + * Underscores and spaces are removed and the first letter after the underscore is uppercased. |
|
630 | + * |
|
631 | + * @param $str string |
|
632 | + * |
|
633 | + * @return string |
|
634 | + */ |
|
635 | + public static function toCamelCase($str) |
|
636 | + { |
|
637 | + $str = strtoupper(substr($str, 0, 1)).substr($str, 1); |
|
638 | + while (true) { |
|
639 | + if (strpos($str, '_') === false && strpos($str, ' ') === false) { |
|
640 | + break; |
|
641 | + } |
|
642 | + |
|
643 | + $pos = strpos($str, '_'); |
|
644 | + if ($pos === false) { |
|
645 | + $pos = strpos($str, ' '); |
|
646 | + } |
|
647 | + $before = substr($str, 0, $pos); |
|
648 | + $after = substr($str, $pos + 1); |
|
649 | + $str = $before.strtoupper(substr($after, 0, 1)).substr($after, 1); |
|
650 | + } |
|
651 | + |
|
652 | + return $str; |
|
653 | + } |
|
654 | + |
|
655 | + /** |
|
656 | + * Tries to put string to the singular form (if it is plural). |
|
657 | + * We assume the table names are in english. |
|
658 | + * |
|
659 | + * @param $str string |
|
660 | + * |
|
661 | + * @return string |
|
662 | + */ |
|
663 | + public static function toSingular($str) |
|
664 | + { |
|
665 | + return Inflector::singularize($str); |
|
666 | + } |
|
667 | + |
|
668 | + /** |
|
669 | + * Put the first letter of the string in lower case. |
|
670 | + * Very useful to transform a class name into a variable name. |
|
671 | + * |
|
672 | + * @param $str string |
|
673 | + * |
|
674 | + * @return string |
|
675 | + */ |
|
676 | + public static function toVariableName($str) |
|
677 | + { |
|
678 | + return strtolower(substr($str, 0, 1)).substr($str, 1); |
|
679 | + } |
|
680 | + |
|
681 | + /** |
|
682 | + * Ensures the file passed in parameter can be written in its directory. |
|
683 | + * |
|
684 | + * @param string $fileName |
|
685 | + * |
|
686 | + * @throws TDBMException |
|
687 | + */ |
|
688 | + private function ensureDirectoryExist($fileName) |
|
689 | + { |
|
690 | + $dirName = dirname($fileName); |
|
691 | + if (!file_exists($dirName)) { |
|
692 | + $old = umask(0); |
|
693 | + $result = mkdir($dirName, 0775, true); |
|
694 | + umask($old); |
|
695 | + if ($result === false) { |
|
696 | + throw new TDBMException("Unable to create directory: '".$dirName."'."); |
|
697 | + } |
|
698 | + } |
|
699 | + } |
|
700 | + |
|
701 | + /** |
|
702 | + * Absolute path to composer json file. |
|
703 | + * |
|
704 | + * @param string $composerFile |
|
705 | + */ |
|
706 | + public function setComposerFile($composerFile) |
|
707 | + { |
|
708 | + $this->rootPath = dirname($composerFile).'/'; |
|
709 | + $this->composerFile = basename($composerFile); |
|
710 | + } |
|
711 | + |
|
712 | + /** |
|
713 | + * Transforms a DBAL type into a PHP type (for PHPDoc purpose). |
|
714 | + * |
|
715 | + * @param Type $type The DBAL type |
|
716 | + * |
|
717 | + * @return string The PHP type |
|
718 | + */ |
|
719 | + public static function dbalTypeToPhpType(Type $type) |
|
720 | + { |
|
721 | + $map = [ |
|
722 | + Type::TARRAY => 'array', |
|
723 | + Type::SIMPLE_ARRAY => 'array', |
|
724 | + Type::JSON_ARRAY => 'array', |
|
725 | + Type::BIGINT => 'string', |
|
726 | + Type::BOOLEAN => 'bool', |
|
727 | + Type::DATETIME => '\DateTimeInterface', |
|
728 | + Type::DATETIMETZ => '\DateTimeInterface', |
|
729 | + Type::DATE => '\DateTimeInterface', |
|
730 | + Type::TIME => '\DateTimeInterface', |
|
731 | + Type::DECIMAL => 'float', |
|
732 | + Type::INTEGER => 'int', |
|
733 | + Type::OBJECT => 'string', |
|
734 | + Type::SMALLINT => 'int', |
|
735 | + Type::STRING => 'string', |
|
736 | + Type::TEXT => 'string', |
|
737 | + Type::BINARY => 'string', |
|
738 | + Type::BLOB => 'string', |
|
739 | + Type::FLOAT => 'float', |
|
740 | + Type::GUID => 'string', |
|
741 | + ]; |
|
742 | + |
|
743 | + return isset($map[$type->getName()]) ? $map[$type->getName()] : $type->getName(); |
|
744 | + } |
|
745 | + |
|
746 | + /** |
|
747 | + * @param string $beanNamespace |
|
748 | + * |
|
749 | + * @return \string[] Returns a map mapping table name to beans name |
|
750 | + */ |
|
751 | + public function buildTableToBeanMap($beanNamespace) |
|
752 | + { |
|
753 | + $tableToBeanMap = []; |
|
754 | + |
|
755 | + $tables = $this->schema->getTables(); |
|
756 | + |
|
757 | + foreach ($tables as $table) { |
|
758 | + $tableName = $table->getName(); |
|
759 | + $tableToBeanMap[$tableName] = $beanNamespace.'\\'.self::getBeanNameFromTableName($tableName); |
|
760 | + } |
|
761 | + |
|
762 | + return $tableToBeanMap; |
|
763 | + } |
|
764 | 764 | } |
@@ -29,267 +29,267 @@ |
||
29 | 29 | */ |
30 | 30 | class InnerResultIterator implements \Iterator, \Countable, \ArrayAccess |
31 | 31 | { |
32 | - /** |
|
33 | - * @var Statement |
|
34 | - */ |
|
35 | - protected $statement; |
|
36 | - |
|
37 | - protected $fetchStarted = false; |
|
38 | - private $objectStorage; |
|
39 | - private $className; |
|
40 | - |
|
41 | - private $tdbmService; |
|
42 | - private $magicSql; |
|
43 | - private $parameters; |
|
44 | - private $limit; |
|
45 | - private $offset; |
|
46 | - private $columnDescriptors; |
|
47 | - private $magicQuery; |
|
48 | - |
|
49 | - /** |
|
50 | - * The key of the current retrieved object. |
|
51 | - * |
|
52 | - * @var int |
|
53 | - */ |
|
54 | - protected $key = -1; |
|
55 | - |
|
56 | - protected $current = null; |
|
57 | - |
|
58 | - private $databasePlatform; |
|
59 | - |
|
60 | - /** |
|
61 | - * @var LoggerInterface |
|
62 | - */ |
|
63 | - private $logger; |
|
64 | - |
|
65 | - public function __construct($magicSql, array $parameters, $limit, $offset, array $columnDescriptors, $objectStorage, $className, TDBMService $tdbmService, MagicQuery $magicQuery, LoggerInterface $logger) |
|
66 | - { |
|
67 | - $this->magicSql = $magicSql; |
|
68 | - $this->objectStorage = $objectStorage; |
|
69 | - $this->className = $className; |
|
70 | - $this->tdbmService = $tdbmService; |
|
71 | - $this->parameters = $parameters; |
|
72 | - $this->limit = $limit; |
|
73 | - $this->offset = $offset; |
|
74 | - $this->columnDescriptors = $columnDescriptors; |
|
75 | - $this->magicQuery = $magicQuery; |
|
76 | - $this->databasePlatform = $this->tdbmService->getConnection()->getDatabasePlatform(); |
|
77 | - $this->logger = $logger; |
|
78 | - } |
|
79 | - |
|
80 | - protected function executeQuery() |
|
81 | - { |
|
82 | - $sql = $this->magicQuery->build($this->magicSql, $this->parameters); |
|
83 | - $sql = $this->tdbmService->getConnection()->getDatabasePlatform()->modifyLimitQuery($sql, $this->limit, $this->offset); |
|
84 | - |
|
85 | - $this->logger->debug('Running SQL request: '.$sql); |
|
86 | - |
|
87 | - $this->statement = $this->tdbmService->getConnection()->executeQuery($sql, $this->parameters); |
|
88 | - |
|
89 | - $this->fetchStarted = true; |
|
90 | - } |
|
91 | - |
|
92 | - /** |
|
93 | - * Counts found records (this is the number of records fetched, taking into account the LIMIT and OFFSET settings). |
|
94 | - * |
|
95 | - * @return int |
|
96 | - */ |
|
97 | - public function count() |
|
98 | - { |
|
99 | - if (!$this->fetchStarted) { |
|
100 | - $this->executeQuery(); |
|
101 | - } |
|
102 | - |
|
103 | - return $this->statement->rowCount(); |
|
104 | - } |
|
105 | - |
|
106 | - /** |
|
107 | - * Fetches record at current cursor. |
|
108 | - * |
|
109 | - * @return AbstractTDBMObject|null |
|
110 | - */ |
|
111 | - public function current() |
|
112 | - { |
|
113 | - return $this->current; |
|
114 | - } |
|
115 | - |
|
116 | - /** |
|
117 | - * Returns the current result's key. |
|
118 | - * |
|
119 | - * @return int |
|
120 | - */ |
|
121 | - public function key() |
|
122 | - { |
|
123 | - return $this->key; |
|
124 | - } |
|
125 | - |
|
126 | - /** |
|
127 | - * Advances the cursor to the next result. |
|
128 | - * Casts the database result into one (or several) beans. |
|
129 | - */ |
|
130 | - public function next() |
|
131 | - { |
|
132 | - $row = $this->statement->fetch(\PDO::FETCH_NUM); |
|
133 | - if ($row) { |
|
134 | - |
|
135 | - // array<tablegroup, array<table, array<column, value>>> |
|
136 | - $beansData = []; |
|
137 | - foreach ($row as $i => $value) { |
|
138 | - $columnDescriptor = $this->columnDescriptors[$i]; |
|
139 | - |
|
140 | - if ($columnDescriptor['tableGroup'] === null) { |
|
141 | - // A column can have no tableGroup (if it comes from an ORDER BY expression) |
|
142 | - continue; |
|
143 | - } |
|
144 | - |
|
145 | - // Let's cast the value according to its type |
|
146 | - $value = $columnDescriptor['type']->convertToPHPValue($value, $this->databasePlatform); |
|
147 | - |
|
148 | - $beansData[$columnDescriptor['tableGroup']][$columnDescriptor['table']][$columnDescriptor['column']] = $value; |
|
149 | - } |
|
150 | - |
|
151 | - $reflectionClassCache = []; |
|
152 | - $firstBean = true; |
|
153 | - foreach ($beansData as $beanData) { |
|
154 | - |
|
155 | - // Let's find the bean class name associated to the bean. |
|
156 | - |
|
157 | - list($actualClassName, $mainBeanTableName, $tablesUsed) = $this->tdbmService->_getClassNameFromBeanData($beanData); |
|
158 | - |
|
159 | - if ($this->className !== null) { |
|
160 | - $actualClassName = $this->className; |
|
161 | - } |
|
162 | - |
|
163 | - // Let's filter out the beanData that is not used (because it belongs to a part of the hierarchy that is not fetched: |
|
164 | - foreach ($beanData as $tableName => $descriptors) { |
|
165 | - if (!in_array($tableName, $tablesUsed)) { |
|
166 | - unset($beanData[$tableName]); |
|
167 | - } |
|
168 | - } |
|
169 | - |
|
170 | - // Must we create the bean? Let's see in the cache if we have a mapping DbRow? |
|
171 | - // Let's get the first object mapping a row: |
|
172 | - // We do this loop only for the first table |
|
173 | - |
|
174 | - $primaryKeys = $this->tdbmService->_getPrimaryKeysFromObjectData($mainBeanTableName, $beanData[$mainBeanTableName]); |
|
175 | - $hash = $this->tdbmService->getObjectHash($primaryKeys); |
|
176 | - |
|
177 | - if ($this->objectStorage->has($mainBeanTableName, $hash)) { |
|
178 | - $dbRow = $this->objectStorage->get($mainBeanTableName, $hash); |
|
179 | - $bean = $dbRow->getTDBMObject(); |
|
180 | - } else { |
|
181 | - // Let's construct the bean |
|
182 | - if (!isset($reflectionClassCache[$actualClassName])) { |
|
183 | - $reflectionClassCache[$actualClassName] = new \ReflectionClass($actualClassName); |
|
184 | - } |
|
185 | - // Let's bypass the constructor when creating the bean! |
|
186 | - $bean = $reflectionClassCache[$actualClassName]->newInstanceWithoutConstructor(); |
|
187 | - $bean->_constructFromData($beanData, $this->tdbmService); |
|
188 | - } |
|
189 | - |
|
190 | - // The first bean is the one containing the main table. |
|
191 | - if ($firstBean) { |
|
192 | - $firstBean = false; |
|
193 | - $this->current = $bean; |
|
194 | - } |
|
195 | - } |
|
196 | - |
|
197 | - ++$this->key; |
|
198 | - } else { |
|
199 | - $this->current = null; |
|
200 | - } |
|
201 | - } |
|
202 | - |
|
203 | - /** |
|
204 | - * Moves the cursor to the beginning of the result set. |
|
205 | - */ |
|
206 | - public function rewind() |
|
207 | - { |
|
208 | - $this->executeQuery(); |
|
209 | - $this->key = -1; |
|
210 | - $this->next(); |
|
211 | - } |
|
212 | - /** |
|
213 | - * Checks if the cursor is reading a valid result. |
|
214 | - * |
|
215 | - * @return bool |
|
216 | - */ |
|
217 | - public function valid() |
|
218 | - { |
|
219 | - return $this->current !== null; |
|
220 | - } |
|
221 | - |
|
222 | - /** |
|
223 | - * Whether a offset exists. |
|
224 | - * |
|
225 | - * @link http://php.net/manual/en/arrayaccess.offsetexists.php |
|
226 | - * |
|
227 | - * @param mixed $offset <p> |
|
228 | - * An offset to check for. |
|
229 | - * </p> |
|
230 | - * |
|
231 | - * @return bool true on success or false on failure. |
|
232 | - * </p> |
|
233 | - * <p> |
|
234 | - * The return value will be casted to boolean if non-boolean was returned |
|
235 | - * |
|
236 | - * @since 5.0.0 |
|
237 | - */ |
|
238 | - public function offsetExists($offset) |
|
239 | - { |
|
240 | - throw new TDBMInvalidOperationException('You cannot access this result set via index because it was fetched in CURSOR mode. Use ARRAY_MODE instead.'); |
|
241 | - } |
|
242 | - |
|
243 | - /** |
|
244 | - * Offset to retrieve. |
|
245 | - * |
|
246 | - * @link http://php.net/manual/en/arrayaccess.offsetget.php |
|
247 | - * |
|
248 | - * @param mixed $offset <p> |
|
249 | - * The offset to retrieve. |
|
250 | - * </p> |
|
251 | - * |
|
252 | - * @return mixed Can return all value types |
|
253 | - * |
|
254 | - * @since 5.0.0 |
|
255 | - */ |
|
256 | - public function offsetGet($offset) |
|
257 | - { |
|
258 | - throw new TDBMInvalidOperationException('You cannot access this result set via index because it was fetched in CURSOR mode. Use ARRAY_MODE instead.'); |
|
259 | - } |
|
260 | - |
|
261 | - /** |
|
262 | - * Offset to set. |
|
263 | - * |
|
264 | - * @link http://php.net/manual/en/arrayaccess.offsetset.php |
|
265 | - * |
|
266 | - * @param mixed $offset <p> |
|
267 | - * The offset to assign the value to. |
|
268 | - * </p> |
|
269 | - * @param mixed $value <p> |
|
270 | - * The value to set. |
|
271 | - * </p> |
|
272 | - * |
|
273 | - * @since 5.0.0 |
|
274 | - */ |
|
275 | - public function offsetSet($offset, $value) |
|
276 | - { |
|
277 | - throw new TDBMInvalidOperationException('You can set values in a TDBM result set.'); |
|
278 | - } |
|
279 | - |
|
280 | - /** |
|
281 | - * Offset to unset. |
|
282 | - * |
|
283 | - * @link http://php.net/manual/en/arrayaccess.offsetunset.php |
|
284 | - * |
|
285 | - * @param mixed $offset <p> |
|
286 | - * The offset to unset. |
|
287 | - * </p> |
|
288 | - * |
|
289 | - * @since 5.0.0 |
|
290 | - */ |
|
291 | - public function offsetUnset($offset) |
|
292 | - { |
|
293 | - throw new TDBMInvalidOperationException('You can unset values in a TDBM result set.'); |
|
294 | - } |
|
32 | + /** |
|
33 | + * @var Statement |
|
34 | + */ |
|
35 | + protected $statement; |
|
36 | + |
|
37 | + protected $fetchStarted = false; |
|
38 | + private $objectStorage; |
|
39 | + private $className; |
|
40 | + |
|
41 | + private $tdbmService; |
|
42 | + private $magicSql; |
|
43 | + private $parameters; |
|
44 | + private $limit; |
|
45 | + private $offset; |
|
46 | + private $columnDescriptors; |
|
47 | + private $magicQuery; |
|
48 | + |
|
49 | + /** |
|
50 | + * The key of the current retrieved object. |
|
51 | + * |
|
52 | + * @var int |
|
53 | + */ |
|
54 | + protected $key = -1; |
|
55 | + |
|
56 | + protected $current = null; |
|
57 | + |
|
58 | + private $databasePlatform; |
|
59 | + |
|
60 | + /** |
|
61 | + * @var LoggerInterface |
|
62 | + */ |
|
63 | + private $logger; |
|
64 | + |
|
65 | + public function __construct($magicSql, array $parameters, $limit, $offset, array $columnDescriptors, $objectStorage, $className, TDBMService $tdbmService, MagicQuery $magicQuery, LoggerInterface $logger) |
|
66 | + { |
|
67 | + $this->magicSql = $magicSql; |
|
68 | + $this->objectStorage = $objectStorage; |
|
69 | + $this->className = $className; |
|
70 | + $this->tdbmService = $tdbmService; |
|
71 | + $this->parameters = $parameters; |
|
72 | + $this->limit = $limit; |
|
73 | + $this->offset = $offset; |
|
74 | + $this->columnDescriptors = $columnDescriptors; |
|
75 | + $this->magicQuery = $magicQuery; |
|
76 | + $this->databasePlatform = $this->tdbmService->getConnection()->getDatabasePlatform(); |
|
77 | + $this->logger = $logger; |
|
78 | + } |
|
79 | + |
|
80 | + protected function executeQuery() |
|
81 | + { |
|
82 | + $sql = $this->magicQuery->build($this->magicSql, $this->parameters); |
|
83 | + $sql = $this->tdbmService->getConnection()->getDatabasePlatform()->modifyLimitQuery($sql, $this->limit, $this->offset); |
|
84 | + |
|
85 | + $this->logger->debug('Running SQL request: '.$sql); |
|
86 | + |
|
87 | + $this->statement = $this->tdbmService->getConnection()->executeQuery($sql, $this->parameters); |
|
88 | + |
|
89 | + $this->fetchStarted = true; |
|
90 | + } |
|
91 | + |
|
92 | + /** |
|
93 | + * Counts found records (this is the number of records fetched, taking into account the LIMIT and OFFSET settings). |
|
94 | + * |
|
95 | + * @return int |
|
96 | + */ |
|
97 | + public function count() |
|
98 | + { |
|
99 | + if (!$this->fetchStarted) { |
|
100 | + $this->executeQuery(); |
|
101 | + } |
|
102 | + |
|
103 | + return $this->statement->rowCount(); |
|
104 | + } |
|
105 | + |
|
106 | + /** |
|
107 | + * Fetches record at current cursor. |
|
108 | + * |
|
109 | + * @return AbstractTDBMObject|null |
|
110 | + */ |
|
111 | + public function current() |
|
112 | + { |
|
113 | + return $this->current; |
|
114 | + } |
|
115 | + |
|
116 | + /** |
|
117 | + * Returns the current result's key. |
|
118 | + * |
|
119 | + * @return int |
|
120 | + */ |
|
121 | + public function key() |
|
122 | + { |
|
123 | + return $this->key; |
|
124 | + } |
|
125 | + |
|
126 | + /** |
|
127 | + * Advances the cursor to the next result. |
|
128 | + * Casts the database result into one (or several) beans. |
|
129 | + */ |
|
130 | + public function next() |
|
131 | + { |
|
132 | + $row = $this->statement->fetch(\PDO::FETCH_NUM); |
|
133 | + if ($row) { |
|
134 | + |
|
135 | + // array<tablegroup, array<table, array<column, value>>> |
|
136 | + $beansData = []; |
|
137 | + foreach ($row as $i => $value) { |
|
138 | + $columnDescriptor = $this->columnDescriptors[$i]; |
|
139 | + |
|
140 | + if ($columnDescriptor['tableGroup'] === null) { |
|
141 | + // A column can have no tableGroup (if it comes from an ORDER BY expression) |
|
142 | + continue; |
|
143 | + } |
|
144 | + |
|
145 | + // Let's cast the value according to its type |
|
146 | + $value = $columnDescriptor['type']->convertToPHPValue($value, $this->databasePlatform); |
|
147 | + |
|
148 | + $beansData[$columnDescriptor['tableGroup']][$columnDescriptor['table']][$columnDescriptor['column']] = $value; |
|
149 | + } |
|
150 | + |
|
151 | + $reflectionClassCache = []; |
|
152 | + $firstBean = true; |
|
153 | + foreach ($beansData as $beanData) { |
|
154 | + |
|
155 | + // Let's find the bean class name associated to the bean. |
|
156 | + |
|
157 | + list($actualClassName, $mainBeanTableName, $tablesUsed) = $this->tdbmService->_getClassNameFromBeanData($beanData); |
|
158 | + |
|
159 | + if ($this->className !== null) { |
|
160 | + $actualClassName = $this->className; |
|
161 | + } |
|
162 | + |
|
163 | + // Let's filter out the beanData that is not used (because it belongs to a part of the hierarchy that is not fetched: |
|
164 | + foreach ($beanData as $tableName => $descriptors) { |
|
165 | + if (!in_array($tableName, $tablesUsed)) { |
|
166 | + unset($beanData[$tableName]); |
|
167 | + } |
|
168 | + } |
|
169 | + |
|
170 | + // Must we create the bean? Let's see in the cache if we have a mapping DbRow? |
|
171 | + // Let's get the first object mapping a row: |
|
172 | + // We do this loop only for the first table |
|
173 | + |
|
174 | + $primaryKeys = $this->tdbmService->_getPrimaryKeysFromObjectData($mainBeanTableName, $beanData[$mainBeanTableName]); |
|
175 | + $hash = $this->tdbmService->getObjectHash($primaryKeys); |
|
176 | + |
|
177 | + if ($this->objectStorage->has($mainBeanTableName, $hash)) { |
|
178 | + $dbRow = $this->objectStorage->get($mainBeanTableName, $hash); |
|
179 | + $bean = $dbRow->getTDBMObject(); |
|
180 | + } else { |
|
181 | + // Let's construct the bean |
|
182 | + if (!isset($reflectionClassCache[$actualClassName])) { |
|
183 | + $reflectionClassCache[$actualClassName] = new \ReflectionClass($actualClassName); |
|
184 | + } |
|
185 | + // Let's bypass the constructor when creating the bean! |
|
186 | + $bean = $reflectionClassCache[$actualClassName]->newInstanceWithoutConstructor(); |
|
187 | + $bean->_constructFromData($beanData, $this->tdbmService); |
|
188 | + } |
|
189 | + |
|
190 | + // The first bean is the one containing the main table. |
|
191 | + if ($firstBean) { |
|
192 | + $firstBean = false; |
|
193 | + $this->current = $bean; |
|
194 | + } |
|
195 | + } |
|
196 | + |
|
197 | + ++$this->key; |
|
198 | + } else { |
|
199 | + $this->current = null; |
|
200 | + } |
|
201 | + } |
|
202 | + |
|
203 | + /** |
|
204 | + * Moves the cursor to the beginning of the result set. |
|
205 | + */ |
|
206 | + public function rewind() |
|
207 | + { |
|
208 | + $this->executeQuery(); |
|
209 | + $this->key = -1; |
|
210 | + $this->next(); |
|
211 | + } |
|
212 | + /** |
|
213 | + * Checks if the cursor is reading a valid result. |
|
214 | + * |
|
215 | + * @return bool |
|
216 | + */ |
|
217 | + public function valid() |
|
218 | + { |
|
219 | + return $this->current !== null; |
|
220 | + } |
|
221 | + |
|
222 | + /** |
|
223 | + * Whether a offset exists. |
|
224 | + * |
|
225 | + * @link http://php.net/manual/en/arrayaccess.offsetexists.php |
|
226 | + * |
|
227 | + * @param mixed $offset <p> |
|
228 | + * An offset to check for. |
|
229 | + * </p> |
|
230 | + * |
|
231 | + * @return bool true on success or false on failure. |
|
232 | + * </p> |
|
233 | + * <p> |
|
234 | + * The return value will be casted to boolean if non-boolean was returned |
|
235 | + * |
|
236 | + * @since 5.0.0 |
|
237 | + */ |
|
238 | + public function offsetExists($offset) |
|
239 | + { |
|
240 | + throw new TDBMInvalidOperationException('You cannot access this result set via index because it was fetched in CURSOR mode. Use ARRAY_MODE instead.'); |
|
241 | + } |
|
242 | + |
|
243 | + /** |
|
244 | + * Offset to retrieve. |
|
245 | + * |
|
246 | + * @link http://php.net/manual/en/arrayaccess.offsetget.php |
|
247 | + * |
|
248 | + * @param mixed $offset <p> |
|
249 | + * The offset to retrieve. |
|
250 | + * </p> |
|
251 | + * |
|
252 | + * @return mixed Can return all value types |
|
253 | + * |
|
254 | + * @since 5.0.0 |
|
255 | + */ |
|
256 | + public function offsetGet($offset) |
|
257 | + { |
|
258 | + throw new TDBMInvalidOperationException('You cannot access this result set via index because it was fetched in CURSOR mode. Use ARRAY_MODE instead.'); |
|
259 | + } |
|
260 | + |
|
261 | + /** |
|
262 | + * Offset to set. |
|
263 | + * |
|
264 | + * @link http://php.net/manual/en/arrayaccess.offsetset.php |
|
265 | + * |
|
266 | + * @param mixed $offset <p> |
|
267 | + * The offset to assign the value to. |
|
268 | + * </p> |
|
269 | + * @param mixed $value <p> |
|
270 | + * The value to set. |
|
271 | + * </p> |
|
272 | + * |
|
273 | + * @since 5.0.0 |
|
274 | + */ |
|
275 | + public function offsetSet($offset, $value) |
|
276 | + { |
|
277 | + throw new TDBMInvalidOperationException('You can set values in a TDBM result set.'); |
|
278 | + } |
|
279 | + |
|
280 | + /** |
|
281 | + * Offset to unset. |
|
282 | + * |
|
283 | + * @link http://php.net/manual/en/arrayaccess.offsetunset.php |
|
284 | + * |
|
285 | + * @param mixed $offset <p> |
|
286 | + * The offset to unset. |
|
287 | + * </p> |
|
288 | + * |
|
289 | + * @since 5.0.0 |
|
290 | + */ |
|
291 | + public function offsetUnset($offset) |
|
292 | + { |
|
293 | + throw new TDBMInvalidOperationException('You can unset values in a TDBM result set.'); |
|
294 | + } |
|
295 | 295 | } |
@@ -12,122 +12,122 @@ |
||
12 | 12 | */ |
13 | 13 | class OrderByAnalyzer |
14 | 14 | { |
15 | - /** |
|
16 | - * The content of the cache variable. |
|
17 | - * |
|
18 | - * @var Cache |
|
19 | - */ |
|
20 | - private $cache; |
|
21 | - |
|
22 | - /** |
|
23 | - * @var string |
|
24 | - */ |
|
25 | - private $cachePrefix; |
|
26 | - |
|
27 | - /** |
|
28 | - * OrderByAnalyzer constructor. |
|
29 | - * |
|
30 | - * @param Cache $cache |
|
31 | - * @param string|null $cachePrefix |
|
32 | - */ |
|
33 | - public function __construct(Cache $cache, $cachePrefix = null) |
|
34 | - { |
|
35 | - $this->cache = $cache; |
|
36 | - $this->cachePrefix = $cachePrefix; |
|
37 | - } |
|
38 | - |
|
39 | - /** |
|
40 | - * Returns an array for each sorted "column" in the form:. |
|
41 | - * |
|
42 | - * [ |
|
43 | - * [ |
|
44 | - * 'type' => 'colref', |
|
45 | - * 'table' => null, |
|
46 | - * 'column' => 'a', |
|
47 | - * 'direction' => 'ASC' |
|
48 | - * ], |
|
49 | - * [ |
|
50 | - * 'type' => 'expr', |
|
51 | - * 'expr' => 'RAND()', |
|
52 | - * 'direction' => 'DESC' |
|
53 | - * ] |
|
54 | - * ] |
|
55 | - * |
|
56 | - * @param string $orderBy |
|
57 | - * |
|
58 | - * @return array |
|
59 | - */ |
|
60 | - public function analyzeOrderBy(string $orderBy) : array |
|
61 | - { |
|
62 | - $key = $this->cachePrefix.'_order_by_analysis_'.$orderBy; |
|
63 | - $results = $this->cache->fetch($key); |
|
64 | - if ($results !== false) { |
|
65 | - return $results; |
|
66 | - } |
|
67 | - $results = $this->analyzeOrderByNoCache($orderBy); |
|
68 | - $this->cache->save($key, $results); |
|
69 | - |
|
70 | - return $results; |
|
71 | - } |
|
72 | - |
|
73 | - private function analyzeOrderByNoCache(string $orderBy) : array |
|
74 | - { |
|
75 | - $sqlParser = new PHPSQLParser(); |
|
76 | - $sql = 'SELECT 1 FROM a ORDER BY '.$orderBy; |
|
77 | - $parsed = $sqlParser->parse($sql, true); |
|
78 | - |
|
79 | - $results = []; |
|
80 | - |
|
81 | - for ($i = 0, $count = count($parsed['ORDER']); $i < $count; ++$i) { |
|
82 | - $orderItem = $parsed['ORDER'][$i]; |
|
83 | - if ($orderItem['expr_type'] === 'colref') { |
|
84 | - $parts = $orderItem['no_quotes']['parts']; |
|
85 | - $columnName = array_pop($parts); |
|
86 | - if (!empty($parts)) { |
|
87 | - $tableName = array_pop($parts); |
|
88 | - } else { |
|
89 | - $tableName = null; |
|
90 | - } |
|
91 | - |
|
92 | - $results[] = [ |
|
93 | - 'type' => 'colref', |
|
94 | - 'table' => $tableName, |
|
95 | - 'column' => $columnName, |
|
96 | - 'direction' => $orderItem['direction'], |
|
97 | - ]; |
|
98 | - } else { |
|
99 | - $position = $orderItem['position']; |
|
100 | - if ($i + 1 < $count) { |
|
101 | - $nextPosition = $parsed['ORDER'][$i + 1]['position']; |
|
102 | - $str = substr($sql, $position, $nextPosition - $position); |
|
103 | - } else { |
|
104 | - $str = substr($sql, $position); |
|
105 | - } |
|
106 | - |
|
107 | - $str = trim($str, " \t\r\n,"); |
|
108 | - |
|
109 | - $results[] = [ |
|
110 | - 'type' => 'expr', |
|
111 | - 'expr' => $this->trimDirection($str), |
|
112 | - 'direction' => $orderItem['direction'], |
|
113 | - ]; |
|
114 | - } |
|
115 | - } |
|
116 | - |
|
117 | - return $results; |
|
118 | - } |
|
119 | - |
|
120 | - /** |
|
121 | - * Trims the ASC/DESC direction at the end of the string. |
|
122 | - * |
|
123 | - * @param string $sql |
|
124 | - * |
|
125 | - * @return string |
|
126 | - */ |
|
127 | - private function trimDirection(string $sql) : string |
|
128 | - { |
|
129 | - preg_match('/^(.*)(\s+(DESC|ASC|))*$/Ui', $sql, $matches); |
|
130 | - |
|
131 | - return $matches[1]; |
|
132 | - } |
|
15 | + /** |
|
16 | + * The content of the cache variable. |
|
17 | + * |
|
18 | + * @var Cache |
|
19 | + */ |
|
20 | + private $cache; |
|
21 | + |
|
22 | + /** |
|
23 | + * @var string |
|
24 | + */ |
|
25 | + private $cachePrefix; |
|
26 | + |
|
27 | + /** |
|
28 | + * OrderByAnalyzer constructor. |
|
29 | + * |
|
30 | + * @param Cache $cache |
|
31 | + * @param string|null $cachePrefix |
|
32 | + */ |
|
33 | + public function __construct(Cache $cache, $cachePrefix = null) |
|
34 | + { |
|
35 | + $this->cache = $cache; |
|
36 | + $this->cachePrefix = $cachePrefix; |
|
37 | + } |
|
38 | + |
|
39 | + /** |
|
40 | + * Returns an array for each sorted "column" in the form:. |
|
41 | + * |
|
42 | + * [ |
|
43 | + * [ |
|
44 | + * 'type' => 'colref', |
|
45 | + * 'table' => null, |
|
46 | + * 'column' => 'a', |
|
47 | + * 'direction' => 'ASC' |
|
48 | + * ], |
|
49 | + * [ |
|
50 | + * 'type' => 'expr', |
|
51 | + * 'expr' => 'RAND()', |
|
52 | + * 'direction' => 'DESC' |
|
53 | + * ] |
|
54 | + * ] |
|
55 | + * |
|
56 | + * @param string $orderBy |
|
57 | + * |
|
58 | + * @return array |
|
59 | + */ |
|
60 | + public function analyzeOrderBy(string $orderBy) : array |
|
61 | + { |
|
62 | + $key = $this->cachePrefix.'_order_by_analysis_'.$orderBy; |
|
63 | + $results = $this->cache->fetch($key); |
|
64 | + if ($results !== false) { |
|
65 | + return $results; |
|
66 | + } |
|
67 | + $results = $this->analyzeOrderByNoCache($orderBy); |
|
68 | + $this->cache->save($key, $results); |
|
69 | + |
|
70 | + return $results; |
|
71 | + } |
|
72 | + |
|
73 | + private function analyzeOrderByNoCache(string $orderBy) : array |
|
74 | + { |
|
75 | + $sqlParser = new PHPSQLParser(); |
|
76 | + $sql = 'SELECT 1 FROM a ORDER BY '.$orderBy; |
|
77 | + $parsed = $sqlParser->parse($sql, true); |
|
78 | + |
|
79 | + $results = []; |
|
80 | + |
|
81 | + for ($i = 0, $count = count($parsed['ORDER']); $i < $count; ++$i) { |
|
82 | + $orderItem = $parsed['ORDER'][$i]; |
|
83 | + if ($orderItem['expr_type'] === 'colref') { |
|
84 | + $parts = $orderItem['no_quotes']['parts']; |
|
85 | + $columnName = array_pop($parts); |
|
86 | + if (!empty($parts)) { |
|
87 | + $tableName = array_pop($parts); |
|
88 | + } else { |
|
89 | + $tableName = null; |
|
90 | + } |
|
91 | + |
|
92 | + $results[] = [ |
|
93 | + 'type' => 'colref', |
|
94 | + 'table' => $tableName, |
|
95 | + 'column' => $columnName, |
|
96 | + 'direction' => $orderItem['direction'], |
|
97 | + ]; |
|
98 | + } else { |
|
99 | + $position = $orderItem['position']; |
|
100 | + if ($i + 1 < $count) { |
|
101 | + $nextPosition = $parsed['ORDER'][$i + 1]['position']; |
|
102 | + $str = substr($sql, $position, $nextPosition - $position); |
|
103 | + } else { |
|
104 | + $str = substr($sql, $position); |
|
105 | + } |
|
106 | + |
|
107 | + $str = trim($str, " \t\r\n,"); |
|
108 | + |
|
109 | + $results[] = [ |
|
110 | + 'type' => 'expr', |
|
111 | + 'expr' => $this->trimDirection($str), |
|
112 | + 'direction' => $orderItem['direction'], |
|
113 | + ]; |
|
114 | + } |
|
115 | + } |
|
116 | + |
|
117 | + return $results; |
|
118 | + } |
|
119 | + |
|
120 | + /** |
|
121 | + * Trims the ASC/DESC direction at the end of the string. |
|
122 | + * |
|
123 | + * @param string $sql |
|
124 | + * |
|
125 | + * @return string |
|
126 | + */ |
|
127 | + private function trimDirection(string $sql) : string |
|
128 | + { |
|
129 | + preg_match('/^(.*)(\s+(DESC|ASC|))*$/Ui', $sql, $matches); |
|
130 | + |
|
131 | + return $matches[1]; |
|
132 | + } |
|
133 | 133 | } |
@@ -20,24 +20,24 @@ |
||
20 | 20 | */ |
21 | 21 | class UncheckedOrderBy |
22 | 22 | { |
23 | - /** |
|
24 | - * @var string |
|
25 | - */ |
|
26 | - private $orderBy; |
|
23 | + /** |
|
24 | + * @var string |
|
25 | + */ |
|
26 | + private $orderBy; |
|
27 | 27 | |
28 | - /** |
|
29 | - * @param $orderBy |
|
30 | - */ |
|
31 | - public function __construct(string $orderBy) |
|
32 | - { |
|
33 | - $this->orderBy = $orderBy; |
|
34 | - } |
|
28 | + /** |
|
29 | + * @param $orderBy |
|
30 | + */ |
|
31 | + public function __construct(string $orderBy) |
|
32 | + { |
|
33 | + $this->orderBy = $orderBy; |
|
34 | + } |
|
35 | 35 | |
36 | - /** |
|
37 | - * @return string |
|
38 | - */ |
|
39 | - public function getOrderBy() : string |
|
40 | - { |
|
41 | - return $this->orderBy; |
|
42 | - } |
|
36 | + /** |
|
37 | + * @return string |
|
38 | + */ |
|
39 | + public function getOrderBy() : string |
|
40 | + { |
|
41 | + return $this->orderBy; |
|
42 | + } |
|
43 | 43 | } |
@@ -11,44 +11,44 @@ |
||
11 | 11 | */ |
12 | 12 | class FindObjectsQueryFactory extends AbstractQueryFactory |
13 | 13 | { |
14 | - private $mainTable; |
|
15 | - private $additionalTablesFetch; |
|
16 | - private $filterString; |
|
17 | - |
|
18 | - public function __construct(string $mainTable, array $additionalTablesFetch, $filterString, $orderBy, TDBMService $tdbmService, Schema $schema, OrderByAnalyzer $orderByAnalyzer) |
|
19 | - { |
|
20 | - parent::__construct($tdbmService, $schema, $orderByAnalyzer, $orderBy); |
|
21 | - $this->mainTable = $mainTable; |
|
22 | - $this->additionalTablesFetch = $additionalTablesFetch; |
|
23 | - $this->filterString = $filterString; |
|
24 | - } |
|
25 | - |
|
26 | - protected function compute() |
|
27 | - { |
|
28 | - list($columnDescList, $columnsList, $orderString) = $this->getColumnsList($this->mainTable, $this->additionalTablesFetch, $this->orderBy); |
|
29 | - |
|
30 | - $sql = 'SELECT DISTINCT '.implode(', ', $columnsList).' FROM MAGICJOIN('.$this->mainTable.')'; |
|
31 | - |
|
32 | - $pkColumnNames = $this->schema->getTable($this->mainTable)->getPrimaryKeyColumns(); |
|
33 | - $pkColumnNames = array_map(function ($pkColumn) { |
|
34 | - return $this->tdbmService->getConnection()->quoteIdentifier($this->mainTable).'.'.$this->tdbmService->getConnection()->quoteIdentifier($pkColumn); |
|
35 | - }, $pkColumnNames); |
|
36 | - |
|
37 | - $countSql = 'SELECT COUNT(DISTINCT '.implode(', ', $pkColumnNames).') FROM MAGICJOIN('.$this->mainTable.')'; |
|
38 | - |
|
39 | - if (!empty($this->filterString)) { |
|
40 | - $sql .= ' WHERE '.$this->filterString; |
|
41 | - $countSql .= ' WHERE '.$this->filterString; |
|
42 | - } |
|
43 | - |
|
44 | - if (!empty($orderString)) { |
|
45 | - $sql .= ' ORDER BY '.$orderString; |
|
46 | - } |
|
47 | - |
|
48 | - $this->magicSql = $sql; |
|
49 | - $this->magicSqlCount = $countSql; |
|
50 | - $this->columnDescList = $columnDescList; |
|
51 | - } |
|
14 | + private $mainTable; |
|
15 | + private $additionalTablesFetch; |
|
16 | + private $filterString; |
|
17 | + |
|
18 | + public function __construct(string $mainTable, array $additionalTablesFetch, $filterString, $orderBy, TDBMService $tdbmService, Schema $schema, OrderByAnalyzer $orderByAnalyzer) |
|
19 | + { |
|
20 | + parent::__construct($tdbmService, $schema, $orderByAnalyzer, $orderBy); |
|
21 | + $this->mainTable = $mainTable; |
|
22 | + $this->additionalTablesFetch = $additionalTablesFetch; |
|
23 | + $this->filterString = $filterString; |
|
24 | + } |
|
25 | + |
|
26 | + protected function compute() |
|
27 | + { |
|
28 | + list($columnDescList, $columnsList, $orderString) = $this->getColumnsList($this->mainTable, $this->additionalTablesFetch, $this->orderBy); |
|
29 | + |
|
30 | + $sql = 'SELECT DISTINCT '.implode(', ', $columnsList).' FROM MAGICJOIN('.$this->mainTable.')'; |
|
31 | + |
|
32 | + $pkColumnNames = $this->schema->getTable($this->mainTable)->getPrimaryKeyColumns(); |
|
33 | + $pkColumnNames = array_map(function ($pkColumn) { |
|
34 | + return $this->tdbmService->getConnection()->quoteIdentifier($this->mainTable).'.'.$this->tdbmService->getConnection()->quoteIdentifier($pkColumn); |
|
35 | + }, $pkColumnNames); |
|
36 | + |
|
37 | + $countSql = 'SELECT COUNT(DISTINCT '.implode(', ', $pkColumnNames).') FROM MAGICJOIN('.$this->mainTable.')'; |
|
38 | + |
|
39 | + if (!empty($this->filterString)) { |
|
40 | + $sql .= ' WHERE '.$this->filterString; |
|
41 | + $countSql .= ' WHERE '.$this->filterString; |
|
42 | + } |
|
43 | + |
|
44 | + if (!empty($orderString)) { |
|
45 | + $sql .= ' ORDER BY '.$orderString; |
|
46 | + } |
|
47 | + |
|
48 | + $this->magicSql = $sql; |
|
49 | + $this->magicSqlCount = $countSql; |
|
50 | + $this->columnDescList = $columnDescList; |
|
51 | + } |
|
52 | 52 | |
53 | 53 | |
54 | 54 | } |
@@ -9,25 +9,25 @@ |
||
9 | 9 | */ |
10 | 10 | interface QueryFactory |
11 | 11 | { |
12 | - /** |
|
13 | - * Sets the ORDER BY directive executed in SQL. |
|
14 | - * |
|
15 | - * For instance: |
|
16 | - * |
|
17 | - * $queryFactory->sort('label ASC, status DESC'); |
|
18 | - * |
|
19 | - * **Important:** TDBM does its best to protect you from SQL injection. In particular, it will only allow column names in the "ORDER BY" clause. This means you are safe to pass input from the user directly in the ORDER BY parameter. |
|
20 | - * If you want to pass an expression to the ORDER BY clause, you will need to tell TDBM to stop checking for SQL injections. You do this by passing a `UncheckedOrderBy` object as a parameter: |
|
21 | - * |
|
22 | - * $queryFactory->sort(new UncheckedOrderBy('RAND()')) |
|
23 | - * |
|
24 | - * @param string|UncheckedOrderBy|null $orderBy |
|
25 | - */ |
|
26 | - public function sort($orderBy); |
|
12 | + /** |
|
13 | + * Sets the ORDER BY directive executed in SQL. |
|
14 | + * |
|
15 | + * For instance: |
|
16 | + * |
|
17 | + * $queryFactory->sort('label ASC, status DESC'); |
|
18 | + * |
|
19 | + * **Important:** TDBM does its best to protect you from SQL injection. In particular, it will only allow column names in the "ORDER BY" clause. This means you are safe to pass input from the user directly in the ORDER BY parameter. |
|
20 | + * If you want to pass an expression to the ORDER BY clause, you will need to tell TDBM to stop checking for SQL injections. You do this by passing a `UncheckedOrderBy` object as a parameter: |
|
21 | + * |
|
22 | + * $queryFactory->sort(new UncheckedOrderBy('RAND()')) |
|
23 | + * |
|
24 | + * @param string|UncheckedOrderBy|null $orderBy |
|
25 | + */ |
|
26 | + public function sort($orderBy); |
|
27 | 27 | |
28 | - public function getMagicSql() : string; |
|
28 | + public function getMagicSql() : string; |
|
29 | 29 | |
30 | - public function getMagicSqlCount() : string; |
|
30 | + public function getMagicSqlCount() : string; |
|
31 | 31 | |
32 | - public function getColumnDescriptors() : array; |
|
32 | + public function getColumnDescriptors() : array; |
|
33 | 33 | } |