SQL Way

Ви можете працювати з базою даних Nette двома способами: писати SQL-запити (спосіб SQL) або дозволити SQL генеруватися автоматично(спосіб Explorer). SQL-спосіб дозволяє безпечно створювати запити, зберігаючи повний контроль над їх структурою.

Детальніше про налаштування з'єднання з базою даних дивіться в розділі Підключення та конфігурація.

Базові запити

Метод query() виконує запити до бази даних і повертає об'єкт ResultSet, що представляє результат. Якщо запит завершився невдало, метод згенерує виключення. Ви можете переглянути результат запиту за допомогою циклу foreach або скористатися однією з допоміжних функцій.

$result = $database->query('SELECT * FROM users');

foreach ($result as $row) {
	echo $row->id;
	echo $row->name;
}

Щоб безпечно вставляти значення в SQL-запити, використовуйте параметризовані запити. Nette Database робить це дуже просто: просто додайте кому і значення до SQL-запиту.

$database->query('SELECT * FROM users WHERE name = ?', $name);

Для декількох параметрів ви можете чергувати SQL-запит з параметрами:

$database->query('SELECT * FROM users WHERE name = ?', $name, 'AND age > ?', $age);

Або спочатку написати весь SQL-запит, а потім додати всі параметри:

$database->query('SELECT * FROM users WHERE name = ? AND age > ?', $name, $age);

Захист від SQL-ін'єкцій

Чому важливо використовувати параметризовані запити? Тому що вони захищають вас від атак SQL-ін'єкцій, коли зловмисники можуть вставляти шкідливі SQL-команди для маніпулювання або доступу до даних бази даних.

**Ніколи не вставляйте змінні безпосередньо в SQL-запит! Завжди використовуйте параметризовані запити, щоб захиститися від SQL-ін'єкцій.

// НЕБЕЗПЕЧНИЙ КОД - вразливий до SQL-ін'єкцій
$database->query("SELECT * FROM users WHERE name = '$name'");

// ✅ Безпечний параметризований запит
$database->query('SELECT * FROM users WHERE name = ?', $name);

Обов'язково ознайомтеся з потенційними ризиками безпеки.

Методи запитів

Де Умови

Ви можете записати умови WHERE у вигляді асоціативного масиву, де ключами є назви стовпців, а значеннями – дані для порівняння. Nette Database автоматично вибирає найбільш підходящий SQL-оператор на основі типу значення.

$database->query('SELECT * FROM users WHERE', [
	'name' => 'John',
	'active' => true,
]);
// WHERE `name` = 'John' AND `active` = 1

Ви також можете явно вказати оператор у ключі:

$database->query('SELECT * FROM users WHERE', [
	'age >' => 25,           // використовує оператор >
	'name LIKE' => '%John%', // використовує оператор LIKE
	'email NOT LIKE' => '%example.com%', // використовує оператор NOT LIKE
]);
// WHERE `age` > 25 AND `name` LIKE '%John%' AND `email` NOT LIKE '%example.com%'

Особливі випадки, такі як null значення або масиви, обробляються автоматично:

$database->query('SELECT * FROM products WHERE', [
	'name' => 'Laptop',         // використовує оператор =
	'category_id' => [1, 2, 3], // використовує IN
	'description' => null,      // uses IS NULL
]);
// WHERE `name` = 'Ноутбук' AND `category_id` IN (1, 2, 3) AND `description` IS NULL

Для від'ємних умов використовуйте оператор NOT:

$database->query('SELECT * FROM products WHERE', [
	'name NOT' => 'Laptop',         // використовує оператор <>
	'category_id NOT' => [1, 2, 3], // використовує NOT IN
	'description NOT' => null,      // uses IS NOT NULL
	'id' => [],                     // пропущено
]);
// WHERE `name` <> 'Ноутбук' AND `category_id` NOT IN (1, 2, 3) AND `description` IS NOT NULL

За замовчуванням умови об'єднуються за допомогою оператора AND. Ви можете змінити цю поведінку за допомогою заповнювача ?or.

