createTable("Settings") ->addString("name", 32) ->addJson("value", true) ->addBool("private", false) // these values are not returned from '/api/settings/get', but can be changed ->addBool("readonly", false) // these values are neither returned, nor can be changed from outside ->primaryKey("name"); $defaultSettings = Settings::loadDefaults(loadEnv()); $settingsQuery = $sql->insert("Settings", ["name", "value", "private", "readonly"]); $defaultSettings->addRows($settingsQuery); $queries[] = $settingsQuery; $queries[] = $sql->createTable("ApiPermission") ->addString("method", 32) ->addJson("groups", true, '[]') ->addString("description", 128, false, "") ->primaryKey("method") ->addBool("is_core", false); self::loadDefaultACL($sql, $queries); self::loadPatches($sql, $queries); return $queries; } private static function loadPatches(SQL $sql, array &$queries): void { $patchFiles = array_merge( glob('Core/Configuration/Patch/*.php'), glob('Site/Configuration/Patch/*.php') ); sort($patchFiles); foreach ($patchFiles as $file) { @include_once $file; } } private static function getCreatedTables(SQL $sql, array $queries): ?array { $createdTables = $sql->listTables(); if ($createdTables !== null) { foreach ($queries as $query) { if ($query instanceof CreateTable) { $tableName = $query->getTableName(); if (!in_array($tableName, $createdTables)) { $createdTables[] = $tableName; } } } } else { self::getLogger($sql)->warning("Error querying existing tables: " . $sql->getLastError()); } return $createdTables; } public static function createEntityQueries(SQL $sql, array $entityClasses, array &$queries, bool $skipExisting = false): void { if (empty($entityClasses)) { return; } // first, check what tables are already created $createdTables = self::getCreatedTables($sql, $queries); if ($createdTables === null) { throw new \Exception("Error querying existing tables"); } // then collect all persistable entities (tables, relations, etc.) $persistables = []; foreach ($entityClasses as $className) { $reflectionClass = new \ReflectionClass($className); if ($reflectionClass->isSubclassOf(DatabaseEntity::class)) { $handler = ("$className::getHandler")($sql, null, true); $persistables[$handler->getTableName()] = $handler; foreach ($handler->getNMRelations() as $nmTableName => $nmRelation) { $persistables[$nmTableName] = $nmRelation; } } else { throw new \Exception("Class '$className' is not a subclass of DatabaseEntity"); } } // now order the persistable entities so all dependencies are met. $tableCount = count($persistables); while (!empty($persistables)) { $prevCount = $tableCount; $unmetDependenciesTotal = []; foreach ($persistables as $tableName => $persistable) { $dependsOn = $persistable->dependsOn(); $unmetDependencies = array_diff($dependsOn, $createdTables); if (empty($unmetDependencies)) { $queries = array_merge($queries, $persistable->getCreateQueries($sql, $skipExisting)); $createdTables[] = $tableName; unset($persistables[$tableName]); } else { $unmetDependenciesTotal = array_merge($unmetDependenciesTotal, $unmetDependencies); } } $tableCount = count($persistables); if ($tableCount === $prevCount) { throw new Exception("Circular or unmet table dependency detected. Unmet dependencies: " . implode(", ", $unmetDependenciesTotal)); } } } private static function loadEntities(SQL $sql, array &$queries): void { $classes = []; $baseDirs = ["Core", "Site"]; foreach ($baseDirs as $baseDir) { $entityDirectory = "./$baseDir/Objects/DatabaseEntity/"; if (file_exists($entityDirectory) && is_dir($entityDirectory)) { $scan_arr = scandir($entityDirectory); $files_arr = array_diff($scan_arr, [".", ".."]); foreach ($files_arr as $file) { $suffix = ".class.php"; if (endsWith($file, $suffix)) { $className = substr($file, 0, strlen($file) - strlen($suffix)); $className = "\\$baseDir\\Objects\\DatabaseEntity\\$className"; $reflectionClass = new \ReflectionClass($className); if ($reflectionClass->isSubclassOf(DatabaseEntity::class)) { $classes[] = $className; } } } } } self::createEntityQueries($sql, $classes, $queries); } public static function loadDefaultACL(SQL $sql, array &$queries, ?array $classes = NULL): void { $query = $sql->insert("ApiPermission", ["method", "groups", "description", "is_core"]); if ($classes === NULL) { $classes = Request::getApiEndpoints(); } foreach ($classes as $class) { if ($class instanceof \ReflectionClass) { $className = $class->getName(); } else if (!is_string($class)) { throw new \Exception("Cannot call loadDefaultACL() for type: " . get_class($class)); } else { $className = $class; } ("$className::loadDefaultACL")($query); } if ($query->hasRows()) { $queries[] = $query; } } }