toString() . ' ', array_map(fn($it) => $it->toWhere(), $fields)); } /** * Create a WHERE clause fragment to implement an ID-based query * * @param string $paramName The parameter name where the value of the ID will be provided (optional; default @id) * @return string The WHERE clause fragment to match by ID * @throws DocumentException If the database mode has not been set */ public static function whereById(string $paramName = ':id'): string { return self::whereByFields([Field::EQ(Configuration::$idField, 0, $paramName)]); } /** * Create an `INSERT` statement for a document * * @param string $tableName The name of the table into which the document will be inserted * @param AutoId|null $autoId The version of automatic ID query to generate (optional, defaults to None) * @return string The `INSERT` statement to insert a document * @throws DocumentException If the database mode is not set */ public static function insert(string $tableName, ?AutoId $autoId = null): string { try { $id = Configuration::$idField; $values = match (Configuration::$mode) { Mode::SQLite => match ($autoId ?? AutoId::None) { AutoId::None => ':data', AutoId::Number => "json_set(:data, '$.$id', " . "(SELECT coalesce(max(data->>'$id'), 0) + 1 FROM $tableName))", AutoId::UUID => "json_set(:data, '$.$id', '" . AutoId::generateUUID() . "')", AutoId::RandomString => "json_set(:data, '$.$id', '" . AutoId::generateRandom() ."')" }, Mode::PgSQL => match ($autoId ?? AutoId::None) { AutoId::None => ':data', AutoId::Number => ":data || ('{\"$id\":' || " . "(SELECT COALESCE(MAX(data->>'$id'), 0) + 1 FROM $tableName) || '}')", AutoId::UUID => ":data || '{\"$id\":\"" . AutoId::generateUUID() . "\"}'", AutoId::RandomString => ":data || '{\"$id\":\"" . AutoId::generateRandom() . "\"}'", }, default => throw new DocumentException('Database mode not set; cannot generate auto-ID INSERT statement'), }; return "INSERT INTO $tableName VALUES ($values)"; } catch (RandomException $ex) { throw new DocumentException('Unable to generate ID: ' . $ex->getMessage(), previous: $ex); } } /** * Query to save a document, inserting it if it does not exist and updating it if it does (AKA "upsert") * * @param string $tableName The name of the table into which a document should be saved * @return string The INSERT...ON CONFLICT query for the document */ public static function save(string $tableName): string { $id = Configuration::$idField; return "INSERT INTO $tableName VALUES (:data) ON CONFLICT ((data->>'$id')) DO UPDATE SET data = EXCLUDED.data"; } /** * Query to update a document * * @param string $tableName The name of the table in which the document should be updated * @return string The UPDATE query for the document * @throws DocumentException If the database mode has not been set */ public static function update(string $tableName): string { return "UPDATE $tableName SET data = :data WHERE " . self::whereById(); } }