Впорядкувати за правилами

Речення ORDER BY можна визначити як масив, де ключами є стовпці, а значеннями – булеві числа, що вказують на порядок зростання:

$database->query('SELECT id FROM author ORDER BY', [
	'id' => true,  // висхідний
	'name' => false, // за спаданням
]);
// SELECT id FROM author ORDER BY `id`, `name` DESC

Вставка даних (INSERT)

Для вставки записів використовуйте інструкцію SQL INSERT.

$values = [
	'name' => 'John Doe',
	'email' => 'john@example.com',
];
$database->query('INSERT INTO users ?', $values);
$userId = $database->getInsertId();

Метод getInsertId() повертає ідентифікатор останнього вставленого рядка. Для деяких баз даних (наприклад, PostgreSQL) ви повинні вказати ім'я послідовності за допомогою $database->getInsertId($sequenceId).

Ви також можете передавати спеціальні значення, такі як файли, об'єкти типу DateTime або перелічувані типи, як параметри.

Вставка декількох записів одночасно:

$database->query('INSERT INTO users ?', [
	['name' => 'User 1', 'email' => 'user1@mail.com'],
	['name' => 'User 2', 'email' => 'user2@mail.com'],
]);

Виконання пакетної INSERT набагато швидше, оскільки виконується лише один запит до бази даних замість кількох окремих запитів.

Примітка з безпеки: Ніколи не використовуйте неперевірені дані як $values. Ознайомтеся з можливими ризиками.

Оновлення даних (UPDATE)

Для оновлення записів використовуйте інструкцію SQL UPDATE.

// Оновлення одного запису
$values = [
	'name' => 'John Smith',
];
$result = $database->query('UPDATE users SET ? WHERE id = ?', $values, 1);

Ви можете перевірити кількість зачеплених рядків за допомогою $result->getRowCount().

Ви можете використовувати оператори += та -= в UPDATE:

$database->query('UPDATE users SET ? WHERE id = ?', [
	'login_count+=' => 1, // інкремент login_count
], 1);

Щоб вставити або оновити запис, якщо він вже існує, використовуйте техніку 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)
//  ПРИ ОНОВЛЕННІ ДВІЧНОГО КЛЮЧА `name` = 'Jim', `year` = 1978

Зверніть увагу, що Nette Database розпізнає контекст SQL-команди, в якій використовується параметр з масивом, і відповідно генерує SQL-код. Наприклад, з першого масиву вона створила (id, name, year) VALUES (123, 'Jim', 1978), а другий перетворила в name = 'Jim', year = 1978. Більш детально це описано в розділі Підказки для побудови SQL.

Видалення даних (DELETE)

Для видалення записів використовуйте інструкцію SQL DELETE. Приклад з кількістю видалених рядків:

$count = $database->query('DELETE FROM users WHERE id = ?', 1)
	->getRowCount();

Підказки щодо побудови SQL

Заповнювачі SQL дозволяють керувати тим, як значення параметрів включаються у вирази SQL:

Підказка Опис Автоматично використовується для
?name Використовується для імен таблиць або стовпців
?values Створює (key, ...) VALUES (value, ...) INSERT ... ?, REPLACE ... ?
?set Створює присвоєння key = value, ... SET ?, KEY UPDATE ?
?and Об'єднує умови в масиві з AND WHERE ?, HAVING ?
?or Об'єднує умови в масив з OR
?order Генерує умову ORDER BY ORDER BY ?, GROUP BY ?

Для динамічної вставки назв таблиць або стовпців використовуйте заповнювач ?name. Nette Database забезпечує правильне екранування відповідно до угод бази даних (наприклад, укладання в зворотні лапки для MySQL).

$table = 'users';
$column = 'name';
$database->query('SELECT ?name FROM ?name WHERE id = 1', $column, $table);
// SELECT `name` FROM `users` WHERE id = 1 (в MySQL)

Попередження: Використовуйте заповнювач ?name лише для перевірених назв таблиць і стовпців. В іншому випадку ви ризикуєте отримати вразливості в системі безпеки.

Інші підказки зазвичай не потрібно вказувати, оскільки Nette використовує розумне автоматичне визначення при побудові SQL-запитів (див. третій стовпчик таблиці). Однак ви можете використовувати їх у ситуаціях, коли ви хочете об'єднати умови, використовуючи OR замість 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'

Особливі цінності

Окрім стандартних скалярних типів (наприклад, string, int, bool), ви також можете передавати спеціальні значення як параметри:

  • Файли: Використовуйте fopen('file.png', 'r'), щоб вставити двійковий вміст файлу.
  • Дата і час: об'єкти DateTime автоматично конвертуються у формат дати бази даних.
  • Перелічувані значення: Екземпляри enum конвертуються у відповідні значення.
  • Літерали SQL: Створені за допомогою Connection::literal('NOW()'), вони вставляються безпосередньо в запит.
$database->query('INSERT INTO articles ?', [
	'title' => 'My Article',
	'published_at' => new DateTime,
	'content' => fopen('image.png', 'r'),
	'state' => Status::Draft,
]);

Для баз даних, які не мають власної підтримки типу datetime (наприклад, SQLite та Oracle), значення DateTime конвертуються відповідно до параметра конфігурації formatDateTime (за замовчуванням: U для мітки часу Unix).

Літерали SQL

У деяких випадках вам може знадобитися вставити необроблений SQL-код як значення, не обробляючи його як рядок і не екрануючи. Для цього використовуйте об'єкти класу Nette\Database\SqlLiteral, які можна створити за допомогою методу Connection::literal().

$result = $database->query('SELECT * FROM users WHERE', [
	'name' => $name,
	'year >' => $database::literal('YEAR()'),
]);
// SELECT * FROM users WHERE (`name` = 'Jim') AND (`year` > YEAR())

Альтернативний варіант:

$result = $database->query('SELECT * FROM users WHERE', [
	'name' => $name,
	$database::literal('year > YEAR()'),
]);
// SELECT * FROM users WHERE (`name` = 'Jim') AND (year > YEAR())

SQL-літерали також можуть містити параметри:

$result = $database->query('SELECT * FROM users WHERE', [
	'name' => $name,
	$database::literal('year > ? AND year &lt; ?', $min, $max),
]);
// SELECT * FROM users WHERE `name` = 'Jim' AND (year > 1978 AND year < 2017)

Це дозволяє створювати гнучкі комбінації:

$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')

Отримання даних

Комбінації клавіш для SELECT-запитів

Для спрощення пошуку даних у класі Connection передбачено кілька методів швидкого доступу, які поєднують виклик query() з наступним викликом fetch*(). Ці методи приймають ті самі параметри, що й query(), тобто SQL-запит і додаткові параметри. Детальний опис методів fetch*() можна знайти нижче.

fetch($sql, ...$params): ?Row Виконує запит і отримує перший рядок як об'єкт Row.
fetchAll($sql, ...$params): array Виконує запит і отримує всі рядки у вигляді масиву об'єктів Row.
fetchPairs($sql, ...$params): array Виконує запит і отримує асоціативний масив, де перший стовпець є ключем, а другий – значенням.
fetchField($sql, ...$params): mixed Виконує запит і отримує значення першої комірки в першому рядку.
fetchList($sql, ...$params): ?array Виконує запит і отримує перший рядок у вигляді індексованого масиву.

Приклад:

// fetchField() - повертає значення першої комірки
$count = $database->query('SELECT COUNT(*) FROM articles')
	->fetchField();

foreach – Ітерація над рядками

Після виконання запиту повертається об'єкт ResultSet, який дозволяє ітераційно переглядати результати різними способами. Найпростіший і найефективніший спосіб отримати рядки – це ітерація в циклі foreach. Цей метод обробляє рядки по одному і дозволяє уникнути зберігання всіх даних в пам'яті одразу.

$result = $database->query('SELECT * FROM users');

foreach ($result as $row) {
	echo $row->id;
	echo $row->name;
	//...
}

