Security has become a foremost concern on the Web in the past few years. Hackers have always been around, but with the increase in computer literacy and the ease of access to virtually any data, the problem has increased exponentially. It is now rare for a new website to not get comment spam withindays of its release, even if it is not promoted at all. |
|
With so many publishers large and small running on the WordPress platform it is a natural target for hackers. The overall security of your site depends on a lot of factors, including the security protocol your hosting company implements at the server level. When someone hacks a website, what are they trying to accomplish? Some just do it for fun while others have more dangerous things in mind and even worse, financial interests. |
|
Don't take their, "we take the security of your site very seriously" corporate nonsense at face value. Some hackers like to show off and will maybe replace your home page with a huge announcement that says your site has been defaced. Not very nice but if you have a backup it can be fairly easy to repair. Other hack attempts are done to improve a website ranking in Google using hidden keywords injected on innocent sites. This is frowned upon and has negative impact on your own search engine ranking. If lots of people complain about their sites being hacked and the common denominator is their host, well – that pretty much speaks for itself. |
|
The following are what I have discovered as extremely useful, there are some rules if you been hacked, and then some tools you can use to find future attacks easier as well as prevent them. First what do you do if you were hacked (assume working in the directory /usr/share/wordpress, but really wherever your wordpress install lives): |
|
- Make a backup of your database
- Get a complete list of your existing themes and plugins
- Download the latest WordPress zip file
- Disable your existing WordPress installation (copy index.php to some other name)
- Move your old WordPress install from /usr/share/wordpress to /usr/share/wordpress.old
- Make a new /usr/share/wordpress directory and install the latest WordPress into this directory
- Restore your existing database
- Use the default theme
- Compare the Exploit-DB for WordPress against your list of plugins and themes
- Search Google for specific 0-day attacks against your version of WordPress, if any exist patch them as explained
- Reinstall your Theme if not on the above list
- Reinstall your Plugins if not on the aforementioned list, if you cannot find your plugins because they have gone the way of the dodo and become available, then inspect every file for any form of malware and then copy from your ‘hacked’ backup directory. However, if you do not know what malware looks like, then get a professional to help.
|
|
There are several plugins I install on every new WordPress site I build. No single plugin covers all the bases, but combined they offer about as much security for a WordPress installation as is possible. |
|
- WP Security Scan – this plugin will help to bring to your attention some basic security vulnerabilities like the use of auto-generated passwords, file permission issues and the existence of an "admin" user account. Very basic, but very useful. While the information in the plugin directory states the plugin is compatible up to version 2.8.4, I continue to use it on my sites without any problems.
|
|
- Secure WordPress – Secure WordPress takes what WP Security Scan began and runs with it. It removes the version information from the site's header, as well as eliminating core, plugin and theme update information for all non-admin users. It also offers an optional free malware and vulnerability scan through sitesecuritymonitor.com.
|
|
- Exploit Scanner – it is important to note that this plugin does not actually remove any hacked or suspicious files. That is left for the user to take care of. It can also throw up false positives from time to time, but your site's security is definitely a better safe than sorry proposition. Exploit Scanner notifications are split into 3 categories: severe, warning and note, which helps you to prioritize what needs to be checked NOW and what might be able to wait until you have some free time.
|
|
- Limit Login Attempts or Login LockDown – both Limit Login Attempts and Login Lockdown offer similar functionality. Essentially they limit the number of login attempts based on IP address and locks out any IP address that exceeds the limit for a specified period of time. You can configure the number of attempts to length of time of the lock out with each plugin. Limit Login Attempts states it is compatible up to WordPress version 3.0.1 while Login LockDown claims to be compatible up to only version 2.8.4, though I should note that I am still using Login Lockdown on several sites without issue.
|
|
- WordPress Firewall – one of the most useful features of this plugin is the ability to have an e-mail sent if there has been a suspected attack. Not only are you alerted via e-mail, but that e-mail includes information about the specific file that was accessed and the IP address of the offender. A particularly noteworthy feature is that when an attack is suspected the offending IP address is redirected either to the site's home page or a 404 page (your choice). You can also whitelist specific IP addresses (like, I don't know – YOURS) so you are still able to make changes to the files associated with your theme or plugins via WordPress without triggering the plugin. Again, the plugin documentation claims the plugin is only compatible up to WordPress version 2.8, but I continue to use it and haven't had any problems with it thus far.
|
|
Securing your WordPress is not and hard task and should be taken into consideration by everyone of you. It will be better to be safe then sorry :) . Below I will put some of the steps you need to do to heave a more secure WordPress site and have a more quiet sleep. |
|
URL-Based Exploits
|
With URL-based exploits, hackers try to find weak spots on your website by making requests that would normally return an error but for some reason are completed. |
http://mysite.com/trying/to/exploit/%2F/config |
|
The above hypothetical URL is essentially a stab in the dark by a hacker. But if the request is met, even though the URL is clearly not meant to go anywhere, the hacker might be able to make use of it. |
|
USING .HTACCESS AS A FIREWALL
|
One of the best methods I’ve found against this kind of threat is an .htaccess firewall. It basically consists of rules that automatically block requests based on strings in the URL.
For example, there is no good reason for an opening bracket ([) to be in a URL. If a request is made using a URL that contains a bracket, then either the user has mistyped something or someone is looking for a security hole. Either way, generating a “403 Forbidden” page is good practice in this case. RedirectMatch 403 \[ Paste the line above in your .htaccess file to block any request that contains an opening bracket. To guard against more than just brackets, you will need a more complex ruleset. Luckily, our awesome editor Jeff Starr has gone out of his way to create a great .htaccess ruleset. The latest iteration is called 5G Firewall and is freely available from Perishable Press for your copy-and-pasting pleasure. The firewall is modular, so you can delete lines from it without breaking the functionality. If something goes wrong when you’re using it, you can usually track down the problem by deleting lines until it starts working again. Once you’ve found the offending line, you can delete it and paste back the rest. |
|
ADDITIONAL SERVER-LEVEL PROTECTION
|
So far, the measures we have taken have nothing to do with our website’s actual code. However secure your code is, you will still need to implement something like what we did above. We don’t have time to look at all tips and tricks for .htaccess, but you can do quite a few other things: |
- Password-protect directories.,
- Use smart redirects,
- Deny access based on IP or an IP range,
- Force downloading of files,
- Disable hotlinking,
- The list goes on.
|
Look at the “Further Reading” section at the end of this article, and become good friends with your.htaccess file. It might seem daunting and confusing at first, but solid knowledge of how to use it will go a long way |
|
Protecting Against Malicious Users
|
The second type of problem that can arise is when someone performs an action that they are not supposed to do. This doesn’t necessarily mean that they intended to harm the website, but it could happen. |
If users are listed somewhere in the admin part of your website, chances are that a link is displayed to delete each user. The link could point to the script in the following location: |
http://mysite.com/admin/scripts/delete_user.php?user_id=5 |
|
This link is relatively obscure; that is, a normal user doesn’t have a good chance of stumbling on it. But if directory listings are enabled, then someone naughty could go tohttp://mysite.com/admin/scripts/, see that you have a delete_user.php file there, and make various requests to try to delete a user. If the script does not check permission or intent, then anyone who visits the link above could delete user 5. |
|
AUTHORITY AND INTENT
|
Whenever a user initiates an action, we need to take two things into consideration. Does the user have authority to perform the action (i.e. do they have permission)? If the user does have authority, do they also intend to complete the action (i.e. do they mean to do what they’re doing)? WordPress has functions to help you make sure that both conditions are met before an action in the script is triggered. We will look at these in detail shortly. If you are building your website from scratch, then you will need to make sure that each user has associated permissions and that you know which action can be performed under which condition. For example, you would probably want only administrators to be able to delete content from the website. Every time a user tries to delete content, you would need to make sure that they are actually an administrator — this is the “authority” part. Intent is best described with an example. Let’s assume you can use the following link to delete a comment: |
http://mysite.com/admin/scripts/delete_comment.php?comment_id=5 |
|
The script itself will check that the user is currently logged in and is an administrator, so it takes care of the authority check. Could someone still wreak havoc? Sure they could! A sneaky hacker could put a link on their own website pointing to the same location: |
<a href="http://mysite.com/admin/scripts/delete_comment.php?comment_id=5">Super-Happy Times Here!</a> |
|
Because everyone likes super-happy times, many users would click the link. In 99% of cases, nothing would happen, because those visitors would not be administrators of mysite.com. But if a logged-in administrator of mysite.com did click on the link, then the action would execute, even though the link was actually clicked from vilehackerperson.com. You might think that the odds of this happening are astronomical. In a way you’d be right, but remember that a hacker can do this extremely easily and can automate it. Millions of people get spam email saying that Bill Gates will take away the Internet unless they pay $1,000. Most recipients don’t see the email or throw it out or mark it as spam or what have you, but perhaps 1 out of every 2 million people is lured in. A thousand bucks for basically doing nothing is not bad at all. And a hacker probably wouldn’t put the link on their own website; all they would need to do is hack a big website and embed the link there without anyone noticing. |
|
Checking For Authority In WordPress
|
WordPress has a permissions system built in referred to as “Roles and Permissions.” Capabilities are the basis of the whole system; roles are just a way to group a set of capabilities together. If a user has the delete_posts capability, then they have the authority to delete posts. If a user has the edit_posts capability, then they can edit their posts. Quite a few capabilities are available, and you can even create your own. Roles are basically groups of capabilities. A user with the role of “contributor” has three capabilities:read, delete_posts and edit_posts. These give the user the authority to read posts and to edit or delete their own posts. These capabilities could be granted individually to any user, but grouping them into frequently used bundles is much easier and more practical. With that in mind, let’s look at how to use WordPress functions to ensure that a user has the authority to complete an action that they initiate |
if(current_user_can("delete_users")) { wp_delete_user(5); } else { die("You naughty, naughty person. Of course, you could just be logged out…"); } |
|
Here, we’ve made sure that the user has the delete_users capability before they are able to complete the action. Don’t make premature assumptions when protecting your scripts; in many cases, especially those of authority, the intent is not malicious. The current_user_can() function takes one argument, which can be a role or a permission. We could let “editors” (who don’t normally have the delete_users capability) delete users in the following way: |
if(current_user_can("editor")) { wp_delete_user(5); } else { die("You must be an editor to delete a user"); } |
|
Be careful with the method above because the roles are not inclusive. This function doesn’t require the user to be at least an editor; it requires them to be exactly an editor (if that makes sense). Because of this, I find it preferable to use capabilities, especially if I have modified the default permissions extensively. Two other similar functions enable you to examine the capabilities of users other than the currently logged-in one. |
if(user_can(5, "manage_links")) { echo "User 5 is allowed to manage links"; } else { echo "Sadness! User 5 may not manage links"; } if(author_can(1879, "update_themes")) { echo "The author of post #1879 is allowed to update themes"; } else { echo "Oh noes, our friend, the author of post #1879 may not update themes"; } |
|
The user_can() function checks whether a given user has the given capability (or role). The first argument is the user’s ID or a user object; the second argument is the name of the capability or role that we want to check for. The author_can() function checks whether the author of a given post has the given capability (or role). The first parameter should be a post ID or a post object; the second is the capability or role that we are examining. |
|
Checking For Intent In WordPress
|
Intent is a bit more difficult to check. In the good ol’ days, a check of $_SERVER['HTTP_REFERER']was the way to go. This stored the page that the user came from. If the domain name was their own, then the user was probably OK… unless, of course, someone had gotten into their files and inserted a link that deleted the user’s database if they clicked on it as an administrator. A newer more secure method was implemented in WordPress 2.03 — quite some time ago — called nonces. Nonce stands for “number used once” and is used frequently in cryptography to secure communications. It is a number that is generated before an action is initialized, then attached to the action’s call, and then checked before the action completes. In WordPress, you would generally use nonces in one of two places: forms and normal links. Let’s look first at how to generate a nonce in a form. |
|
NONCES IN FORMS
|
<form id="myform" method="post" action="myawesomescript.php"> <h2>Enter an awesome word here</h2> <input type='text' name='word'> <?php wp_nonce_field( 'awesome_name_nonce') ?> </form> |
|
This will generate a hidden input field containing your generated nonce, which will be sent along with all of the form’s other data. The wp_nonce_field function takes four parameters:
- The first parameter is optional, but recommended because it gives the nonce a unique identifier.
- The second parameter is the name of the field. This is also optional and defaults to
_wpnonce .
- The third parameter is boolean. If set to
true , it will also send the referrer for validation.
- The fourth parameter is also a boolean and controls whether the field is echoed right then and there.
The resulting hidden field would look something like this: |
<input type="hidden" id="_wpnonce" name="_wpnonce" value="d6d71w4664"> |
|
Setting all of this up won’t make a huge difference if it isn’t used when the form is actually processed. We need to check for the presence and the value of the nonce before allowing any actions to be performed. Here is one way to do that: |
if (!wp_verify_nonce($_POST['_wpnonce'],'awesome_name_nonce') ) { die('Oops, your nonce didn\'t verify. So there.'); } else { awesome_word_inserter($_POST["word"]); } |
|
Here, we’ve used the wp_verify_nonce() function to make sure that our nonce is correct. This function takes two parameters: the first is the value of the nonce field, and the second is the name of the action that we defined (this was the first parameter for the wp_nonce_field() function). If the nonce is verified, then the function will return true; otherwise, it will return false. |
|
NONCES IN LINKS
|
In some cases, you will want a link, instead of a form, to perform an action. This would typically look like our previous examples: |
http://mysite.com/admin/scripts/deletethatthing.php?thing_id=231 |
|
To generate a nonce for a link, we can use the following method:; |
$base_url = "http://mysite.com/admin/scripts/deletethatthing.php?thing_id=231"; $nonce_url = wp_nonce_url( $base_url, "thingdeleter_nonce"); echo "<a href='".$nonce_url."'>Delete that thing</a>"; |
|
The resulting link would be something like this: |
http://mysite.com/admin/scripts/deletethatthing.php?thing_id=231&_wpnonce=d6f77f1364 |
|
When we actually go to the script, we can check the nonce using the same method as before: |
if (!wp_verify_nonce($_GET['_wpnonce'],'thingdeleter_nonce') ) { die('Oops, your nonce didn\'t verify. So there.'); } else { delete_that_thing($_GET["thing_id"]); } |
|
Checking Authority And Intent At The Same Time
|
|
We need to look at both aspects at once; although, now that we’ve looked at all of the components, it won’t exactly be rocket science! Let’s take a simple link that lets the user delete a comment. We would have this on the page that lists comments: |
$nonce_url = wp_nonce_url("http://mysite.com/scripts/delete_comment.php?comment_id=1451", "delete_comment_nonce"); echo "<a href='".$nonce_url."'>dispose of this comment</a>"; |
|
Data Security
|
Our work might seem to be done, but if you’ve been developing for a while, then you know it never is. An additional layer of security needs to be added to stop insecure data (or erroneous data) from entering our database. Adding this additional layer is called data sanitization. A quick clarification: data sanitization refers to the process of cleaning up our data to make sure that nothing suspicious gets sent to the database. Validation refers to all of the checks we perform on data to make sure they are the types of data we need; it is typically done to ensure that the user has entered a valid email address, a well-formed URL, etc. The two terms are sometimes used interchangeably, and other methods may be similar, but they are quite different things. For our purposes, sanitization is a bit more important, because it has more to do with security. The main thing we are trying to protect against is SQL injection. SQL injection is a technique used by hackers to exploit a database’s weaknesses. Take the following example: |
// A hacker goes to your search field and searches for elephant' - note the apostrophe at the end. In the script, the following SQL is run: SELECT ID, post_title FROM wp_posts WHERE post_title LIKE '%elephant'%' |
|
In the example above, the user’s search for elephant' has resulted in unclosed quotes in your script. While the hacker might not be able to do much with this, an error message would be generated, indicating to them that at the very least you are not sanitizing your data. In some cases, the SQL itself could be harmful or could give the hacker much more information than you’d like. Take the example of an administrator being able to enter a user’s log-in name in a form and getting back the user’s email address. |
SELECT user_email FROM wp_users WHERE user_login = 'danielp' |
|
If the hacker manages to perform an SQL injection attack, they could type ' OR 1=1 ' in the form, which would result in the following query: |
SELECT user_email FROM wp_users WHERE user_login = '' OR 1=1 '' |
|
This would return all email addresses in the database, because we would be retrieving all addresses for which the user’s log-in name is an empty string, or 1=1 , which is always true . There are two ways to protect against this kind of problem — implementing both is good practice. In round one, we validate the data. If an email address needs to be entered, we can filter out user data that does not conform to the format of email addresses. We simply make sure that the format is right, otherwise we redirect the user, stating that the address is invalid. If the data passes round one, we move to round two, where we remove all characters that could mess up the query. This usually entails escaping quotes so that they can be used only as actual quotes in the SQL query. When working without a framework, you would typically use addslashes() or something similar, but WordPress offers its own solution |
|
DATA SANITIZATION IN WORDPRESS
|
When communicating with the database in WordPress, the preferred method is to use the $wpdb class. You can read all about this in “WordPress Essentials: Interacting With the WordPress Database.” The class offers a number of methods to alleviate your SQL injection worries. Before jumping in, let’s look at some examples to get a basic understanding of how the class works. |
// Perform any query $wpdb->query("DELETE FROM wp_users WHERE user_id = 5"); // Get one column of data $posts = $wpdb->get_col("SELECT post_title FROM wp_posts WHERE post_status = 'publish' ORDER BY comment_count DESC LIMIT 0,10"); // Get a row of data $post = $wpdb->get_row("SELECT * FROM wp_posts WHERE ID = 1453"); // Get multiple rows and columns $posts = $wpdb->get_results("SELECT ID, post_title, post_date FROM wp_posts WHERE post_type = 'publish' ORDER BY post_date DESC LIMIT 0, 12 "); // Get a single value $author_id = $wpdb->get_var("SELECT post_author FROM wp_posts WHERE ID = 2563"); // Insert a record $wpdb->insert("wp_postmeta", array("post_id" => 2323, "meta_key" => "favorite_count", "meta_value" => 224 ), array("%d", "%s", "%d")); // Update a record $wpdb->update("wp_postmeta", array("meta_value" => 225), array("meta_key" => "favorite_count", "post_id" => 2323), array("%d"), array("%s", "%d")); |
|
The insert() and update() methods are helper methods, and they’re great because, apart from modularizing your database interactions a bit, they also take care of sanitization for you. If you want to use the general query() method, though, you will need to take care of it on your own. The easier way is just to use the escape() method: |
$data = $wpdb->escape($_POST[about_me]); $wpdb->query("UPDATE wp_usermeta SET meta_value = '$data' WHERE meta_key = 'description' AND user_id = 154 "); |
|
A slightly harder but better way to go about this is to use the prepare() method. An example from the WordPress Codex illustrates this perfectly: |
$metakey = "Harriet's Adages"; $metavalue = "WordPress' database interface is like Sunday Morning: Easy."; $wpdb->query( $wpdb->prepare( " INSERT INTO $wpdb->postmeta ( post_id, meta_key, meta_value ) VALUES ( %d, %s, %s ) ", 10, $metakey, $metavalue ) ); |
|
FURTHER PROTECTION USING SANITIZATION
|
Sanitization is a fairly big topic and requires quite some time to master. For now, you’ll be busy mostly determining which characters to allow and which to disallow, and then finding ways to parse out the latter. Some common needs are to parse HTML out of addresses, filter numbers out of strings, validate email addresses and so on, but you will need to implement your own solutions for more complex needs. See the “Further Reading” section for more on this topic. |
|
Final Thoughts
|
The measures needed to secure a website cannot be discussed in a single book, let alone a poor article. There are many methods and topics we did not look at, such as advanced password encryption, salts and so on. But hopefully, by implementing what we’ve discussed, your website will be much safer. Hackers usually go for the weakest link, so if your website is not insanely popular and is fairly secure, you should be OK. While I have a lot of experience in this field, I am far from being a security expert. If you know of any other or better methods, do share them in the comments. There is always something new to learn about website security. |
Support-Agent
Comments
Karim Chowdhury