| 
									
										
										
										
											2020-03-15 21:21:11 +00:00
										 |  |  | <?php | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-21 20:18:05 +00:00
										 |  |  | /* | 
					
						
							|  |  |  |  * This file is part of GNU social - https://www.gnu.org/software/social | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * GNU social is free software: you can redistribute it and/or modify | 
					
						
							|  |  |  |  * it under the terms of the GNU Affero General Public License as published by | 
					
						
							|  |  |  |  * the Free Software Foundation, either version 3 of the License, or | 
					
						
							|  |  |  |  * (at your option) any later version. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * GNU social is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							|  |  |  |  * GNU Affero General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU Affero General Public License | 
					
						
							|  |  |  |  * along with GNU social.  If not, see <http://www.gnu.org/licenses/>. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Doctrine metadata driver which implements our old `schemaDef` interface | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @package GNUsocial | 
					
						
							|  |  |  |  * @category DB | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @author    Hugo Sales <hugo@fc.up.pt> | 
					
						
							|  |  |  |  * @copyright 2020 Free Software Foundation, Inc http://www.fsf.org | 
					
						
							|  |  |  |  * @license   https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-15 21:21:11 +00:00
										 |  |  | namespace App\Util; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | use Doctrine\Persistence\Mapping\ClassMetadata; | 
					
						
							|  |  |  | use Doctrine\Persistence\Mapping\Driver\StaticPHPDriver; | 
					
						
							| 
									
										
										
										
											2020-03-28 15:40:28 +00:00
										 |  |  | use Functional as F; | 
					
						
							| 
									
										
										
										
											2020-03-15 21:21:11 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | class SchemaDefDriver extends StaticPHPDriver | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-03-18 10:52:08 +00:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * PEAR DB type => Doctrine type | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     private const types = [ | 
					
						
							|  |  |  |         'varchar'  => 'string', | 
					
						
							| 
									
										
										
										
											2020-03-30 17:15:30 +00:00
										 |  |  |         'char'     => 'string', // char is a fixed witdh varchar
 | 
					
						
							| 
									
										
										
										
											2020-03-18 10:52:08 +00:00
										 |  |  |         'int'      => 'integer', | 
					
						
							| 
									
										
										
										
											2020-03-30 17:15:30 +00:00
										 |  |  |         'serial'   => 'integer', | 
					
						
							| 
									
										
										
										
											2020-03-18 10:52:08 +00:00
										 |  |  |         'tinyint'  => 'smallint', // no portable tinyint
 | 
					
						
							|  |  |  |         'bigint'   => 'bigint', | 
					
						
							|  |  |  |         'bool'     => 'boolean', | 
					
						
							|  |  |  |         'numeric'  => 'decimal', | 
					
						
							|  |  |  |         'text'     => 'text', | 
					
						
							|  |  |  |         'datetime' => 'datetime', | 
					
						
							|  |  |  |         // Unused in V2, but might start being used
 | 
					
						
							|  |  |  |         'date'        => 'date', | 
					
						
							|  |  |  |         'time'        => 'time', | 
					
						
							|  |  |  |         'datetimez'   => 'datetimez', | 
					
						
							|  |  |  |         'object'      => 'object', | 
					
						
							|  |  |  |         'array'       => 'array', | 
					
						
							|  |  |  |         'simplearray' => 'simplearray', | 
					
						
							|  |  |  |         'json_array'  => 'json_array', | 
					
						
							|  |  |  |         'float'       => 'float', | 
					
						
							|  |  |  |         'guid'        => 'guid', | 
					
						
							|  |  |  |         'blob'        => 'blob', | 
					
						
							|  |  |  |     ]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							| 
									
										
										
										
											2020-03-30 17:15:30 +00:00
										 |  |  |      * Fill in the database $metadata for $class_name | 
					
						
							| 
									
										
										
										
											2020-03-18 10:52:08 +00:00
										 |  |  |      * | 
					
						
							| 
									
										
										
										
											2020-03-30 17:15:30 +00:00
										 |  |  |      * @param string        $class_name | 
					
						
							| 
									
										
										
										
											2020-03-18 10:52:08 +00:00
										 |  |  |      * @param ClassMetadata $metadata | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2020-03-30 17:15:30 +00:00
										 |  |  |     public function loadMetadataForClass($class_name, ClassMetadata $metadata) | 
					
						
							| 
									
										
										
										
											2020-03-15 21:21:11 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2020-03-30 17:15:30 +00:00
										 |  |  |         $schema = $class_name::schemaDef(); | 
					
						
							| 
									
										
										
										
											2020-03-18 10:52:08 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         $metadata->setPrimaryTable(['name' => $schema['name'], | 
					
						
							| 
									
										
										
										
											2020-03-30 17:15:30 +00:00
										 |  |  |             'indexes'                      => self::kv_to_name_col($schema['indexes'] ?? []), | 
					
						
							|  |  |  |             'uniqueConstraints'            => self::kv_to_name_col($schema['unique keys'] ?? []), | 
					
						
							| 
									
										
										
										
											2020-03-30 14:43:45 +00:00
										 |  |  |             'options'                      => ['comment' => $schema['description'] ?? ''], | 
					
						
							|  |  |  |         ]); | 
					
						
							| 
									
										
										
										
											2020-03-18 10:52:08 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         foreach ($schema['fields'] as $name => $opts) { | 
					
						
							| 
									
										
										
										
											2020-03-30 17:15:30 +00:00
										 |  |  |             // TODO
 | 
					
						
							| 
									
										
										
										
											2020-03-18 10:52:08 +00:00
										 |  |  |             // Convert old to new types
 | 
					
						
							| 
									
										
										
										
											2020-03-28 15:40:28 +00:00
										 |  |  |             $type = // $name === 'date'
 | 
					
						
							|  |  |  |                   // // Old date fields were stored as int, store as datetime/timestamp
 | 
					
						
							|  |  |  |                   // ? 'datetime'
 | 
					
						
							|  |  |  |                   // // For ints, prepend the size (smallint)
 | 
					
						
							|  |  |  |                   // // The size field doesn't exist otherwise
 | 
					
						
							|  |  |  |                   // :
 | 
					
						
							|  |  |  |                   self::types[($opts['size'] ?? '') . $opts['type']]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             $unique = null; | 
					
						
							| 
									
										
										
										
											2020-03-30 17:15:30 +00:00
										 |  |  |             foreach ($schema['unique keys'] ?? [] as $key => $uniq_arr) { | 
					
						
							| 
									
										
										
										
											2020-03-28 15:40:28 +00:00
										 |  |  |                 if (in_array($name, $uniq_arr)) { | 
					
						
							|  |  |  |                     $unique = $key; | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-18 10:52:08 +00:00
										 |  |  |             $field = [ | 
					
						
							| 
									
										
										
										
											2020-03-28 15:40:28 +00:00
										 |  |  |                 // boolean, optional
 | 
					
						
							|  |  |  |                 'id' => in_array($name, $schema['primary key']), | 
					
						
							|  |  |  |                 // string
 | 
					
						
							| 
									
										
										
										
											2020-03-18 10:52:08 +00:00
										 |  |  |                 'fieldName' => $name, | 
					
						
							| 
									
										
										
										
											2020-03-28 15:40:28 +00:00
										 |  |  |                 // string
 | 
					
						
							|  |  |  |                 'type' => $type, | 
					
						
							|  |  |  |                 // stringn, optional
 | 
					
						
							|  |  |  |                 'unique' => $unique, | 
					
						
							|  |  |  |                 // String length, ignored if not a string
 | 
					
						
							|  |  |  |                 // int, optional
 | 
					
						
							|  |  |  |                 'length' => $opts['length'] ?? null, | 
					
						
							|  |  |  |                 // boolean, optional
 | 
					
						
							|  |  |  |                 'nullable' => !($opts['not null'] ?? false), | 
					
						
							|  |  |  |                 // Numeric precision and scale, ignored if not a number
 | 
					
						
							|  |  |  |                 // integer, optional
 | 
					
						
							|  |  |  |                 'precision' => $opts['precision'] ?? null, | 
					
						
							|  |  |  |                 // integer, optional
 | 
					
						
							|  |  |  |                 'scale'   => $opts['scale'] ?? null, | 
					
						
							|  |  |  |                 'options' => [ | 
					
						
							| 
									
										
										
										
											2020-03-30 17:15:30 +00:00
										 |  |  |                     'comment'  => $opts['description'] ?? null, | 
					
						
							|  |  |  |                     'default'  => $opts['default']     ?? null, | 
					
						
							|  |  |  |                     'unsigned' => $opts['unsigned']    ?? null, | 
					
						
							|  |  |  |                     // bool, optional
 | 
					
						
							|  |  |  |                     'fixed' => $opts['type'] === 'char', | 
					
						
							| 
									
										
										
										
											2020-03-18 10:52:08 +00:00
										 |  |  |                     // 'collation' => string, unused
 | 
					
						
							|  |  |  |                     // 'check', unused
 | 
					
						
							|  |  |  |                 ], | 
					
						
							|  |  |  |                 // 'columnDefinition', unused
 | 
					
						
							|  |  |  |             ]; | 
					
						
							|  |  |  |             // The optional feilds from earlier were populated with null, remove them
 | 
					
						
							| 
									
										
										
										
											2020-03-28 15:40:28 +00:00
										 |  |  |             $field            = array_filter($field,            F\not('is_null')); | 
					
						
							|  |  |  |             $field['options'] = array_filter($field['options'], F\not('is_null')); | 
					
						
							| 
									
										
										
										
											2020-03-18 10:52:08 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |             $metadata->mapField($field); | 
					
						
							| 
									
										
										
										
											2020-03-30 17:15:30 +00:00
										 |  |  |             if ($opts['type'] === 'serial') { | 
					
						
							|  |  |  |                 $metadata->setIdGeneratorType($metadata::GENERATOR_TYPE_AUTO); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2020-03-18 10:52:08 +00:00
										 |  |  |         } | 
					
						
							|  |  |  |         // TODO foreign keys
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Override StaticPHPDriver's method, | 
					
						
							|  |  |  |      * we care about classes that have the method `schemaDef`, | 
					
						
							|  |  |  |      * instead of `loadMetadata`. | 
					
						
							|  |  |  |      * | 
					
						
							| 
									
										
										
										
											2020-03-30 17:15:30 +00:00
										 |  |  |      * @param string $class_name | 
					
						
							| 
									
										
										
										
											2020-05-10 21:43:15 +01:00
										 |  |  |      * @return bool | 
					
						
							| 
									
										
										
										
											2020-03-18 10:52:08 +00:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2020-03-30 17:15:30 +00:00
										 |  |  |     public function isTransient($class_name) | 
					
						
							| 
									
										
										
										
											2020-03-18 10:52:08 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2020-03-30 17:15:30 +00:00
										 |  |  |         return !method_exists($class_name, 'schemaDef'); | 
					
						
							| 
									
										
										
										
											2020-03-18 10:52:08 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Convert [$key => $val] to ['name' => $key, 'columns' => $val] | 
					
						
							| 
									
										
										
										
											2020-05-10 21:43:15 +01:00
										 |  |  |      * @param array $arr | 
					
						
							|  |  |  |      * @return array | 
					
						
							| 
									
										
										
										
											2020-03-18 10:52:08 +00:00
										 |  |  |      */ | 
					
						
							|  |  |  |     private static function kv_to_name_col(array $arr): array | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $res = []; | 
					
						
							|  |  |  |         foreach ($arr as $name => $cols) { | 
					
						
							|  |  |  |             $res[] = ['name' => $name, 'columns' => $cols]; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return $res; | 
					
						
							| 
									
										
										
										
											2020-03-15 21:21:11 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | } |