Objective 12) Frost Tower Website Checkup
Difficulty:
Investigate Frost Tower's website for security issues. This source code will be useful in your analysis. In Jack Frost's TODO list, what job position does Jack plan to offer Santa? Ribb Bonbowford, in Santa's dining room, may have some pointers for you.
Hints and Resources
Hints provided after helping Rib Bonbowford and completing the Elf Code Python Terminal Challenge
SQL Injection with Source
When you have the source code, API documentation becomes tremendously valuable.
Troll Introduction
Talk to Ingreta Tude in Jack's Studio
Hey there! I’m Ingreta Tude. I really don’t like the direction Jack Frost is leading us. He seems obsessed with beating Santa and taking over the holiday season. It just doesn’t seem right. Why can’t we work together with Santa and the elves instead of trying to beat them? But, I do have an Objective for you. We’re getting ready to launch a new website for Frost Tower, and the big guy has charged me with making sure it’s secure. My sister, Ruby Cyster, created this site, and I don’t trust the results. Can you please take a look at it to find flaws? Here is the source code if you need it.
Solution
Step 1: Finding the SQL Injection Point
Opening the web site we see only a home page with a place to enter an email contact addres. However, looking through the main file of the source code, server.js, we see that there are a number of additional endpoints available.
Inspecting the code we find that the /detail endpoint provides us with an opportunitty to inject SQL
server.js | |
---|---|
195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 |
|
This endpoint allows the user to specify one or more contact IDs to view by supplying them as the last element of the URL, such as
https://staging.jackfrosttower.com/detail/1234
If a single ID is requested then the code at line 212 constructs the query as
SELECT * FROM uniquecontact WHERE id=?
and the '?' placeholder is substituted with the ID when the query is executed on line 219. MySQLjs automatically escapes any values passed this way, so injecting SQL using this method will not work.
However, if more than one ID is requested by separating them with commas, then the code on lines 204-210 build the query with whatever was used for the IDs built in. For example, the URL
https://staging.jackfrosttower.com/detail/1,2
will result in the following query being built
SELECT # FROM uniquecontact WHERE id=1 or ID=2 OR id=?
This gives us an opportunity to inject code, but first we have to bypass the authentication.
Step 2: Authentication Bypass
In order to use the /detail endpoint we need to pass the authentication check on line 200. On a sucessful login the session.uniqueID value is defined to be our username, but if we haven't logged in it will be undefined.
But further inspection of the code reveals a flaw in the /postcontact endpoint. This endpoint is called after pressing 'Save' on the 'Contact Us' page, and includes the following bit of code.
server.js | |
---|---|
141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 |
|
Here we see that if the email address entered into the 'Contact Us' page doesn't exist (line 149 evaluates as False), then line 157 adds a new record to the uniquecontact table.
If however the email address already exists in the table, then line 151 will set session.uniqueID to that value. Since the check in the /detail endpoint is just looking to see if session.uniqueID is set to something, this means if we submit a 'Contact Us' form twice using the same email then we will pass the authentication check.
Step 3: SQL Injection Constraints
Now that we are 'authenticated' to the system we can try to inject code using the /details endpoint.
One of the more common SQL Injection techniques involves the use of a UNION SELECT statement. This statement will execute a second SELECT and combines its results with the first. One of the requiements however when using a UNION SELECT is that the second SELECT statement must return the same number of columns as the first.
For example, the URL
https://staging.jackfrosttower.com/detail/1 UNION SELECT name FROM users;--,2
will be expanded by the code into the full query
SELECT * FROM uniquecontact WHERE id=1 UNION SELECT name FROM users;-- OR ID=2 OR ID=?
Note that I'm adding the comment characters ;--
at the end of my injection so everything after that will be ignored.
This query will result in an error because the first select, SELECT * FROM uniquecontact WHERE id=1
, returns 7 columns and SELECT name FROM users
only returns 1 column.
Additionally, after the /detail endpoint retreives values from the database it passes them to the detail.ejs file for rendering (see line 236 in server.js). The detail.ejs file requires that the values passed to it 'fit' into the structure of the uniquecontact table, shown below from the script used to initialize the database.
encontact_db.sql | |
---|---|
8 9 10 11 12 13 14 15 16 17 |
|
So when we inject code into the /detail endpoint we need to make sure not only that we return 7 colunmns, but that the types of those values match as well. Fortunately NULL values match anything (see below).
Step 4: Finding Jack's TODO List
From what we see in the source there does not appear to be any table or field in the encontact database that might contain Jack's TODO list. However, there could be another table that we are not aware of.
The MySQL query that would show use all the tables in the encontact database is
SELECT table_name FROM information_schema.tables WHERE table_schema='encontact'
.
But as mentioned above, our injection needs to return 7 values and this query only returns one. To get around this we can use the JOIN
statement to combine multiple SELECT
statements into one result.
SELECT * FROM (select NULL)c1 join (SELECT table_name FROM information_schema.tables WHERE table_schema='encontact')c2 join (select NULL)c3 join (select NULL)c4 join (select NULL)c5 join (select NULL)c6 join (select NULL)c7
This query would return 7 columns, the 2nd of which will be the names of the tables in the encontact database and the others all NULL. This query can be executed through SQL injection with the following URL
https://staging.jackfrosttower.com/detail/1 UNION SELECT * FROM (select NULL)c1 join (SELECT table_name FROM information_schema.tables WHERE table_schema='encontact')c2 join (select NULL)c3 join (select NULL)c4 join (select NULL)c5 join (select NULL)c6 join (select NULL)c7;--,2
Screenshot
Executing this injection tells us that there is another table named "todo" in the database. The MySQL query that would show all the column names for this table is
SELECT column_name FROM information_schema.columns WHERE table_name='todo'
which we can inject into the web site as well.
https://staging.jackfrosttower.com/detail/1 UNION SELECT * FROM (select NULL)c1 join (SELECT column_name FROM information_schema.columns WHERE table_name='todo')c2 join (select NULL)c3 join (select NULL)c4 join (select NULL)c5 join (select NULL)c6 join (select NULL)c7;--,1
Screenshot
Now we know there are three columns in the todo table, 'id', 'note', and 'completed', so we can compose our final query to retrieve all the rows of this table. Note that in this query we are using fewer join statements since the select from the todo table will return three columns.
https://staging.jackfrosttower.com/detail/1 UNION SELECT * FROM (select NULL)c1 join (select * from todo)c2 join (select NULL)c5 join (select NULL)c6 join (select NULL)c7;--,1
Screenshot
Completion
Answer
clerk
Ingreta Tude
Oh wow - I thought we left SQL injection in the last decade.
Thanks for your help finding this!
Extras
For Jack's complete TODO list, see this table in the Extras section.