SQL Way
Vous pouvez travailler avec Nette Database de deux manières : en écrivant des requêtes SQL (méthode SQL) ou en laissant SQL être généré automatiquement(méthode Explorer). La méthode SQL vous permet de construire des requêtes en toute sécurité tout en gardant un contrôle total sur leur structure.
Voir Connexion et configuration pour plus de détails sur la configuration de la connexion à la base de données.
Interrogation de base
La méthode query()
exécute des requêtes de base de données et renvoie un objet ResultSet représentant le résultat. Si la
requête échoue, la méthode lève une exception. Vous pouvez
parcourir le résultat de la requête en boucle à l'aide d'une boucle foreach
ou utiliser l'une des fonctions d'aide.
$result = $database->query('SELECT * FROM users');
foreach ($result as $row) {
echo $row->id;
echo $row->name;
}
Pour insérer en toute sécurité des valeurs dans les requêtes SQL, utilisez des requêtes paramétrées. Nette Database rend cette opération très simple : il suffit d'ajouter une virgule et la valeur à la requête SQL.
$database->query('SELECT * FROM users WHERE name = ?', $name);
Pour les paramètres multiples, vous pouvez soit intercaler la requête SQL avec les paramètres :
$database->query('SELECT * FROM users WHERE name = ?', $name, 'AND age > ?', $age);
soit écrire d'abord la requête SQL complète, puis ajouter tous les paramètres :
$database->query('SELECT * FROM users WHERE name = ? AND age > ?', $name, $age);
Protection contre l'injection SQL
Pourquoi est-il important d'utiliser des requêtes paramétrées ? Parce qu'elles vous protègent contre les attaques par injection SQL, où les attaquants peuvent injecter des commandes SQL malveillantes pour manipuler ou accéder aux données de la base de données.
N'insérez jamais de variables directement dans une requête SQL Utilisez toujours des requêtes paramétrées pour vous protéger contre les injections SQL.
// ❌ UNSAFE CODE - vulnérable à l'injection SQL
$database->query("SELECT * FROM users WHERE name = '$name'");
// ✅ Requête paramétrée sûre
$database->query('SELECT * FROM users WHERE name = ?', $name);
Veillez à vous familiariser avec les risques de sécurité potentiels.
Techniques d'interrogation
Conditions WHERE
Vous pouvez écrire les conditions WHERE
sous la forme d'un tableau associatif, où les clés sont des noms de
colonnes et les valeurs sont les données à comparer. Nette Database sélectionne automatiquement l'opérateur SQL le plus
approprié en fonction du type de valeur.
$database->query('SELECT * FROM users WHERE', [
'name' => 'John',
'active' => true,
]);
// WHERE `name` = 'John' AND `active` = 1
Vous pouvez également spécifier explicitement l'opérateur dans la clé :
$database->query('SELECT * FROM users WHERE', [
'age >' => 25, // utilise l'opérateur >
'name LIKE' => '%John%', // utilise l'opérateur LIKE
'email NOT LIKE' => '%example.com%', // utilise l'opérateur NOT LIKE
]);
// WHERE `age` > 25 AND `name` LIKE '%John%' AND `email` NOT LIKE '%example.com%'
Les cas particuliers tels que les valeurs null
ou les tableaux sont traités automatiquement :
$database->query('SELECT * FROM products WHERE', [
'name' => 'Laptop', // utilise l'opérateur =
'category_id' => [1, 2, 3], // utilise IN
'description' => null, // uses IS NULL
]);
// WHERE `name` = 'Laptop' AND `category_id` IN (1, 2, 3) AND `description` IS NULL
Pour les conditions négatives, utilisez l'opérateur NOT
:
$database->query('SELECT * FROM products WHERE', [
'name NOT' => 'Laptop', // utilise l'opérateur <>
'category_id NOT' => [1, 2, 3], // utilise NOT IN
'description NOT' => null, // utilise IS NOT NULL
'id' => [], // ignoré
]);
// WHERE `name` <> 'Laptop' AND `category_id` NOT IN (1, 2, 3) AND `description` IS NOT NULL
Par défaut, les conditions sont combinées à l'aide de l'opérateur AND
. Vous pouvez modifier ce comportement à
l'aide de l'espace réservé ?or.
Règles ORDER BY
La clause ORDER BY
peut être définie comme un tableau, où les clés représentent les colonnes et les valeurs
sont des booléens indiquant l'ordre croissant :
$database->query('SELECT id FROM author ORDER BY', [
'id' => true, // ascendante
'name' => false, // décroissant
]);
// SELECT id FROM auteur ORDER BY `id`, `name` DESC
Insérer des données (INSERT)
Pour insérer des enregistrements, utilisez l'instruction SQL INSERT
.
$values = [
'name' => 'John Doe',
'email' => 'john@example.com',
];
$database->query('INSERT INTO users ?', $values);
$userId = $database->getInsertId();
La méthode getInsertId()
renvoie l'ID de la dernière ligne insérée. Pour certaines bases de données (par
exemple, PostgreSQL), vous devez spécifier le nom de la séquence en utilisant
$database->getInsertId($sequenceId)
.
Vous pouvez également transmettre des valeurs spéciales, telles que des fichiers, des objets DateTime ou des types d'énumération, en tant que paramètres.
Insertion de plusieurs enregistrements à la fois :
$database->query('INSERT INTO users ?', [
['name' => 'User 1', 'email' => 'user1@mail.com'],
['name' => 'User 2', 'email' => 'user2@mail.com'],
]);
L'insertion par lots est beaucoup plus rapide car une seule requête est exécutée au lieu de plusieurs.
Note de sécurité: N'utilisez jamais de données non validées en tant que $values
. Familiarisez-vous avec
les risques possibles.
Mise à jour des données (UPDATE)
Pour mettre à jour des enregistrements, utilisez l'instruction SQL UPDATE
.
// Mise à jour d'un seul enregistrement
$values = [
'name' => 'John Smith',
];
$result = $database->query('UPDATE users SET ? WHERE id = ?', $values, 1);
Vous pouvez vérifier le nombre de lignes affectées en utilisant $result->getRowCount()
.
Vous pouvez utiliser les opérateurs +=
et -=
dans UPDATE
:
$database->query('UPDATE users SET ? WHERE id = ?', [
'login_count+=' => 1, // incrémenter login_count
], 1);
Pour insérer ou mettre à jour un enregistrement s'il existe déjà, utilisez la technique
ON DUPLICATE KEY UPDATE
:
$values = [
'name' => $name,
'year' => $year,
];
$database->query('INSERT INTO users ? ON DUPLICATE KEY UPDATE ?',
$values + ['id' => $id],
$values,
);
// INSERT INTO users (`id`, `name`, `year`) VALUES (123, 'Jim', 1978)
// ON DUPLICATE KEY UPDATE `name` = 'Jim', `year` = 1978
Notez que Nette Database reconnaît le contexte de la commande SQL dans lequel un paramètre avec un tableau est utilisé et
génère le code SQL en conséquence. Par exemple, il a construit (id, name, year) VALUES (123, 'Jim', 1978)
à
partir du premier tableau, tandis qu'il a converti le second en name = 'Jim', year = 1978
. Ce point est abordé plus
en détail dans la section Conseils pour la construction du code SQL.
Suppression de données (DELETE)
Pour supprimer des enregistrements, utilisez l'instruction SQL DELETE
. Exemple avec le nombre de lignes
supprimées :
$count = $database->query('DELETE FROM users WHERE id = ?', 1)
->getRowCount();
Conseils de construction SQL
Les caractères de remplissage SQL permettent de contrôler la façon dont les valeurs des paramètres sont incorporées dans les expressions SQL :
Les caractères de remplacement SQL permettent de contrôler la façon dont les valeurs des paramètres sont incorporées dans les expressions SQL. | ||
---|---|---|
?name |
Utilisé pour les noms de table ou de colonne | |
?values |
Génère (key, ...) VALUES (value, ...) |
INSERT ... ? , REPLACE ... ? |
?set |
Génère les affectations key = value, ... |
SET ? , KEY UPDATE ? |
?and |
Joint les conditions dans un tableau avec AND |
WHERE ? , HAVING ? |
?or |
Joint les conditions d'un tableau avec OR |
– -. |
?order |
Génère la clause ORDER BY |
ORDER BY ? , GROUP BY ? |
Pour insérer dynamiquement des noms de tables ou de colonnes, utilisez l'espace réservé ?name
. Nette Database
assure un échappement correct selon les conventions de la base de données (par exemple, en enfermant dans des crochets pour
MySQL).
$table = 'users';
$column = 'name';
$database->query('SELECT ?name FROM ?name WHERE id = 1', $column, $table);
// SELECT `nom` FROM `users` WHERE id = 1 (dans MySQL)
Avertissement: N'utilisez l'espace réservé ?name
que pour les noms de tables et de colonnes validés.
Dans le cas contraire, vous risquez des failles de
sécurité.
Il n'est généralement pas nécessaire de spécifier d'autres indices, car Nette utilise une auto-détection intelligente lors
de la construction des requêtes SQL (voir la troisième colonne du tableau). Toutefois, vous pouvez les utiliser dans les
situations où vous souhaitez combiner des conditions en utilisant OR
au lieu de AND
:
$database->query('SELECT * FROM users WHERE ?or', [
'name' => 'John',
'email' => 'john@example.com',
]);
// SELECT * FROM users WHERE `name` = 'John' OR `email` = 'john@example.com'
Valeurs spéciales
Outre les types scalaires standard (par exemple, string
, int
, bool
), vous pouvez
également transmettre des valeurs spéciales en tant que paramètres :
- Fichiers : Utilisez
fopen('file.png', 'r')
pour insérer le contenu binaire d'un fichier. - Date et heure : les objets
DateTime
sont automatiquement convertis au format de date de la base de données. - Valeurs d'énumération : Les instances de
enum
sont converties en valeurs correspondantes. - Littéraux SQL : Créés à l'aide de
Connection::literal('NOW()')
, ils sont insérés directement dans la requête.
$database->query('INSERT INTO articles ?', [
'title' => 'My Article',
'published_at' => new DateTime,
'content' => fopen('image.png', 'r'),
'state' => Status::Draft,
]);
Pour les bases de données qui ne prennent pas en charge le type datetime
(par exemple, SQLite et Oracle), les
valeurs DateTime
sont converties conformément à l'option de configuration formatDateTime
(par défaut
: U
pour les horodatages Unix).
Litres SQL
Dans certains cas, vous pouvez avoir besoin d'insérer du code SQL brut comme valeur sans le traiter comme une chaîne de
caractères ou l'escamoter. Pour cela, utilisez des objets de la classe Nette\Database\SqlLiteral
, qui peuvent être
créés à l'aide de la méthode Connection::literal()
.
$result = $database->query('SELECT * FROM users WHERE', [
'name' => $name,
'year >' => $database::literal('YEAR()'),
]);
// SELECT * FROM users WHERE (`name` = 'Jim') AND (`year` > YEAR())
Autre solution :
$result = $database->query('SELECT * FROM users WHERE', [
'name' => $name,
$database::literal('year > YEAR()'),
]);
// SELECT * FROM users WHERE (`name` = 'Jim') AND (year > YEAR())
Les littéraux SQL peuvent également contenir des paramètres :
$result = $database->query('SELECT * FROM users WHERE', [
'name' => $name,
$database::literal('year > ? AND year < ?', $min, $max),
]);
// SELECT * FROM users WHERE `name` = 'Jim' AND (year > 1978 AND year < 2017)
Cela permet des combinaisons flexibles :
$result = $database->query('SELECT * FROM users WHERE', [
'name' => $name,
$database::literal('?or', [
'active' => true,
'role' => $role,
]),
]);
// SELECT * FROM users WHERE `name` = 'Jim' AND (`active` = 1 OR `role` = 'admin')
Récupération de données
Raccourcis pour les requêtes SELECT
Pour simplifier la recherche de données, la classe Connection
fournit plusieurs raccourcis qui combinent un appel
à query()
avec un appel ultérieur à fetch*()
. Ces méthodes acceptent les mêmes paramètres que
query()
, c'est-à-dire une requête SQL et des paramètres facultatifs. Une description détaillée des méthodes
fetch*()
se trouve ci-dessous.
fetch($sql, ...$params): ?Row |
Exécute la requête et récupère la première ligne sous la forme d'un objet Row . |
fetchAll($sql, ...$params): array |
Exécute la requête et récupère toutes les lignes sous la forme d'un tableau d'objets Row . |
fetchPairs($sql, ...$params): array |
Exécute la requête et récupère un tableau associatif dont la première colonne est la clé et la seconde la valeur. |
fetchField($sql, ...$params): mixed |
Exécute la requête et récupère la valeur de la première cellule de la première ligne. |
fetchList($sql, ...$params): ?array |
Exécute la requête et récupère la première ligne sous forme de tableau indexé. |
Exemple :
// fetchField() - renvoie la valeur de la première cellule
$count = $database->query('SELECT COUNT(*) FROM articles')
->fetchField();
foreach
– Itération sur les lignes
Après l'exécution d'une requête, un objet ResultSet est renvoyé, ce qui vous permet
d'itérer sur les résultats de différentes manières. La méthode la plus simple et la plus efficace en termes de mémoire pour
récupérer des lignes consiste à itérer dans une boucle foreach
. Cette méthode traite les lignes une par une et
évite de stocker toutes les données en mémoire en même temps.
$result = $database->query('SELECT * FROM users');
foreach ($result as $row) {
echo $row->id;
echo $row->name;
//...
}
Le site ResultSet
ne peut être itéré qu'une seule fois. Si vous devez l'itérer plusieurs fois,
vous devez d'abord charger les données dans un tableau, par exemple en utilisant la méthode fetchAll()
.
fetch(): ?Row
Exécute la requête et récupère une seule ligne sous la forme d'un objet Row
. S'il n'y a plus de lignes
disponibles, elle renvoie null
. Cette méthode fait passer le pointeur interne à la ligne suivante.
$result = $database->query('SELECT * FROM users');
$row = $result->fetch(); // récupère la première ligne
if ($row) {
echo $row->name;
}
fetchAll(): array
Récupère toutes les lignes restantes de ResultSet
sous la forme d'un tableau d'objets Row
.
$result = $database->query('SELECT * FROM users');
$rows = $result->fetchAll(); // récupère toutes les lignes
foreach ($rows as $row) {
echo $row->name;
}
fetchPairs (string|int|null $key = null, string|int|null $value = null): array
Récupère les résultats sous la forme d'un tableau associatif. Le premier argument spécifie la colonne à utiliser comme clé, et le second spécifie la colonne à utiliser comme valeur :
$result = $database->query('SELECT id, name FROM users');
$names = $result->fetchPairs('id', 'name');
// [1 => 'John Doe', 2 => 'Jane Doe', ...]
Si seul le premier paramètre est fourni, la valeur sera la ligne entière (sous la forme d'un objet Row
) :
$rows = $result->fetchPairs('id');
// [1 => Row(id: 1, name: 'John'), 2 => Row(id: 2, name: 'Jane'), ...]
Si null
est fourni comme clé, le tableau sera indexé numériquement à partir de zéro :
$names = $result->fetchPairs(null, 'name');
// [0 => 'John Doe', 1 => 'Jane Doe', ...]
fetchPairs (Closure $callback): array
Vous pouvez également fournir un rappel qui détermine les paires clé-valeur ou les valeurs pour chaque ligne.
$result = $database->query('SELECT * FROM users');
$items = $result->fetchPairs(fn($row) => "$row->id - $row->name");
// ['1 - John', '2 - Jane', ...]
// Le callback peut également renvoyer un tableau avec une paire clé/valeur:
$names = $result->fetchPairs(fn($row) => [$row->name, $row->age]);
// ['John' => 46, 'Jane' => 21, ...]
fetchField(): mixed
Récupère la valeur de la première cellule de la ligne actuelle. S'il n'y a plus de lignes disponibles, elle renvoie
null
. Cette méthode fait passer le pointeur interne à la ligne suivante.
$result = $database->query('SELECT name FROM users');
$name = $result->fetchField(); // récupère le nom de la première ligne
fetchList(): ?array
Récupère la ligne sous la forme d'un tableau indexé. S'il n'y a plus de lignes disponibles, elle renvoie null
.
Cette méthode fait passer le pointeur interne à la ligne suivante.
$result = $database->query('SELECT name, email FROM users');
$row = $result->fetchList(); // ['John', 'john@example.com']
getRowCount(): ?int
Renvoie le nombre de lignes affectées par la dernière requête UPDATE
ou DELETE
. Pour les requêtes
SELECT
, il renvoie le nombre de lignes récupérées, mais ce nombre n'est pas toujours connu – dans ce cas, il
renvoie null
.
getColumnCount(): ?int
Renvoie le nombre de colonnes dans ResultSet
.
Informations sur les requêtes
Pour obtenir des détails sur la dernière requête exécutée, utilisez :
echo $database->getLastQueryString(); // sort la requête SQL
$result = $database->query('SELECT * FROM articles');
echo $result->getQueryString(); // affiche la requête SQL
echo $result->getTime(); // affiche le temps d'exécution en secondes
Pour afficher le résultat sous forme de tableau HTML, utilisez :
$result = $database->query('SELECT * FROM articles');
$result->dump();
Vous pouvez également obtenir des informations sur les types de colonnes à partir du site ResultSet
:
$result = $database->query('SELECT * FROM articles');
$types = $result->getColumnTypes();
foreach ($types as $column => $type) {
echo "$column is of type $type->type"; // par exemple, "id is of type int" (l'identifiant est de type int)
}
Enregistrement des requêtes
Vous pouvez mettre en œuvre une journalisation personnalisée des requêtes. L'événement onQuery
est un tableau
de rappels qui sont invoqués après chaque exécution de requête :
$database->onQuery[] = function ($database, $result) use ($logger) {
$logger->info('Query: ' . $result->getQueryString());
$logger->info('Time: ' . $result->getTime());
if ($result->getRowCount() > 1000) {
$logger->warning('Large result set: ' . $result->getRowCount() . ' rows');
}
};