Vía SQL
Puede trabajar con Nette Database de dos formas: escribiendo consultas SQL (forma SQL) o dejando que SQL se genere automáticamente(forma Explorer). El modo SQL le permite crear consultas de forma segura manteniendo un control total sobre su estructura.
Ver Conexión y Configuración para más detalles sobre la configuración de la conexión a la base de datos.
Consultas básicas
El método query()
ejecuta consultas a la base de datos y devuelve un objeto ResultSet que representa el resultado. Si la consulta
falla, el método lanza una excepción. Puede recorrer el resultado de la consulta mediante un bucle
foreach
o utilizar una de las funciones de ayuda.
$result = $database->query('SELECT * FROM users');
foreach ($result as $row) {
echo $row->id;
echo $row->name;
}
Para insertar valores de forma segura en las consultas SQL, utilice consultas parametrizadas. Nette Database lo hace muy sencillo: basta con añadir una coma y el valor a la consulta SQL.
$database->query('SELECT * FROM users WHERE name = ?', $name);
Para varios parámetros, puede intercalar la consulta SQL con los parámetros:
$database->query('SELECT * FROM users WHERE name = ?', $name, 'AND age > ?', $age);
O escribir primero toda la consulta SQL y luego intercalar todos los parámetros:
$database->query('SELECT * FROM users WHERE name = ? AND age > ?', $name, $age);
Protección contra la inyección SQL
¿Por qué es importante utilizar consultas parametrizadas? Porque le protegen de los ataques de inyección SQL, en los que los atacantes pueden inyectar comandos SQL maliciosos para manipular o acceder a los datos de la base de datos.
**Utiliza siempre consultas parametrizadas para protegerte de la inyección SQL.
// ❌ CÓDIGO INSEGURO - vulnerable a la inyección SQL.
$database->query("SELECT * FROM users WHERE name = '$name'");
// ✅ Consulta parametrizada segura
$database->query('SELECT * FROM users WHERE name = ?', $name);
Familiarícese con los posibles riesgos de seguridad.
Técnicas de consulta
Condiciones WHERE
Puede escribir las condiciones de WHERE
como una matriz asociativa, donde las claves son nombres de columnas y los
valores son los datos a comparar. Nette Database selecciona automáticamente el operador SQL más apropiado en función del tipo
de valor.
$database->query('SELECT * FROM users WHERE', [
'name' => 'John',
'active' => true,
]);
// WHERE `name` = 'John' AND `active` = 1
También puede especificar explícitamente el operador en la clave:
$database->query('SELECT * FROM users WHERE', [
'age >' => 25, // utiliza el operador >.
'name LIKE' => '%John%', // utiliza el operador LIKE
'email NOT LIKE' => '%example.com%', // utiliza el operador NOT LIKE
]);
// WHERE `edad` > 25 AND `nombre` LIKE '%John%' AND `email` NOT LIKE '%ejemplo.com%'
Los casos especiales, como los valores de null
o las matrices, se gestionan automáticamente:
$database->query('SELECT * FROM products WHERE', [
'name' => 'Laptop', // utiliza el operador =
'category_id' => [1, 2, 3], // utiliza IN
'description' => null, // usos IS NULL
]);
// WHERE `name` = 'Laptop' AND `category_id` IN (1, 2, 3) AND `description` IS NULL
Para condiciones negativas, utilice el operador NOT
:
$database->query('SELECT * FROM products WHERE', [
'name NOT' => 'Laptop', // utiliza el operador <>
'category_id NOT' => [1, 2, 3], // utiliza NOT IN
'description NOT' => null, // utiliza IS NOT NULL
'id' => [], // omitido
]);
// WHERE `name` <> 'Laptop' AND `category_id` NOT IN (1, 2, 3) AND `description` IS NOT NULL
Por defecto, las condiciones se combinan utilizando el operador AND
. Puede cambiar este comportamiento utilizando
el marcador de posición ?or.
Reglas ORDER BY
La cláusula ORDER BY
puede definirse como una matriz, donde las claves representan columnas y los valores son
booleanos que indican un orden ascendente:
$database->query('SELECT id FROM author ORDER BY', [
'id' => true, // ascendente
'name' => false, // descendente
]);
// SELECT id FROM autor ORDER BY `id`, `nombre` DESC
Inserción de datos (INSERT)
Para insertar registros, utilice la sentencia SQL INSERT
.
$values = [
'name' => 'John Doe',
'email' => 'john@example.com',
];
$database->query('INSERT INTO users ?', $values);
$userId = $database->getInsertId();
El método getInsertId()
devuelve el ID de la última fila insertada. Para determinadas bases de datos (por
ejemplo, PostgreSQL), debe especificar el nombre de la secuencia utilizando
$database->getInsertId($sequenceId)
.
También puede pasar valores especiales, como archivos, objetos DateTime o tipos enum, como parámetros.
Inserción de varios registros a la vez:
$database->query('INSERT INTO users ?', [
['name' => 'User 1', 'email' => 'user1@mail.com'],
['name' => 'User 2', 'email' => 'user2@mail.com'],
]);
Realizar un INSERT por lotes es mucho más rápido porque sólo se ejecuta una consulta a la base de datos en lugar de varias consultas individuales.
Nota de seguridad: Nunca utilice datos no validados como $values
. Familiarícese con los posibles riesgos.
Actualización de datos (UPDATE)
Para actualizar registros, utilice la sentencia SQL UPDATE
.
// Actualizar un único registro
$values = [
'name' => 'John Smith',
];
$result = $database->query('UPDATE users SET ? WHERE id = ?', $values, 1);
Puede comprobar el número de filas afectadas utilizando $result->getRowCount()
.
Puede utilizar los operadores +=
y -=
en UPDATE
:
$database->query('UPDATE users SET ? WHERE id = ?', [
'login_count+=' => 1, // increment login_count
], 1);
Para insertar o actualizar un registro si ya existe, utilice la técnica 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 `nombre` = 'Jim', `año` = 1978
Observe que Nette Database reconoce el contexto del comando SQL en el que se utiliza un parámetro con una matriz y genera el
código SQL en consecuencia. Por ejemplo, construyó (id, name, year) VALUES (123, 'Jim', 1978)
a partir del primer
array, mientras que convirtió el segundo en name = 'Jim', year = 1978
. Esto se explica con más detalle en la
sección Sugerencias para la construcción de SQL.
Borrado de datos (DELETE)
Para borrar registros, utilice la sentencia SQL DELETE
. Ejemplo con el número de filas borradas:
$count = $database->query('DELETE FROM users WHERE id = ?', 1)
->getRowCount();
Sugerencias de construcción SQL
Los marcadores de posición SQL permiten controlar cómo se incorporan los valores de los parámetros en las expresiones SQL:
Sugerencia | Descripción | Usado Automáticamente Para |
---|---|---|
?name |
Se utiliza para nombres de tabla o columna | – |
?values |
Genera (key, ...) VALUES (value, ...) |
INSERT ... ? , REPLACE ... ? |
?set |
Genera asignaciones key = value, ... |
SET ? , KEY UPDATE ? |
?and |
Une condiciones en un array con AND |
WHERE ? , HAVING ? |
?or |
Une condiciones en una matriz con OR |
– |
?order |
Genera la cláusula ORDER BY |
ORDER BY ? , GROUP BY ? |
Para insertar dinámicamente nombres de tabla o columna, utilice el marcador de posición ?name
. Nette Database
garantiza el correcto escape de acuerdo con las convenciones de la base de datos (p. ej., encierre entre corchetes para
MySQL).
$table = 'users';
$column = 'name';
$database->query('SELECT ?name FROM ?name WHERE id = 1', $column, $table);
// SELECT `name` FROM `users` WHERE id = 1 (en MySQL)
Advertencia: Utilice el marcador de posición ?name
únicamente para nombres de tabla y columna validados.
De lo contrario, se arriesga a sufrir vulnerabilidades de seguridad.
Normalmente no es necesario especificar otras sugerencias, ya que Nette utiliza la autodetección inteligente al construir
consultas SQL (véase la tercera columna de la tabla). Sin embargo, puede utilizarlas en situaciones en las que desee combinar
condiciones utilizando OR
en lugar de AND
:
$database->query('SELECT * FROM users WHERE ?or', [
'name' => 'John',
'email' => 'john@example.com',
]);
// SELECT * FROM usuarios WHERE `nombre` = 'Juan' OR `email` = 'john@example.com'
Valores especiales
Además de los tipos escalares estándar (por ejemplo, string
, int
, bool
), también
puede pasar valores especiales como parámetros:
- Ficheros: Utilice
fopen('file.png', 'r')
para insertar el contenido binario de un archivo. - Fecha y hora: los objetos
DateTime
se convierten automáticamente al formato de fecha de la base de datos. - Valores Enum: Las instancias de
enum
se convierten a sus valores correspondientes. - Literales SQL: Creados con
Connection::literal('NOW()')
, se insertan directamente en la consulta.
$database->query('INSERT INTO articles ?', [
'title' => 'My Article',
'published_at' => new DateTime,
'content' => fopen('image.png', 'r'),
'state' => Status::Draft,
]);
Para las bases de datos que carecen de soporte nativo para el tipo datetime
(por ejemplo, SQLite y Oracle), los
valores de DateTime
se convierten según la opción de configuración formatDateTime
(por defecto:
U
para Unix timestamp).
Literales SQL
En algunos casos, puede necesitar insertar código SQL sin procesar como un valor sin tratarlo como una cadena o escaparlo.
Para ello, utilice objetos de la clase Nette\Database\SqlLiteral
, que pueden crearse utilizando el método
Connection::literal()
.
$result = $database->query('SELECT * FROM users WHERE', [
'name' => $name,
'year >' => $database::literal('YEAR()'),
]);
// SELECT * FROM usuarios WHERE (`nombre` = 'Jim') AND (`año` > AÑO())
Como alternativa:
$result = $database->query('SELECT * FROM users WHERE', [
'name' => $name,
$database::literal('year > YEAR()'),
]);
// SELECT * FROM usuarios WHERE (`nombre` = 'Jim') AND (año > YEAR())
Los literales SQL también pueden contener parámetros:
$result = $database->query('SELECT * FROM users WHERE', [
'name' => $name,
$database::literal('year > ? AND year < ?', $min, $max),
]);
// SELECT * FROM usuarios WHERE `nombre` = 'Jim' AND (año > 1978 AND año < 2017)
Esto permite combinaciones 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')
Obtención de datos
Atajos para consultas SELECT
Para simplificar la recuperación de datos, la clase Connection
proporciona varios métodos abreviados que
combinan una llamada a query()
con otra posterior a fetch*()
. Estos métodos aceptan los mismos
parámetros que query()
, es decir, una consulta SQL y parámetros opcionales.
A continuación se ofrece una descripción detallada de los métodos de fetch*()
.
fetch($sql, ...$params): ?Row Ejecuta la consulta y obtiene la primera fila como un objeto
Row . |
fetchAll($sql, ...$params): array Ejecuta la consulta y obtiene todas las filas como una matriz de objetos
Row . |
fetchPairs($sql, ...$params): array Ejecuta la consulta y obtiene una matriz asociativa en la que la primera
columna es la clave y la segunda el valor. |
fetchField($sql, ...$params): mixed Ejecuta la consulta y obtiene el valor de la primera celda de la
primera fila. |
fetchList($sql, ...$params): ?array Ejecuta la consulta y obtiene la primera fila como una matriz
indexada. |
Ejemplo:
// fetchField() - devuelve el valor de la primera celda
$count = $database->query('SELECT COUNT(*) FROM articles')
->fetchField();
foreach
– Iteración sobre filas
Después de ejecutar una consulta, se devuelve un objeto ResultSet, que permite iterar sobre los resultados de
varias maneras. La forma más sencilla y eficiente de recuperar filas es iterar en un bucle foreach
. Este método
procesa las filas de una en una y evita almacenar todos los datos en memoria a la vez.
$result = $database->query('SELECT * FROM users');
foreach ($result as $row) {
echo $row->id;
echo $row->name;
//...
}
El bucle ResultSet
sólo puede iterarse una vez. Si necesita iterar sobre él varias veces, primero
debe cargar los datos en un array, por ejemplo, utilizando el método fetchAll()
.
fetch(): ?Row
Ejecuta la consulta y obtiene una única fila como objeto Row
. Si no hay más filas disponibles, devuelve
null
. Este método avanza el puntero interno a la siguiente fila.
$result = $database->query('SELECT * FROM users');
$row = $result->fetch(); // recupera la primera fila
if ($row) {
echo $row->name;
}
fetchAll(): array
Obtiene todas las filas restantes de ResultSet
como una matriz de objetos Row
.
$result = $database->query('SELECT * FROM users');
$rows = $result->fetchAll(); // recupera todas las filas
foreach ($rows as $row) {
echo $row->name;
}
fetchPairs(string|int|null $key = null, string|int|null $value = null): array
Obtiene los resultados como una matriz asociativa. El primer argumento especifica la columna a utilizar como clave, y el segundo especifica la columna a utilizar como valor:
$result = $database->query('SELECT id, name FROM users');
$names = $result->fetchPairs('id', 'name');
// [1 => 'Juan Pérez', 2 => 'Juana Pérez', ...]
Si sólo se proporciona el primer parámetro, el valor será la fila completa (como objeto Row
):
$rows = $result->fetchPairs('id');
// [1 => Fila(id: 1, nombre: 'Juan'), 2 => Fila(id: 2, nombre: 'Juana'), ...]
Si se pasa null
como clave, la matriz se indexará numéricamente empezando por cero:
$names = $result->fetchPairs(null, 'name');
// [0 => 'Juan Nadie', 1 => 'Juana Nadie', ...]
fetchPairs(Closure $callback): array
Como alternativa, puede proporcionar una llamada de retorno que determine los pares clave-valor o los valores de cada fila.
$result = $database->query('SELECT * FROM users');
$items = $result->fetchPairs(fn($row) => "$row->id - $row->name");
// ['1 - Juan', '2 - Juana', ...]
// La llamada de retorno también puede devolver un array con un par clave y valor:
$names = $result->fetchPairs(fn($row) => [$row->name, $row->age]);
// ['Juan' => 46, 'Juana' => 21, ...]
fetchField(): mixed
Obtiene el valor de la primera celda de la fila actual. Si no hay más filas disponibles, devuelve null
. Este
método avanza el puntero interno a la siguiente fila.
$result = $database->query('SELECT name FROM users');
$name = $result->fetchField(); // obtiene el nombre de la primera fila
fetchList(): ?array
Obtiene la fila como una matriz indexada. Si no hay más filas disponibles, devuelve null
. Este método avanza el
puntero interno a la siguiente fila.
$result = $database->query('SELECT name, email FROM users');
$row = $result->fetchList(); // ['John', 'john@example.com']
getRowCount(): ?int
Devuelve el número de filas afectadas por la última consulta UPDATE
o DELETE
. En el caso de las
consultas a SELECT
, devuelve el número de filas obtenidas, pero es posible que no siempre se conozca, en cuyo caso
devuelve null
.
getColumnCount(): ?int
Devuelve el número de columnas de ResultSet
.
Información de consulta
Para obtener detalles sobre la última consulta ejecutada, utilice:
echo $database->getLastQueryString(); // genera la consulta SQL
$result = $database->query('SELECT * FROM articles');
echo $result->getQueryString(); // muestra la consulta SQL
echo $result->getTime(); // muestra el tiempo de ejecución en segundos
Para mostrar el resultado en forma de tabla HTML, utilice:
$result = $database->query('SELECT * FROM articles');
$result->dump();
También puede obtener información sobre los tipos de columnas en ResultSet
:
$result = $database->query('SELECT * FROM articles');
$types = $result->getColumnTypes();
foreach ($types as $column => $type) {
echo "$column is of type $type->type"; // por ejemplo, 'id es de tipo int'
}
Registro de consultas
Puede implementar un registro de consultas personalizado. El evento onQuery
es un array de callbacks que se
invocan después de cada ejecución de consulta:
$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');
}
};