Цикл ResultSet можна виконати лише один раз. Якщо вам потрібно виконати його кілька разів, ви повинні спочатку завантажити дані в масив, наприклад, за допомогою методу fetchAll().

fetch(): ?Row

Виконує запит і отримує один рядок як об'єкт Row. Якщо рядків більше немає, повертається null. Цей метод просуває внутрішній вказівник на наступний рядок.

$result = $database->query('SELECT * FROM users');
$row = $result->fetch(); // отримує перший рядок
if ($row) {
	echo $row->name;
}

fetchAll(): array

Отримує всі рядки, що залишилися, з ResultSet у вигляді масиву об'єктів Row.

$result = $database->query('SELECT * FROM users');
$rows = $result->fetchAll(); // отримує всі рядки
foreach ($rows as $row) {
	echo $row->name;
}

fetchPairs (string|int|null $key = null, string|int|null $value = null)array

Отримує результати у вигляді асоціативного масиву. Перший аргумент вказує стовпець, який потрібно використати як ключ, а другий – як значення:

$result = $database->query('SELECT id, name FROM users');
$names = $result->fetchPairs('id', 'name');
// [1 => 'John Doe', 2 => 'Jane Doe', ...]

Якщо вказано лише перший параметр, значенням буде весь рядок (як об'єкт Row ):

$rows = $result->fetchPairs('id');
// [1 => Row(id: 1, name: 'John'), 2 => Row(id: 2, name: 'Jane'), ...]

Якщо в якості ключа передано null, масив буде проіндексовано чисельно, починаючи з нуля:

$names = $result->fetchPairs(null, 'name');
// [0 => 'John Doe', 1 => 'Jane Doe', ...]

fetchPairs (Closure $callback)array

Крім того, ви можете передбачити зворотний виклик, який визначає пари ключ-значення або значення для кожного рядка.

$result = $database->query('SELECT * FROM users');
$items = $result->fetchPairs(fn($row) => "$row->id - $row->name");
// ['1 - John', '2 - Jane', ...]

// Зворотний виклик також може повертати масив з парою ключ & значення:
$names = $result->fetchPairs(fn($row) => [$row->name, $row->age]);
// ['John' => 46, 'Jane' => 21, ...]

fetchField(): mixed

Отримує значення першої комірки у поточному рядку. Якщо рядків більше немає, повертає null. Цей метод пересуває внутрішній вказівник на наступний рядок.

$result = $database->query('SELECT name FROM users');
$name = $result->fetchField(); // отримує ім'я з першого рядка

fetchList(): ?array

Отримує рядок як індексований масив. Якщо рядків більше немає, повертає null. Цей метод просуває внутрішній вказівник на наступний рядок.

$result = $database->query('SELECT name, email FROM users');
$row = $result->fetchList(); // ['John', 'john@example.com']

getRowCount(): ?int

Повертає кількість рядків, на які вплинув останній запит UPDATE або DELETE. Для запитів SELECT повертає кількість отриманих рядків, але вона не завжди може бути відомою – у таких випадках повертається null.

getColumnCount(): ?int

Повертає кількість стовпців у файлі ResultSet.

Інформація про запит

Щоб отримати інформацію про останній виконаний запит, скористайтеся кнопкою :

echo $database->getLastQueryString(); // виводить SQL-запит

$result = $database->query('SELECT * FROM articles');
echo $result->getQueryString();    // виводить SQL-запит
echo $result->getTime();           // виводить час виконання в секундах

Щоб відобразити результат у вигляді HTML-таблиці, скористайтеся::

$result = $database->query('SELECT * FROM articles');
$result->dump();

Ви також можете отримати інформацію про типи стовпців з сайту ResultSet:

$result = $database->query('SELECT * FROM articles');
$types = $result->getColumnTypes();

foreach ($types as $column => $type) {
	echo "$column is of type $type->type"; // наприклад, 'id має тип int'
}

Логування запитів

Ви можете реалізувати власне логування запитів. Подія onQuery – це масив зворотних викликів, які викликаються після виконання кожного запиту:

$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');
	}
};