<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0"
	xmlns:content="http://purl.org/rss/1.0/modules/content/"
	xmlns:wfw="http://wellformedweb.org/CommentAPI/"
	xmlns:dc="http://purl.org/dc/elements/1.1/"
	xmlns:atom="http://www.w3.org/2005/Atom"
	xmlns:sy="http://purl.org/rss/1.0/modules/syndication/"
	xmlns:slash="http://purl.org/rss/1.0/modules/slash/"
	>

<channel>
	<title>AskApache &#187; PHP</title>
	<atom:link href="http://www.askapache.com/php/feed/" rel="self" type="application/rss+xml" />
	<link>http://www.askapache.com</link>
	<description>Advanced Web Development</description>
	<lastBuildDate>Thu, 19 Nov 2009 12:21:49 +0000</lastBuildDate>
	
	<language>en</language>
	<sy:updatePeriod>hourly</sy:updatePeriod>
	<sy:updateFrequency>1</sy:updateFrequency>
			<item>
		<title>Advanced WordPress wp-config.php Tweaks</title>
		<link>http://www.askapache.com/wordpress/advanced-wp-config-php-tweaks.html</link>
		<comments>http://www.askapache.com/wordpress/advanced-wp-config-php-tweaks.html#comments</comments>
		<pubDate>Sat, 03 Oct 2009 07:23:37 +0000</pubDate>
		<dc:creator>AskApache</dc:creator>
				<category><![CDATA[CSS]]></category>
		<category><![CDATA[Cache]]></category>
		<category><![CDATA[Featured]]></category>
		<category><![CDATA[Hacking]]></category>
		<category><![CDATA[Javascript]]></category>
		<category><![CDATA[Linux Unix BSD]]></category>
		<category><![CDATA[PHP]]></category>
		<category><![CDATA[SEO]]></category>
		<category><![CDATA[Security]]></category>
		<category><![CDATA[Server Administration]]></category>
		<category><![CDATA[Shell Scripting]]></category>
		<category><![CDATA[Web Design]]></category>
		<category><![CDATA[Web Hosting]]></category>
		<category><![CDATA[Webmaster]]></category>
		<category><![CDATA[WordPress]]></category>
		<category><![CDATA[WordPress Plugins]]></category>
		<category><![CDATA[htaccess]]></category>
		<category><![CDATA[mod_rewrite]]></category>
		<category><![CDATA[admin]]></category>
		<category><![CDATA[advanced]]></category>
		<category><![CDATA[Cookies]]></category>
		<category><![CDATA[debugging]]></category>
		<category><![CDATA[phpinfo]]></category>
		<category><![CDATA[wp-config.php]]></category>

		<guid isPermaLink="false">http://www.askapache.com/?p=3341</guid>
		<description><![CDATA[<p>The bottom line for this article is that I want to make WordPress as fast, secure, and easy to install, run, and manage because I am using it more and more for client production sites, I will work for days in order to solve an issue so that I never have to spend time on that issue again. Time is money in this industry and that is ultimately (time) what there is to gain by tweaking WordPress.</p>

<p class="cnote"><strong>Note:</strong> I spent no time on readability, this is primarily a read the code and figure it out article.. This is for advanced users looking for a reference or discussion and for those of you looking to advance.  Feedback would be great <em>if you make it that far..</em></p>]]></description>
			<content:encoded><![CDATA[<p>The bottom line for this article is that I want to make WordPress as fast, secure, and easy to install, run, and manage because I am using it more and more for client production sites, I will work for days in order to solve an issue so that I never have to spend time on that issue again. Time is money in this industry and that is ultimately (time) what there is to gain by tweaking WordPress.</p>
<p class="cnote"><strong>Note:</strong> I spent no time on readability, this is primarily a read the code and figure it out article.. This is for advanced users looking for a reference or discussion and for those of you looking to advance.  Feedback would be great <em>if you make it that far..</em></p>
<p>For a better handle on the way I like to structure web site directories, see <a href="http://www.askapache.com/htaccess/optimize-website-files-cache-security.html">Optimize a Website for Speed, Security, and Easy Management</a> but note it is a bit outdated compared to what I&#8217;m doing now.  I don&#8217;t have the luxury of using only one type of server, or hosting provider anymore, so I have been working towards making things even more portable in order to move from host to host from server to server without issues i.e. my portable <a href="http://www.askapache.com/linux-unix/bash_profile-functions-advanced-shell.html">.bash_profile</a>.</p>
<p>So I&#8217;ve been basically experimenting various ways to accomplish that and thought I would share what I am currently doing for my benefit and hopefully get some input.  All of my WP installs run the development version, and one main idea with my setups is that upgrading is automated.  So I really keep the WordPress install clean and use plugins and wp-config.php to do all the customization.</p>
<ul>
<li>Portability &#8211; Hands-free upgrades and easy to move</li>
<li>Security &#8211; Additional security and protection</li>
<li>Speed &#8211; Less CPU and Disk I/O</li>
<li>Customization &#8211; All my favorite customizations</li>
</ul>
<h2>wp-config.php</h2>
<p>These are the main settings I use.. Seriously this is more like an interactive article, because to understand it you will need to do some code grepping.  You may want to grab a jolt.</p>
<h3>ASKAPACHE_ROOT</h3>
<p>The ASKAPACHE_ROOT variable is just a better way for me to be able to include and access all the different files in my site tree.  For instance, in my non-wp php files, I can do this:</p>
<pre>
!defined(&#039;ASKAPACHE_ROOT&#039;) &#038;&#038; require $_SERVER[&#039;DOCUMENT_ROOT&#039;] . &#039;/wp-config.php&#039;;
include(ASKAPACHE_ROOT . &#039;/includes/custom-download.inc.php&#039;);
</pre>
<h3>ASKAPACHE_LOCK</h3>
<p>This is one of my all-time favorite hacks, that I think is one of the most useful methods I employ as a web developer.  This allows me to use far-future-expire headers for optimum caching, while still forcing browsers to re-validate every day or so automatically, or forcing them to re-validate whenever I change the suffix.  This takes advantage of the <a href="http://www.askapache.com/htaccess/mod_rewrite-fix-for-caching-updated-files.html">mod_rewrite trick</a> that I use on EVERY site I run, definately worth learning. Because I practice best-practice web-standards, for every web site I create a single css file and javascript file, which I then add to the template like:</p>
<pre>
&lt;link rel="stylesheet" type="text/css" media="all" href="http://z.askapache.com/c/apache-0&lt;?php echo ASKAPACHE_LOCK?&gt;.css" /&gt;
&lt;script src="http://z.askapache.com/j/apache-0&lt;?php echo ASKAPACHE_LOCK;?&gt;.js" type="text/javascript"&gt;&lt;/script&gt;
</pre>
<pre>
&lt;?php
/**
 * The base configurations of the WordPress.
 *
 * This file has the following configurations: MySQL settings, Table Prefix,
 * Secret Keys, WordPress Language, and ABSPATH. You can find more information by
 * visiting {@link http://codex.wordpress.org/Editing_wp-config.php Editing
 * wp-config.php} Codex page. You can get the MySQL settings from your web host.
 *
 * This file is used by the wp-config.php creation script during the
 * installation. You don&#039;t have to use the web site, you can just copy this file
 * to "wp-config.php" and fill in the values.
 *
 * @package WordPress
 */
/* http://codex.wordpress.org/Editing_wp-config.php */
&nbsp;
/** /home/liet/askapache.com */
!defined(&#039;ASKAPACHE_ROOT&#039;) &#038;&#038; define(&#039;ASKAPACHE_ROOT&#039;, str_replace(&#039;/htdocs&#039;,&#039;&#039;, $_SERVER[&#039;DOCUMENT_ROOT&#039;]));
&nbsp;
/** The 008 at the end is for manual tweaking.  time() returns seconds since &#039;00:00:00 1970-01-01 UTC&#039;. */
// http://www.askapache.com/htaccess/mod_rewrite-fix-for-caching-updated-files.html
!defined(&#039;ASKAPACHE_LOCK&#039;) &#038;&#038; define(ASKAPACHE_LOCK&#039;, substr(time(),0,5).&#039;008&#039;); // 12533001
&nbsp;
/** absolute path to the Wordpress directory */
!defined(&#039;ABSPATH&#039;) &#038;&#038; define(&#039;ABSPATH&#039;, ASKAPACHE_ROOT .&#039;/htdocs/&#039;);
&nbsp;
/**
 * WP_SITEURL, defined since WordPress Version 2.2, allows the WordPress address (URL) to be defined. The valued defined is the address where your WordPress core files reside.
 * It should include the http:// part too. Do not put a slash "/" at the end.
 * Setting this value in wp-config.php overrides the wp_options table value for siteurl and disables the WordPress address (URL) field in the Administration &gt; Settings &gt; General panel.
 */
!defined(&#039;WP_SITEURL&#039;) &#038;&#038; define(&#039;WP_SITEURL&#039;, &#039;http://&#039;.$_SERVER[&#039;SERVER_NAME&#039;]);
&nbsp;
/**
 * WP_HOME is another wp-config.php option added in WordPress Version 2.2. Similar to WP_SITEURL,
 * WP_HOME overrides the wp_options table value for home but does not change it permanently.
 * home is the address you want people to type in their browser to reach your WordPress blog. It should include the http:// part. Also, do not put a slash "/" at the end.
 */
!defined(&#039;WP_HOME&#039;) &#038;&#038; define(&#039;WP_HOME&#039;, WP_SITEURL);
&nbsp;
/** no trailing slash, full paths only */
!defined(&#039;WP_CONTENT_DIR&#039;) &#038;&#038; define( &#039;WP_CONTENT_DIR&#039;, ABSPATH . &#039;wp-content&#039; );
&nbsp;
// full url - WP_CONTENT_DIR is defined further up
!defined(&#039;WP_CONTENT_URL&#039;) &#038;&#038; define( &#039;WP_CONTENT_URL&#039;, WP_SITEURL . &#039;/wp-content&#039;);
&nbsp;
/** Allows for the plugins directory to be moved from the default location. @since 2.6.0 */
// full path, no trailing slash
!defined(&#039;WP_PLUGIN_DIR&#039;) &#038;&#038; define( &#039;WP_PLUGIN_DIR&#039;, WP_CONTENT_DIR . &#039;/plugins&#039; );
&nbsp;
/** Allows for the plugins directory to be moved from the default location. @since 2.6.0 */
// full url, no trailing slash
!defined(&#039;WP_PLUGIN_URL&#039;) &#038;&#038; define( &#039;WP_PLUGIN_URL&#039;, WP_CONTENT_URL . &#039;/plugins&#039; );
&nbsp;
/** Allows for the plugins directory to be moved from the default location. @since 2.1.0 */
// Relative to ABSPATH.  For back compat.
//!defined(&#039;PLUGINDIR&#039;) &#038;&#038; define( &#039;PLUGINDIR&#039;, &#039;wp-content/plugins&#039; );
&nbsp;
/** Number of autosaves to save. TRUE is default and enables post revisions, FALSE disables revisions completely. */
!defined(&#039;WP_POST_REVISIONS&#039;) &#038;&#038; define(&#039;WP_POST_REVISIONS&#039;, 150);
&nbsp;
/* ini_set(&#039;memory_limit&#039;, WP_MEMORY_LIMIT); */
!defined(&#039;WP_MEMORY_LIMIT&#039;) &#038;&#038; define(&#039;WP_MEMORY_LIMIT&#039;, &#039;64M&#039;);
&nbsp;
/** Only check at this interval for new messages. Default is 5min */
/** @since 2.9  */
!defined(&#039;WP_MAIL_INTERVAL&#039;) &#038;&#038; define(&#039;WP_MAIL_INTERVAL&#039;, 3600); // 1 hour
&nbsp;
/** Saves updated post values to post from edit window every x seconds. (default 60)
 * When editing a post, WordPress uses Ajax to auto-save revisions to the post as you edit. You may want to increase this setting for longer delays in between auto-saves, or decrease the setting to make sure you never lose changes.
 * @since 2.5.0 */
!defined( &#039;AUTOSAVE_INTERVAL&#039; ) &#038;&#038; define( &#039;AUTOSAVE_INTERVAL&#039;, 60 );
&nbsp;
/** @since 2.9.0  */
/** Permanently deletes posts, pages, attachments, and comments which have been in the trash for EMPTY_TRASH_DAYS. */
!defined( &#039;EMPTY_TRASH_DAYS&#039; ) &#038;&#038; define( &#039;EMPTY_TRASH_DAYS&#039;, 300 );
</pre>
<hr class="C" />
<h2>Debugging WordPress</h2>
<p>One of my secrets for getting really good at this stuff is to master debugging.  There is really not ever a time when I am working on a site that I don&#8217;t have <a href="http://www.askapache.com/security/elite-log-file-scrolling-with-color-syntax.html">color-highlighted logs scrolling automatically in an ssh window</a>.  It&#8217;s really almost impossible to fix problems with wordpress or do any kind of advanced anything without being able to view debugging info.  At first I relied heavily on a <a href="http://www.askapache.com/php/custom-phpini-tips-and-tricks.html">custom php.ini</a> being available on the server, but after having to deal with many hosts who don&#8217;t allow <code>php.ini</code> files I now rely completely on setting values using <a href="http://php.net/manual/en/function.ini-set.php">ini_set</a> for ultimate portability. Detailed towards the end of this article and is also included in this <code>wp-config.php</code></p>
<pre>
/**#@+
 * DEBUGGING STUFF
 */
/** display of notices during development. if false, error_reporting is E_ERROR | E_WARNING | E_PARSE | E_USER_ERROR | E_USER_WARNING | E_RECOVERABLE_ERROR otherwise E_ALL */
!defined(&#039;WP_DEBUG&#039;) &#038;&#038; define(&#039;WP_DEBUG&#039;, false);
&nbsp;
/** The SAVEQUERIES definition saves the database queries to a array and that array can be displayed to help analyze those queries.
 *  The information saves each query, what function called it, and how long that query took to execute.  */
!defined(&#039;SAVE_QUERIES&#039;) &#038;&#038; define(&#039;SAVE_QUERIES&#039;, WP_DEBUG);
&nbsp;
!defined(&#039;ACTION_DEBUG&#039;) &#038;&#038; define(&#039;ACTION_DEBUG&#039;, WP_DEBUG);
&nbsp;
/** This will allow you to edit the scriptname.dev.js files in the wp-includes/js and wp-admin/js directories.  */
!defined(&#039;SCRIPT_DEBUG&#039;) &#038;&#038; define(&#039;SCRIPT_DEBUG&#039;, WP_DEBUG);

&nbsp;
/** Add define(&#039;WP_DEBUG_LOG&#039;, true); to enable php debug logging to WP_CONTENT_DIR/debug.log */
//!defined(&#039;WP_DEBUG_LOG&#039;) &#038;&#038; define(&#039;WP_DEBUG_LOG&#039;, true);
&nbsp;
/** This determines whether errors should be printed to the screen as part of the output or if they should be hidden from the user.
 *  Add define(&#039;WP_DEBUG_DISPLAY&#039;, false); to wp-config.php to use the globally configured setting for display_errors and not force it to On */
!defined(&#039;WP_DEBUG_DISPLAY&#039;) &#038;&#038; define(&#039;WP_DEBUG_DISPLAY&#039;, false);
</pre>
<hr class="C" />
<h2>Ultimate Security Tweaks</h2>
<p>Well, ultimate for WP&#8217;s built-in keys and password functions, this is all for wp-config.php keep in mind.  This is a very neccessary and recommended step, and is one of the only things I modify for each new installation.</p>
<h3>Security KEYS</h3>
<p>If like me you are familiar with password-cracking software like John the ripper, rainbow hash tables, l0pht-crack, etc.. then you will like to know that you can specify your own keys and salts for the encryption used by WP.  They are <code>AUTH_KEY</code>, <code>AUTH_SALT</code>, <code>SECURE_AUTH_KEY</code>, <code>SECURE_AUTH_SALT</code>, <code>LOGGED_IN_KEY</code>, <code>LOGGED_IN_SALT</code>, <code>NONCE_KEY</code>, <code>NONCE_SALT</code>,  <code>SECRET_KEY</code> and <code>SECRET_SALT</code>.</p>
<p>A random and long key gives you better encryption, and exponentially increasing that is using a random and long salt for the encryption.  Encryptions with known salts are incredibly easy to decrypt compared to encryptions with secure salts, because the salt + key individually need to be guessed in order to find a matching hash, vs. just the key if the salt is known.  See: <a href="http://www.askapache.com/security/locating-weak-passwords.html">Locating weak passwords</a>.</p>
<blockquote>
<p>A secret key is a hashing salt which makes your site harder to hack and access harder to crack by adding random elements to the password.</p>
<p>In simple terms, a secret key is a password with elements that make it harder to generate enough options to break through your security barriers. A password like &#8220;password&#8221; or &#8220;test&#8221; is simple and easily broken. A random, unpredictable password such as &#8220;88a7da62429ba6ad3cb3c76a09641fc&#8221; takes years to come up with the right combination.</p>
</blockquote>
<p>For more information on the technical background and breakdown of secret keys and secure passwords, see: </p>
<ul>
<li><a href="http://wordpress.org/support/topic/170987">WordPress Support Forum &#8211; HOWTO: Set up secret keys in WordPress 2.6+</a></li>
<li><a href="http://en.wikipedia.org/wiki/Password_cracking">Wikipedia&#8217;s explanation of Password Cracking</a></li>
</ul>
<p>I like to use the <a href="https://api.wordpress.org/secret-key/1.1/">WordPress.org secret-key service</a> 4 times.  That&#8217;s because for each key and salt I like to do: (1 key from api +random keyboard input+1 key from api).</p>
<pre>
/**#@+
 * Authentication Unique Keys.
 *
 * Change these to different unique phrases!
 * You can generate these using the {@link https://api.wordpress.org/secret-key/1.1/ WordPress.org secret-key service}
 * You can change these at any point in time to invalidate all existing cookies.
 * This will force all users to have to log in again.
 *
 * @since 2.6.0
 *
 * Get salt to add to hashes to help prevent attacks.
 *
 * The secret key is located in two places: the database in case the secret key
 * isn&#039;t defined in the second place, which is in the wp-config.php file. If you
 * are going to set the secret key, then you must do so in the wp-config.php
 * file.
 *
 * The secret key in the database is randomly generated and will be appended to
 * the secret key that is in wp-config.php file in some instances. It is
 * important to have the secret key defined or changed in wp-config.php.
 *
 * If you have installed WordPress 2.5 or later, then you will have the
 * SECRET_KEY defined in the wp-config.php already. You will want to change the
 * value in it because hackers will know what it is. If you have upgraded to
 * WordPress 2.5 or later version from a version before WordPress 2.5, then you
 * should add the constant to your wp-config.php file.
 *
 * Below is an example of how the SECRET_KEY constant is defined with a value.
 * You must not copy the below example and paste into your wp-config.php. If you
 * need an example, then you can have a
 * {@link https://api.wordpress.org/secret-key/1.1/ secret key created} for you.
 *
 * Salting passwords helps against tools which has stored hashed values of
 * common dictionary strings. The added values makes it harder to crack if given
 * salt string is not weak.
 *
 * @since 2.5
 * @link https://api.wordpress.org/secret-key/1.1/ Create a Secret Key for wp-config.php
 *
 * @return string Salt value from either &#039;SECRET_KEY&#039; or &#039;secret&#039; option
 */
define(&#039;AUTH_KEY&#039;,        &#039;jflkhaskljdfhkljasdhflkjashd;flkjhas;djfh;kajshdflkjashdlfkjhasdlkfhal?p[B+GR{@&gt;{Yq`c|LnG;dvq#| %OA_cbBSU6,rICC1o/c)-|&#039;);
define(&#039;SECURE_AUTH_KEY&#039;, &#039;jflkhaskljdfhkljasdhflkjashd;flkjhas;djfh;kajshdflkjashdlfkjhasdlkfhal?Vp[Bb15baar8&#038;R-r&lt;[T|?(xhJJABGq+Ux+U$)-Hltp/&#039;);
define(&#039;LOGGED_IN_KEY&#039;,   &#039;jflkhaskljdfhkljasdhflkjashd;flkjhas;djfh;kajshdflkjashdlfkjhasdlkfhal?Vp[B&lt;5n6DG|YWnJ9tY2!M1L)`{-$LW~~Ia%.uCbn!P. 41o2$Z$4&#039;);
define(&#039;NONCE_KEY&#039;,       &#039;jflkhaskljdfhkljasdhflkjashd;flkjhas;djfh;kajshdflkjashdlfkjhasdlkfhal?Vp[Bgu&lt;wM*zewR0.{+m:bmrB?wj!B,4]Wo+4 Avk ApR-D?E&#039;);
define(&#039;SECRET_KEY&#039;,     &#039;jflkhaskljdfhkljasdhflkjashd;flkjhas;djfh;kajshdflkjashdlfkjhasdlkfhal?Vp[B52ugH6muE9r4._iZwoYKUybrqLPpv|d Xr+|yrqhUE&#039;);
&nbsp;
define(&#039;AUTH_SALT&#039;,        &#039;123423190847olqkfhladhfsldshafasdfasdf09a7f-90a87df98adfyapoiyaf9asd8f70a9s8d7f908a7sdf97W4qCdm~Ky%+%~PPa5b YEmDI%U[W!-B&#039;);
define(&#039;SECURE_AUTH_SALT&#039;, &#039;123423190847olqkfhladhfsldshafasdfasdf09a7f-90a87df98adfyapoiyaf9asd8f70a9s8d7f908a7sdf97W4qCdmad/7o6.AU3%9o-|Kqm]+eUqr-n~:ag&#039;);
define(&#039;LOGGED_IN_SALT&#039;,   &#039;123423190847olqkfhladhfsldshafasdfasdf09a7f-90a87df98adfyapoiyaf9asd8f70a9s8d7f908a7sdf97W4qCdmsLiCv@KJ{#wd(?qe(KcH3!&#039;);
define(&#039;NONCE_SALT&#039;,       &#039;123423190847olqkfhladhfsldshafasdfasdf09a7f-90a87df98adfyapoiyaf9asd8f70a9s8d7f908a7sdf97W4qCdmG9&gt;+wm 2)bS0Pd_+1rx0brX]ND8|&#039;);
define(&#039;SECRET_SALT&#039;,      &#039;123423190847olqkfhladhfsldshafasdfasdf09a7f-90a87df98adfyapoiyaf9asd8f70a9s8d7f908a7sdf97W4qCdm2&lt;&gt;))U|sty)+4vpWooKls/^[vN&#039;);
/**#@-*/
</pre>
<hr class="C" />
<h2>Using SSL for Admin and Login</h2>
<p>SSL is kinda required from my point of view, it is just way to easy to sniff data off the wire otherwise.  At least with SSL you force them to use tools like burpsuite, paros proxy, webscarab, etc..</p>
<pre>
/** @since 2.6.0  */
!defined(&#039;FORCE_SSL_ADMIN&#039;) &#038;&#038; define(&#039;FORCE_SSL_ADMIN&#039;, true);
&nbsp;
/** @since 2.6.0  */
!defined(&#039;FORCE_SSL_LOGIN&#039;) &#038;&#038; define(&#039;FORCE_SSL_LOGIN&#039;, true);
</pre>
<h3>Mod_Rewrite to Force SSL</h3>
<p>This is pretty cool, it forces non-https for all urls except for /wp-admin and wp-login.php, which both require https.  It also checks for the logged_in_cookie, and if that is present in the request then it doesn't force non-https.  Kinda confusing if you don't have a <a href="http://www.askapache.com/htaccess/mod_rewrite-variables-cheatsheet.html">mod_rewrite cheatsheet</a>.</p>
<pre>
RewriteCond %{THE_REQUEST} ^$ [OR]
RewriteCond %{REQUEST_URI} ^/(wp-admin|wp-login\.php).*$ [NC,OR]
RewriteCond %{HTTP_COOKIE} ^.*wp_li_sadfsdfasdf11b361cdsdfasdfasd=.*$ [NC]
RewriteRule .* - [S=1]
&nbsp;
RewriteCond %{HTTPS} =on [OR]
RewriteCond %{HTTP_HOST} !^www\.askapache\.com$ [NC]
RewriteRule .* http://%{SERVER_NAME}%{REQUEST_URI} [R=301,L]
&nbsp;
RewriteCond %{THE_REQUEST} ^[A-Z]{3,9}\ /(wp-admin/.*|wp-login\.php.*)\ HTTP/ [NC]
RewriteCond %{HTTPS} !=on
RewriteRule .* https://%{SERVER_NAME}%{REQUEST_URI} [R=301,L]
</pre>
<hr class="C" />
<h2>File System Permissions</h2>
<p><a class="IFR" href="http://www.askapache.com/security/chmod-umask-fileperms-stat-tricks.html"><img src="/wp-content/uploads/2008/11/danger-chmod-screenshot.png" alt="chmod, umask, file permissions test" title="chmod, umask, file permissions test" /></a>You can get a basic and solid intro on file permissions by reading: <a href="http://codex.wordpress.org/Changing_File_Permissions">Changing File Permissions</a>, or you can check out some of my <a href="http://www.askapache.com/security/chmod-umask-fileperms-stat-tricks.html">file permission research</a>.<br class="C" /></p>
<pre>
/** The permissions as octal number, usually 0644 for files, 0755 for dirs.
 *  http://codex.wordpress.org/Changing_File_Permissions
 *  if ( !$wp_filesystem-&gt;mkdir($remote_destination, FS_CHMOD_DIR) )
 */
!defined(&#039;FS_CHMOD_DIR&#039;) &#038;&#038; define(&#039;FS_CHMOD_DIR&#039;, (0755 &#038; ~ umask()));
!defined(&#039;FS_CHMOD_FILE&#039;) &#038;&#038; define(&#039;FS_CHMOD_FILE&#039;, (0644 &#038; ~ umask()));
/**#@-*/
&nbsp;
/** Define the timeouts for the connections. Only available after the construct is called to allow for per-transport overriding of the default. */
//stream_set_timeout( $stream, FS_TIMEOUT );
//!defined(&#039;FS_TIMEOUT&#039;) &#038;&#038; define(&#039;FS_TIMEOUT&#039;, 30);
&nbsp;
//$this-&gt;link = @ftp_connect($this-&gt;options[&#039;hostname&#039;], $this-&gt;options[&#039;port&#039;], FS_CONNECT_TIMEOUT);
//!defined(&#039;FS_CONNECT_TIMEOUT&#039;) &#038;&#038; define(&#039;FS_CONNECT_TIMEOUT&#039;, 30);
&nbsp;
// function get_filesystem_method($args = array(), $context = false) {
//  $method = defined(&#039;FS_METHOD&#039;) ? FS_METHOD : false; //Please ensure that this is either &#039;direct&#039;, &#039;ssh&#039;, &#039;ftpext&#039; or &#039;ftpsockets&#039;
//!defined(&#039;FS_METHOD&#039;) &#038;&#038; define(&#039;FS_METHOD&#039;, &#039;direct&#039;);
&nbsp;
/** These methods for the WordPress core, plugin, and theme upgrades try to determine the WordPress path, as reported by PHP, but symlink trickery can sometimes
 * &#039;muck this up&#039; so if you know the paths to the various folders on the server, as seen via your FTP user, you can manually define them in the wp-config.php file.
 * FS_METHOD forces the filesystem method. It should only be "direct", "ssh", "ftpext", or "ftpsockets".
 * FTP_BASE is the full path to the "base" folder of the WordPress installation.
 * FTP_CONTENT_DIR is the full path to the wp-content folder of the WordPress installation.
 * FTP_PLUGIN_DIR is the full path to the plugins folder of the WordPress installation.
 * FTP_PUBKEY is the full path to your SSH public key.
 * FTP_PRIKEY is the full path to your SSH private key.
 * FTP_USER is either user FTP or SSH username. Most likely these are the same, but use the appropriate one for the type of update you wish to do.
 * FTP_PASS is the password for the username entered for FTP_USER. If you are using SSH public key authentication this can be omitted.
 * FTP_HOST is the hostname:port combination for your SSH/FTP server. The standard FTP port is 21 and the standard SSH port is 22.
 */
//define(&#039;FS_METHOD&#039;, &#039;ftpext&#039;);
//define(&#039;FTP_BASE&#039;, &#039;/path/to/wordpress/&#039;);
//define(&#039;FTP_CONTENT_DIR&#039;, &#039;/path/to/wordpress/wp-content/&#039;);
//define(&#039;FTP_PLUGIN_DIR &#039;, &#039;/path/to/wordpress/wp-content/plugins/&#039;);
//define(&#039;FTP_PUBKEY&#039;, &#039;/home/username/.ssh/id_rsa.pub&#039;);
//define(&#039;FTP_PRIKEY&#039;, &#039;/home/username/.ssh/id_rsa&#039;);
//define(&#039;FTP_USER&#039;, &#039;username&#039;);
//define(&#039;FTP_PASS&#039;, &#039;password&#039;);
//define(&#039;FTP_HOST&#039;, &#039;ftp.example.org:21&#039;);
&nbsp;
/**
 * Block requests through the proxy.
 *
 * Those who are behind a proxy and want to prevent access to certain hosts may do so. This will
 * prevent plugins from working and core functionality, if you don&#039;t include api.wordpress.org.
 *
 * You block external URL requests by defining WP_HTTP_BLOCK_EXTERNAL in your wp-config.php file
 * and this will only allow localhost and your blog to make requests.
 * The constant WP_ACCESSIBLE_HOSTS will allow additional hosts to go through for requests. The format of the
 * WP_ACCESSIBLE_HOSTS constant is a comma separated list of hostnames to allow.
 *
 * @since 2.8.0
 * @link http://core.trac.wordpress.org/ticket/8927 Allow preventing external requests.
/** @since 2.9  */
//!defined(&#039;WP_HTTP_BLOCK_EXTERNAL&#039;) &#038;&#038; define( &#039;WP_HTTP_BLOCK_EXTERNAL&#039;, false );
&nbsp;
/*
 * The constant WP_ACCESSIBLE_HOSTS will allow additional hosts to go through for requests. The format of the
 * WP_ACCESSIBLE_HOSTS constant is a comma separated list of hostnames to allow.
 *
 * @since 2.8.0
 * @link http://core.trac.wordpress.org/ticket/8927 Allow preventing external requests.
 * $accessible_hosts = preg_split(&#039;|,\s*|&#039;, WP_ACCESSIBLE_HOSTS);
 * return !in_array( $check[&#039;host&#039;], $accessible_hosts ); //Inverse logic, If its in the array, then we can&#039;t access it.
 */
//!defined(&#039;WP_ACCESSIBLE_HOSTS&#039;) &#038;&#038; define( &#039;WP_ACCESSIBLE_HOSTS&#039;, &#039;askapache.com,askapache.org&#039; );
</pre>
<hr class="C" />
<h3>Cookies!</h3>
<p>There&#8217;s always a little comfort in having non-default cookies for security (against auto-bots), and using shorter names also means smaller HTTP Packets.</p>
<p>The <code>$cookie_hash</code> is my hack to get around the fact that <code>COOKIEHASH</code> isn&#8217;t definable in <code>wp-config</code>.</p>
<pre>
/**#@+
 * COOKIES
 * Used to guarantee unique hash cookies @since 1.5 */
$cookie_hash=md5(WP_SITEURL);
&nbsp;
/** Set a cookie now to see if they are supported by the browser.
 * setcookie(TEST_COOKIE, &#039;WP Cookie check&#039;, 0, COOKIEPATH, COOKIE_DOMAIN);
 * @since 2.3.0 */
!defined(&#039;TEST_COOKIE&#039;) &#038;&#038; define(&#039;TEST_COOKIE&#039;, &#039;wp_tc&#039;);
&nbsp;
/* @since 2.6.0 */
!defined(&#039;LOGGED_IN_COOKIE&#039;) &#038;&#038; define(&#039;LOGGED_IN_COOKIE&#039;, &#039;wp_li_&#039; . $cookie_hash);
&nbsp;
/* @since 2.6.0 */
!defined(&#039;SECURE_AUTH_COOKIE&#039;) &#038;&#038; define(&#039;SECURE_AUTH_COOKIE&#039;, &#039;wp_sa_&#039; . $cookie_hash);
&nbsp;
/* @since 2.5.0 */
!defined(&#039;AUTH_COOKIE&#039;) &#038;&#038; define(&#039;AUTH_COOKIE&#039;, &#039;wp_a_&#039; . $cookie_hash);
&nbsp;
/* @since 2.0.0 */
!defined(&#039;PASS_COOKIE&#039;) &#038;&#038; define(&#039;PASS_COOKIE&#039;, &#039;wp_p_&#039; . $cookie_hash);
&nbsp;
/* @since 2.0.0 */
!defined(&#039;USER_COOKIE&#039;) &#038;&#038; define(&#039;USER_COOKIE&#039;, &#039;wp_u_&#039; . $cookie_hash);
&nbsp;
/* ok unset this var, its not needed as COOKIEHASH will have this value, but is not definable in wp-config.php */
unset($cookie_hash);
&nbsp;
/** @since 1.2.0 */
!defined(&#039;COOKIEPATH&#039;) &#038;&#038; define(&#039;COOKIEPATH&#039;, preg_replace(&#039;|https?://[^/]+|i&#039;, &#039;&#039;, WP_HOME . &#039;/&#039; ) );
&nbsp;
/** @since 1.5.0 */
!defined(&#039;SITECOOKIEPATH&#039;) &#038;&#038; define(&#039;SITECOOKIEPATH&#039;, preg_replace(&#039;|https?://[^/]+|i&#039;, &#039;&#039;, WP_SITEURL . &#039;/&#039; ) );
&nbsp;
/** @since 2.6.0 */
!defined(&#039;ADMIN_COOKIE_PATH&#039;) &#038;&#038; define( &#039;ADMIN_COOKIE_PATH&#039;, SITECOOKIEPATH . &#039;wp-admin&#039; );
&nbsp;
/** @since 2.6.0 */
!defined(&#039;PLUGINS_COOKIE_PATH&#039;) &#038;&#038; define( &#039;PLUGINS_COOKIE_PATH&#039;, preg_replace(&#039;|https?://[^/]+|i&#039;, &#039;&#039;, WP_PLUGIN_URL)  );
&nbsp;
/** @since 2.0.0 */
!defined(&#039;COOKIE_DOMAIN&#039;) &#038;&#038; define(&#039;COOKIE_DOMAIN&#039;, $_SERVER[&#039;SERVER_NAME&#039;]);
</pre>
<hr class="C" />
<pre>
/**
  * The WP_CACHE setting, if true, includes the wp-content/advanced-cache.php script, when executing wp-settings.php.
  * For an advanced caching plugin to use, static because you would only want one
  * if ( defined(&#039;WP_CACHE&#039;) )@include WP_CONTENT_DIR . &#039;/advanced-cache.php&#039;;
  */
!defined(&#039;WP_CACHE&#039;) &#038;&#038; define(&#039;WP_CACHE&#039;, true);
&nbsp;
/** WordPress Localized Language, defaults to en_US.
 *
 * Change this to localize WordPress.  A corresponding MO file for the chosen
 * language must be installed to wp-content/languages. For example, install
 * de.mo to wp-content/languages and set WPLANG to &#039;de&#039; to enable German
 * language support. */
!defined(&#039;WPLANG&#039;) &#038;&#038; define (&#039;WPLANG&#039;, &#039;en_US&#039;);
&nbsp;
/** Stores the location of the language directory. First looks for language folder in WP_CONTENT_DIR
 *   and uses that folder if it exists. Or it uses the "languages" folder in WPINC. @since 2.1.0 */
//!defined(&#039;WP_LANG_DIR&#039;) &#038;&#038; define(&#039;WP_LANG_DIR&#039;, ABSPATH . WPINC . &#039;/languages&#039;);
&nbsp;
/** LANGDIR defines what directory the WPLANG .mo file resides. If LANGDIR is not defined WordPress looks first to wp-content/languages and then wp-includes/languages for the .mo defined by WPLANG file.  Old static relative path maintained for limited backwards compatibility - won&#039;t work in some cases*/
//!defined(&#039;LANGDIR&#039;) &#038;&#038; define(&#039;LANGDIR&#039;, &#039;wp-content/languages&#039;);
&nbsp;
/** Stores the location of the WordPress directory of functions, classes, and core content. @since 1.0.0 */
//!defined(&#039;WPINC&#039;) &#038;&#038; define(&#039;WPINC&#039;, &#039;wp-includes&#039;);
</pre>
<hr class="C" />
<h2>WPMU Stuff</h2>
<p>I personally don&#8217;t use.</p>
<pre>
/** Allows for the mu-plugins directory to be moved from the default location. @since 2.8.0 */
//!defined(&#039;WPMU_PLUGIN_DIR&#039;) &#038;&#038; define( &#039;WPMU_PLUGIN_DIR&#039;, WP_CONTENT_DIR . &#039;/mu-plugins&#039; ); // full path, no trailing slash
&nbsp;
/** Allows for the mu-plugins directory to be moved from the default location. @since 2.8.0 */
//!defined(&#039;WPMU_PLUGIN_URL&#039;) &#038;&#038; define( &#039;WPMU_PLUGIN_URL&#039;, WP_CONTENT_URL . &#039;/mu-plugins&#039; ); // full url, no trailing slash
&nbsp;
/** Allows for the mu-plugins directory to be moved from the default location. @since 2.8.0 */
//!defined( &#039;MUPLUGINDIR&#039; ) &#038;&#038; define( &#039;MUPLUGINDIR&#039;, &#039;wp-content/mu-plugins&#039; ); // Relative to ABSPATH.  For back compat.
</pre>
<hr class="C" />
<h2>WordPress Database</h2>
<p>This is usually the only thing I have to manually edit when creating a new site, unless I just use the same DB and modify the $table_prefix, (farther down). I run everything I possibly can in UTF-8, but if you don&#8217;t already know alot about character sets, wow it is one of the most confusing things so you may want to save learning about that topic for another day.  Otherwise the following are helpful (<em>and show how confusing character sets are!</em>)</p>
<ul>
<li><a href="http://dev.mysql.com/doc/refman/5.0/en/charset-charsets.html">Character Sets and Collations MySQL Support</a></li>
<li><a href="http://codex.wordpress.org/Converting_Database_Character_Sets">Converting Database Character Sets</a></li>
<li><a href="http://dev.mysql.com/doc/refman/5.0/en/charset-unicode-sets.html">UTF-8 character sets</a> (<a href="http://en.wikipedia.org/wiki/UTF-8">UTF-8</a>)</li>
</ul>
<p>If you ever setup WP to use the builtin membership features, make sure you learn about the <code>CUSTOM_USER_TABLE</code> and <code>CUSTOM_USER_META_TABLE</code> constants, I&#8217;ve found them very helpful.</p>
<pre>
/**#@+
 * MySQL settings
 */
/** The name of the database for WordPress */
define(&#039;DB_NAME&#039;, &#039;askapachewpblog75&#039;);
&nbsp;
/** The username to access the database */
define(&#039;DB_USER&#039;, &#039;askapache245d&#039;);
&nbsp;
/** The password for the username to access the database */
define(&#039;DB_PASSWORD&#039;, &#039;asdfklj2340&#039;);
&nbsp;
/** The hostname to connect to the database at */
define(&#039;DB_HOST&#039;, &#039;mysql.askapache.com&#039;);
&nbsp;
/** The charset of the database */
define(&#039;DB_CHARSET&#039;, &#039;utf8&#039;);
&nbsp;
/** The collation of the database */
define(&#039;DB_COLLATE&#039;, &#039;utf8_general_ci&#039;);
</pre>
<hr class="C" />
<h2>$table_prefix</h2>
<p>The <code>$table_prefix</code> is the value placed in the front of your database tables. Change the value if you want to use something other than wp_ for your database prefix. Typically this is changed if you are <a href="http://codex.wordpress.org/Installing_Multiple_Blogs">installing multiple WordPress blogs</a> in the same database, and also for enhanced security.</p>
<p>Its a safe and good idea to change this value pre-installation to add more security to your WordPress blog. Exploits attempted against your WordPress blog by malicious crackers often are built with the premise that your blog uses the prefix wp_, by changing the value you mitigate some attack vectors. </p>
<pre>
/**
 * WordPress Database Table prefix.
 *
 * You can have multiple installations in one database if you give each a unique
 * prefix. Only numbers, letters, and underscores please!
 */
$table_prefix  = &#039;ar15_&#039;;
&nbsp;
/** CUSTOM_USER_TABLE and CUSTOM_USER_META_TABLE are used to designated that the user and usermeta tables normally utilized by WordPress are not used, instead these values/tables are used to store your user information. */
//!defined(&#039;CUSTOM_USER_TABLE&#039;) &#038;&#038; define(&#039;CUSTOM_USER_TABLE&#039;, $table_prefix . &#039;my_users&#039;);
//!defined(&#039;CUSTOM_USER_META_TABLE&#039;) &#038;&#038; define(&#039;CUSTOM_USER_META_TABLE&#039;, $table_prefix . &#039;my_usermeta&#039;);
</pre>
<h2>Setup PHP Ini Settings</h2>
<pre>
&nbsp;
/** Turns the output of errors on or off, you really never want this on, you should only view errors by reading the log file. */
ini_set(&#039;display_errors&#039;, WP_DEBUG_DISPLAY);
&nbsp;
/** Tells whether script error messages should be logged to the server&#039;s error log or error_log. */
ini_set(&#039;log_errors&#039;, &#039;On&#039;);
&nbsp;
/** http://us.php.net/manual/en/timezones.php */
ini_set(&#039;date.timezone&#039;, &#039;America/Indianapolis&#039;);
&nbsp;
/** Where to log php errors */
ini_set(&#039;error_log&#039;, ASKAPACHE_ROOT . &#039;/logs/php_error.log&#039;);
&nbsp;
/** Set the memory limit, otherwise defaults to &#039;32M&#039; */
ini_set(&#039;memory_limit&#039;, WP_MEMORY_LIMIT);
</pre>
<h2>Sessions are slow</h2>
<p>So I only use sessions when I have a specific use&#8230; In this case I need sessions only when one of the tools in the /online-tools/ directory is being used.  And that is for the <a href="http://www.askapache.com/security/php-captcha-anti-spam-example.html">captcha image</a>.  In the future I won&#8217;t ever use sessions.</p>
<pre>
if(preg_match( &#039;#^/online-tools/#&#039;,$_SERVER[&#039;REQUEST_URI&#039;])) session_start();
</pre>
<h2>Include Custom Files</h2>
<p>Sure you could use the my-hacks.php that WP allows, or you can just stick your functions in your <code>TEMPLATEPATH/functions.php</code> file, but they are executed only after the wp-settings.php file, which may be too late for your file.</p>
<p>In the past I&#8217;ve also used the <a href="http://us2.php.net/manual/en/ini.core.php#ini.auto-prepend-file">auto_prepend_file</a> settings to run my script before anything (index.php) but I ran into some issues on different hosts, and it wasn&#8217;t as portable.</p>
<p>This is useful because you can have a file with globally available functions that you can use in non-WP areas as well as WP areas.  I am moving away from this more and more as I learn more about classes and build plugins instead for portability.</p>
<pre>
include_once ASKAPACHE_ROOT . &#039;/includes/myfunctions.inc&#039;;
&nbsp;
/** Sets up WordPress vars and included files. */
require_once(ABSPATH . &#039;wp-settings.php&#039;);
?&gt;
</pre>
<h2>Some Useful PHP</h2>
<p>I am constantly trying to make my sites and code more portable, so I am using plugins alot more to accomplish things that I use to do with separate php.  Here are some examples of minimal php.</p>
<pre>
add_filter("the_generator", create_function(&#039;$a&#039;,&#039;return "";&#039;));
add_filter(&#039;the_content&#039;, create_function(&#039;$a&#039;, &#039;return ((is_feed())? $a."&lt;p&gt;&lt;a href=\"".get_permalink()."\"&gt;".get_the_title()."&lt;/a&gt; originally appeared on ".get_bloginfo("name").".&lt;/p&gt;" : $a);&#039;), 99999);
add_filter(&#039;excerpt_length&#039;, create_function(&#039;$a&#039;, &#039;return 300;&#039;),99);
add_filter(&#039;excerpt_more&#039;, create_function(&#039;$a&#039;, &#039;return "&hellip;";&#039;),99);
add_action( &#039;wp_head&#039;, create_function(&#039;$a&#039;,&#039;echo "&lt;link rel=\"pingback\" href=\"&#039;.get_bloginfo(&#039;pingback_url&#039;).&#039;\" /&gt;\n";&#039;), 95 );
add_action( &#039;wp_head&#039;, create_function(&#039;$a&#039;,&#039;echo "&lt;link rel=\"schema.rss\" href=\"http://purl.org/rss/1.0/\" /&gt;\n";&#039;), 96 );
add_action( &#039;wp_head&#039;, create_function(&#039;$a&#039;,&#039;echo "&lt;link rel=\"schema.rel\" href=\"http://purl.org/vocab/relationship/\" /&gt;\n";&#039;), 97 );
add_action( &#039;wp_head&#039;, create_function(&#039;$a&#039;,&#039;echo "&lt;link rel=\"meta\" type=\"application/rdf+xml\" href=\"/foaf.rdf\" /&gt;\n";&#039;), 98 );
add_action( &#039;wp_head&#039;, create_function(&#039;$a&#039;,&#039;echo "&lt;link href=\"/favicon.ico\" rel=\"shortcut icon\" type=\"image/x-icon\" /&gt;\n";&#039;), 99 );
</pre>
<h2>Debugging Note</h2>
<p><a href="http://wordpress.org/extend/plugins/askapache-debug-viewer/screenshots/"><img alt="AskApache Advanced Debugging Output" src="http://s.wordpress.org/extend/plugins/askapache-debug-viewer/screenshot-1.png?r=160129" title="AskApache Advanced Debugging Output" width="625" height="548" /></a>If you read this far than you probably know how important debugging is, but I sometimes like to stick the best tips deep in my articles to make sure only YOU find it.  GRTFM isn&#8217;t used on this site, it&#8217;s mostly a requirement because my writing can get pretty bad..  The point, debugging is more than a crucial requirement if you want to do anything cool.  Don&#8217;t worry I got you.. check my <a href="http://wordpress.org/extend/plugins/askapache-debug-viewer/">AskApache Debug Viewer Plugin from the official WP site</a>.  It&#8217;s pretty close to providing as verbose amount of information that I could possibly figure out how to get out of php, probably more than you have ever seen at least, I focused on quantity.  I use it all the time on new installs as there is no setup required and it tells me advanced information about the setup of the server, hacker code for sure.<br class="C" /></p>
<p>Here&#8217;s a quick function to see set global vars, I just think this is interesting code.</p>
<pre>
function askapache_global_debug(){
  global $_GET,$_POST,$_COOKIE,$_SESSION,$_ENV,$_FILES,$_SERVER,$_REQUEST,$HTTP_POST_FILES,$HTTP_POST_VARS,$HTTP_SERVER_VARS,$HTTP_RAW_POST_DATA,$HTTP_GET_VARS,$HTTP_COOKIE_VARS,$HTTP_ENV_VARS;
  $gv=create_function(&#039;$n&#039;,&#039;global $$n; ob_start(); if ( is_array($$n) &#038;&#038; sizeof($$n)&gt;0 &#038;&#038; print("[{$n}]\n") ) print_r($$n);return ob_get_clean();&#039;);
  foreach (array(&#039;_GET&#039;,&#039;_POST&#039;,&#039;_COOKIE&#039;,&#039;_SESSION&#039;,&#039;_ENV&#039;,&#039;_FILES&#039;,&#039;_SERVER&#039;,&#039;_REQUEST&#039;,&#039;HTTP_POST_FILES&#039;,&#039;HTTP_POST_VARS&#039;,&#039;HTTP_SERVER_VARS&#039;,&#039;HTTP_RAW_POST_DATA&#039;,&#039;HTTP_GET_VARS&#039;,&#039;HTTP_COOKIE_VARS&#039;,&#039;HTTP_ENV_VARS&#039;) as $k)echo $gv($k);
  print_r(get_defined_constants());
}
</pre>
<p class="anote">Also check the WordPress Codex page: <a href="http://codex.wordpress.org/Editing_wp-config.php">Editing wp-config.php</a></p>
<p><a href="http://www.askapache.com/wordpress/advanced-wp-config-php-tweaks.html">Advanced WordPress wp-config.php Tweaks</a> originally appeared on AskApache.</p>]]></content:encoded>
			<wfw:commentRss>http://www.askapache.com/wordpress/advanced-wp-config-php-tweaks.html/feed/</wfw:commentRss>
		<slash:comments>4</slash:comments>
		</item>
		<item>
		<title>An AskApache Plugin Upgrade to Rule them All</title>
		<link>http://www.askapache.com/wordpress/an-askapache-plugin-upgrade-to-rule-them-all.html</link>
		<comments>http://www.askapache.com/wordpress/an-askapache-plugin-upgrade-to-rule-them-all.html#comments</comments>
		<pubDate>Wed, 29 Jul 2009 17:59:07 +0000</pubDate>
		<dc:creator>AskApache</dc:creator>
				<category><![CDATA[Apache]]></category>
		<category><![CDATA[Apache Modules]]></category>
		<category><![CDATA[Cache]]></category>
		<category><![CDATA[Featured]]></category>
		<category><![CDATA[Hacking]]></category>
		<category><![CDATA[Linux Unix BSD]]></category>
		<category><![CDATA[PHP]]></category>
		<category><![CDATA[Security]]></category>
		<category><![CDATA[Web Hosting]]></category>
		<category><![CDATA[Web Tools]]></category>
		<category><![CDATA[Webmaster]]></category>
		<category><![CDATA[Windows]]></category>
		<category><![CDATA[WordPress]]></category>
		<category><![CDATA[WordPress Plugins]]></category>
		<category><![CDATA[htaccess]]></category>
		<category><![CDATA[mod_rewrite]]></category>
		<category><![CDATA[AskApache Google 404]]></category>
		<category><![CDATA[AskApache Password Protection]]></category>
		<category><![CDATA[Fsockopen]]></category>
		<category><![CDATA[wordpress security]]></category>

		<guid isPermaLink="false">http://www.askapache.com/?p=3062</guid>
		<description><![CDATA[<p><a class="IFL" href="/wp-content/uploads/2009/07/apache-server-status.png"><img src="/wp-content/uploads/2009/07/apache-server-status-350x164.png" alt="apache-server-status" title="apache-server-status" width="350" height="164" class="alignnone size-medium wp-image-3070" /></a>So my blog as been rather quiet for almost a year now, and very few updates if any have been released for my Password Protection PLugin, my Google 404 Plugin, and definately not for my AskApache CrazyCache plugin, which I will be releasing last...  So for all of you who've helped me out by sending me suggestions and notifying me of errors and sticking with it...  Just wanted to <strong>say sorry about that, and thanks for all the great ideas.. </strong> Well, I've been sticking with it as well believe it our not.  I manage to get free days once in a while, and then its <strong>time to jam</strong>.<br class="C" /></p>]]></description>
			<content:encoded><![CDATA[<p><a class="IFL" href="/wp-content/uploads/2009/07/apache-server-status.png"><img src="/wp-content/uploads/2009/07/apache-server-status-350x164.png" alt="apache-server-status" title="apache-server-status" width="350" height="164" class="alignnone size-medium wp-image-3070" /></a>So my blog as been rather quiet for almost a year now, and very few updates if any have been released for my Password Protection PLugin, my Google 404 Plugin, and definately not for my AskApache CrazyCache plugin, which I will be releasing last&#8230;  So for all of you who&#8217;ve helped me out by sending me suggestions and notifying me of errors and sticking with it&#8230;  Just wanted to <strong>say sorry about that, and thanks for all the great ideas.. </strong> Well, I&#8217;ve been sticking with it as well believe it our not.  I manage to get free days once in a while, and then its <strong>time to jam</strong>.</p>
<p>I&#8217;ve used just about every CMS/Blog/Forum/Trac/Gallery/etc) and really didn&#8217;t like a lot of the way they coded&#8230;  I could use php but I didn&#8217;t KNOW php.. so I&#8217;ve had to learn php also, and it was tough to learn the advanced class usage and all the other language specific (but similar) constructs for php.  It was especially difficult (but fun and challenging) to program so as to be compatible with php4 and php5 (Such is WordPress).    But I kept at it, and soon you can decide for yourself what to make of it.</p>
<p>I can code in plenty of languages (bash, lua, windows .bat and vbs,  ocaml, big fan of all things shell) and can work my way through C and even sorta somewhat with assembly.  Assembly is the hardest, by far,  I&#8217;m into easy and powerful languages like Python, Javascript, perl, php, ruby, and CGI. I&#8217;ve used PHP for a long time to do various things,  but never to build software projects like this.  Once I noticed WordPress&#8217;s core .php files and the excellent programming I wanted to try and learn hot to do it.   The WordPress code is some of the best I&#8217;ve seen.  It appears the way they built it was planned, and not just dreamt up while typing that I can&#8217;t help but do.    Every time I read through the core code I learn a new trick or very nice way to do something.  Those guys are really good, and I think WordPress is going to dominate for a long long time.</p>
<h2>The Strategy</h2>
<p>The Password Protection (passpro) plugin has a lot of complex stuff going on, especially for a newbie to PHP and WordPress like me, so after refactoring the whole thing at least 5 times I decided to modify my approach, and wrote the AskApache Google 404 Plugin as a way to practice on a simpler piece of code, while at the same time providing a plugin of value.   Eventually I stopped thinking I could just code the whole thing in one sit-down with a stream-of-consciousness, and had to instead modularize the code and focus in on each part before moving to the next (I go without a plan because its fun, just not the most productive, but again, I&#8217;m not a programmer in the scientific sense.).</p>
<p>So I decided I had to really learn how WordPress Plugins work, filters, hooks, actions, and basically comfortability at reverse-engineering code, (Im a beginner for the last time), and so with the upcoming release of the AskApache Google 404 Plugin I have succeeded in making an incredibly stable plugin.  That way I only have to worry about what the aapasspro plugin is doing, instead of trying to fit it into a framework.  </p>
<h2>AskApache Google 404 Upgrade</h2>
<p>I think its rather unusual to develop a nice plugin like this 404 handler merely for the purpose of improving upon another plugin, but hey it worked.  As of <em>08/03/2009 14:06PM EST</em> I have about 1 hour left of finishing touches to release this upgrade.  But as you cantell by my badly edited posts, I don&#8217;t have a lot of time to myself.  An hour here and there is about it.  So it could be up to 2 weeks before I actually have the time to commit the release to the repo.  On a sidenote, have you checked out <a href="http://windows7news.com/" title="Windows 7 News">Windows 7 News</a>?  I&#8217;ve been contracted to do some technical work for them and thought they had an excellent site.</p>
<p><a href="/wp-content/uploads/2009/07/askapache-google-upgrade-ss1.png"><img src="/wp-content/uploads/2009/07/askapache-google-upgrade-ss1-344x350.png" alt="askapache-google-upgrade-ss1" title="askapache-google-upgrade-ss1" width="344" height="350" class="alignnone size-medium wp-image-3139" /></a></p>
<p><a href="/wp-content/uploads/2009/07/askapache-google-upgrade-ss2.png"><img src="/wp-content/uploads/2009/07/askapache-google-upgrade-ss2-293x350.png" alt="askapache-google-upgrade-ss2" title="askapache-google-upgrade-ss2" width="293" height="350" class="alignnone size-medium wp-image-3138" /></a></p>
<p>But keep in mind, the 404 PLugin is just where I practice for the passpro plugin, which truly does have features that no other software like it has ever had.  I understand the technology behind this plugin, and know it would really have a great impact on improving the Web (esp. WordPress) for all of us, I&#8217;ve just had to learn how to make it.</p>
<h2>AskApache Password Protection</h2>
<p>Probably still a couple weeks away, this plugin is the ultimate culmination of apache hackers dreams, at least those on shared servers (who may be interested in learning how to bypass security of said servers)..  So this is something I have much too fun with doing what I like to do.. network/protocol-level security.  I&#8217;ve examined the source code for many software packages that I use or have used to audit a server&#8217;s security, and this simple php plugin in most instances can enumerate with accuraccy most of the server&#8217;s setup in about a minute.  The catch (and the file permission problems I had to find a workaround too) is that this software is launched on the server, not remotely against the server.</p>
<p>Some of the software I examined was whiskers, nessus, nmap, hping, mozilla source, wireshark, ncftp, netcat, etc..  The closest comparison to the socket-level class I&#8217;ve hacked together to those is wireshark.  Except that wireshark only interprets (captures) the data passing over the wire, while this class does that and in fact sends and receives the data like netcat or nmap.  Its really more similar to metasploit, and can easily be used to send hex, binary, ascii, or any type of payload to the remote or local host.</p>
<h2>The Upgrades Begin</h2>
<p>Well I started working on them a long time ago.  Both the Password Protection plugin and the Google 404 plugin needed serious work.   And I finally have it all figured out.  Essentially I would work on one and finish an upgrade, but I just wasn&#8217;t happy with it and I wold start all over again, refactoring the code.   So as I put the finishing touches on those 2 plugins keep an eye out.  They are major upgrades.   I was able to meet all the goals I had for them, and came up with a lot of more improvements during the process.One of the main things I needed was a socket-level class to perform all kinds of checks and tests on.  I need this also for my crazy cache plugin, which my blog is currently using ,  and I have a 2 more really nice pplugins I use that also needed  access to a network class.  I wrote about what I was doing with fsockopen, and I&#8217;ve been improving on that example ever since.  I use this class to do some really powerful and exciting stuff, but you&#8217;ll see it soon enough.  As an indication of &#8216;getting it right&#8217; for the Password Protection plugin, the plugin will now work on Windows, Apache, IIS, Lighthttpd, and will even work running on a blackberry web server.  So now everyone using wordpress can at least get some security()</p>
<p class="enote">Many of the the other improvements focus on using the fsockopen class and .htaccess tricks to basically enumerate and discover all the different capabilities of your particular server;  That way you can learn about all the features and security that are possible for your specific server, and the securty modules wi8ll be geared for that as well.  FINALLY this plugin is going to be stable, and I just cant wait to see how people react when they learn all great capability their Apache-based Server has that they didn&#8217;t have a clue about.   Its amazing in that sense, and hackers will love theh way it works.. but your server admins will love it even more because its entirely 100% focused on helping you to set your site up (if you have Apache) to keep spammers out, to keep virii-serving robots and their log-hogging exploit requests and CPU/Mem robiing 404 errors off of your servers for real.  This will have a noticeable affect to whoever is running the server.   As you can tell.. I am pumped!</br></p>
<hr class="C" />
Apache is easy to configure and use, but only when you have root access.  Most people on shared and private hosting aren&#8217;t even able to view the main config file, let alone execute the Apache binaries to see what features are available and what configuration is being used.<br class="C" /></p>
<p>Apache can only be influenced by the main server configs and by .htaccess files.  Not by php, not by perl, and the main configs are almost never accessible to the masses.  But .htaccess files are.  And many hosting providers allow and enable .htaccess files, a configuration file for your web server.  The advanced features and capabilities of Apache were out of reach for most of us, it just wasn&#8217;t possible to enumerate or access, and most hosting providers are infamous for their lack of .htaccess (customer) support.  This plugin goes around those problems to give the power back to the people.<br class="C" /></p>
<p>y creating custom .htaccess files containing unpublished .htaccess tricks and techniques and combining that with the use of socket-level networking from WordPress (PHP) using <a href="http://www.askapache.com/php/fsockopen-socket.html">fsockopen</a>, we can effectively enumerate and discover an incredible amount of features and settings you will be able to control and use with this plugin.</p>
<p>Here are a few examples of the capabilities of this plugin, some of which I believe no other software can do..  <em>(Open source free to copy!)</em>.</p>
<ol>
<li>Current Version of Apache (<strong>Down to the API Version</strong>)</li>
<li>List of <strong>ALL Modules currently enabled</strong> by Apache (Such as Mod_Rewrite)</li>
<li>List of <strong>ALL Directives enabled by EACH enabled Module.</strong></li>
<li>Enumerate .htaccess Overrides, Context Permissions</li>
<li>Test for any builtin Handlers (like the <a href="/wp-content/uploads/2009/07/apache-server-status.png">status handler screenshot</a>)</li>
<li>Configure SSI (<a href="http://www.askapache.com/htaccess/advanced-htaccess-ssi.html#htaccess-ssi-security">http://www.askapache.com/htaccess/advanced-htaccess-ssi.html#htaccess-ssi-security</a>)</li>
</ol>
<blockquote cite="http://www.askapache.com/htaccess/password-protection-plugin-status.html"><div class="inote"><cite><a href="http://www.askapache.com/htaccess/password-protection-plugin-status.html"></a></cite>
<p><strong>March 1, 2009</strong><br /><strong>I would focus on the method that WordPress uses</strong>.  The code they have now (2.8 bleeding-edge) still isn&#8217;t where it needs to be, but this is some difficult stuff and <strong>they have a brilliant start, it&#8217;ll work.. just a question of when</strong>.</p>
<p><a class="IFL" href="/wp-content/uploads/2009/03/apache-security-model-tall1.png"><img src="/wp-content/uploads/2009/03/apache-security-model-tall1-250x123.png" alt="Apache Security Model - In Color" title="apache-security-model-wide" width="250" height="123" /></a><strong>The main issue</strong> with the password protection plugin working for some people and not others is due to <a title="detailed file permission article" href="http://www.askapache.com/security/chmod-stat.html">file permission configurations</a>.  The plugin attempts to write/modify files in your blog&#8217;s root directory.<br class="C" /></p>
</div>
</blockquote>
<hr class="C" />
<blockquote cite="http://www.askapache.com/htaccess/htaccess-plugin-blocks-spam-hackers-and-password-protects-blog.html"><div class="inote"><cite><a href="http://www.askapache.com/htaccess/htaccess-plugin-blocks-spam-hackers-and-password-protects-blog.html"></a></cite>
<p><strong>November 05, 2008</strong><br />To make a long story short, I downloaded each major release of the apache httpd source code starting at version 1.3.0 and finishing with version 2.2.11, I then compiled each version and built a HTTPD from source for all these apache versions.</p>
<div>
<div style="width:100px;overflow:hidden;float:left;">
<ul>
<li>1.3.0</li>
<li>1.3.1</li>
<li>1.3.11</li>
<li>1.3.12</li>
<li>1.3.14</li>
<li>1.3.17</li>
<li>1.3.19</li>
<li>1.3.2</li>
<li>1.3.20</li>
<li>1.3.22</li>
<li>1.3.23</li>
<li>1.3.24</li>
<li>1.3.27</li>
<li>1.3.28</li>
</ul>
</div>
<div style="width:100px;overflow:hidden;float:left;">
<ul>
<li>1.3.29</li>
<li>1.3.3</li>
<li>1.3.31</li>
<li>1.3.32</li>
<li>1.3.33</li>
<li>1.3.34</li>
<li>1.3.35</li>
<li>1.3.36</li>
<li>1.3.37</li>
<li>1.3.39</li>
<li>1.3.4</li>
<li>1.3.41</li>
<li>1.3.6</li>
<li>1.3.9</li>
</ul>
</div>
<div style="width:100px;overflow:hidden;float:left;">
<ul>
<li>2.0.35</li>
<li>2.0.36</li>
<li>2.0.39</li>
<li>2.0.40</li>
<li>2.0.42</li>
<li>2.0.43</li>
<li>2.0.44</li>
<li>2.0.45</li>
<li>2.0.46</li>
<li>2.0.47</li>
<li>2.0.48</li>
<li>2.0.49</li>
<li>2.0.50</li>
<li>2.0.51</li>
</ul>
</div>
<div style="width:150px;overflow:hidden;float:left;">
<ul>
<li>2.0.52</li>
<li>2.0.53</li>
<li>2.0.54</li>
<li>2.0.55</li>
<li>2.0.58</li>
<li>2.0.59</li>
<li>2.0.61</li>
<li>2.0.63</li>
<li>2.1.3-beta</li>
<li>2.1.6-alpha</li>
<li>2.1.7-beta</li>
<li>2.1.8-beta</li>
<li>2.1.9-beta</li>
</ul>
</div>
<div style="width:100px;overflow:hidden;float:left;">
<ul>
<li>2.2.0</li>
<li>2.2.10</li>
<li>2.2.2</li>
<li>2.2.3</li>
<li>2.2.4</li>
<li>2.2.6</li>
<li>2.2.8</li>
<li>2.2.9</li>
<li><strong>2.2.10</strong></li>
<li><strong>2.2.11</strong></li>
</ul>
</div>
<p><br class="C" /></div>
<p>Then I went through each version and determined the compatible modules for that version, and I&#8217;m pretty confident that I was also able to find each and every directive allowed by the compatible modules for that version (including core directives).  See <a href="http://www.askapache.com/htaccess/htaccess.html#htaccess-directives">.htaccess directive list</a>.  Basically I can now test a server using a variety of methods and determine almost 100% accurately what version of Apache (down to the API) is running, what modules (and versions) are enabled, and each and every directive that is allowed or disallowed for that version.  So this is so awesome because now we can enable all sorts of additional security features.</p>
</div>
</blockquote>
<hr class="C" />
<blockquote cite="http://www.askapache.com/htaccess/htaccess.html#htaccess-modules"><p><cite><a href="http://www.askapache.com/htaccess/htaccess.html#htaccess-modules">Htaccess enabled Modules</a></cite>
<p>Here are most of the modules that come with Apache.  Each one can have new commands that can be used in .htaccess file scopes.</p>
<p><a href="http://www.askapache.com/servers/mod_actions.c.html">mod_actions</a>, <a href="http://www.askapache.com/servers/mod_alias.c.html">mod_alias</a>, <a href="http://www.askapache.com/servers/mod_asis.c.html">mod_asis</a>, <a href="http://www.askapache.com/servers/mod_auth_basic.c.html">mod_auth_basic</a>, <a href="http://www.askapache.com/servers/mod_auth_digest.c.html">mod_auth_digest</a>, <a href="http://www.askapache.com/servers/mod_authn_anon.c.html">mod_authn_anon</a>, <a href="http://www.askapache.com/servers/mod_authn_dbd.c.html">mod_authn_dbd</a>, <a href="http://www.askapache.com/servers/mod_authn_dbm.c.html">mod_authn_dbm</a>, <a href="http://www.askapache.com/servers/mod_authn_default.c.html">mod_authn_default</a>, <a href="http://www.askapache.com/servers/mod_authn_file.c.html">mod_authn_file</a>, <a href="http://www.askapache.com/servers/mod_authz_dbm.c.html">mod_authz_dbm</a>, <a href="http://www.askapache.com/servers/mod_authz_default.c.html">mod_authz_default</a>, <a href="http://www.askapache.com/servers/mod_authz_groupfile.c.html">mod_authz_groupfile</a>, <a href="http://www.askapache.com/servers/mod_authz_host.c.html">mod_authz_host</a>, <a href="http://www.askapache.com/servers/mod_authz_owner.c.html">mod_authz_owner</a>, <a href="http://www.askapache.com/servers/mod_authz_user.c.html">mod_authz_user</a>, <a href="http://www.askapache.com/servers/mod_autoindex.c.html">mod_autoindex</a>, <a href="http://www.askapache.com/servers/mod_cache.c.html">mod_cache</a>, <a href="http://www.askapache.com/servers/mod_cern_meta.c.html">mod_cern_meta</a>, <a href="http://www.askapache.com/servers/mod_cgi.c.html">mod_cgi</a>, <a href="http://www.askapache.com/servers/mod_dav.c.html">mod_dav</a>, <a href="http://www.askapache.com/servers/mod_dav_fs.c.html">mod_dav_fs</a>, <a href="http://www.askapache.com/servers/mod_dbd.c.html">mod_dbd</a>, <a href="http://www.askapache.com/servers/mod_deflate.c.html">mod_deflate</a>, <a href="http://www.askapache.com/servers/mod_dir.c.html">mod_dir</a>, <a href="http://www.askapache.com/servers/mod_disk_cache.c.html">mod_disk_cache</a>, <a href="http://www.askapache.com/servers/mod_dumpio.c.html">mod_dumpio</a>, <a href="http://www.askapache.com/servers/mod_env.c.html">mod_env</a>, <a href="http://www.askapache.com/servers/mod_expires.c.html">mod_expires</a>, <a href="http://www.askapache.com/servers/mod_ext_filter.c.html">mod_ext_filter</a>, <a href="http://www.askapache.com/servers/mod_file_cache.c.html">mod_file_cache</a>, <a href="http://www.askapache.com/servers/mod_filter.c.html">mod_filter</a>, <a href="http://www.askapache.com/servers/mod_headers.c.html">mod_headers</a>, <a href="http://www.askapache.com/servers/mod_ident.c.html">mod_ident</a>, <a href="http://www.askapache.com/servers/mod_imagemap.c.html">mod_imagemap</a>, <a href="http://www.askapache.com/servers/mod_include.c.html">mod_include</a>, <a href="http://www.askapache.com/servers/mod_info.c.html">mod_info</a>, <a href="http://www.askapache.com/servers/mod_log_config.c.html">mod_log_config</a>, <a href="http://www.askapache.com/servers/mod_log_forensic.c.html">mod_log_forensic</a>, <a href="http://www.askapache.com/servers/mod_logio.c.html">mod_logio</a>, <a href="http://www.askapache.com/servers/mod_mem_cache.c.html">mod_mem_cache</a>, <a href="http://www.askapache.com/servers/mod_mime.c.html">mod_mime</a>, <a href="http://www.askapache.com/servers/mod_mime_magic.c.html">mod_mime_magic</a>, <a href="http://www.askapache.com/servers/mod_negotiation.c.html">mod_negotiation</a>, <a href="http://www.askapache.com/servers/mod_proxy.c.html">mod_proxy</a>, <a href="http://www.askapache.com/servers/mod_proxy_ajp.c.html">mod_proxy_ajp</a>, <a href="http://www.askapache.com/servers/mod_proxy_balancer.c.html">mod_proxy_balancer</a>, <a href="http://www.askapache.com/servers/mod_proxy_connect.c.html">mod_proxy_connect</a>, <a href="http://www.askapache.com/servers/mod_proxy_ftp.c.html">mod_proxy_ftp</a>, <a href="http://www.askapache.com/servers/mod_proxy_http.c.html">mod_proxy_http</a>, <a href="http://www.askapache.com/servers/mod_rewrite.c.html">mod_rewrite</a>, <a href="http://www.askapache.com/servers/mod_setenvif.c.html">mod_setenvif</a>, <a href="http://www.askapache.com/servers/mod_speling.c.html">mod_speling</a>, <a href="http://www.askapache.com/servers/mod_ssl.c.html">mod_ssl</a>, <a href="http://www.askapache.com/servers/mod_status.c.html">mod_status</a>, <a href="http://www.askapache.com/servers/mod_substitute.c.html">mod_substitute</a>, <a href="http://www.askapache.com/servers/mod_unique_id.c.html">mod_unique_id</a>, <a href="http://www.askapache.com/servers/mod_userdir.c.html">mod_userdir</a>, <a href="http://www.askapache.com/servers/mod_usertrack.c.html">mod_usertrack</a>, <a href="http://www.askapache.com/servers/mod_version.c.html">mod_version</a>, <a href="http://www.askapache.com/servers/mod_vhost_alias.c.html">mod_vhost_alias</a></p>
</blockquote>
<hr class="C" />
<hr class="C" />
<h2>Debugging HTTP protocol</h2>
<p>Check this out!  I&#8217;m particularly happy about this feature, which outputs an exact trace of any requests made by the plugin (such as during the testing phase) by saving the actual raw data sent out on the wire using fsockopen, RX and TX.  This is useful for a number of reasons, viewing your headers, finding Redirect Loops, testing RewriteRules, and following the request hop-by-hop for debugging.  The below example shows 2 requests for 2 URIs.  The first URI is protected using Digest Authentication, the 2nd shows Basic.</p>
<pre>
 ______________
|  RAW TRACE   |
==================================================================================================================================
GET /htaccess/index.txt?testing=query HTTP/1.1
Host: www.askapache.com
User-Agent: Mozilla/4.0 (compatible; MSIE 8.0; Windows NT 5.1) AA_PassPro/1.9 (http://www.askapache.com/)
Accept: text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8
Accept-Language: en-us
Accept-Encoding: gzip, deflate
Connection: close
Referer: http://www.askapache.com/
&nbsp;
HTTP/1.1 401 Authorization Required
Date: Wed, 22 Jul 2009 06:29:58 GMT
Server: Apache
WWW-Authenticate: Digest realm="do or die", nonce="03328f3ec7c7b", algorithm=MD5, domain="/", qop="auth"
Accept-Ranges: bytes
Vary: Accept-Encoding
Content-Encoding: gzip
Content-Length: 882
Connection: close
Content-Type: text/html; charset=UTF-8
&nbsp;
GET /htaccess/index.txt?testing=query HTTP/1.1
Host: www.askapache.com
User-Agent: Mozilla/4.0 (compatible; MSIE 8.0; Windows NT 5.1) AA_PassPro/1.9 (http://www.askapache.com/)
Accept: text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8
Accept-Language: en-us
Accept-Encoding: gzip, deflate
Connection: close
Referer: http://www.askapache.com/
Authorization: Digest username="test",realm="do or die",nonce="03328f3ec7c7b",uri="/htaccess/index.txt?testing=query",
cnonce="82d057852a9dc497",nc=00000001,algorithm=MD5,response="9d476e9ea3",qop="auth"
&nbsp;
HTTP/1.1 200 OK
Date: Wed, 22 Jul 2009 06:29:58 GMT
Server: Apache
Authentication-Info: rspauth="9051b01ee26dd62b3e2b40dada694f45", cnonce="82d057852a9dc497", nc=00000001, qop=auth
Last-Modified: Tue, 21 Jul 2009 23:56:00 GMT
Accept-Ranges: bytes
Cache-Control: max-age=3600
Expires: Wed, 22 Jul 2009 07:29:58 GMT
Vary: Accept-Encoding
Content-Encoding: gzip
Content-Length: 27
Connection: close
Content-Type: text/plain; charset=UTF-8
&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;``
&nbsp;
 ______________
|  RAW TRACE   |
==================================================================================================================================
GET /htaccess/po.txt?testing=query HTTP/1.1
Host: www.askapache.com
User-Agent: Mozilla/4.0 (compatible; MSIE 8.0; Windows NT 5.1) AA_PassPro/1.9 (http://www.askapache.com/)
Accept: text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8
Accept-Language: en-us
Accept-Encoding: gzip, deflate
Connection: close
Referer: http://www.askapache.com/
&nbsp;
HTTP/1.1 401 Authorization Required
Date: Wed, 22 Jul 2009 06:29:58 GMT
Server: Apache
WWW-Authenticate: Basic realm="Po Pimping"
Accept-Ranges: bytes
Vary: Accept-Encoding
Content-Encoding: gzip
Content-Length: 878
Connection: close
Content-Type: text/html; charset=UTF-8
&nbsp;
GET /htaccess/po.txt?testing=query HTTP/1.1
Host: www.askapache.com
User-Agent: Mozilla/4.0 (compatible; MSIE 8.0; Windows NT 5.1) AA_PassPro/1.9 (http://www.askapache.com/)
Accept: text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8
Accept-Language: en-us
Accept-Encoding: gzip, deflate
Connection: close
Referer: http://www.askapache.com/
Authorization: Basic adfAGAltcA==
&nbsp;
HTTP/1.1 200 OK
Date: Wed, 22 Jul 2009 06:29:58 GMT
Server: Apache
Last-Modified: Wed, 22 Jul 2009 05:54:39 GMT
Accept-Ranges: bytes
Cache-Control: max-age=3600
Expires: Wed, 22 Jul 2009 07:29:58 GMT
Vary: Accept-Encoding
Content-Encoding: gzip
Content-Length: 27
Connection: close
Content-Type: text/plain; charset=UTF-8
&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;`&#96;``
</pre>
<h2>.htaccess Directives</h2>
<p>AcceptFilter, AcceptMutex, AcceptPathInfo, AccessFileName, Action, AddAlt, AddAltByEncoding, AddAltByType, AddCharset, AddDefaultCharset, AddDescription, AddEncoding, AddHandler, AddIcon, AddIconByEncoding, AddIconByType, AddInputFilter, AddLanguage, AddModuleInfo, AddOutputFilter, AddOutputFilterByType, AddType, Alias, AliasMatch, AllowCONNECT, AllowEncodedSlashes, AllowOverride, Anonymous, Anonymous_Authoritative, Anonymous_LogEmail, Anonymous_MustGiveEmail, Anonymous_NoUserID, Anonymous_NoUserId, Anonymous_VerifyEmail, AssignUserId, AuthAuthoritative, AuthBasicAuthoritative, AuthBasicProvider, AuthDBDUserPWQuery, AuthDBDUserRealmQuery, AuthDBM, AuthDBMAuthoritative, AuthDBMGroupFile, AuthDBMType, AuthDBMUserFile, AuthDefaultAuthoritative, AuthDigestAlgorithm, AuthDigestDomain, AuthDigestFile, AuthDigestGroupFile, AuthDigestNcCheck, AuthDigestNonceFormat, AuthDigestNonceLifetime, AuthDigestProvider, AuthDigestQop, AuthDigestShmemSize, AuthGroupFile, AuthLDAPAuthzEnabled, AuthLDAPBindDN, AuthLDAPBindON, AuthLDAPBindPassword, AuthLDAPCharsetConfig, AuthLDAPCompareDNOnServer, AuthLDAPDereferenceAliases, AuthLDAPEnabled, AuthLDAPFrontPageHack, AuthLDAPGroupAttribute, AuthLDAPGroupAttributeIsDN, AuthLDAPRemoteUserAttribute, AuthLDAPRemoteUserIsDN, AuthLDAPStartTLS, AuthLDAPURL, AuthLDAPUrl, AuthName, AuthType, AuthUserFile, AuthzDBMAuthoritative, AuthzDBMType, AuthzDefaultAuthoritative, AuthzGroupFileAuthoritative, AuthzLDAPAuthoritative, AuthzOwnerAuthoritative, AuthzUserAuthoritative, BS2000Account, BalancerMember, BrowserMatch, BrowserMatchNoCase, BufferedLogs, CGIMapExtension, CacheDefaultExpire, CacheDirLength, CacheDirLevels, CacheDisable, CacheEnable, CacheExpiryCheck, CacheFile, CacheForceCompletion, CacheGcClean, CacheGcDaily, CacheGcInterval, CacheGcMemUsage, CacheGcUnused, CacheIgnoreCacheControl, CacheIgnoreHeaders, CacheIgnoreNoLastMod, CacheLastModifiedFactor, CacheMaxExpire, CacheMaxFileSize, CacheMaxStreamingBuffer, CacheMinFileSize, CacheNegotiatedDocs, CacheRoot, CacheSize, CacheStoreNoStore, CacheStorePrivate, CacheTimeMargin, CharsetDefault, CharsetOptions, CharsetSourceEnc, CheckCaseOnly, CheckSpelling, ChildPerUserId, ContentDigest, CookieDomain, CookieExpires, CookieLog, CookieName, CookieStyle, CookieTracking, CoreDumpDirectory, CustomLog, DAV, DAVDepthInfinity, DAVGenericLockDB, DAVMinTimeout, DBDExptime, DBDKeep, DBDMax, DBDMin, DBDParams, DBDPersist, DBDPrepareSQL, DBDriver, Dav, DavDepthInfinity, DavGenericLockDB, DavLockDB, DavMinTimeout, DefaultIcon, DefaultLanguage, DefaultType, DeflateBufferSize, DeflateCompressionLevel, DeflateFilterNote, DeflateMemLevel, DeflateWindowSize, Directory, DirectoryIndex, DirectoryMatch, DirectorySlash, DocumentRoot, DumpIOInput, DumpIOOutput, EnableExceptionHook, EnableMMAP, EnableSendfile, ErrorDocument, ErrorLog, Example, ExpiresActive, ExpiresByType, ExpiresDefault, ExtFilterDefine, ExtFilterOptions, ExtendedStatus, FancyIndexing, FileETag, Files, FilesMatch, FilterChain, FilterDeclare, FilterProtocol, FilterProvider, FilterTrace, ForceLanguagePriority, ForceType, ForensicLog, GprofDir, GracefulShutdownTimeout, Group, Header, HeaderName, HostNameLookups, HostnameLookups, ISAIPFakeAsync, ISAPIAppendLogToErrors, ISAPIAppendLogToQuery, ISAPICacheFile, ISAPIFakeAsync, ISAPILogNotSupported, ISAPIReadAheadBuffer, IdentityCheck, IdentityCheckTimeout, IfDefine, IfModule, IfVersion, ImapBase, ImapDefault, ImapMenu, Include, IndexIgnore, IndexOptions, IndexOrderDefault, IndexStyleSheet, KeepAlive, KeepAliveTimeout, LDAPCacheEntries, LDAPCacheTTL, LDAPCertDBPath, LDAPConnectionTimeout, LDAPOpCacheEntries, LDAPOpCacheTTL, LDAPSharedCacheFile, LDAPSharedCacheSize, LDAPTrustedClientCert, LDAPTrustedGlobalCert, LDAPTrustedMode, LDAPVerifyServerCert, LanguagePriority, Limit, LimitExcept, LimitInternalRecursion, LimitRequestBody, LimitRequestFields, LimitRequestFieldsize, LimitRequestLine, LimitXMLRequestBody, Listen, ListenBacklog, LoadFile, LoadModule, Location, LocationMatch, LockFile, LogFormat, LogLevel, MCacheMaxObjectCount, MCacheMaxObjectSize, MCacheMaxStreamingBuffer, MCacheMinObjectSize, MCacheRemovalAlgorithm, MCacheSize, MMapFile, MaxClients, MaxKeepAliveRequests, MaxMemFree, MaxRequestsPerChild, MaxSpareServers, MaxSpareThreads, MaxSpareThreadsPerChild, MaxThreads, MetaDir, MetaFiles, MetaSuffix, MimeMagicFile, MinSpareServers, MinSpareThreads, ModMimeUsePathInfo, MultiviewsMatch, NWSSLTrustedCerts, NWSSLUpgradeable, NameVirtualHost, NoProxy, NumServers, Options, PassEnv, PerlAccessHandler, PerlAuthenHandler, PerlAuthzHandler, PerlChildExitHandler, PerlChildInitHandler, PerlCleanupHandler, PerlDispatchHandler, PerlFixupHandler, PerlFreshRestart, PerlHandler, PerlHeaderParserHandler, PerlInitHandler, PerlLogHandler, PerlModule, PerlPassEnv, PerlPostReadRequestHandler, PerlRequire, PerlRestartHandler, PerlSendHeader, PerlSetEnv, PerlSetVar, PerlSetupEnv, PerlTaintCheck, PerlTransHandler, PerlTypeHandler, PerlWarn, PidFile, Port, Protocol, ProtocolEcho, Proxy, ProxyBadHeader, ProxyBlock, ProxyDomain, ProxyErrorOverride, ProxyFtpDirCharset, ProxyIOBufferSize, ProxyMatch, ProxyMaxForwards, ProxyPass, ProxyPassInterpolateEnv, ProxyPassMatch, ProxyPassReverse, ProxyPassReverseCookieDomain, ProxyPassReverseCookiePath, ProxyPreserveHost, ProxyReceiveBufferSize, ProxyRemote, ProxyRemoteMatch, ProxyRequests, ProxySet, ProxyStatus, ProxyTimeout, ProxyVia, RLimitCPU, RLimitMEM, RLimitNPROC, ReadmeName, Redirect, RedirectMatch, RedirectPermanent, RedirectTemp, RemoveCharset, RemoveEncoding, RemoveHandler, RemoveInputFilter, RemoveLanguage, RemoveOutputFilter, RemoveType, RequestHeader, Require, RewriteBase, RewriteCond, RewriteEngine, RewriteLock, RewriteLog, RewriteLogLevel, RewriteMap, RewriteOptions, RewriteRule, SSIAccessEnable, SSIEndTag, SSIErrorMsg, SSIStartTag, SSITimeFormat, SSIUndefinedEcho, SSLCACertificateFile, SSLCACertificatePath, SSLCADNRequestFile, SSLCADNRequestPath, SSLCARevocationFile, SSLCARevocationPath, SSLCertificateChainFile, SSLCertificateFile, SSLCertificateKeyFile, SSLCipherSuite, SSLCryptoDevice, SSLEngine, SSLHonorCipherOrder, SSLLog, SSLLogLevel, SSLMutex, SSLOptions, SSLPassPhraseDialog, SSLProtocol, SSLProxyCACertificateFile, SSLProxyCACertificatePath, SSLProxyCARevocationFile, SSLProxyCARevocationPath, SSLProxyCipherSuite, SSLProxyEngine, SSLProxyMachineCertificateFile, SSLProxyMachineCertificatePath, SSLProxyProtocol, SSLProxyVerify, SSLProxyVerifyDepth, SSLRandomSeed, SSLRequire, SSLRequireSSL, SSLSessionCache, SSLSessionCacheTimeout, SSLUserName, SSLVerifyClient, SSLVerifyDepth, Satisfy, ScoreBoardFile, Script, ScriptAlias, ScriptAliasMatch, ScriptInterpreterSource, ScriptLog, ScriptLogBuffer, ScriptLogLength, ScriptStock, SecureListen, SendBufferSize, ServerAdmin, ServerAlias, ServerLimit, ServerName, ServerPath, ServerRoot, ServerSignature, ServerTokens, SetEnv, SetEnvIf, SetEnvIfNoCase, SetHandler, SetInputFilter, SetOutputFilter, StartServers, StartThreads, Substitute, SuexecUserGroup, ThreadLimit, ThreadStackSize, ThreadsPerChild, TimeOut, Timeout, TraceEnable, TransferLog, TypeAuthDBMUserFile, TypesConfig, UnsetEnv, UseCanonicalName, UseCanonicalPhysicalPort, User, UserDir, VirtualDocumentRoot, VirtualDocumentRootIP, VirtualHost, VirtualScriptAlias, VirtualScriptAliasIP, Win32DisableAcceptEx, XBitHack, allow, deny, order, php_admin_flag, php_admin_value, php_flag, php_value</p>
<p class="anote">You can view the <a href="http://www.askapache.com/htaccess/htaccess-security-block-spam-hackers.html">plugins home page</a>, <a href="http://www.askapache.com/wordpress/htaccess-password-protect.html#aadl">old</a>, or <a href="http://wordpress.org/extend/plugins/askapache-password-protect/">view it on the wordpress.org site</a>.</p>
<p><a href="http://www.askapache.com/wordpress/an-askapache-plugin-upgrade-to-rule-them-all.html">An AskApache Plugin Upgrade to Rule them All</a> originally appeared on AskApache.</p>]]></content:encoded>
			<wfw:commentRss>http://www.askapache.com/wordpress/an-askapache-plugin-upgrade-to-rule-them-all.html/feed/</wfw:commentRss>
		<slash:comments>1</slash:comments>
		</item>
		<item>
		<title>PHP and AJAX shell console</title>
		<link>http://www.askapache.com/ajax/php-and-ajax-shell-console.html</link>
		<comments>http://www.askapache.com/ajax/php-and-ajax-shell-console.html#comments</comments>
		<pubDate>Sun, 14 Jun 2009 01:01:15 +0000</pubDate>
		<dc:creator>AskApache</dc:creator>
				<category><![CDATA[Ajax]]></category>
		<category><![CDATA[DreamHost]]></category>
		<category><![CDATA[Featured]]></category>
		<category><![CDATA[Hacking]]></category>
		<category><![CDATA[Javascript]]></category>
		<category><![CDATA[Linux Unix BSD]]></category>
		<category><![CDATA[PHP]]></category>
		<category><![CDATA[Security]]></category>
		<category><![CDATA[Server Administration]]></category>
		<category><![CDATA[Shell Scripting]]></category>
		<category><![CDATA[Web Hosting]]></category>
		<category><![CDATA[Web Tools]]></category>
		<category><![CDATA[Webmaster]]></category>
		<category><![CDATA[htaccess]]></category>
		<category><![CDATA[mod_rewrite]]></category>
		<category><![CDATA[bash]]></category>
		<category><![CDATA[console]]></category>
		<category><![CDATA[linux]]></category>
		<category><![CDATA[Shell]]></category>
		<category><![CDATA[shell console]]></category>

		<guid isPermaLink="false">http://www.askapache.com/tools/php-and-ajax-shell-console.html</guid>
		<description><![CDATA[<p><a class="IFL" href='http://www.askapache.com/tools/php-and-ajax-shell-console.html' title='PHP AJAX shell console'><img src='/wp-content/uploads/2007/08/php-ajax-shell-console.thumbnail.png' alt='PHP AJAX shell console' /></a> Ever wanted to execute commands on your server through php? Now you can.  I'm calling this file (see below) shell.php and it allows you to run commands on your web server with the same permissions that your php executable has.<br class="C" /></p>]]></description>
			<content:encoded><![CDATA[<p>Ever wanted to execute commands on your server through php to mimick a shell login? <strong>Now you can.</strong>  I&#8217;m calling this file (see below) shell.php and it allows you to run commands on your web server with the same permissions that your php executable has.</p>
<p id="aaflash"><span class="v640"><span id="phpajaxshell"><span class="load">Loading Video</span><a rel="nofollow" class="getFlash" href="http://www.askapache.com/getflash/"></a></span></span></p>
<h2>PHP for <code>shell.php</code></h2>
<p>Substitue 1.1.1.1 for your IP address.. or see below for password authentication methods.</p>
<pre>
&lt;?php
 if ($_SERVER[&#039;REMOTE_ADDR&#039;] !== &#039;1.1.1.1&#039;) die();
 ob_start();
 if (!empty($_GET[&#039;cmd&#039;])){
 $ff=$_GET[&#039;cmd&#039;];
 #shell_exec($ff);
 system($ff);
 #exec($ff);
 #passthru($ff);
 }
 else {
?&gt;
&lt;!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd"&gt;
&lt;html&gt;
&lt;head&gt;
&lt;meta http-equiv="Content-Type" content="text/html; charset=utf-8"&gt;
&lt;title&gt;PHP AJAX Shell&lt;/title&gt;
&lt;script type="text/javascript" language="javascript"&gt;var CommHis=new Array();var HisP;
function doReq(_1,_2,_3){var HR=false;if(window.XMLHttpRequest){HR=new XMLHttpRequest();if(HR.overrideMimeType){HR.overrideMimeType("text/xml");}}
else{if(window.ActiveXObject){try{HR=new ActiveXObject("Msxml2.XMLHTTP");}catch(e){try{HR=new ActiveXObject("Microsoft.XMLHTTP");}
catch(e){}}}}if(!HR){return false;}HR.onreadystatechange=function(){if(HR.readyState==4){
if(HR.status==200){if(_3){eval(_2+"(HR.responseXML)");}else{eval(_2+"(HR.responseText)");}}}};HR.open("GET",_1,true);HR.send(null);}
function pR(rS){var _6=document.getElementById("outt");var _7=rS.split("\n\n");
var _8=document.getElementById("cmd").value;_6.appendChild(document.createTextNode(_8));
_6.appendChild(document.createElement("br"));for(var _9 in _7){var _a=document.createElement("pre");
_a.style.display="inline";line=document.createTextNode(_7[_9]);_a.appendChild(line);_6.appendChild(_a);
_6.appendChild(document.createElement("br"));}_6.appendChild(document.createTextNode(":-&gt; "));_6.scrollTop=_6.scrollHeight;
document.getElementById("cmd").value="";}function keyE(_b){switch(_b.keyCode){
case 13:var _c=document.getElementById("cmd").value;if(_c){CommHis[CommHis.length]=_c;HisP=CommHis.length;var _d=document.location.href+"?cmd="+escape(_c);
doReq(_d,"pR");}break;
case 38:if(HisP&gt;0){HisP&#45;-;document.getElementById("cmd").value=CommHis[HisP];}break;
case 40:if(HisP&lt;CommHis.length-1){HisP++;document.getElementById("cmd").value=CommHis[HisP];}break;default:break;}}
&lt;/script&gt;&lt;/head&gt;&lt;body style="font-family:courier"&gt;
&lt;form onsubmit="return false" style="color:#3F0;background:#000;position:relative;min-height:450px;max-height:490px"&gt;
&lt;div id="outt" style="overflow:auto;padding:5px;height:90%;min-height:450px;max-height:490px"&gt;:-&gt;&lt;/div&gt;
&lt;input tabindex="1" onkeyup="keyE(event)" style="color:#FFF;background:#333;width:100%;" id="cmd" type="text" /&gt;
&lt;/form&gt;
&lt;/body&gt;
&lt;/html&gt;
&lt;?php } ?&gt;
</pre>
<h2>Read this</h2>
<p><strong>Note</strong>:  The history feature works by remembering the last commands that you typed.. Access them by pressing the up or down arrows on your keyboard.</p>
<p>This is not an interactive session, so you cannot cd to a directory and then do stuff in that directory..  You may however be able to do stuff like <code>/bin/bash -c "cd ../../;mv this there;ls -la;"</code> or you could try exporting your current dir or something..</p>
<p>Writing shell scripts and serving them on your web server works by renaming the file.sh to file.cgi and chmodding it to 750 or +x.   Also make sure you try <code>dos2unix -dv file.cgi</code> If you can&#8217;t get it to work.. </p>
<h3>Example shell script as cgi</h3>
<pre>
#!/bin/sh
export MYBNAME=`date +%mx%dx%y-%Hx%M.tgz`
tar -czf ${HOME}/backups/${MYBNAME} ${HOME}/site1/
exit 0;
</pre>
<h2>Locking Down Access to your shell.php</h2>
<p>Thanks to the comment by Andrew Ramsden, Here are a couple ways to secure your shell.php file so that only you can run this script.</p>
<h3>Secure your remote shell by adding this to your shell.php</h3>
<p>Add this line to the very top of your shell.php file to make sure that only you can access this script.  Everyone else sees a blank screen.</p>
<pre>
if ($_SERVER[&#039;REMOTE_ADDR&#039;] !== &#039;1.1.1.1&#039;) die();
</pre>
<h3>Secure your remote shell with htaccess</h3>
<p>This only allows access from IP 1.1.1.1 and redirects everyone else.  See <a href="http://www.askapache.com/htaccess/apache-authentication-in-htaccess.html#using-allow-directive-in-apache" title="allow directive in apache htaccess">Using the Allow Directive in Apache htaccess</a> for more info.</p>
<pre>
Order deny,allow
Deny from all
Allow from 1.1.1.1
ErrorDocument 403 http://www.askapache.com
</pre>
<h2>Secure your remote shell with mod_rewrite and htaccess</h2>
<p>Based on the code from <a href="http://www.askapache.com/htaccess/htaccess-for-webmasters.html#redirect-except-1-ip-mod-rewrite" title="Apache htaccess tutorial">htaccess article</a>  This only allows access from user with IP of 1.1.1.1 and redirects everyone else.</p>
<pre>
RewriteEngine On
RewriteBase /
RewriteCond %{REMOTE_HOST} !^1\.1\.1\.1
RewriteRule .* http://www.askapache.com [R=302,L]
</pre>
<p><a href="http://www.askapache.com/ajax/php-and-ajax-shell-console.html">PHP and AJAX shell console</a> originally appeared on AskApache.</p>]]></content:encoded>
			<wfw:commentRss>http://www.askapache.com/ajax/php-and-ajax-shell-console.html/feed/</wfw:commentRss>
		<slash:comments>13</slash:comments>
		</item>
		<item>
		<title>Advanced Htaccess Demo/Example using Cookies, Headers, Rewrites</title>
		<link>http://www.askapache.com/htaccess/pdf-cookies-headers-rewrites.html</link>
		<comments>http://www.askapache.com/htaccess/pdf-cookies-headers-rewrites.html#comments</comments>
		<pubDate>Wed, 01 Apr 2009 03:07:53 +0000</pubDate>
		<dc:creator>AskApache</dc:creator>
				<category><![CDATA[Apache]]></category>
		<category><![CDATA[Featured]]></category>
		<category><![CDATA[Javascript]]></category>
		<category><![CDATA[PHP]]></category>
		<category><![CDATA[htaccess]]></category>
		<category><![CDATA[mod_rewrite]]></category>
		<category><![CDATA[htaccess rewrites]]></category>
		<category><![CDATA[http headers]]></category>
		<category><![CDATA[pdf]]></category>

		<guid isPermaLink="false">http://www.askapache.com/?p=2328</guid>
		<description><![CDATA[<p><a class="IFL" href="http://www.askapache.com/htaccess/pdf-cookies-headers-rewrites.html"><img src="/wp-content/uploads/2009/03/nsa-350x272.jpg" alt="Advanced Htaccess NSA" title="Advanced Htaccess NSA" width="350" height="272" /></a><strong>Whoa pretty sweet huh?</strong>  Bet you've never seen that before!  As I explain the htaccess code that achieves this, keep in mind this is merely one simple application for this code.  It's much more advanced than your basic htaccess trick, notice how this htaccess acts like a php script, very unusual..  I really wanted to share this trick after I created it for one of my clients because this is the tip of the iceberg.  Another use would be to display an alternate style sheet depending on a users theme preference.  The coolest thing about this example <acronym title="In My Humble Opinion">IMHO</acronym> is that it uses multiple advanced .htaccess ideas in order for it to work, most htaccess code on the net is very singular.  This code uses mod_headers to set the Content-Disposition header for forcing a download and uses mod_rewrite to do the rest.<br class="C" /></p>]]></description>
			<content:encoded><![CDATA[<p><a class="IFL" href="/wp-content/uploads/2009/03/nsa.jpg"><img src="/wp-content/uploads/2009/03/nsa-350x272.jpg" alt="Advanced Htaccess NSA" title="Advanced Htaccess NSA" width="350" height="272" /></a></p>
<p><strong>Welcome to the first generation of the new .htaccess tutorials/articles</strong>.  Basically these are articles detailing my BEST (almost) successful .htaccess experiments, meaning this is the only place on the net you&#8217;ll find this technique.  It&#8217;s home-grown .htaccess, and its some kush, good kush. Instead of just publishing my cryptic results and code in days past, whoah.. a working demo <em>and the NSA building</em>  <strong>;)</strong></p>
<p>Using the <a href="http://www.askapache.com/htaccess/mod_rewrite-variables-cheatsheet.html">Mod_Rewrite Variables Cheatsheet</a> makes this example, and all advanced .htaccess code easier to understand.  Well, advanced for me at least.  I know alot of AskApache visitors are some of the top gurus in many IT fields, but mostly web developers or website owners like me..   This demo is pretty self-explanatory.. Try it out to see how simple of a task this <a href="http://www.askapache.com/htaccess/htaccess.html">.htaccess trick</a> performs.  And make sure you read the whole article as this htaccess technique can be used to do a heck of a lot more than this simple demo.</p>
<h2>Set PDF Viewing Mode</h2>
<div class="cnote">
<p id="pdfi" class="BTN FL"><span class="y"><span class="t"></span><span class="text">Inline</span></span><span class="b"><span></span></span></p>
<p id="pdfa" class="BTN FL"><span class="y"><span class="t"></span><span class="text">Download</span></span><span class="b"><span></span></span></p>
<p id="pdfs" class="BTN FL"><span class="y"><span class="t"></span><span class="text">Save As</span></span><span class="b"><span></span></span></p>
<p class="FL" style="margin-left:50px;"><kbd id="pdfr" style="border:2px solid #BFBFCC;padding:10px;line-height:1.1em;">Please make a selection, defaults to "Save As" mode.</kbd></p>
<p><br class="C" /><br />
<a style="font-size:1.2em;line-height:1.2em;width:22em;" class="BTN" href="http://www.askapache.com/storage/pdf/AskApache-Test.pdf"><span class="y"><span class="t"></span><span class="text">View PDF using selected mode &raquo;</span></span><span class="b"><span></span></span></a>
</div>
<p>Whoa pretty sweet huh?  Bet you&#8217;ve never seen that before!  As I explain the htaccess code that achieves this, keep in mind this is merely one simple application for this code.  It&#8217;s much more advanced than your basic htaccess trick, notice how this htaccess acts like a php script, very unusual..  I really wanted to share this trick after I created it for one of my clients because this is the tip of the iceberg.  Another use would be to display an alternate style sheet depending on a users theme preference.  The coolest thing about this example <acronym title="In My Humble Opinion">IMHO</acronym> is that it uses multiple advanced .htaccess ideas in order for it to work, most htaccess code on the net is very singular.  This code uses mod_headers to set the Content-Disposition header for forcing a download and uses mod_rewrite to:</p>
<ol>
<li>Send different Content-Type headers</li>
<li>Check the value of a cookie</li>
<li>Set environment variables for use later by mod_headers header directive</li>
</ol>
<h2>What&#8217;s Going On</h2>
<p>There are 3 different ways for a server to send a pdf file in response to a request for one.  This causes 3 different ways to open/view the pdf file in the clients browser.</p>
<ol>
<li>The browser display&#8217;s a <strong>&#8220;Save File As&#8221;</strong> dialog, allowing you to save the file or open.</li>
<li>The browser opens the pdf file <strong>&#8220;Inline&#8221;</strong>, opening the pdf file in the browser like a web page.</li>
<li>The browser &#8220;<strong>Downloads</strong>&#8221; the pdf file automatically as an &#8220;<strong>Attachment</strong>&#8221; and then causes an external pdf reader program like adobe reader to open the file.</li>
</ol>
<p>Some people prefer to have the option of saving the file to view later, some prefer opening it with an external program, and some just like the pdf file to load right in the browser&#8230;  The point is that by using .htaccess, we can let them choose any of the 3 methods and save their preference for all further pdf files requested from our site by that user.</p>
<h2>How It Works</h2>
<p>When you click on one of the 3 demo buttons above, &#8220;Inline&#8221;, &#8220;Save As&#8221;, or &#8220;Download&#8221;, a cookie named <code>askapache_pdf</code> is saved in your browser using the javascript below, with the value being set to which button you clicked.  Then when you request the pdf file the .htaccess code below uses mod_rewrite to read the value of the askapache_pdf cookie, and depending on which was your preference it will send alternate HTTP Headers that control how your browser handles the file.</p>
<h3>Htaccess Demo File</h3>
<p>For the demo I created the folder /storage/pdf/ and this is the .htaccess file at /storage/pdf/.htaccess</p>
<pre class='brushhtaccess'>
#
# The default Content-Type for .pdf files
# This will make .pdf files default Content-Type header have
# the value &#039;application/pdf&#039; - but the default can be overridden by
# using RewriteRule with the [T=&#039;different/type&#039;]
#
AddType application/pdf .pdf
&nbsp;
#
# Turn on the rewrite engine
# if its already on you dont need this
#
RewriteEngine On
&nbsp;
#
# Skip RewriteRules if not .pdf request, like autoindexing
# The next [2] RewriteRule directives are specific for .pdf files
# so if the filename requested does not end in .pdf
# then the [S=2] instructs the next 2 RewriteRule
# directives to be completely skipped
#
RewriteRule !.*\.pdf$ - [S=2]
&nbsp;
#
# The first RewriteCond checks to see if the askapache_pdf cookie
# is NOT set.  The second RewriteCond checks to see if the askapche_pdf
# cookie has the value of s, which is the value corresponding to
# someone clicking the "Save As" button.
#
# The [NC,OR] flag means that if the cookie askapache_pdf does not
# exist, OR (next cond) if the askapache_pdf cookie does exist and is set to &#039;s&#039;
# then process the RewriteRule.  If neither cond is true the rewriterule is skipped.
#
# If one of the RewriteCond is true, then the RewriteRule is processed.
# The RewriteRule applies to any/all requests (.*) but doesn&#039;t rewrite anything (-)
# This RewriteRule sets an Apache environment variable ASKAPACHE_PDFS to have the
# value of 1 if either rewritecond is true.  The variable can be checked by any directives
# following the rewriterule in the whole htaccess file.  The ASKAPACHE_PDFS ends in S
# because if this variable exists then it means the users preference is &#039;Save As&#039;
#
# Notice that if the user requested the pdf file without selecting a preference
# i.e. no cookie exists, then the ASKAPACHE_PDFS variable is still set.
# This just lets us pick the default preference for them, in this example the
# default is &#039;Save As&#039;
#
RewriteCond %{HTTP_COOKIE} !^.*askapache_pdf.*$ [NC,OR]
RewriteCond %{HTTP_COOKIE} ^.*askapache_pdf=s.*$ [NC]
RewriteRule .* - [E=ASKAPACHE_PDFS:1]
&nbsp;
#
# The RewriteCond checks the askapache_pdf cookie for the value &#039;a&#039;
# which &#039;a&#039; represents &#039;Download&#039;
#
# If the cookies value is &#039;a&#039; then the RewriteRule overrides the default
# Content-Type from &#039;application/pdf&#039; set with AddType earlier, to
# &#039;application/octet-stream&#039;, which is a special content-type that tells the browser
# that the file cannot be loaded by the browser &#039;Inline&#039;, but must be saved
# which will be opened by an external viewer depending on browser
#
RewriteCond %{HTTP_COOKIE} ^.*askapache_pdf=a.*$
RewriteRule .* - [T=application/octet-stream]
&nbsp;
#
# This is superfly.  If the cookie/users-preference was &#039;Save As&#039; (s)
# then the RewriteRule above the last one set the environment
# variable ASKAPACHE_PDFS to have the value 1.  The Header directive here
# is ONLY processed in that variable ASKAPACHE_PDFS exists.  That is what
# the end &#039;env=ASKAPACHE_PDFS&#039; does, it is the condition that must be met or
# the Header directive is skipped.
#
# If the ASKAPACHE_PDFS environment variable set by RewriteRule does exist
# then the header directive adds the header &#039;Content-Disposition: attachment&#039; to
# the normal Response Headers.  The &#039;Content-Disposition: attachment&#039; header
# instructs your browser to present you with the &#039;Save As&#039; dialog box
# allowing you to choose whether you want to save or open
#
Header set Content-Disposition "attachment" env=ASKAPACHE_PDFS
</pre>
<h3>Unique HTTP Headers Returned</h3>
<p>When it comes down to it, the following information is the 3 modes.  Notice each one is different, because these headers are the only thing controlling how your browser handles the file.</p>
<p>Save As Mode (askapache_pdf=s)</p>
<pre class='brushhtaccess'>
Content-Disposition: attachment
Content-Type: application/pdf
</pre>
<p>Inline Mode (askapache_pdf=i)</p>
<pre class='brushhtaccess'>
Content-Type: application/pdf
</pre>
<p>Download Mode (askapache_pdf=a)</p>
<pre class='brushhtaccess'>
Content-Type: application/octet-stream
</pre>
<h2>Javascript used by Demo</h2>
<p>The best place for javascript is quirksmode, here is a definitive article on setting, reading, parsing, etc.. <a title="I am a javascript cookie monster" href="http://www.quirksmode.org/js/cookies.html">COOKIES</a>.</p>
<pre>
  if(!gi(&#039;pdfr&#039;))return;
  var pdfr=gi(&#039;pdfr&#039;);
  var cval=getCookie(&#039;askapache_pdf&#039;);

  if(cval==&#039;i&#039;){pdfr.innerHTML=&#039;Currently set to "Inline".&#039;;}
  else if(cval==&#039;a&#039;){pdfr.innerHTML=&#039;Currently set to "Download" mode.&#039;;}
  else if(cval==&#039;s&#039;){pdfr.innerHTML=&#039;Currently set to "Save As" mode.&#039;;}
&nbsp;
  addMyEvent(gi(&#039;pdfi&#039;),"mousedown",function(){setCookie("askapache_pdf", "i", "", "/", "www.askapache.com"); gi(&#039;pdfr&#039;).innerHTML = &#039;Changed mode to "Inline".&#039;; return false; });
  addMyEvent(gi(&#039;pdfa&#039;),"mousedown",function(){setCookie("askapache_pdf", "a", "", "/", "www.askapache.com"); gi(&#039;pdfr&#039;).innerHTML = &#039;Changed mode to "Download".&#039;; return false; });
  addMyEvent(gi(&#039;pdfs&#039;),"mousedown",function(){setCookie("askapache_pdf", "s", "", "/", "www.askapache.com"); gi(&#039;pdfr&#039;).innerHTML = &#039;Changed mode to "Save As".&#039;; return false; });
</pre>
<hr class="C" />
<hr class="C" />
<h2>Alternative Method &#8211; No Cookies + PHP</h2>
<p>This is what I came up with first for my client, and then while programming the php I noticed.. Hey!  I think I can do the same thing using .htaccess, which would save me on cpu/memory/potential security/etc.. but this works great too.  Though you will need to hack the code to get it working probably..</p>
<p>Note that the .htaccess rewrite code I used here used FILENAME-i.pdf or FILENAME-s.pdf to pass the preference to the pdf-dl.php script, it also worked for FILENAME.pdf?i=i</p>
<h3>pdf-dl.php</h3>
<p><?php<br />
if (<br />
  !isset($_GET['file'])<br />
  || ($f=$_GET['file'])===false<br />
  || ($fp=@fopen($f,"rb"))===false<br />
  || ($fi=pathinfo($f))===false<br />
  || ($fi['fsize']=filesize($f))===false<br />
  || strtolower($fi["extension"])!='pdf'<br />
) die('Failed');</p>
<p>ob_start();<br />
header('Accept-Ranges: bytes');<br />
header("Content-Length: {$fi['fsize']}");<br />
header('Content-Type: application/pdf');<br />
if(!isset($_GET['i'])) header("Content-Disposition: attachment; filename=\"{$fi['basename']}\"");</p>
<p>$sent = 0;<br />
while ( !feof($fp) &#038;&#038; $sent < $fi['fsize'] &#038;&#038; ($buf = fread($fp, 8192)) != '' ){<br />
  echo $buf;<br />
  $sent += strlen($buf);<br />
  flush();<br />
  ob_flush();<br />
}<br />
fclose($fp);<br />
exit;<br />
?>
</pre>
<h3>Alternate Method .htaccess</h3>
<pre class='brushhtaccess'>
#
# Deny direct request to pdf-dl.php file
#
RewriteCond %{THE_REQUEST} ^.*pdf-dl\.php.*$ [NC]
RewriteRule .* - [F]
&nbsp;
#
# Handle PDF files named anything-i.pdf as inline
#
RewriteCond %{REQUEST_FILENAME} !-f
RewriteRule ([^/]*)-i\.pdf$  /cgi-bin/pdf-dl.php?i=i&#038;file=%{DOCUMENT_ROOT}/storage/pdf/$1.pdf [L,NC,QSA,S=1]
&nbsp;
#
# Handle PDF files without -i.pdf as attachments
#
RewriteCond %{REQUEST_FILENAME} !-f
RewriteRule ([^/]*)\.pdf$  /cgi-bin/pdf-dl.php?file=%{DOCUMENT_ROOT}/storage/pdf/$1.pdf [L,NC,QSA]
</pre>
<h2>More Info</h2>
<p>The following is more information about the Content-Dispositon header and related headers, and will make you an expert at this if you read it all.. (no thanks)</p>
<h3>Interesting Reading</h3>
<p>Here is the thread of the original draft proposal for the Content-Disposition header.</p>
<ul>
<li><a href="http://www.imc.org/ietf-822/old-archive1/msg03620.html">Content-Disposition Header</a>, <em>Rens Troost - 22 Jun 1993</em>
<ul>
<li><a href="http://www.imc.org/ietf-822/old-archive1/msg03629.html">Re: Content-Disposition Header</a>, <em>Nathaniel Borenstein</em>
<ul>
<li><a href="http://www.imc.org/ietf-822/old-archive1/msg03630.html">Re: Content-Disposition Header</a>, <em>Gabe Beged-Dov</em>
<ul>
<li><a href="http://www.imc.org/ietf-822/old-archive1/msg03631.html">Re: Content-Disposition Header</a>, <em>Rens Troost</em></li>
<li><a href="http://www.imc.org/ietf-822/old-archive1/msg03635.html">Re: Content-Disposition Header</a>, <em>Gabe Beged-Dov</em></li>
<li><a href="http://www.imc.org/ietf-822/old-archive1/msg03641.html">Content-Disposition Header and multipart/alternative</a>, <em>Rens Troost</em></li>
<li><a href="http://www.imc.org/ietf-822/old-archive1/msg03645.html">Re: Content-Disposition Header and multipart/alternative</a>, <em>Nathaniel Borenstein</em></li>
</ul>
</li>
<li><a href="http://www.imc.org/ietf-822/old-archive1/msg03632.html">Re: Content-Disposition Header</a>, <em>Keith Moore</em>
<ul>
<li><a href="http://www.imc.org/ietf-822/old-archive1/msg03633.html">Re: Content-Disposition Header</a>, <em>Nathaniel Borenstein</em></li>
<li><a href="http://www.imc.org/ietf-822/old-archive1/msg03634.html">Re: Content-Disposition Header</a>, <em>Ed Levinson (Contractor)</em></li>
<li><a href="http://www.imc.org/ietf-822/old-archive1/msg03636.html">Re: Content-Disposition Header</a>, <em>Keith Moore</em></li>
<li><a href="http://www.imc.org/ietf-822/old-archive1/msg03640.html">Re: Content-Disposition Header</a>, <em>Rens Troost</em></li>
<li><a href="http://www.imc.org/ietf-822/old-archive1/msg03650.html">Re: Content-Disposition Header</a>, <em>Harald Tveit Alvestrand</em></li>
</ul>
</li>
<li><a href="http://www.imc.org/ietf-822/old-archive1/msg03621.html">Re: Content-Disposition Header</a>, <em>Steve Dorner</em>
<ul>
<li><a href="http://www.imc.org/ietf-822/old-archive1/msg03622.html">Re: Content-Disposition Header</a>, <em>Rens Troost</em>
<ul>
<li><a href="http://www.imc.org/ietf-822/old-archive1/msg03624.html">Re: Content-Disposition Header</a>, <em>Keith Moore</em></li>
</ul>
</li>
</ul>
</li>
<li><a href="http://www.imc.org/ietf-822/old-archive1/msg03652.html">Re: Content-Disposition Header</a>, <em>Carlyn M. Lowery</em></li>
</ul>
</li>
</ul>
</li>
</ul>
<h3>Intense Reading</h3>
<ul>
<li><a href="http://www2.roguewave.com/support/docs/leif/sourcepro/html/protocolsug/10-1.html">Using the MIME Headers Effectively</a></li>
<li><a href="http://www.iana.org/assignments/mail-cont-disp">Mail Content Disposition Values and Parameters</a></li>
<li><cite><a href="http://rfc.askapache.com/rfc1766/rfc1766.txt">Tags for the Identification of Languages</a></cite></li>
<li><cite><a href="http://rfc.askapache.com/rfc1436/rfc1436.txt">The Internet Gopher Protocol (a distributed document search and retrieval protocol)</a></cite></li>
<li><cite><a href="http://rfc.askapache.com/rfc1630/rfc1630.txt">Universal Resource Identifiers in WWW</a></cite></li>
<li><cite><a href="http://rfc.askapache.com/rfc1738/rfc1738.txt">Uniform Resource Locators (URL)</a></cite></li>
<li><cite><a href="http://rfc.askapache.com/rfc1866/rfc1866.txt">Hypertext Markup Language - 2.0</a></cite></li>
<li><cite><a href="http://rfc.askapache.com/rfc1945/rfc1945.txt">Hypertext Transfer Protocol -- HTTP/1.0</a></cite></li>
<li><cite><a href="http://rfc.askapache.com/rfc2045/rfc2045.txt">Multipurpose Internet Mail Extensions (MIME) Part One: Format of Internet Message Bodies</a></cite></li>
<li><cite><a href="http://rfc.askapache.com/rfc1123/rfc1123.txt">Requirements for Internet Hosts -- Communication Layers</a></cite></li>
<li><cite><a href="http://rfc.askapache.com/rfc822/rfc822.txt">Standard for The Format of ARPA Internet Text Messages</a></cite></li>
<li><cite><a href="http://rfc.askapache.com/rfc1808/rfc1808.txt">Relative Uniform Resource Locators</a></cite></li>
<li><cite><a href="http://rfc.askapache.com/rfc1036/rfc1036.txt">Standard for Interchange of USENET Messages</a></cite></li>
<li><cite><a href="http://rfc.askapache.com/rfc977/rfc977.txt">Network News Transfer Protocol</a></cite></li>
<li><cite><a href="http://rfc.askapache.com/rfc2047/rfc2047.txt">MIME (Multipurpose Internet Mail Extensions) Part Three: Message Header Extensions for Non-ASCII Text</a></cite></li>
<li><cite><a href="http://rfc.askapache.com/rfc1867/rfc1867.txt">Form-based File Upload in HTML</a></cite></li>
<li><cite><a href="http://rfc.askapache.com/rfc821/rfc821.txt">Simple Mail Transfer Protocol</a></cite></li>
<li><cite><a href="http://rfc.askapache.com/rfc1590/rfc1590.txt">Media Type Registration Procedure</a></cite></li>
<li><cite><a href="http://rfc.askapache.com/rfc959/rfc959.txt">File Transfer Protocol</a></cite></li>
<li><cite><a href="http://rfc.askapache.com/rfc1700/rfc1700.txt">Assigned Numbers</a></cite></li>
<li><cite><a href="http://rfc.askapache.com/rfc1737/rfc1737.txt">Functional Requirements for Uniform Resource Names</a></cite></li>
<li><cite><a href="http://rfc.askapache.com/rfc1864/rfc1864.txt">The Content-MD5 Header Field</a></cite></li>
<li><cite><a href="http://rfc.askapache.com/rfc1900/rfc1900.txt">Renumbering Needs Work</a></cite></li>
<li><cite><a href="http://rfc.askapache.com/rfc1952/rfc1952.txt">GZIP file format specification version 4.3</a></cite></li>
<li><cite>Improving HTTP Latency</cite></li>
<li><cite><a href="http://www.isi.edu/touch/pubs/http-perf96/">Analysis of HTTP Performance</a></cite></li>
<li><cite><a href="http://rfc.askapache.com/rfc1305/rfc1305.txt">Network Time Protocol (Version 3) Specification, Implementation and Analysis</a></cite></li>
<li><cite><a href="http://rfc.askapache.com/rfc1951/rfc1951.txt">DEFLATE Compressed Data Format Specification version 1.3</a></cite></li>
<li><cite><a href="http://sunsite.unc.edu/mdma-release/http-prob.html">Analysis of HTTP Performance Problems,</a></cite></li>
<li><cite><a href="http://rfc.askapache.com/rfc1950/rfc1950.txt">ZLIB Compressed Data Format Specification version 3.3</a></cite></li>
<li><cite><a href="http://rfc.askapache.com/rfc2069/rfc2069.txt">An Extension to HTTP: Digest Access Authentication</a></cite></li>
<li><cite><a href="http://rfc.askapache.com/rfc2068/rfc2068.txt">Hypertext Transfer Protocol -- HTTP/1.1</a></cite></li>
<li><cite><a href="http://rfc.askapache.com/rfc2119/rfc2119.txt">Key words for use in RFCs to Indicate Requirement Levels</a></cite></li>
<li><cite><a href="http://rfc.askapache.com/rfc1806/rfc1806.txt">Communicating Presentation Information in Internet Messages: The Content-Disposition Header</a></cite></li>
<li><cite><a href="http://rfc.askapache.com/rfc2145/rfc2145.txt">Use and Interpretation of HTTP Version Numbers</a></cite></li>
<li><cite><a href="http://rfc.askapache.com/rfc2076/rfc2076.txt">Common Internet Message Headers</a></cite></li>
<li><cite><a href="http://rfc.askapache.com/rfc2279/rfc2279.txt">UTF-8, a transformation format of Unicode and ISO-10646</a></cite></li>
<li><cite><a href="http://rfc.askapache.com/rfc2046/rfc2046.txt">Multipurpose Internet Mail Extensions (MIME) Part Two: Media Types</a></cite></li>
<li><cite><a href="http://rfc.askapache.com/rfc2277/rfc2277.txt">IETF Policy on Character Sets and Languages</a></cite></li>
<li><cite><a href="http://rfc.askapache.com/rfc2396/rfc2396.txt">Uniform Resource Identifiers (URI): Generic Syntax and Semantics</a></cite></li>
<li><cite><a href="http://rfc.askapache.com/rfc2617/rfc2617.txt">HTTP Authentication: Basic and Digest Access Authentication</a></cite></li>
<li><cite><a href="http://rfc.askapache.com/rfc2110/rfc2110.txt">MIME E-mail Encapsulation of Aggregate Documents, such as HTML (MHTML)</a></cite></li>
<li><cite><a href="http://rfc.askapache.com/rfc2026/rfc2026.txt">The Internet Standards Process -- Revision 3</a></cite></li>
<li><cite><a href="http://rfc.askapache.com/rfc2324/rfc2324.txt">Hyper Text Coffee Pot Control Protocol (HTCPCP/1.0)</a></cite></li>
<li><cite><a href="http://rfc.askapache.com/rfc2049/rfc2049.txt">Multipurpose Internet Mail Extensions (MIME) Part Five: Conformance Criteria and Examples</a></cite></li>
<li><cite><a href="http://rfc.askapache.com/rfc2183/rfc2183.txt">Communicating Presentation Information in Internet Messages: The Content-Disposition Header Field</a></cite></li>
</ul>
<p><a href="http://www.askapache.com/htaccess/pdf-cookies-headers-rewrites.html">Advanced Htaccess Demo/Example using Cookies, Headers, Rewrites</a> originally appeared on AskApache.</p>]]></content:encoded>
			<wfw:commentRss>http://www.askapache.com/htaccess/pdf-cookies-headers-rewrites.html/feed/</wfw:commentRss>
		<slash:comments>6</slash:comments>
		</item>
		<item>
		<title>Htaccess SEO Trends by Google</title>
		<link>http://www.askapache.com/htaccess/htaccess-seo-trends.html</link>
		<comments>http://www.askapache.com/htaccess/htaccess-seo-trends.html#comments</comments>
		<pubDate>Sun, 29 Mar 2009 06:10:03 +0000</pubDate>
		<dc:creator>AskApache</dc:creator>
				<category><![CDATA[Apache]]></category>
		<category><![CDATA[Google]]></category>
		<category><![CDATA[PHP]]></category>
		<category><![CDATA[SEO]]></category>
		<category><![CDATA[htaccess]]></category>

		<guid isPermaLink="false">http://www.askapache.com/?p=1059</guid>
		<description><![CDATA[<p><strong><span class="Gred">htaccess</span> vs. <span class="Gblue">httpd.conf</span></strong></p>
<p><a class="IFL hs hs24" href="http://www.askapache.com/htaccess/htaccess-seo-trends.html" title=".htaccess seo vs. httpd.conf seo"></a><br class="C" /></p>]]></description>
			<content:encoded><![CDATA[<h2>How Relevant Is <span class="Gblue">.htaccess</span>?</h2>
<p><strong>Scale is based on the average worldwide traffic of htaccess in all years.</strong></p>
<p><img src="http://www.google.com/trends/viz?q=.htaccess&#038;graph=weekly_img&#038;sa=N" alt=".htaccess usage and seo" width="580" height="260" /><br class="C" /></p>
<h3><strong>.</strong><span class="Gred">htaccess</span> vs. <span class="Gblue">htaccess</span></h3>
<p><a href="http://google.com/trends/viz?q=.htaccess,+htaccess&#038;date=all&#038;geo=all&#038;graph=weekly_img&#038;sort=0&#038;sa=N"><img src="http://www.google.com/trends/viz?q=.htaccess,+htaccess&#038;date=all&#038;geo=all&#038;graph=weekly_img&#038;sort=0&#038;sa=N" alt="comparing .htaccess and htaccess seo" /></a><br class="C" /></p>
<h3><span class="Gred">htaccess</span> vs. <span class="Gblue">httpd.conf</span></h3>
<p><a href="http://www.google.com/trends?q=httpd.conf%2C+.htaccess&#038;ctab=0&#038;geo=all&#038;date=all&#038;sort=0" title="htaccess vs. httpd.conf"><img src="http://www.google.com/trends/viz?q=httpd.conf,+.htaccess&#038;date=all&#038;geo=all&#038;graph=weekly_img&#038;sort=0&#038;sa=N" alt="htaccess vs. httpd.conf" /></a><br class="C" /></p>
<h3><span class="Gblue">mod_rewrite</span> vs. <span class="Gred">mod_security</span></h3>
<p><a href="http://google.com/trends?q=mod_rewrite%2Cmod_security&#038;ctab=0&#038;geo=all&#038;date=all&#038;sort=0"><img src="http://www.google.com/trends/viz?q=mod_rewrite,mod_security&#038;date=all&#038;geo=all&#038;graph=weekly_img&#038;sort=0&#038;sa=N" alt="comparing mod_rewrite and mod_security" /></a><br class="C" /></p>
<hr class="C" />
<h2>Web Server Comparisons</h2>
<p>Of course IIS is well-known to someone like me&#8230; but who cares about the big servers when small, light, <a href="http://en.wikipedia.org/wiki/Comparison_of_lightweight_web_servers">super-fast servers</a> are on the rise?  See this <a href="http://en.wikipedia.org/wiki/Comparison_of_web_server_software">Comparison of Web Servers</a>.</p>
<h3><span class="Gblue">Apache HTTP</span> vs. <span class="Gred">Windows IIS</span></h3>
<p><a href="http://www.google.com/trends?q=%22apache+HTTP%22%2C+%22windows+IIS%22&#038;ctab=0&#038;geo=all&#038;date=all&#038;sort=0" alt="Apache vs. IIS"><img src='http://www.google.com/trends/viz?q=%22apache+HTTP%22,+%22windows+IIS%22&#038;date=all&#038;geo=all&#038;graph=weekly_img&#038;sort=0&#038;sa=N' width="580" height="260" alt="Apache or Microsoft IIS" /></a><br class="C" /></p>
<h3><span class="Gred">Fnord</span>, Nginx, <span class="Gblue">LightHttpd</span></h3>
<p>Fnord what is going on?</p>
<p><a href="http://google.com/trends?q=lighttpd%2Cnginx%2Cfnord&#038;ctab=0&#038;geo=all&#038;date=ytd&#038;sort=0"><img src="http://www.google.com/trends/viz?q=lighttpd,nginx,fnord&#038;date=ytd&#038;geo=all&#038;graph=weekly_img&#038;sort=0&#038;sa=N" alt="Which light server is popular" /></a><br class="C" /></p>
<hr class="C" />
<h2>Web Programming Comparisons</h2>
<p>An interesting look at the search engine trends of programming languages out there today&#8230;</p>
<h3>Did <span class="Gred">Ruby</span> Pass <span class="Gblue">Perl</span>!</h3>
<p>Keep in mind that Google may be interpreting &#8220;ruby&#8221; to be a precious stone&#8230; and what is a &#8220;perl&#8221; exactly?  ;)</p>
<p><a href="http://google.com/trends?q=perl%2Cruby&#038;ctab=0&#038;geo=all&#038;date=all&#038;sort=0"><img src="http://www.google.com/trends/viz?q=perl,+ruby&#038;date=all&#038;geo=all&#038;graph=weekly_img&#038;sort=0&#038;sa=N" alt="Ruby vs Perl Programming Language" /></a><br class="C" /></p>
<h3><span class="Gblue">Perl</span> vs. <span class="Gred">PHP</span>, Old Debate</h3>
<p><a href="http://google.com/trends?q=perl%2Cphp&#038;ctab=0&#038;geo=all&#038;date=all&#038;sort=0"><img src="http://www.google.com/trends/viz?q=perl,+php&#038;date=all&#038;geo=all&#038;graph=weekly_img&#038;sort=0&#038;sa=N" alt="Is it Perl or PHP" /></a><br class="C" /></p>
<hr class="C" />
<h2><span class="Gblue">SEO</span></h2>
<p>Wow. Big surpise there ;)</p>
<p><a href="http://google.com/trends?q=seo&#038;ctab=0&#038;geo=all&#038;date=all&#038;sort=0"><img src="http://www.google.com/trends/viz?q=seo&#038;date=all&#038;geo=all&#038;graph=weekly_img&#038;sort=0&#038;sa=N" alt="Well at least something is drastically up" /></a><br class="C" /></p>
<hr class="C" />
<h2><span class="Gblue">Treo</span> vs. <span class="Gred">Blackberry</span> vs. <span class="Gorange">Palm</span></h2>
<p>I love my Blackberry Curve!  I used to love my Sony Clie, and I won&#8217;t forget my many Palms!</p>
<p><a href="http://google.com/trends?q=treo%2Cblackberry%2Cpalm&#038;ctab=0&#038;geo=all&#038;date=all&#038;sort=0"><img src="http://www.google.com/trends/viz?q=treo,blackberry,palm&#038;date=all&#038;geo=all&#038;graph=weekly_img&#038;sort=0&#038;sa=N" alt="Treo vs. Blackberry vs. Palm" /></a><br class="C" /></p>
<hr class="C" />
<h2>AskApache WebSite</h2>
<p>Google Trends also lets you compare websites, I can&#8217;t believe its been over a year for askapache.com, thanks for stopping by!</p>
<p><a href="http://trends.google.com/websites?q=askapache.com&#038;geo=all&#038;date=all&#038;sort=0"><img src='http://z.askapache.com/i/google-rank.png' alt='AskApache.com Google Rank' class='alignnone' /></a><br class="C" /></p>
<hr class="C" />
<h2>Google Trends &#8211; Rocks!</h2>
<p><strong>Fun, free, and helpful tool from who else?</strong><br />
<a href='http://google.com/trends'><img src="/wp-content/uploads/2008/07/logo-200x79.gif" alt="Google Trends by Google Labs" title="logo" width="200" height="79" /></a><br class="C" /></p>
<p><a href="http://www.askapache.com/htaccess/htaccess-seo-trends.html">Htaccess SEO Trends by Google</a> originally appeared on AskApache.</p>]]></content:encoded>
			<wfw:commentRss>http://www.askapache.com/htaccess/htaccess-seo-trends.html/feed/</wfw:commentRss>
		<slash:comments>1</slash:comments>
		</item>
		<item>
		<title>Password Protection Plugin Status</title>
		<link>http://www.askapache.com/htaccess/password-protection-plugin-status.html</link>
		<comments>http://www.askapache.com/htaccess/password-protection-plugin-status.html#comments</comments>
		<pubDate>Sun, 01 Mar 2009 17:39:57 +0000</pubDate>
		<dc:creator>AskApache</dc:creator>
				<category><![CDATA[Apache]]></category>
		<category><![CDATA[Hacking]]></category>
		<category><![CDATA[PHP]]></category>
		<category><![CDATA[Security]]></category>
		<category><![CDATA[Web Hosting]]></category>
		<category><![CDATA[WordPress]]></category>
		<category><![CDATA[WordPress Plugins]]></category>
		<category><![CDATA[htaccess]]></category>
		<category><![CDATA[mod_rewrite]]></category>
		<category><![CDATA[.htaccess plugin]]></category>
		<category><![CDATA[askapache]]></category>
		<category><![CDATA[password protection]]></category>

		<guid isPermaLink="false">http://www.askapache.com/?p=1966</guid>
		<description><![CDATA[<p><a class="IFL" href="/wp-content/uploads/2009/03/apache-security-model-tall1.png"><img src="/wp-content/uploads/2009/03/apache-security-model-tall1-250x123.png" alt="Apache Security Model - In Color" title="apache-security-model-wide" width="250" height="123" /></a>I wanted to address why the update to the AskApache Password Protection plugin didn't happen pre-2009 as I had hoped.. Mostly due to my job but I thought I could at least fill you in.  Oh and this is going to get very boring very fast, unless you're ready to rumble in the zone.<br class="C" /></p>]]></description>
			<content:encoded><![CDATA[<p>I wanted to address why the update to the AskApache Password Protection plugin didn&#8217;t happen pre-2009 as I had hoped.. Mostly due to my job but I thought I could at least fill you in.  Oh and this is going to get very boring very fast, unless you&#8217;re ready to rumble in the zone.</p>
<h2>File Permissions!</h2>
<p>The main issue with the password protection plugin working for some people and not others is due to <a title="detailed file permission article" href="http://www.askapache.com/security/chmod-stat.html">file permission configurations</a>.  WordPress is simply a group of .php files saved on your server.  The actual program that is in fact running WordPress is the <a title="SuEXEC and php.cgi" href="http://www.askapache.com/htaccess/php-cgi-redirect_status.html">PHP interpreter</a>, which is in turn controlled by the Apache Server.  Almost all computers are running at least 2 servers, the Web Server which serves and displays your files, and a FTP server.</p>
<p><a class="IFL" href="/wp-content/uploads/2009/03/apache-security-model-tall1.png"><img src="/wp-content/uploads/2009/03/apache-security-model-tall1-250x123.png" alt="Apache Security Model - In Color" title="apache-security-model-wide" width="250" height="123" /></a>  Here&#8217;s a detailed look at the Apache Security Model, from <a href="http://www.apachesecurity.net/blog/2009/02/apache_security_model.html">ApacheSecurity.net</a>, a blog maintained by <em>Ivan Ristic</em>, the author of <a href="http://www.modsecurity.org/">ModSecurity</a>.<br class="C" /></p>
<p>The problem is happening because when you login to your FTP server with your username and password, the files that you upload are then  owned by that username and password, which is almost always an actual user account on the server system.  But the Apache Server is an executable file itself, and it is not owned by your FTP username, for security reasons.  Apache controls the PHP Interpreter, which parses and executes the WordPress and plugin files as a separate user.  ( <a href="http://httpd.apache.org/docs/trunk/suexec.html">SuEXEC</a>, <a href="http://httpd.apache.org/docs/trunk/misc/security_tips.html">Apache Security Tips</a> ) </p>
<p>So what happens is the <code>askapache-password-protect.php</code> file saved on your server and is owned by the user that created it (if you downloaded it to your computer then used ftp to transfer, your ftp user owns it..  if you used a php downloader script, then the php process owner owns it)   So when you click on the Run Tests button from the WordPress administration website what you are doing is sending a request via HTTP to your Apache Server process, which sees the requested file is .php so it then runs the php interpreter to execute the askapache-password-protect.php file, then that file uses programming to attempt and write/modify a file in your blog&#8217;s root directory.</p>
<h2>Process Owner vs. File Owner</h2>
<p>So who owns your blog&#8217;s root directory?  Your ftp user account/ you do..   but who owns the process that is trying to write/modify a file that is owned by your ftp user?  The PHP Process that is actually executing the file access/write requests.  This is the core way that 99% of all web sites get cracked into.. All these malicious robots and exploit bots do is attempt to write a file onto your server so that it can then be used to take over your site.  If they can save a file on your blog&#8217;s directory (uploads, insecure plugin code, not filtering user input, etc..) it inherits the permissions of the process that actually wrote the data bits onto the hard-drive.</p>
<p>So some server-admins/web hosts configure the php interpreter to not have write access anywhere except for a couple neccessary locations like /tmp.  They have auto-installation&#8217;s available through their online web panels, meaning instead of executing .php scripts in your user directory as the php process they force you to use, they can bypass all that because the installation scripts they use are all on their systems, not on your &#8220;locked-down&#8221; cluster.</p>
<p>This is the fundamental security battle that network server security is all based on..   Apache is owned by a powerful user because it owns the server process, so apache is often run as the user dhapache or nobody..  If a cracker is able to find a way to get a file saved on your server with the dhapache user as the owner then they&#8217;ve basically just gotten control of the whole thing.  When you upload a file to your server using the add attachment form in wordpress, the file first goes through the dhapache user which passes the file to the php process owner which has much less permissions.  Apache has been in open-source development for many many years now, its the safest most secure server in the world, windows servers are hackable, apache servers are hacked usually only when the sysadmin configures it wrong or accidentally.</p>
<p>Believe it or not, as confusing as my feeble explanation was, this is only like .1% of whats going on.. I&#8217;ve basically spent the last several months developing the new version specifically to be able to work no matter what configuration you have.  What I ended up doing was finding ways to bypass this security on a couple hosting providers that are setup in this way, but even though I got it to work in most instances it basically was hacking their systems, and if I published that code to automatically bypass web-hosts security setups I think I&#8217;d be in big trouble and they would just close those specific holes and the plugin would not work again.   So I decided instead of exploiting host-specifics hacks to get the plugin to work that I would focus on the method that WordPress sorta uses.  The code they have now (2.8 bleeding-edge) still isn&#8217;t where it needs to be, but this is some difficult stuff and they have a brilliant start, it&#8217;ll work.. just a question of when.</p>
<h2><code>wp-admin/includes/file.php</code></h2>
<p>Ok so this function <code>get_filesystem_method</code> is a brilliant bit of code that would&#8217;ve been beyond my current PHP skills to come up with.  It determines which if any of the following methods can be used to modify files on your server from within WordPress, which is exactly what the new version of the passpro plugin needs to use.   The first test simply creates a file from within php using wp_tempnam, a function that attempts to locate and write to a temporary location on your server that has the best chance of having write access.  If it is successfully created (this code assumes that it will be, something they need to fix) then the fileowner (uses stat internally) of the temp file just created is compared to the owner of the php script&#8230;  Normally this works and then the plugin woks too, but on some hosts the script is running as a separate user than that of the file which means you can&#8217;t directly access the local file system.  That is what is occurring for most of you who experience permission problems while testing the plugin.  There are thousands of caveats for each little part depending on your php version, php setup, server setup, server version, which Server API you are using, the type of SAPI being used, and on and on..</p>
<pre>
function get_filesystem_method($args = array()) {
  $method = false;
  if( function_exists(&#039;getmyuid&#039;) &#038;&#038; function_exists(&#039;fileowner&#039;) ){
    $temp_file = wp_tempnam();
    if ( getmyuid() == fileowner($temp_file) ) $method = &#039;direct&#039;;
    unlink($temp_file);
  }
  if ( ! $method &#038;&#038; isset($args[&#039;connection_type&#039;]) &#038;&#038; &#039;ssh&#039; == $args[&#039;connection_type&#039;] &#038;&#038; extension_loaded(&#039;ssh2&#039;) )
          $method = &#039;ssh2&#039;;
  if ( ! $method &#038;&#038; extension_loaded(&#039;ftp&#039;) )
          $method = &#039;ftpext&#039;;
  if ( ! $method &#038;&#038; ( extension_loaded(&#039;sockets&#039;) || function_exists(&#039;fsockopen&#039;) ) )
          $method = &#039;ftpsockets&#039;; //Sockets: Socket extension; PHP Mode: FSockopen / fwrite / fread
  return apply_filters(&#039;filesystem_method&#039;, $method);
}
</pre>
<h2>Enumerating Permissions can be Annoying</h2>
<p>This was part of some tests I did to see what kind of access I had with the very helpful posix functions which are very accurate as well since they were designed for a system with file permissions, ie. not Win.  Don&#8217;t ask me how because I won&#8217;t tell you, but on one of the hosts I was testing on that did not allow direct access I was able to get the Apache server running as dhapache to erroneously write a file into my users blog directory.  This is a big security no-no and I now have my .htaccess file written into the blog directory where it should go, but instead of my php script&#8217;s user having write access to the file so I can modify it, its owned by dhapache!  Because the file is owned by dhapache I shouldn&#8217;t even be allowed to know it exists, but there it is.  So the next step was to try and take ownership of the .htaccess file so that I could modify it.  I tried and tried but was unsuccessful, I couldn&#8217;t modify it so that was another dead end.  Actually it took me awhile to figure out how to remove the file from my directory.  Being that it was owned by dhapache I couldn&#8217;t delete or modify it using my php process or even through ftp/ssh!  Sysadmins regularly run find commands that search the servers for any files owned by dhapache that should not be there as this is a big red flag that someone has found a way to manipulate dhapache which could potentially lead to modifying dhapche-owned server config files.. Luckily I was able to delete it by basically running the hack again to overwrite the file.</p>
<pre>
  if ((posix_setgid(getmygid())) !== false) $this-&gt;to_log(&#039;&#039;, 1,
      "Success Changing SETGID of {$file}  to " . getmygid(), 3);
  elseif ((posix_setgid(filegroup(__file__))) !== false) $this-&gt;to_log(&#039;&#039;, 1,
      "Success Changing SETUID of {$file} to " . filegroup(__file__), 3);
  if ((posix_setegid(getmygid())) !== false) $this-&gt;to_log(&#039;&#039;, 1,
      "Success Changing SETEGID of {$file} to " . getmygid(), 3);
  elseif ((posix_setegid(filegroup(__file__))) !== false) $this-&gt;to_log(&#039;&#039;, 1,
      "Success Changing SETEGID of {$file} to " . filegroup(__file__), 3);
  if ((posix_setuid(getmyuid())) !== false) $this-&gt;to_log(&#039;&#039;, 1,
      "Success Changing SETUID of {$file}  to " . getmyuid(), 3);
  elseif ((posix_setuid(get_current_user())) !== false) $this-&gt;to_log(&#039;&#039;, 1,
      "Success Changing SETUID of {$file} to " . get_current_user(), 3);
  if ((posix_seteuid(getmyuid())) !== false) $this-&gt;to_log(&#039;&#039;, 1,
      "Success Changing SETEUID of {$file} to " . getmyuid(), 3);
  elseif ((posix_seteuid(get_current_user())) !== false) $this-&gt;to_log(&#039;&#039;, 1,
      "Success Changing SETEUID of {$file} to " . get_current_user(), 3);
  if ((chmod($file, FS_CHMOD_DIR) || chmod($file, 0776) || chmod($file, 0766) || chmod($file,
    FS_CHMOD_FILE)) !== false) $this-&gt;to_log(&#039;&#039;, 1, "Success Changing Mode of {$file}", 3);
  if ((chown($file, getmyuid())) !== false) $this-&gt;to_log(&#039;&#039;, 1,
      "Success Changing Ownership of {$file} to " . getmyuid(), 3);
  elseif ((chown($file, get_current_user())) !== false) $this-&gt;to_log(&#039;&#039;, 1,
      "Success Changing Ownership of {$file} to " . get_current_user(), 3);
  if ((chgrp($file, getmygid())) !== false) $this-&gt;to_log(&#039;&#039;, 1,
      "Success Changing Group of {$file} to " . getmygid(), 3);
  elseif ((chgrp($file, filegroup(__file__))) !== false) $this-&gt;to_log(&#039;&#039;, 1,
      "Success Changing Group of {$file} to " . filegroup(__file__), 3);
  if ((chmod($file, FS_CHMOD_DIR) || chmod($file, 0776) || chmod($file, 0766) || chmod($file,
    FS_CHMOD_FILE)) !== false) $this-&gt;to_log(&#039;&#039;, 1, "Success Changing Mode of {$file}", 3);
  if ((chown($file, getmyuid())) !== false) $this-&gt;to_log(&#039;&#039;, 1,
      "Success Changing Ownership of {$file} to " . getmyuid(), 3);
  elseif ((chown($file, get_current_user())) !== false) $this-&gt;to_log(&#039;&#039;, 1,
      "Success Changing Ownership of {$file} to " . get_current_user(), 3);
  if ((chgrp($file, getmygid())) !== false) $this-&gt;to_log(&#039;&#039;, 1,
      "Success Changing Group of {$file} to " . getmygid(), 3);
  elseif ((chgrp($file, filegroup(__file__))) !== false) $this-&gt;to_log(&#039;&#039;, 1,
      "Success Changing Group of {$file} to " . filegroup(__file__), 3);
  return (!$this-&gt;_fclose($fh)) ? $this-&gt;to_log(__function__ . &#039;:&#039; . __line__ .
    " Error closing {$mode} handle for {$file}", 0) : $total;
</pre>
<p>If php process isn&#8217;t allowed to write to your web directory but you have an ftp account that is, then we request your ftp username/password in wordpress and if the php process running the <code>askapache-password-protect.php</code> plugin script is allowed access to <a href="http://www.askapache.com/php/fsockopen-socket.html">raw networking sockets using fsockopen</a> then we can basically access and write to your blog&#8217;s <code>.htaccess</code> file by using php to mimick an ftp client session. There are also other protocols and options available using php if ftp/fsockopen isn&#8217;t allowed, but you run out of alternatives quick.  Using the curl extension is one option.</p>
<p>So I wrote my own ftp library for a fsockopen class I had already developed for specific test requirements in unreleased versions, so the release of the new askapache password protect plugin will work for 75% or so of the people who have trouble now.. not to mention the insane logging and debugging I&#8217;ve added while looking for the reasons some web-hosts still don&#8217;t work.  Some use custom php security modules, wrappers, and custom virtual servers that are akin to a vmware server.  So for maybe 10% of those running apache who have had problems they would still have them. I&#8217;m still playing with some ssh capability from within the plugin similar to the ftp technique..  I really hope WordPress just adds this functionality by updating their current filesystem classes..</p>
<h2>Fsockopen Payload Class</h2>
<p>Here&#8217;s what I had several versions ago.. Just sticking it up here in case anyone is curious, one cool thing this version starts to incorporate is being able to send direct data payloads across the socket so it can be used like <a href="http://metasploit.com/" title="Metasploit provides useful information to people who perform penetration testing, IDS signature development, and exploit research.">the metasploit framework</a> to send payloads of exploits, but of course we&#8217;re using it to mimick other protocols like ftp, which can be setup by feeding hex into the socket direct from a real ftp client, and piping the output. Keep in mind that this is my first time using php classes, so the learning curve has been incredible&#8230;</p>
<pre>
&lt;?php
error_log("RUNNING " . basename(__file__) . "\n");
if (!in_array(&#039;AskApache_Net&#039;, get_declared_classes()) &#038;&#038; !class_exists(&#039;AskApache_Net&#039;)):
 class AskApache_Net extends AskApachePassPro
 {
  var $_fp = null;
  var $_socket = array(&#039;protocol&#039; =&gt; &#039;1.0&#039;, &#039;method&#039; =&gt; &#039;GET&#039;, &#039;referer&#039; =&gt;
   &#039;http://www.askapache.com/&#039;, &#039;port&#039; =&gt; &#039;80&#039;, &#039;ua&#039; =&gt;
   &#039;Mozilla/5.0 (compatible; AskApache_Net/1.6; http://www.askapache.com/)&#039;, &#039;scheme&#039; =&gt;
   &#039;http&#039;, &#039;transport&#039; =&gt; &#039;&#039;, &#039;host&#039; =&gt; &#039;&#039;, &#039;user&#039; =&gt; &#039;&#039;, &#039;pass&#039; =&gt; &#039;&#039;, &#039;path&#039; =&gt; &#039;/&#039;,
   &#039;query&#039; =&gt; &#039;&#039;, &#039;fragment&#039; =&gt; &#039;&#039;);
  var $authtype = &#039;Basic&#039;;
  var $timeout = 15;
  var $_dh = &#039;&#039;;
  var $_digest = array(&#039;realm&#039; =&gt; &#039;&#039;, &#039;nonce&#039; =&gt; &#039;&#039;, &#039;uri&#039; =&gt; &#039;&#039;, &#039;algorithm&#039; =&gt; &#039;MD5&#039;,
   &#039;qop&#039; =&gt; &#039;auth&#039;, &#039;opaque&#039; =&gt; &#039;&#039;, &#039;domain&#039; =&gt; &#039;&#039;, &#039;nc&#039; =&gt; &#039;00000001&#039;, &#039;cnonce&#039; =&gt;
   &#039;82d057852a9dc497&#039;, &#039;A1&#039; =&gt; &#039;&#039;, &#039;A2&#039; =&gt; &#039;&#039;, &#039;response&#039; =&gt; &#039;&#039;);
  var $_ACLF = "\r\n";
  var $_request_body = &#039;&#039;;
  var $_request_headers = array();
  var $_response_headers = array();
  var $my_headers;
  var $_response_header = &#039;&#039;;
  var $_response_protocol = &#039;&#039;;
  var $_response_version = &#039;&#039;;
  var $_response_code = &#039;&#039;;
  var $_response_message = &#039;&#039;;
  var $_response_body = &#039;&#039;;
  var $_errs = array(3 =&gt; &#039;Socket creation failed&#039;, 4 =&gt; &#039;DNS lookup failure&#039;, 5 =&gt;
   &#039;Connection refused or timed out&#039;, 111 =&gt; &#039;Connection refused&#039;, 113 =&gt;
   &#039;No route to host&#039;, 110 =&gt; &#039;Connection timed out&#039;, 104 =&gt; &#039;Connection reset by client&#039;);

  /**
   * AskApache_Net::AskApache_Net()
   */
  function AskApache_Net()
  {
   return $this-&gt;__construct();
  }

  /**
   * AskApache_Net::__destruct()
   */
  function __destruct()
  {
   $this-&gt;_timer(&#039;class&#039;);
   return true;
  }

  /**
   * AskApache_Net::__construct()
   */
  function __construct()
  {
   $this-&gt;_timer(&#039;class&#039;);
   $this-&gt;_ACLF = chr(13) . chr(10);
   @set_time_limit(60);
   return true;
  }

  /**
   * AskApache_Net::hsockit()
   */
  function hsockit($URI)
  {
   $this-&gt;msg(__function__ . &#039;:&#039; . __line__, 3);
   $this-&gt;_socket[&#039;method&#039;] = &#039;HEAD&#039;;
   return $this-&gt;sockit($URI);
  }

  /**
   * AskApache_Net::sockit()
   */
  function sockit($URI = &#039;&#039;)
  {
   $this-&gt;msg(__function__ . &#039;:&#039; . __line__, 3);
   if (!$this-&gt;_build_sock($URI)) return $this-&gt;msg(__function__ . &#039;:&#039; . __line__,
     "Failed!", 0);
   if (!$this-&gt;_connect()) return $this-&gt;msg(__function__ . &#039;:&#039; . __line__, "Failed!", 0);
   $this-&gt;_build_request();
   if (!$this-&gt;_build_request()) return $this-&gt;msg(__function__ . &#039;:&#039; . __line__,
     "Failed!", 0);
   if (!$this-&gt;_tx()) return $this-&gt;msg(__function__ . &#039;:&#039; . __line__, "tx Failed!", 0);
   if (!$this-&gt;_rx()) return $this-&gt;msg(__function__ . &#039;:&#039; . __line__, "rx Failed!", 0);
   if (!$this-&gt;_disconnect()) return $this-&gt;msg(__function__ . &#039;:&#039; . __line__,
     "disconnect Failed!", 0);
   if ((bool)$this-&gt;net_debug === true) {
    foreach (array(&#039;out_payload&#039;, &#039;_request_body&#039;, &#039;_response_header&#039;, &#039;_response_body&#039;) as
     $nam) {
     if (is_array($this-&gt;$nam)) {
      if (sizeof($this-&gt;$nam) &gt; 1) {
       echo "\n\n{$nam}\n";
       print_r($this-&gt;$nam);
      }
     } else {
&nbsp;
      if (!empty($this-&gt;$nam)) {
       echo "\n\n{$nam}\n";
       echo $this-&gt;$nam;
      }
     }
    }
    $this-&gt;tcp_trace(1);
   }
   return (int)$this-&gt;_response_code;
  }

  /**
   * AskApache_Net::_build_sock()
   */
  function _build_sock($url)
  {
   $this-&gt;msg(__function__ . &#039;:&#039; . __line__, 3);
   $socket_info = &#038;$this-&gt;_socket;
   if (!$u_bits = parse_url($url)) return false;
   if (empty($u_bits[&#039;method&#039;])) $u_bits[&#039;method&#039;] = &#039;GET&#039;;
   if (empty($u_bits[&#039;protocol&#039;])) $u_bits[&#039;protocol&#039;] = &#039;1.0&#039;;
   if (empty($u_bits[&#039;host&#039;])) $u_bits[&#039;host&#039;] = $_SERVER[&#039;HTTP_HOST&#039;];
   if (empty($u_bits[&#039;scheme&#039;])) $u_bits[&#039;scheme&#039;] = &#039;http&#039;;
   if (empty($u_bits[&#039;port&#039;])) $u_bits[&#039;port&#039;] = $_SERVER[&#039;SERVER_PORT&#039;];
   $u_bits[&#039;path&#039;] = (empty($u_bits[&#039;path&#039;]) ? &#039;/&#039; : $u_bits[&#039;path&#039;]) . (!empty($u_bits[&#039;query&#039;]) ?
    &#039;?&#039; . $u_bits[&#039;query&#039;] : &#039;&#039;);
   if (empty($u_bits[&#039;ua&#039;])) $u_bits[&#039;ua&#039;] =
     &#039;Mozilla/5.0 (compatible; AskApache_Net/1.0; http://www.askapache.com)&#039;;
   if (empty($u_bits[&#039;referer&#039;])) $u_bits[&#039;referer&#039;] = &#039;http://www.askapache.com&#039;;
   if (empty($u_bits[&#039;fragment&#039;])) unset($u_bits[&#039;fragment&#039;]);
   if (empty($u_bits[&#039;user&#039;])) unset($u_bits[&#039;user&#039;]);
   if (empty($u_bits[&#039;pass&#039;])) unset($u_bits[&#039;pass&#039;]);
   if ($u_bits[&#039;scheme&#039;] == &#039;https&#039; || $this-&gt;_socket[&#039;scheme&#039;] == &#039;https&#039;) $u_bits[&#039;transport&#039;] =
     &#039;ssl://&#039;;
   if ($u_bits[&#039;scheme&#039;] == &#039;https&#039; || $this-&gt;_socket[&#039;scheme&#039;] == &#039;https&#039;) $u_bits[&#039;port&#039;] =
     &#039;443&#039;;
   $socket_info = $this-&gt;_parse_args($u_bits, $socket_info);
   extract($socket_info, EXTR_SKIP);
   return true;
  }

  /**
   * AskApache_Net::_build_auth_header()
   */
  function _build_auth_header()
  {
   $this-&gt;msg(__function__ . &#039;:&#039; . __line__, 3);
   if ($this-&gt;authtype == &#039;Basic&#039;) $this-&gt;_request_headers[] = &#039;Authorization: Basic &#039; .
     base64_encode($this-&gt;_socket[&#039;user&#039;] . ":" . $this-&gt;_socket[&#039;pass&#039;]);
   elseif ($this-&gt;authtype == &#039;Digest&#039;) {
    $this-&gt;msg(__function__ . &#039;:&#039; . __line__, 3);
    $this-&gt;_socket[&#039;protocol&#039;] = &#039;1.1&#039;;
    $hdr = $mtx = array();
    preg_match_all(&#039;/(\w+)=(?:"([^"]+)"|([^\s,]+))/&#039;, $this-&gt;_dh, $mtx, PREG_SET_ORDER);
    foreach ($mtx as $m) $hdr[$m[1]] = $m[2] ? $m[2] : $m[3];
    foreach ($hdr as $key =&gt; $val)
     if (array_key_exists($key, $this-&gt;_digest) &#038;&#038; !empty($val)) $this-&gt;_digest[$key] = $val;
    $this-&gt;_digest[&#039;uri&#039;] = $this-&gt;_socket[&#039;path&#039;];
    $this-&gt;_digest[&#039;A1&#039;] = md5($this-&gt;_socket[&#039;user&#039;] . &#039;:&#039; . $this-&gt;_digest[&#039;realm&#039;] .
     &#039;:&#039; . $this-&gt;_socket[&#039;pass&#039;]);
    $this-&gt;_digest[&#039;A2&#039;] = md5($this-&gt;_socket[&#039;method&#039;] . &#039;:&#039; . $this-&gt;_socket[&#039;path&#039;]);
    $this-&gt;_digest[&#039;response&#039;] = md5($this-&gt;_digest[&#039;A1&#039;] . &#039;:&#039; . $this-&gt;_digest[&#039;nonce&#039;] .
     &#039;:&#039; . $this-&gt;_digest[&#039;nc&#039;] . &#039;:&#039; . $this-&gt;_digest[&#039;cnonce&#039;] . &#039;:&#039; . $this-&gt;_digest[&#039;qop&#039;] .
     &#039;:&#039; . $this-&gt;_digest[&#039;A2&#039;]);
    $this-&gt;_request_headers[] = sprintf(&#039;Authorization: Digest username="%1$s", realm="%2$s", nonce="%3$s",&#039;.
     &#039;uri="%4$s", algorithm=%5$s, response="%6$s", qop="%7$s", nc="%8$s"%9$s%10$s&#039;,
     $this-&gt;_socket[&#039;user&#039;], $this-&gt;_digest[&#039;realm&#039;], $this-&gt;_digest[&#039;nonce&#039;], $this-&gt;
     _digest[&#039;uri&#039;], $this-&gt;_digest[&#039;algorithm&#039;], $this-&gt;_digest[&#039;response&#039;], $this-&gt;
     _digest[&#039;qop&#039;], $this-&gt;_digest[&#039;nc&#039;], !empty($this-&gt;_digest[&#039;cnonce&#039;]) ? &#039;, cnonce="&#039; .
     $this-&gt;_digest[&#039;cnonce&#039;] . &#039;"&#039; : &#039;&#039;, !empty($this-&gt;_digest[&#039;opaque&#039;]) ? &#039;, opaque="&#039; .
     $this-&gt;_digest[&#039;opaque&#039;] . &#039;"&#039; : &#039;&#039;);
   }
   return true;
  }

  /**
   * AskApache_Net::_build_request()
   */
  function _build_request()
  {
   $this-&gt;msg(__function__ . &#039;:&#039; . __line__, 3);
   $this-&gt;_request_headers[] = $this-&gt;_socket[&#039;method&#039;] . " " . $this-&gt;_socket[&#039;path&#039;] .
    " HTTP/" . $this-&gt;_socket[&#039;protocol&#039;];
   if (is_array($this-&gt;my_headers) &#038;&#038; sizeof($this-&gt;my_headers) &gt; 0) $this-&gt;
     _request_headers = array_merge($this-&gt;_request_headers, $this-&gt;my_headers);
   else {
    $this-&gt;_request_headers[] = "Host: " . $this-&gt;_socket[&#039;host&#039;];
    $this-&gt;_request_headers[] = "User-Agent: " . $this-&gt;_socket[&#039;ua&#039;];
    $this-&gt;_request_headers[] = &#039;Accept: application/xhtml+xml,text/html;q=0.9,*/*;q=0.5&#039;;
    $this-&gt;_request_headers[] = &#039;Accept-Language: en-us,en;q=0.5&#039;;
    $this-&gt;_request_headers[] = &#039;Accept-Encoding: none&#039;;
    $this-&gt;_request_headers[] = &#039;Accept-Charset: ISO-8859-1,utf-8;q=0.7,*;q=0.7&#039;;
    $this-&gt;_request_headers[] = &#039;Referer: &#039; . $this-&gt;_socket[&#039;referer&#039;];
   }
   if (!empty($this-&gt;_socket[&#039;user&#039;]) &#038;&#038; !empty($this-&gt;_socket[&#039;pass&#039;])) $this-&gt;
     _build_auth_header();
   if ($this-&gt;out_payload !== false) $this-&gt;_request_body = $this-&gt;out_payload;
   else  $this-&gt;_request_body = join($this-&gt;_ACLF, $this-&gt;_request_headers) . $this-&gt;
     _ACLF . $this-&gt;_ACLF;
   return true;
  }

  /**
   * AskApache_Net::_tx()
   */
  function _tx()
  {
   $this-&gt;msg(__function__ . &#039;:&#039; . __line__, 3);
   return (bool)(is_resource($this-&gt;_fp) &#038;&#038; $this-&gt;_netwrite($this-&gt;_fp, $this-&gt;
    _request_body));
  }

  /**
   * AskApache_Net::_rx()
   */
  function _rx()
  {
   $this-&gt;msg(__function__ . &#039;:&#039; . __line__, 3);
   if (!is_resource($this-&gt;_fp)) return false;
   $this-&gt;_response = $this-&gt;_netread($this-&gt;_fp, 500000);
   $parts = explode($this-&gt;_ACLF . $this-&gt;_ACLF, ltrim($this-&gt;_response), 2);
   $this-&gt;_response_header = trim($parts[0]);
   $this-&gt;_response_body = trim($parts[1]);
   if (preg_match(&#039;#([^/]*)/([\d\.]+) ([\d]*?) (.*)#&#039;, $this-&gt;_response_header, $htx)) {
    $this-&gt;_response_protocol = trim($htx[1]);
    $this-&gt;_response_version = trim($htx[2]);
    $this-&gt;_response_code = trim($htx[3]);
    $this-&gt;_response_message = trim($htx[4]);
   }
   if (preg_match_all(&#039;#([^:]+)\:?(.*)#&#039;, str_replace($htx, &#039;&#039;, $this-&gt;_response_header),
    $mtx, PREG_SET_ORDER)) {
    foreach ($mtx as $m) {
     $this-&gt;_headers[strtolower(trim($m[1]))] = trim($m[2]);
     if (preg_match(&#039;/(WWW|Proxy)-Authenticate:.*Digest/i&#039;, trim($m[1]))) $this-&gt;_dh =
       trim($m[1]);
    }
   }
   return true;
  }

  /**
   * AskApache_Net::tcp_trace()
   */
  function tcp_trace($p = false)
  {
   $this-&gt;_timer(__function__ );
   $ret = join("\n", array_merge((array )$this-&gt;_request_headers, array(&#039;&#039;), (array )$this-&gt;
    _response_headers));
   if ($p !== false) {
    echo $ret;
    $ret = true;
   }
   $this-&gt;_timer(__function__ );
   return $ret;
  }

  /**
   * AskApache_Net::_get_ip()
   */
  function _get_ip($host)
  {
   $this-&gt;msg(__function__ . &#039;:&#039; . __line__, 3);
&nbsp;
   if (!preg_match(&#039;/^[\t ]*[0-9]+\.[0-9]+\.[0-9]+\.[0-9]+[\t ]*$/&#039;, $host)) $hostip =
     gethostbyname($host);
   $ip = ($hostip == $host) ? $host : long2ip(ip2long($hostip));
   return $ip;
  }

  /**
   * AskApache_Net::_connect()
   */
  function _connect()
  {
   $this-&gt;msg(__function__ . &#039;:&#039; . __line__, 3);
   if (false === ($this-&gt;_fp = fsockopen($this-&gt;_get_ip($this-&gt;_socket[&#039;host&#039;]), $this-&gt;
    _socket[&#039;port&#039;], $errno, $errstr, $this-&gt;timeout)) || !is_resource($this-&gt;_fp)) {
    $err = (array_key_exists($errno, $this-&gt;_errs)) ? $this-&gt;_errs[$errno] :
     &#039;Connection failed&#039;;
    return $this-&gt;msg(__function__ . &#039;:&#039; . __line__ . " Fsockopen failed! [{$errno}] {$err} ({$errstr})",
     0);
   }
   if (function_exists("socket_set_timeout")) socket_set_timeout($this-&gt;_fp, $this-&gt;
     timeout);
   elseif (function_exists("stream_set_timeout")) stream_set_timeout($this-&gt;_fp, $this-&gt;
     timeout);
   usleep(10000);
   return true;
  }

  /**
   * AskApache_Net::_disconnect()
   */
  function _disconnect()
  {
   $this-&gt;msg(__function__ . &#039;:&#039; . __line__, 3);
   if (is_resource($this-&gt;_fp)) return $this-&gt;_fclose($this-&gt;_fp);
   else  $this-&gt;_fp = null;
   return true;
  }

  /**
   * AskApache_Net::get_response_headers()
   */
  function get_response_headers($header = false)
  {
   $this-&gt;msg(__function__ . &#039;:&#039; . __line__, 3);
   if ($header !== false &#038;&#038; array_key_exists($header, $this-&gt;_response_headers)) return $this-&gt;
     _response_headers[$header];
   return $this-&gt;_response_headers;
  }

  /**
   * AskApache_Net::get_response_body()
   */
  function get_response_body()
  {
   $this-&gt;msg(__function__ . &#039;:&#039; . __line__, 3);
   return $this-&gt;_response_body;
  }

  /**
   * AskApache_Net::_netread()
   */
  function _netread(&#038;$fh, $ts = 50000000, $bs = 124)
  {
   $this-&gt;_timer(__function__ );
   for ($d = $b = &#039;&#039;, $rt = $at = $r = 0; ($fh !== false &#038;&#038; !feof($fh) &#038;&#038; $b !== false &#038;&#038;
    $at &lt; 50000000 &#038;&#038; $rt &lt; $ts); $r = $ts - $rt, $bs = (($bs &gt; $r) ? $r : $bs), $this-&gt;
    _timer("R: {$rt}"), $b = fread($fh, $bs), $br = strlen($b), $d .= $b, $this-&gt;_timer("R: {$rt}"),
    $rt += $br, $at++, $this-&gt;msg("[RT: {$rt}]\t[BR: {$br}" . (($ts != 50000000) ? "]\t\t [{$r} / {$ts}]" :
    " : {$bs}]\t[{$at}]"))) ;
   $this-&gt;_timer(__function__ );
   return ((strlen($d) != 0)) ? $d : false;
  }

  /**
   * AskApache_Net::_netwrite()
   */
  function _netwrite(&#038;$fh, $d = &#039;&#039;, $bs = 512)
  {
   $this-&gt;_timer(__function__ );
&nbsp;
   for ($bw = $wt = $at = 0, $dat = &#039;&#039;, $ts = strlen($d); ($fh !== false &#038;&#038; $bw !== false &#038;&#038;
    $at &lt; 50000000 &#038;&#038; $wt &lt; $ts); $r = $ts - $wt, $bs = (($bs &gt; $r) ? $r : $bs), $dat =
    substr($d, $wt, $bs), $bw = fwrite($fh, $dat), $wt += $bw, $this-&gt;msg("[WT: {$wt}]\t[BW: {$bw}]\t\t[I: {$r} / {$ts}:{$bs}] - {$at}"),
    $at++) ;
   $this-&gt;msg("[WT: {$wt}]\t[BW: {$bw}]\t\t[I: {$r} / {$ts}:{$bs}] - {$at}");
   $this-&gt;_timer(__function__ );
   return ($wt == $ts) ? true : false;
  }
 }
endif;
?&gt;
</pre>
<p>So I decided to finally give in to what I&#8217;ve been avoiding all along and added a php-software-based method that will work on everycomputer, windows, blackberrys, etc.. That took me about 15minutes as its just a few lines of code.. The problem I have with it is that php is what is actually controlling the sending, receiving, and verifying of the authentication headers instead of using the builtin super-secure apache method.</p>
<p>Here&#8217;s how you would block someone using the apache/askapache way:</p>
<pre>
[Exploit Request] =&gt; ([BLOCKED]-AskApache)
</pre>
<p>This prevents the exploit from even reaching PHP, saving your computer a lot of CPU/memory and bandwdith, and obviously can&#8217;t exploit wordpress if php isn&#8217;t even loading.</p>
<p>Here&#8217;s how the php-software-based method blocks the same request:</p>
<pre>
[Exploit Request] =&gt; (AskApache) =&gt; (PHP) =&gt; (WordPress) =&gt; ([BLOCKED]-askapache-password-protect.php)
</pre>
<p>So the last bit of programming and research I&#8217;m doing at the moment is how to cause the askapache-password-protect plugin to execute as soon as possible, ideally it would execute before WordPress starts..   And I am still crazy swamped at work, this was the longest non-posting period of the blog to date!</p>
<p><a href="http://www.askapache.com/htaccess/password-protection-plugin-status.html">Password Protection Plugin Status</a> originally appeared on AskApache.</p>]]></content:encoded>
			<wfw:commentRss>http://www.askapache.com/htaccess/password-protection-plugin-status.html/feed/</wfw:commentRss>
		<slash:comments>1</slash:comments>
		</item>
		<item>
		<title>.htaccess Plugin Blocks Spam, Hackers, and Password Protects Blog</title>
		<link>http://www.askapache.com/htaccess/htaccess-plugin-blocks-spam-hackers-and-password-protects-blog.html</link>
		<comments>http://www.askapache.com/htaccess/htaccess-plugin-blocks-spam-hackers-and-password-protects-blog.html#comments</comments>
		<pubDate>Sat, 22 Nov 2008 14:18:12 +0000</pubDate>
		<dc:creator>AskApache</dc:creator>
				<category><![CDATA[Apache]]></category>
		<category><![CDATA[Cache]]></category>
		<category><![CDATA[Featured]]></category>
		<category><![CDATA[Hacking]]></category>
		<category><![CDATA[PHP]]></category>
		<category><![CDATA[SEO]]></category>
		<category><![CDATA[Security]]></category>
		<category><![CDATA[WordPress]]></category>
		<category><![CDATA[WordPress Plugins]]></category>
		<category><![CDATA[htaccess]]></category>
		<category><![CDATA[htaccess rewrites]]></category>
		<category><![CDATA[mod_rewrite]]></category>

		<guid isPermaLink="false">http://www.askapache.com/?p=1053</guid>
		<description><![CDATA[<p><a class="IFL" href="http://www.askapache.com/htaccess/htaccess-plugin-blocks-spam-hackers-and-password-protects-blog.html"><img src="/wp-content/uploads/2008/11/htaccess-plugin-2.png" alt=".htaccess security plugin 2" title=".htaccess security plugin 2" /></a><br /><br />Well what can I say, <strong>other than this is sooo DOPE</strong>!  Here is a list of the modules this plugin (version 4.7 unreleased) will automatically detect.  I compiled the list myself using every module included with any default Apache installation for ALL the versions listed below, 1.3 to 2.2+<br /><br />Want to know something else I'm including in this plugin?  For each and every module that is detected, this plugin can then detect ALL of the modules .htaccess Directives!  For instance, <code>RewriteRule, AccessFileName, AddHandler, etc..</code> are each a directive belonging to a module that is allowed to be used from within .htaccess files.<br /><br /><strong>Talk about sick.. these tricks have the diamond disease!</strong><br class="C" /></p>]]></description>
			<content:encoded><![CDATA[<p><a class="IFL" href="/wp-content/uploads/2008/11/htaccess-plugin-2.png"><img src="/wp-content/uploads/2008/11/htaccess-plugin-2.png" alt=".htaccess security plugin 2" title=".htaccess security plugin 2" /></a></p>
<p>Well what can I say, <strong>other than this is sooo DOPE</strong>!  Here is a <a href="#htaccess-module-list">list of the modules</a> this plugin (version 4.7 unreleased) will automatically detect.  I compiled the list myself using every module included with any default Apache installation for ALL the versions listed below, 1.3 to 2.2+</p>
<p>Want to know something else I&#8217;m including in this plugin?  For each and every module that is detected, this plugin can then detect ALL of the modules .htaccess Directives!  For instance, <code>RewriteRule, AccessFileName, AddHandler, etc..</code> are each a directive belonging to a module that is allowed to be used from within .htaccess files.</p>
<p><strong>Talk about sick.. these tricks have the diamond disease!</strong><br class="C" /></p>
<h2>Screenshot Unreleased 4.7</h2>
<p>I&#8217;ve been making a lot of progress as these screenshots illustrate, including the ability to detect 100% accurately the modules that are enabled on your server.  Big deal!  you might say&#8230; &#8220;How does knowing the modules help?&#8221;</p>
<p>Well it just so happens that in addition to detecting which modules are loaded on your server, this plugin will also detect which Directives are enabled for each module that are allowed to be used from within your .htaccess file!  Future release will provide the ability to explore the different .htaccess directives allowed by your server, so you can do all sorts of cool Apache .htaccess tricks to secure your blog and make it run better.</p>
<p><a href="/wp-content/uploads/2008/11/htaccess-plugin-1.png"><img src="/wp-content/uploads/2008/11/htaccess-plugin-1.png" alt=".htaccess security plugin 1" title=".htaccess security plugin 1" /></a><br /><a href="/wp-content/uploads/2008/11/htaccess-plugin-3.png"><img src="/wp-content/uploads/2008/11/htaccess-plugin-3.png" alt=".htaccess security plugin 3" title=".htaccess security plugin 3" /></a><br /><a href="/wp-content/uploads/2008/11/htaccess-plugin-4.png"><img src="/wp-content/uploads/2008/11/htaccess-plugin-4.png" alt=".htaccess security plugin 4" title=".htaccess security plugin 4" /></a></p>
<h2><a id="htaccess-module-list">Apache Module Detection</a></h2>
<p>Future releases of this plugin will also let you search for non-default modules, wild, beta, and others.</p>
<ul>
<li>mod_access</li>
<li>mod_actions</li>
<li>mod_alias</li>
<li>mod_asis</li>
<li>mod_auth</li>
<li>mod_auth_anon</li>
<li>mod_auth_basic</li>
<li>mod_auth_dbm</li>
<li>mod_auth_digest</li>
<li>mod_auth_ldap</li>
<li>mod_authn_alias</li>
<li>mod_authn_anon</li>
<li>mod_authn_dbd</li>
<li>mod_authn_dbm</li>
<li>mod_authn_default</li>
<li>mod_authn_file</li>
<li>mod_authnz_ldap</li>
<li>mod_authz_dbm</li>
<li>mod_authz_default</li>
<li>mod_authz_groupfile</li>
<li>mod_authz_host</li>
<li>mod_authz_owner</li>
<li>mod_authz_user</li>
<li>mod_autoindex</li>
<li>mod_bucketeer</li>
<li>mod_cache</li>
<li>mod_case_filter</li>
<li>mod_case_filter_in</li>
<li>mod_cern_meta</li>
<li>mod_cgi</li>
<li>mod_cgid</li>
<li>mod_charset_lite</li>
<li>mod_dav</li>
<li>mod_dav_fs</li>
<li>mod_dav_lock</li>
<li>mod_dbd</li>
<li>mod_deflate</li>
<li>mod_dir</li>
<li>mod_disk_cache</li>
<li>mod_dumpio</li>
<li>mod_echo</li>
<li>mod_env</li>
<li>mod_example</li>
<li>mod_expires</li>
<li>mod_ext_filter</li>
<li>mod_file_cache</li>
<li>mod_filter</li>
<li>mod_headers</li>
<li>mod_ident</li>
<li>mod_imagemap</li>
<li>mod_imap</li>
<li>mod_include</li>
<li>mod_info</li>
<li>mod_isapi</li>
<li>mod_log_config</li>
<li>mod_log_forensic</li>
<li>mod_logio</li>
<li>mod_mem_cache</li>
<li>mod_mime</li>
<li>mod_mime_magic</li>
<li>mod_mycore</li>
<li>mod_negotiation</li>
<li>mod_netware</li>
<li>mod_nw_ssl</li>
<li>mod_optional_fn_export</li>
<li>mod_optional_fn_import</li>
<li>mod_optional_hook_export</li>
<li>mod_optional_hook_import</li>
<li>mod_proxy</li>
<li>mod_proxy_ajp</li>
<li>mod_proxy_balancer</li>
<li>mod_proxy_connect</li>
<li>mod_proxy_ftp</li>
<li>mod_proxy_http</li>
<li>mod_rewrite</li>
<li>mod_security</li>
<li>mod_setenvif</li>
<li>mod_so</li>
<li>mod_speling</li>
<li>mod_ssl</li>
<li>mod_status</li>
<li>mod_substitute</li>
<li>mod_suexec</li>
<li>mod_test</li>
<li>mod_unique_id</li>
<li>mod_userdir</li>
<li>mod_usertrack</li>
<li>mod_version</li>
<li>mod_vhost_alias</li>
<li>mod_win32</li>
</ul>
<p><a rel="lb" class="IFL hs hs35" href='/wp-content/uploads/2008/07/http-security-askapache.png' title="http-security-askapache"></a>The original plugin page and description <a href="http://www.askapache.com/wordpress/htaccess-password-protect.html">can be found here</a>.<br class="C" /></p>
<h2>UPDATE: 11/22/08</h2>
<p><a rel="lb" class="IFL hs hs35" href='/wp-content/uploads/2008/07/http-security-askapache.png' title="http-security-askapache"></a><strong>To make a long story short,</strong> I downloaded each major release of the apache httpd source code from version 1.3.0 to version 2.2.10, then I configured and compiled each for a custom HTTPD installation built from source.  This allowed me to find every directive allowed in .htaccess files for each particular version.  <strong style="font-weight:bold;">YES!</strong><br class="C" /></p>
<blockquote cite="http://wordpress.org/support/topic/214390"><p><cite><a href="http://wordpress.org/support/rss/topic/214390">http://wordpress.org/support/rss/topic/214390</a></cite><br />
I&#8217;ve been working on a completely improved version on/off for about a month with the specific goal of finally ending all the little errors that can crop up when dealing with .htaccess.</p>
<p>To that effect I am succeeding marvelously, first I&#8217;ve converted the plugin to a class (4+5 compat), I&#8217;ve replaced my error_handling with WordPress&#8217;s WP_Error class, and the coolest change is the new tests I&#8217;ve added.</p>
<p>To make a long story short, I downloaded each major release of the apache httpd source code starting at version 1.3.0 and finishing with version 2.2.10, I then compiled each version and built a HTTPD from source for all the apache versions.  </p>
<p><em><code>1.3.0</code>, <code>1.3.1</code>, <code>1.3.11</code>, <code>1.3.12</code>, <code>1.3.14</code>, <code>1.3.17</code>, <code>1.3.19</code>, <code>1.3.2</code>, <code>1.3.20</code>, <code>1.3.22</code>, <code>1.3.23</code>, <code>1.3.24</code>, <code>1.3.27</code>, <code>1.3.28</code>, <code>1.3.29</code>, <code>1.3.3</code>, <code>1.3.31</code>, <code>1.3.32</code>, <code>1.3.33</code>, <code>1.3.34</code>, <code>1.3.35</code>, <code>1.3.36</code>, <code>1.3.37</code>, <code>1.3.39</code>, <code>1.3.4</code>, <code>1.3.41</code>, <code>1.3.6</code>, <code>1.3.9</code>, <code>2.0.35</code>, <code>2.0.36</code>, <code>2.0.39</code>, <code>2.0.40</code>, <code>2.0.42</code>, <code>2.0.43</code>, <code>2.0.44</code>, <code>2.0.45</code>, <code>2.0.46</code>, <code>2.0.47</code>, <code>2.0.48</code>, <code>2.0.49</code>, <code>2.0.50</code>, <code>2.0.51</code>, <code>2.0.52</code>, <code>2.0.53</code>, <code>2.0.54</code>, <code>2.0.55</code>, <code>2.0.58</code>, <code>2.0.59</code>, <code>2.0.61</code>, <code>2.0.63</code>, <code>2.1.3-beta</code>, <code>2.1.6-alpha</code>, <code>2.1.7-beta</code>, <code>2.1.8-beta</code>, <code>2.1.9-beta</code>, <code>2.2.0</code>, <code>2.2.10</code>, <code>2.2.2</code>, <code>2.2.3</code>, <code>2.2.4</code>, <code>2.2.6</code>, <code>2.2.8</code>, <code>2.2.9</code></em></p>
<p>Then I went through each version and determined the compatible modules for that version, and I&#8217;m pretty confident that I was also able to find each and every directive allowed by the compatible modules for that version (including core directives).  See <a href="http://www.askapache.com/htaccess/htaccess.html#htaccess-directives">.htaccess directive list</a>.</p>
<p>Basically I can now test a server using a variety of methods and determine almost 100% accurately what version of Apache (down to the API) is running, what modules (and versions) are enabled, and each and every directive that is allowed or disallowed for that version.</p>
<p>So this is so awesome because now we can enable all sorts of additional security features.</p>
<p>Other big changes are:</p>
<ul>
<li>Completely hands-off updates, so that updating the plugin keeps all your settings.</li>
<li>making each SID module have its own configuration and options (like protecting individual files, individual request, and custom exploit strings).</li>
<li>Advanced ErrorDocument usage and handling (like tracking repeat offenders and suggesting they be blocked, emailing admin with custom info, etc..)</li>
<li>Multi User/Group password Control</li>
</ul>
<p>And this time I am developing the plugin using a plethora of wordpress installations and configurations, to make sure that it will work regardless of a custom siteurl, blogid, etc..</p>
<p><strong>Release will come before 2009.. I have some vacations to take and business to finish first. </strong>
</p></blockquote>
<h2>.htaccess Security Modules</h2>
<h3><a id="htaccess-sid700" title="Directory Protection">Directory Protection</a></h3>
<p>Enable the DirectoryIndex Protection, preventing directory index listings and defaulting. [<a href="http://www.askapache.com/htaccess/htaccess.html">Disable</a>]</p>
<pre class='brushhtaccess'>
Options -Indexes
DirectoryIndex index.html index.php /index.php
</pre>
<h3><a id="htaccess-sid800" title="Password Protect wp-login.php">Password Protect wp-login.php</a></h3>
<p>Requires a valid user/pass to access the login page <strong>- *** Safe, Use</strong> [<a href="http://www.askapache.com/htaccess/apache-status-code-headers-errordocument.html#status-401">401</a>]</p>
<pre class='brushhtaccess'>
&lt;Files wp-login.php&gt;
Order Deny,Allow
Deny from All
Satisfy Any
&nbsp;
AuthName "Protected By AskApache"
AuthUserFile /home/askapache.com/.htpasswda1
AuthType Basic
Require valid-user
&lt;/Files&gt;
</pre>
<h3><a id="htaccess-sid900" title="Password Protect wp-admin">Password Protect wp-admin</a></h3>
<p>Requires a valid user/pass to access any non-static (css, js, images) file in this directory. <strong>- *** Safe, Use</strong> [<a href="http://www.askapache.com/htaccess/apache-status-code-headers-errordocument.html#status-401">401</a>]</p>
<pre class='brushhtaccess'>
Options -ExecCGI -Indexes +FollowSymLinks -Includes
DirectoryIndex index.php /index.php
&nbsp;
Order Deny,Allow
&nbsp;
Deny from All
Satisfy Any
&nbsp;
AuthName "Protected By AskApache"
AuthUserFile /home/askapache.com/.htpasswda1
AuthType Basic
Require valid-user
&nbsp;
&lt;FilesMatch "\.(ico|pdf|flv|jpg|jpeg|mp3|mpg|mp4|mov|wav|wmv|png|gif|swf|css|js)$"&gt;
Allow from All
&lt;/FilesMatch&gt;
&nbsp;
&lt;FilesMatch "(async-upload)\.php$"&gt;
&lt;IfModule mod_security.c&gt;
SecFilterEngine Off
&lt;/IfModule&gt;
Allow from All
&lt;/FilesMatch&gt;
</pre>
<h3><a id="htaccess-sid1000" title="Protect wp-content">Protect wp-content</a></h3>
<p>Denies any Direct request for files ending in .php with a 403 Forbidden.. May break plugins/themes [<a href="http://www.askapache.com/htaccess/apache-status-code-headers-errordocument.html#status-401">401</a>]</p>
<pre class='brushhtaccess'>
RewriteCond %{THE_REQUEST} ^[A-Z]{3,9}\ /wp-content/.*$ [NC]
RewriteCond %{REQUEST_FILENAME} !^.+flexible-upload-wp25js.php$
RewriteCond %{REQUEST_FILENAME} ^.+\.(php|html|htm|txt)$
RewriteRule .* - [F,NS,L]
</pre>
<h3><a id="htaccess-sid1010" title="Protect wp-includes">Protect wp-includes</a></h3>
<p>Denies any Direct request for files ending in .php with a 403 Forbidden.. May break plugins/themes [<a href="http://www.askapache.com/htaccess/apache-status-code-headers-errordocument.html#status-403">403</a>]</p>
<pre class='brushhtaccess'>
RewriteCond %{THE_REQUEST} ^[A-Z]{3,9}\ /wp-includes/.*$ [NC]
RewriteCond %{THE_REQUEST} !^[A-Z]{3,9}\ /wp-includes/js/.+/.+\ HTTP/ [NC]
RewriteCond %{REQUEST_FILENAME} ^.+\.php$
RewriteRule .* - [F,NS,L]
</pre>
<h3><a id="htaccess-sid1011" title="Common Exploits">Common Exploits</a></h3>
<p>Block common exploit requests with 403 Forbidden. These can help alot, may break some plugins. [<a href="http://www.askapache.com/htaccess/apache-status-code-headers-errordocument.html#status-403">403</a>]</p>
<pre class='brushhtaccess'>
RewriteCond %{REQUEST_URI} !^/(wp-login.php|wp-admin/|wp-content/plugins/|wp-includes/).* [NC]
RewriteCond %{THE_REQUEST} ^[A-Z]{3,9}\ ///.*\ HTTP/ [NC,OR]
RewriteCond %{THE_REQUEST} ^[A-Z]{3,9}\ /.*\?\=?(http|ftp|ssl|https):/.*\ HTTP/ [NC,OR]
RewriteCond %{THE_REQUEST} ^[A-Z]{3,9}\ /.*\?\?.*\ HTTP/ [NC,OR]
RewriteCond %{THE_REQUEST} ^[A-Z]{3,9}\ /.*\.(asp|ini|dll).*\ HTTP/ [NC,OR]
RewriteCond %{THE_REQUEST} ^[A-Z]{3,9}\ /.*\.(htpasswd|htaccess|aahtpasswd).*\ HTTP/ [NC]
RewriteRule .* - [F,NS,L]
</pre>
<h3><a id="htaccess-sid1012" title="Stop Hotlinking">Stop Hotlinking</a></h3>
<p>Denies any request for static files (images, css, etc) if referrer is not local site or empty. [<a href="http://www.askapache.com/htaccess/apache-status-code-headers-errordocument.html#status-403">403</a>]</p>
<pre class='brushhtaccess'>
RewriteCond %{HTTP_REFERER} !^$
RewriteCond %{REQUEST_URI} !^/(wp-login.php|wp-admin/|wp-content/plugins/|wp-includes/).* [NC]
RewriteCond %{HTTP_REFERER} !^http://www.askapache.com.*$ [NC]
RewriteRule \.(ico|pdf|flv|jpg|jpeg|mp3|mpg|mp4|mov|wav|wmv|png|gif|swf|css|js)$ - [F,NS,L]
</pre>
<h3><a id="htaccess-sid1015" title="Safe Request Methods">Safe Request Methods</a></h3>
<p>Denies any request not using <a href="http://www.askapache.com/online-tools/request-method-scanner/">GET,PROPFIND,POST,OPTIONS,PUT,HEAD</a> <strong>- *** Safe, Use</strong> [<a href="http://www.askapache.com/htaccess/apache-status-code-headers-errordocument.html#status-403">403</a>]</p>
<pre class='brushhtaccess'>
RewriteCond %{REQUEST_METHOD} !^(GET|HEAD|POST|PROPFIND|OPTIONS|PUT)$ [NC]
RewriteRule .* - [F,NS,L]
</pre>
<h3><a id="htaccess-sid1017" title="Forbid Proxies">Forbid Proxies</a></h3>
<p>Denies any POST Request using a Proxy Server. Can still access site, but not comment.  See <a href="http://perishablepress.com/press/2008/04/20/how-to-block-proxy-servers-via-htaccess/">Perishable Press</a> [<a href="http://www.askapache.com/htaccess/apache-status-code-headers-errordocument.html#status-403">403</a>]</p>
<pre class='brushhtaccess'>
RewriteCond %{REQUEST_METHOD} =POST
RewriteCond %{HTTP:VIA}%{HTTP:FORWARDED}%{HTTP:USERAGENT_VIA}%{HTTP:X_FORWARDED_FOR}%{HTTP:PROXY_CONNECTION} !^$ [OR]
RewriteCond %{HTTP:XPROXY_CONNECTION}%{HTTP:HTTP_PC_REMOTE_ADDR}%{HTTP:HTTP_CLIENT_IP} !^$
RewriteCond %{REQUEST_URI} !^/(wp-login.php|wp-admin/|wp-content/plugins/|wp-includes/).* [NC]
RewriteRule .* - [F,NS,L]
</pre>
<h3><a id="htaccess-sid1018" title="Real wp-comments-post.php">Real wp-comments-post.php</a></h3>
<p>Denies any POST attempt made to a non-existing wp-comments-post.php <strong>- *** Safe, Use</strong> [<a href="http://www.askapache.com/htaccess/apache-status-code-headers-errordocument.html#status-403">403</a>]</p>
<pre class='brushhtaccess'>
RewriteCond %{THE_REQUEST} ^[A-Z]{3,9}\ /.*/wp-comments-post\.php.*\ HTTP/ [NC]
RewriteRule .* - [F,NS,L]
</pre>
<h3><a id="htaccess-sid1019" title="HTTP PROTOCOL">HTTP PROTOCOL</a></h3>
<p>Denies any badly formed HTTP PROTOCOL in the request, 0.9, 1.0, and 1.1 only  <strong>- *** Safe, Use</strong> [<a href="http://www.askapache.com/htaccess/apache-status-code-headers-errordocument.html#status-403">403</a>]</p>
<pre class='brushhtaccess'>
RewriteCond %{THE_REQUEST} !^[A-Z]{3,9}\ .+\ HTTP/(0\.9|1\.0|1\.1) [NC]
RewriteRule .* - [F,NS,L]
</pre>
<h3><a id="htaccess-sid1020" title="SPECIFY CHARACTERS">SPECIFY CHARACTERS</a></h3>
<p>Denies any request for a url containing characters other than &#8220;a-zA-Z0-9.+/-?=&#038;&#8221;  &#8211; REALLY helps but may break your site depending on your links. [<a href="http://www.askapache.com/htaccess/apache-status-code-headers-errordocument.html#status-403">403</a>]</p>
<pre class='brushhtaccess'>
RewriteCond %{REQUEST_URI} !^/(wp-login.php|wp-admin/|wp-content/plugins/|wp-includes/).* [NC]
RewriteCond %{THE_REQUEST} !^[A-Z]{3,9}\ [a-zA-Z0-9\.\+_/\-\?\=\&#038;]+\ HTTP/ [NC]
RewriteRule .* - [F,NS,L]
</pre>
<h3><a id="htaccess-sid1021" title="BAD Content Length">BAD Content Length</a></h3>
<p>Denies any POST request that doesnt have a Content-Length Header <strong>- *** Safe, Use</strong> [<a href="http://www.askapache.com/htaccess/apache-status-code-headers-errordocument.html#status-403">403</a>]</p>
<pre class='brushhtaccess'>
RewriteCond %{REQUEST_METHOD} =POST
RewriteCond %{HTTP:Content-Length} ^$
RewriteCond %{REQUEST_URI} !^/(wp-admin/|wp-content/plugins/|wp-includes/).* [NC]
RewriteRule .* - [F,NS,L]
</pre>
<h3><a id="htaccess-sid1022" title="BAD Content Type">BAD Content Type</a></h3>
<p>Denies any POST request with a content type other than application/x-www-form-urlencoded|multipart/form-data <strong>- *** Safe, Use</strong> [<a href="http://www.askapache.com/htaccess/apache-status-code-headers-errordocument.html#status-403">403</a>]</p>
<pre class='brushhtaccess'>
RewriteCond %{REQUEST_METHOD} =POST
RewriteCond %{HTTP:Content-Type} !^(application/x-www-form-urlencoded|multipart/form-data.*(boundary.*)?)$ [NC]
RewriteCond %{REQUEST_URI} !^/(wp-login.php|wp-admin/|wp-content/plugins/|wp-includes/).* [NC]
RewriteRule .* - [F,NS,L]
</pre>
<h3><a id="htaccess-sid1023" title="Directory Traversal">Directory Traversal</a></h3>
<p>Denies Requests containing ../ or ./. which is a directory traversal exploit attempt <strong>- *** Safe, Use</strong> [<a href="http://www.askapache.com/htaccess/apache-status-code-headers-errordocument.html#status-403">403</a>]</p>
<h3><a id="htaccess-sid1024" title="PHPSESSID Cookie">PHPSESSID Cookie</a></h3>
<p>Only blocks when a PHPSESSID cookie is sent by the user and it contains characters other than 0-9a-z <strong>- *** Safe, Use</strong> [<a href="http://www.askapache.com/htaccess/apache-status-code-headers-errordocument.html#status-403">403</a>]</p>
<h3><a id="htaccess-sid1025" title="NO HOST:">NO HOST:</a></h3>
<p>Denies requests that dont contain a HTTP HOST Header. <strong>- *** Safe, Use</strong> [<a href="http://www.askapache.com/htaccess/apache-status-code-headers-errordocument.html#status-403">403</a>]</p>
<pre class='brushhtaccess'>
RewriteCond %{REQUEST_URI} !^/(wp-login.php|wp-admin/|wp-content/plugins/|wp-includes/).* [NC]
RewriteCond %{HTTP_HOST} ^$
RewriteRule .* - [F,NS,L]
</pre>
<h3><a id="htaccess-sid1026" title="Bogus Graphics Exploit">Bogus Graphics Exploit</a></h3>
<p>Denies obvious exploit using bogus graphics  <strong>- *** Safe, Use</strong> [<a href="http://www.askapache.com/htaccess/apache-status-code-headers-errordocument.html#status-403">403</a>]</p>
<pre class='brushhtaccess'>
RewriteCond %{HTTP:Content-Disposition} \.php [NC]
RewriteCond %{HTTP:Content-Type} image/.+ [NC]
RewriteRule .* - [F,NS,L]
</pre>
<h3><a id="htaccess-sid1027" title="No UserAgent, No Post">No UserAgent, No Post</a></h3>
<p>Denies POST requests by blank user-agents.  May prevent a small number of visitors from POSTING. [<a href="http://www.askapache.com/htaccess/apache-status-code-headers-errordocument.html#status-403">403</a>]</p>
<pre class='brushhtaccess'>
RewriteCond %{REQUEST_METHOD} =POST
RewriteCond %{HTTP_USER_AGENT} ^-?$
RewriteCond %{REQUEST_URI} !^/(wp-login.php|wp-admin/|wp-content/plugins/|wp-includes/).* [NC]
RewriteRule .* - [F,NS,L]
</pre>
<h3><a id="htaccess-sid1028" title="No Referer, No Comment">No Referer, No Comment</a></h3>
<p>Denies any comment attempt with a blank HTTP_REFERER field, highly indicative of spam.  May prevent some visitors from POSTING. [<a href="http://www.askapache.com/htaccess/apache-status-code-headers-errordocument.html#status-403">403</a>]</p>
<pre class='brushhtaccess'>
RewriteCond %{THE_REQUEST} ^[A-Z]{3,9}\ /.*/wp-comments-post\.php.*\ HTTP/ [NC]
RewriteCond %{HTTP_REFERER} ^-?$
RewriteRule .* - [F,NS,L]
</pre>
<h3><a id="htaccess-sid1029" title="Trackback Spam">Trackback Spam</a></h3>
<p>Denies obvious trackback spam.   See <a href="http://ocaoimh.ie/2008/07/03/more-ways-to-stop-spammers-and-unwanted-traffic/">Holy Shmoly!</a> [<a href="http://www.askapache.com/htaccess/apache-status-code-headers-errordocument.html#status-403">403</a>]</p>
<pre class='brushhtaccess'>
RewriteCond %{REQUEST_METHOD} =POST
RewriteCond %{HTTP_USER_AGENT} ^.*(opera|mozilla|firefox|msie|safari).*$ [NC]
RewriteCond %{THE_REQUEST} ^[A-Z]{3,9}\ /.+/trackback/?\ HTTP/ [NC]
RewriteRule .* - [F,NS,L]
</pre>
<h3><a id="htaccess-sid1030" title="SSL-Only Site">SSL-Only Site</a></h3>
<p>Redirects all non-SSL (https) requests to your https-enabled url [<a href="http://www.askapache.com/htaccess/apache-status-code-headers-errordocument.html#status-301">301</a>]</p>
<h3><a id="htaccess-sid2000" title="Anti-Spam, Anti-Exploits">Anti-Spam, Anti-Exploits</a></h3>
<p>Denies Obvious Spam and uses advanced mod_security protection [<a href="http://www.askapache.com/htaccess/mod_security-htaccess-tricks.html">Read More</a>]</p>
<h2>.htaccess Security Module Screenshot</h2>
<p><a rel="lb" href='/wp-content/uploads/2008/07/http-security-askapache.png'><img src="/wp-content/uploads/2008/07/http-security-askapache1.png" alt="" title="http-security-askapache1" /></a></p>
<p><a href="http://www.askapache.com/htaccess/htaccess-plugin-blocks-spam-hackers-and-password-protects-blog.html">.htaccess Plugin Blocks Spam, Hackers, and Password Protects Blog</a> originally appeared on AskApache.</p>]]></content:encoded>
			<wfw:commentRss>http://www.askapache.com/htaccess/htaccess-plugin-blocks-spam-hackers-and-password-protects-blog.html/feed/</wfw:commentRss>
		<slash:comments>37</slash:comments>
		</item>
		<item>
		<title>Chmod, Umask, Stat, Fileperms, and File Permissions</title>
		<link>http://www.askapache.com/security/chmod-stat.html</link>
		<comments>http://www.askapache.com/security/chmod-stat.html#comments</comments>
		<pubDate>Wed, 19 Nov 2008 10:16:56 +0000</pubDate>
		<dc:creator>AskApache</dc:creator>
				<category><![CDATA[Apache]]></category>
		<category><![CDATA[Featured]]></category>
		<category><![CDATA[Hacking]]></category>
		<category><![CDATA[Linux Unix BSD]]></category>
		<category><![CDATA[PHP]]></category>
		<category><![CDATA[Security]]></category>
		<category><![CDATA[Server Administration]]></category>
		<category><![CDATA[Shell Scripting]]></category>
		<category><![CDATA[Web Hosting]]></category>
		<category><![CDATA[Webmaster]]></category>
		<category><![CDATA[chmod]]></category>
		<category><![CDATA[File Permissions]]></category>
		<category><![CDATA[fileperms]]></category>
		<category><![CDATA[stat]]></category>
		<category><![CDATA[umask]]></category>

		<guid isPermaLink="false">http://www.askapache.com/?p=1800</guid>
		<description><![CDATA[<p><a class="IFL" href="http://www.askapache.com/security/chmod-umask-fileperms-stat-tricks.html"><img src="/wp-content/uploads/2008/11/danger-fire-250x150.jpg" alt="that fire.. chmod, umask, and file permission heat" title="ITs that fire.. chmod, umask, and file permission heat" width="250" height="150" /></a>Unix file permissions are one of the more difficult subjects to grasp.. Well, ok maybe "grasp" isn't the word.. Master is the right word.. Unix file permissions is a hard topic to fully master, mainly I think because there aren't many instances when a computer user encounters them.  I've done a lot of research on it the past couple weeks...  and now here's everything I've learned so far.. cuz you guys <em>AskApache Regs</em> Rock! <br class="C" /></p>]]></description>
			<content:encoded><![CDATA[<p><a class="IFL" href="/wp-content/uploads/2008/11/danger-fire.jpg"><img src="/wp-content/uploads/2008/11/danger-fire-250x150.jpg" alt="that fire.. chmod, umask, and file permission heat" title="ITs that fire.. chmod, umask, and file permission heat" width="250" height="150" /></a>Unix file permissions are one of the more difficult subjects to grasp.. Well, ok maybe &#8220;grasp&#8221; isn&#8217;t the word.. Master is the right word.. Unix file permissions is a hard topic to fully master, mainly I think because there aren&#8217;t many instances when a computer user encounters them.   Windows has been trying to figure it out for decades with little progress, so don&#8217;t feel bad if you don&#8217;t know much about it.  <strong>Unless you&#8217;re with the program</strong> and running Mac or any other <a href="http://www.archlinux.org/">BSD/Unix</a> based OS you&#8217;ve never had the ability to secure your system in this most basic and fundamental way.  Usually the first time someone encounters file permissions it&#8217;s because their website was cracked.. <br class="C" /></p>
<h3>.htaccess</h3>
<p><kbd>$ chmod 604 .htaccess</kbd></p>
<pre class='brushbash'>
604 -rw&#45;&#45;&#45;-r&#45;-  /home/askapache/cgi-bin/.htaccess
</pre>
<h3>php.cgi</h3>
<p><kbd>$ chmod 711 php.cgi</kbd></p>
<pre class='brushbash'>
$ 711 -rwx&#45;-x&#45;-x  /home/askapache/cgi-bin/php.cgi
</pre>
<h3>.php.ini</h3>
<p><kbd>$ chmod 600 php.ini</kbd></p>
<pre class='brushbash'>
$ 600 -rw&#45;&#45;-&#45;&#45;&#45;-  /home/askapache/cgi-bin/php.ini
</pre>
<p>I&#8217;m in the process of developing an updated version of the .htaccess security plugin, and one thing I have been working on is file permissions.  Some people had problems trying to create files on their server and I realized it was bad programming on my part..  so I began researching permissions in detail. I went deep into the source code of Apache (<em>which is why this site is called AskApache, BTW</em>), PHP, Python, Ocaml, Perl, Ruby, and POSIX operating systems and got a pretty good handle on it now..</p>
<h2>Tips before we dig in</h2>
<p>Here&#8217;s a few things I&#8217;ve learned that I didn&#8217;t know before (using php).</p>
<h3>Deleting Files and Directories</h3>
<p>Deleting a file may require chmodding the file to 666 or even 777 before you are able to delete it.  You also might have to chmod the parent directory of the file as well.  Also, you may have to chdir to the directory the file is in.  And lastly you may have to change the owner or group of the file.  Further than that you can try renaming the file first then deleting it..</p>
<p>Deleting a directory means you need to remove every file in it first.  It needs to be empty.  And if your file system uses NFS or some other networked FS you might have even more problems deleting files.  If the file you are trying to delete is being used by say, Apache or php then you might have to kill that process first.</p>
<h3>Creating Files in Restrictive Environments</h3>
<p>My research has been geared to try and make my code as robust as possible, I&#8217;m throwing everything but the kitchen sink into some of these functions because so many people are on such different types of servers.  To create a file in a restrictive environment is a fun excercise to take.. You can write a file using many different functions, but there are some tricks if they all fail.  One trick is instead of trying to &#8220;write&#8221; the data to the file, you can UPLOAD the data to the server and let PHP handle the file as if you used an upload form.  I like to use fsockopen to do it, as some installations have been setup to prevent this type of fake upload.</p>
<p>Then there are the various other hacks like using an ftp connection (if you know the user/pass) to send the file from php, using ssh from php, whatever is available on the hosts php installation.  In addition to those more involved workarounds you can often get around this problem by doing little hacks discussed at php.net in the comments for various functions.  Such as changing the umask, changing directories with chdir first, creating a temporary file using a function like tempfile and then renaming or copying the tempfile to your desired file which sometimes gives you the permissions needed to write to the location.</p>
<p>If the php installation is newer than you can also look into creating your own stream context to pass write the data direct.</p>
<h2>Stat Function</h2>
<p>I&#8217;ve created a stat function in php that goes farther than the normal stat function&#8230; Just give the function a file to stat, and it returns an array of information.  </p>
<p>function askapache_stat($filename) {<br />
 clearstatcache();<br />
 $ss=@stat($filename);<br />
 if(!$ss) die(&#8220;Couldnt stat {$filename}&#8221;);<br />
 $file_convert=array(0140000=>&#8217;ssocket&#8217;,0120000=>&#8217;llink&#8217;,0100000=>&#8217;-file&#8217;,0060000=>&#8217;bblock&#8217;,0040000=>&#8217;ddir&#8217;,0020000=>&#8217;cchar&#8217;,0010000=>&#8217;pfifo&#8217;);<br />
 $p=$ss['mode'];<br />
 $t=decoct($ss['mode'] &#038; 0170000);<br />
 $str = (array_key_exists(octdec($t),$file_convert)) ? $file_convert[octdec($t)]{0} : &#8216;u&#8217;;<br />
 $str.=(($p&#038;0&#215;0100)?&#8217;r':&#8217;-').(($p&#038;0&#215;0080)?&#8217;w':&#8217;-').(($p&#038;0&#215;0040)?(($p&#038;0&#215;0800)?&#8217;s&#8217;:'x&#8217;):(($p&#038;0&#215;0800)?&#8217;S':&#8217;-'));<br />
 $str.=(($p&#038;0&#215;0020)?&#8217;r':&#8217;-').(($p&#038;0&#215;0010)?&#8217;w':&#8217;-').(($p&#038;0&#215;0008)?(($p&#038;0&#215;0400)?&#8217;s&#8217;:'x&#8217;):(($p&#038;0&#215;0400)?&#8217;S':&#8217;-'));<br />
 $str.=(($p&#038;0&#215;0004)?&#8217;r':&#8217;-').(($p&#038;0&#215;0002)?&#8217;w':&#8217;-').(($p&#038;0&#215;0001)?(($p&#038;0&#215;0200)?&#8217;t':&#8217;x'):(($p&#038;0&#215;0200)?&#8217;T':&#8217;-'));</p>
<p> $s=array(<br />
 &#8216;perms&#8217;=>array(<br />
  &#8216;umask&#8217;=>sprintf(&#8220;%04o&#8221;,umask()),<br />
  &#8216;human&#8217;=>$str,<br />
  &#8216;octal1&#8242;=>sprintf(&#8220;%o&#8221;, ($ss['mode'] &#038; 000777)),<br />
  &#8216;octal2&#8242;=>sprintf(&#8220;0%o&#8221;, 0777 &#038; $p),<br />
  &#8216;decimal&#8217;=>sprintf(&#8220;%04o&#8221;, $p),<br />
  &#8216;fileperms&#8217;=>@fileperms($filename),<br />
  &#8216;mode1&#8242;=>$p,<br />
  &#8216;mode2&#8242;=>$ss['mode']),</p>
<p> &#8216;filetype&#8217;=>array(<br />
  &#8216;type&#8217;=>substr($file_convert[octdec($t)],1),<br />
  &#8216;type_octal&#8217;=>sprintf(&#8220;%07o&#8221;, octdec($t)),<br />
  &#8216;is_file&#8217;=>@is_file($filename),<br />
  &#8216;is_dir&#8217;=>@is_dir($filename),<br />
  &#8216;is_link&#8217;=>@is_link($filename),<br />
  &#8216;is_readable&#8217;=> @is_readable($filename),<br />
  &#8216;is_writable&#8217;=> @is_writable($filename)),</p>
<p> &#8216;owner&#8217;=>array(<br />
  &#8216;fileowner&#8217;=>$ss['uid'],<br />
  &#8216;filegroup&#8217;=>$ss['gid'],<br />
  &#8216;owner_name&#8217;=>(function_exists(&#8216;posix_getpwuid&#8217;)) ? @reset(@posix_getpwuid($ss['uid'])) : &#8221;,<br />
  &#8216;group_name&#8217;=>(function_exists(&#8216;posix_getgrgid&#8217;)) ? @reset(@posix_getgrgid($ss['gid'])) : &#8221;),</p>
<p> &#8216;file&#8217;=>array(<br />
  &#8216;filename&#8217;=>$filename,<br />
  &#8216;realpath&#8217;=>(@realpath($filename) != $filename) ? @realpath($filename) : &#8221;,<br />
  &#8216;dirname&#8217;=>@dirname($filename),<br />
  &#8216;basename&#8217;=>@basename($filename)),</p>
<p> &#8216;device&#8217;=>array(<br />
  &#8216;device&#8217;=>$ss['dev'], //Device<br />
  &#8216;device_number&#8217;=>$ss['rdev'], //Device number, if device.<br />
  &#8216;inode&#8217;=>$ss['ino'], //File serial number<br />
  &#8216;link_count&#8217;=>$ss['nlink'], //link count<br />
  &#8216;link_to&#8217;=>($s['type']==&#8217;link&#8217;) ? @readlink($filename) : &#8221;),</p>
<p> &#8217;size&#8217;=>array(<br />
  &#8217;size&#8217;=>$ss['size'], //Size of file, in bytes.<br />
  &#8216;blocks&#8217;=>$ss['blocks'], //Number 512-byte blocks allocated<br />
  &#8216;block_size&#8217;=> $ss['blksize']), //Optimal block size for I/O.</p>
<p> &#8216;time&#8217;=>array(<br />
  &#8216;mtime&#8217;=>$ss['mtime'], //Time of last modification<br />
  &#8216;atime&#8217;=>$ss['atime'], //Time of last access.<br />
  &#8216;ctime&#8217;=>$ss['ctime'], //Time of last status change<br />
  &#8216;accessed&#8217;=>@date(&#8216;Y M D H:i:s&#8217;,$ss['atime']),<br />
  &#8216;modified&#8217;=>@date(&#8216;Y M D H:i:s&#8217;,$ss['mtime']),<br />
  &#8216;created&#8217;=>@date(&#8216;Y M D H:i:s&#8217;,$ss['ctime'])),<br />
 );</p>
<p> clearstatcache();<br />
 return $s;<br />
}
</pre>
<h3>PHP Stat Function Output</h2>
<p>Example output, say from <code>print_r(askapache_stat( __FILE__ ) );</code></p>
<p>Array(<br />
[perms] => Array<br />
  (<br />
  [umask] => 0022<br />
  [human] => -rw-r--r--<br />
  [octal1] => 644<br />
  [octal2] => 0644<br />
  [decimal] => 100644<br />
  [fileperms] => 33188<br />
  [mode1] => 33188<br />
  [mode2] => 33188<br />
  )</p>
<p>[filetype] => Array<br />
  (<br />
  [type] => file<br />
  [type_octal] => 0100000<br />
  [is_file] => 1<br />
  [is_dir] =><br />
  [is_link] =><br />
  [is_readable] => 1<br />
  [is_writable] => 1<br />
  )</p>
<p>[owner] => Array<br />
  (<br />
  [fileowner] => 035483<br />
  [filegroup] => 23472<br />
  [owner_name] => askapache<br />
  [group_name] => grp22558<br />
  )</p>
<p>[file] => Array<br />
  (<br />
  [filename] => /home/askapache/askapache-stat/htdocs/ok/g.php<br />
  [realpath] =><br />
  [dirname] => /home/askapache/askapache-stat/htdocs/ok<br />
  [basename] => g.php<br />
  )</p>
<p>[device] => Array<br />
  (<br />
  [device] => 25<br />
  [device_number] => 0<br />
  [inode] => 92455020<br />
  [link_count] => 1<br />
  [link_to] =><br />
  )</p>
<p>[size] => Array<br />
  (<br />
  [size] => 2652<br />
  [blocks] => 8<br />
  [block_size] => 8192<br />
  )</p>
<p>[time] => Array<br />
  (<br />
  [mtime] => 1227685253<br />
  [atime] => 1227685138<br />
  [ctime] => 1227685253<br />
  [accessed] => 2008 Nov Tue 23:38:58<br />
  [modified] => 2008 Nov Tue 23:40:53<br />
  [created] => 2008 Nov Tue 23:40:53<br />
  )<br />
)
</pre>
<h2><a id="chmod-0-to-7777"></a>Every Permission 0000 to 0777</h2>
<p><a class="IFL" href="/wp-content/uploads/2008/11/danger-chmod-screenshot.png"><img src="/wp-content/uploads/2008/11/danger-chmod-screenshot.png" alt="chmod, umask, file permissions test" title="chmod, umask, file permissions test" /></a>This shows what each numeric permission does to a REGULAR file.  I'll provide the code to do this below so you can do the same thing on your server.<br class="C" /></p>
<dl class="dlsm" style="border-right:1px solid #CCC;">
<dt><kbd>chmod 0</kbd></dt>
<dd><code>&#45;&#45;-&#45;&#45;-&#45;&#45;&#45;-</code></dd>
<dt><kbd>chmod 1</kbd></dt>
<dd><code>&#45;&#45;-&#45;&#45;-&#45;&#45;-x</code></dd>
<dt><kbd>chmod 2</kbd></dt>
<dd><code>&#45;&#45;-&#45;&#45;&#45;--w-</code></dd>
<dt><kbd>chmod 3</kbd></dt>
<dd><code>&#45;&#45;-&#45;&#45;&#45;--wx</code></dd>
<dt><kbd>chmod 4</kbd></dt>
<dd><code>&#45;&#45;-&#45;&#45;&#45;-r&#45;-</code></dd>
<dt><kbd>chmod 5</kbd></dt>
<dd><code>&#45;&#45;-&#45;&#45;&#45;-r-x</code></dd>
<dt><kbd>chmod 6</kbd></dt>
<dd><code>&#45;&#45;-&#45;&#45;&#45;-rw-</code></dd>
<dt><kbd>chmod 7</kbd></dt>
<dd><code>&#45;&#45;-&#45;&#45;&#45;-rwx</code></dd>
<dt><kbd>chmod 10</kbd></dt>
<dd><code>&#45;&#45;-&#45;&#45;-x&#45;&#45;-</code></dd>
<dt><kbd>chmod 11</kbd></dt>
<dd><code>&#45;&#45;-&#45;&#45;-x&#45;-x</code></dd>
<dt><kbd>chmod 12</kbd></dt>
<dd><code>&#45;&#45;-&#45;&#45;-x-w-</code></dd>
<dt><kbd>chmod 13</kbd></dt>
<dd><code>&#45;&#45;-&#45;&#45;-x-wx</code></dd>
<dt><kbd>chmod 14</kbd></dt>
<dd><code>&#45;&#45;-&#45;&#45;-xr&#45;-</code></dd>
<dt><kbd>chmod 15</kbd></dt>
<dd><code>&#45;&#45;-&#45;&#45;-xr-x</code></dd>
<dt><kbd>chmod 16</kbd></dt>
<dd><code>&#45;&#45;-&#45;&#45;-xrw-</code></dd>
<dt><kbd>chmod 17</kbd></dt>
<dd><code>&#45;&#45;-&#45;&#45;-xrwx</code></dd>
<dt><kbd>chmod 20</kbd></dt>
<dd><code>&#45;&#45;&#45;--w&#45;&#45;&#45;-</code></dd>
<dt><kbd>chmod 21</kbd></dt>
<dd><code>&#45;&#45;&#45;--w&#45;&#45;-x</code></dd>
<dt><kbd>chmod 22</kbd></dt>
<dd><code>&#45;&#45;&#45;--w&#45;-w-</code></dd>
<dt><kbd>chmod 23</kbd></dt>
<dd><code>&#45;&#45;&#45;--w&#45;-wx</code></dd>
<dt><kbd>chmod 24</kbd></dt>
<dd><code>&#45;&#45;&#45;--w-r&#45;-</code></dd>
<dt><kbd>chmod 25</kbd></dt>
<dd><code>&#45;&#45;&#45;--w-r-x</code></dd>
<dt><kbd>chmod 26</kbd></dt>
<dd><code>&#45;&#45;&#45;--w-rw-</code></dd>
<dt><kbd>chmod 27</kbd></dt>
<dd><code>&#45;&#45;&#45;--w-rwx</code></dd>
<dt><kbd>chmod 30</kbd></dt>
<dd><code>&#45;&#45;&#45;--wx&#45;&#45;-</code></dd>
<dt><kbd>chmod 31</kbd></dt>
<dd><code>&#45;&#45;&#45;--wx&#45;-x</code></dd>
<dt><kbd>chmod 32</kbd></dt>
<dd><code>&#45;&#45;&#45;--wx-w-</code></dd>
<dt><kbd>chmod 33</kbd></dt>
<dd><code>&#45;&#45;&#45;--wx-wx</code></dd>
<dt><kbd>chmod 34</kbd></dt>
<dd><code>&#45;&#45;&#45;--wxr&#45;-</code></dd>
<dt><kbd>chmod 35</kbd></dt>
<dd><code>&#45;&#45;&#45;--wxr-x</code></dd>
<dt><kbd>chmod 36</kbd></dt>
<dd><code>&#45;&#45;&#45;--wxrw-</code></dd>
<dt><kbd>chmod 37</kbd></dt>
<dd><code>&#45;&#45;&#45;--wxrwx</code></dd>
<dt><kbd>chmod 40</kbd></dt>
<dd><code>&#45;&#45;&#45;-r&#45;&#45;&#45;--</code></dd>
<dt><kbd>chmod 41</kbd></dt>
<dd><code>&#45;&#45;&#45;-r&#45;&#45;&#45;-x</code></dd>
<dt><kbd>chmod 42</kbd></dt>
<dd><code>&#45;&#45;&#45;-r&#45;&#45;-w-</code></dd>
<dt><kbd>chmod 43</kbd></dt>
<dd><code>&#45;&#45;&#45;-r&#45;&#45;-wx</code></dd>
<dt><kbd>chmod 44</kbd></dt>
<dd><code>&#45;&#45;&#45;-r&#45;-r&#45;-</code></dd>
<dt><kbd>chmod 45</kbd></dt>
<dd><code>&#45;&#45;&#45;-r&#45;-r-x</code></dd>
<dt><kbd>chmod 46</kbd></dt>
<dd><code>&#45;&#45;&#45;-r&#45;-rw-</code></dd>
<dt><kbd>chmod 47</kbd></dt>
<dd><code>&#45;&#45;&#45;-r&#45;-rwx</code></dd>
<dt><kbd>chmod 50</kbd></dt>
<dd><code>&#45;&#45;&#45;-r-x&#45;&#45;-</code></dd>
<dt><kbd>chmod 51</kbd></dt>
<dd><code>&#45;&#45;&#45;-r-x&#45;-x</code></dd>
<dt><kbd>chmod 52</kbd></dt>
<dd><code>&#45;&#45;&#45;-r-x-w-</code></dd>
<dt><kbd>chmod 53</kbd></dt>
<dd><code>&#45;&#45;&#45;-r-x-wx</code></dd>
<dt><kbd>chmod 54</kbd></dt>
<dd><code>&#45;&#45;&#45;-r-xr&#45;-</code></dd>
<dt><kbd>chmod 55</kbd></dt>
<dd><code>&#45;&#45;&#45;-r-xr-x</code></dd>
<dt><kbd>chmod 56</kbd></dt>
<dd><code>&#45;&#45;&#45;-r-xrw-</code></dd>
<dt><kbd>chmod 57</kbd></dt>
<dd><code>&#45;&#45;&#45;-r-xrwx</code></dd>
<dt><kbd>chmod 60</kbd></dt>
<dd><code>&#45;&#45;&#45;-rw&#45;&#45;&#45;-</code></dd>
<dt><kbd>chmod 61</kbd></dt>
<dd><code>&#45;&#45;&#45;-rw&#45;&#45;-x</code></dd>
<dt><kbd>chmod 62</kbd></dt>
<dd><code>&#45;&#45;&#45;-rw&#45;-w-</code></dd>
<dt><kbd>chmod 63</kbd></dt>
<dd><code>&#45;&#45;&#45;-rw&#45;-wx</code></dd>
<dt><kbd>chmod 64</kbd></dt>
<dd><code>&#45;&#45;&#45;-rw-r&#45;-</code></dd>
<dt><kbd>chmod 65</kbd></dt>
<dd><code>&#45;&#45;&#45;-rw-r-x</code></dd>
<dt><kbd>chmod 66</kbd></dt>
<dd><code>&#45;&#45;&#45;-rw-rw-</code></dd>
<dt><kbd>chmod 67</kbd></dt>
<dd><code>&#45;&#45;&#45;-rw-rwx</code></dd>
<dt><kbd>chmod 70</kbd></dt>
<dd><code>&#45;&#45;&#45;-rwx&#45;&#45;-</code></dd>
<dt><kbd>chmod 71</kbd></dt>
<dd><code>&#45;&#45;&#45;-rwx&#45;-x</code></dd>
<dt><kbd>chmod 72</kbd></dt>
<dd><code>&#45;&#45;&#45;-rwx-w-</code></dd>
<dt><kbd>chmod 73</kbd></dt>
<dd><code>&#45;&#45;&#45;-rwx-wx</code></dd>
<dt><kbd>chmod 74</kbd></dt>
<dd><code>&#45;&#45;&#45;-rwxr&#45;-</code></dd>
<dt><kbd>chmod 75</kbd></dt>
<dd><code>&#45;&#45;&#45;-rwxr-x</code></dd>
<dt><kbd>chmod 76</kbd></dt>
<dd><code>&#45;&#45;&#45;-rwxrw-</code></dd>
<dt><kbd>chmod 77</kbd></dt>
<dd><code>&#45;&#45;&#45;-rwxrwx</code></dd>
<dt><kbd>chmod 100</kbd></dt>
<dd><code>&#45;&#45;-x&#45;&#45;-&#45;&#45;-</code></dd>
<dt><kbd>chmod 101</kbd></dt>
<dd><code>&#45;&#45;-x&#45;&#45;&#45;--x</code></dd>
<dt><kbd>chmod 102</kbd></dt>
<dd><code>&#45;&#45;-x&#45;&#45;&#45;-w-</code></dd>
<dt><kbd>chmod 103</kbd></dt>
<dd><code>&#45;&#45;-x&#45;&#45;&#45;-wx</code></dd>
<dt><kbd>chmod 104</kbd></dt>
<dd><code>&#45;&#45;-x&#45;&#45;-r&#45;-</code></dd>
<dt><kbd>chmod 105</kbd></dt>
<dd><code>&#45;&#45;-x&#45;&#45;-r-x</code></dd>
<dt><kbd>chmod 106</kbd></dt>
<dd><code>&#45;&#45;-x&#45;&#45;-rw-</code></dd>
<dt><kbd>chmod 107</kbd></dt>
<dd><code>&#45;&#45;-x&#45;&#45;-rwx</code></dd>
<dt><kbd>chmod 110</kbd></dt>
<dd><code>&#45;&#45;-x&#45;-x&#45;&#45;-</code></dd>
<dt><kbd>chmod 111</kbd></dt>
<dd><code>&#45;&#45;-x&#45;-x&#45;-x</code></dd>
<dt><kbd>chmod 112</kbd></dt>
<dd><code>&#45;&#45;-x&#45;-x-w-</code></dd>
<dt><kbd>chmod 113</kbd></dt>
<dd><code>&#45;&#45;-x&#45;-x-wx</code></dd>
<dt><kbd>chmod 114</kbd></dt>
<dd><code>&#45;&#45;-x&#45;-xr&#45;-</code></dd>
<dt><kbd>chmod 115</kbd></dt>
<dd><code>&#45;&#45;-x&#45;-xr-x</code></dd>
<dt><kbd>chmod 116</kbd></dt>
<dd><code>&#45;&#45;-x&#45;-xrw-</code></dd>
<dt><kbd>chmod 117</kbd></dt>
<dd><code>&#45;&#45;-x&#45;-xrwx</code></dd>
<dt><kbd>chmod 120</kbd></dt>
<dd><code>&#45;&#45;-x-w&#45;&#45;&#45;-</code></dd>
<dt><kbd>chmod 121</kbd></dt>
<dd><code>&#45;&#45;-x-w&#45;&#45;-x</code></dd>
<dt><kbd>chmod 122</kbd></dt>
<dd><code>&#45;&#45;-x-w&#45;-w-</code></dd>
<dt><kbd>chmod 123</kbd></dt>
<dd><code>&#45;&#45;-x-w&#45;-wx</code></dd>
<dt><kbd>chmod 124</kbd></dt>
<dd><code>&#45;&#45;-x-w-r&#45;-</code></dd>
<dt><kbd>chmod 125</kbd></dt>
<dd><code>&#45;&#45;-x-w-r-x</code></dd>
<dt><kbd>chmod 126</kbd></dt>
<dd><code>&#45;&#45;-x-w-rw-</code></dd>
<dt><kbd>chmod 127</kbd></dt>
<dd><code>&#45;&#45;-x-w-rwx</code></dd>
<dt><kbd>chmod 130</kbd></dt>
<dd><code>&#45;&#45;-x-wx&#45;&#45;-</code></dd>
<dt><kbd>chmod 131</kbd></dt>
<dd><code>&#45;&#45;-x-wx&#45;-x</code></dd>
<dt><kbd>chmod 132</kbd></dt>
<dd><code>&#45;&#45;-x-wx-w-</code></dd>
<dt><kbd>chmod 133</kbd></dt>
<dd><code>&#45;&#45;-x-wx-wx</code></dd>
<dt><kbd>chmod 134</kbd></dt>
<dd><code>&#45;&#45;-x-wxr&#45;-</code></dd>
<dt><kbd>chmod 135</kbd></dt>
<dd><code>&#45;&#45;-x-wxr-x</code></dd>
<dt><kbd>chmod 136</kbd></dt>
<dd><code>&#45;&#45;-x-wxrw-</code></dd>
<dt><kbd>chmod 137</kbd></dt>
<dd><code>&#45;&#45;-x-wxrwx</code></dd>
<dt><kbd>chmod 140</kbd></dt>
<dd><code>&#45;&#45;-xr&#45;&#45;&#45;--</code></dd>
<dt><kbd>chmod 141</kbd></dt>
<dd><code>&#45;&#45;-xr&#45;&#45;&#45;-x</code></dd>
<dt><kbd>chmod 142</kbd></dt>
<dd><code>&#45;&#45;-xr&#45;&#45;-w-</code></dd>
<dt><kbd>chmod 143</kbd></dt>
<dd><code>&#45;&#45;-xr&#45;&#45;-wx</code></dd>
<dt><kbd>chmod 144</kbd></dt>
<dd><code>&#45;&#45;-xr&#45;-r&#45;-</code></dd>
<dt><kbd>chmod 145</kbd></dt>
<dd><code>&#45;&#45;-xr&#45;-r-x</code></dd>
<dt><kbd>chmod 146</kbd></dt>
<dd><code>&#45;&#45;-xr&#45;-rw-</code></dd>
<dt><kbd>chmod 147</kbd></dt>
<dd><code>&#45;&#45;-xr&#45;-rwx</code></dd>
<dt><kbd>chmod 150</kbd></dt>
<dd><code>&#45;&#45;-xr-x&#45;&#45;-</code></dd>
<dt><kbd>chmod 151</kbd></dt>
<dd><code>&#45;&#45;-xr-x&#45;-x</code></dd>
<dt><kbd>chmod 152</kbd></dt>
<dd><code>&#45;&#45;-xr-x-w-</code></dd>
<dt><kbd>chmod 153</kbd></dt>
<dd><code>&#45;&#45;-xr-x-wx</code></dd>
<dt><kbd>chmod 154</kbd></dt>
<dd><code>&#45;&#45;-xr-xr&#45;-</code></dd>
<dt><kbd>chmod 155</kbd></dt>
<dd><code>&#45;&#45;-xr-xr-x</code></dd>
<dt><kbd>chmod 156</kbd></dt>
<dd><code>&#45;&#45;-xr-xrw-</code></dd>
<dt><kbd>chmod 157</kbd></dt>
<dd><code>&#45;&#45;-xr-xrwx</code></dd>
<dt><kbd>chmod 160</kbd></dt>
<dd><code>&#45;&#45;-xrw&#45;&#45;&#45;-</code></dd>
<dt><kbd>chmod 161</kbd></dt>
<dd><code>&#45;&#45;-xrw&#45;&#45;-x</code></dd>
<dt><kbd>chmod 162</kbd></dt>
<dd><code>&#45;&#45;-xrw&#45;-w-</code></dd>
<dt><kbd>chmod 163</kbd></dt>
<dd><code>&#45;&#45;-xrw&#45;-wx</code></dd>
<dt><kbd>chmod 164</kbd></dt>
<dd><code>&#45;&#45;-xrw-r&#45;-</code></dd>
<dt><kbd>chmod 165</kbd></dt>
<dd><code>&#45;&#45;-xrw-r-x</code></dd>
<dt><kbd>chmod 166</kbd></dt>
<dd><code>&#45;&#45;-xrw-rw-</code></dd>
<dt><kbd>chmod 167</kbd></dt>
<dd><code>&#45;&#45;-xrw-rwx</code></dd>
<dt><kbd>chmod 170</kbd></dt>
<dd><code>&#45;&#45;-xrwx&#45;&#45;-</code></dd>
<dt><kbd>chmod 171</kbd></dt>
<dd><code>&#45;&#45;-xrwx&#45;-x</code></dd>
<dt><kbd>chmod 172</kbd></dt>
<dd><code>&#45;&#45;-xrwx-w-</code></dd>
<dt><kbd>chmod 173</kbd></dt>
<dd><code>&#45;&#45;-xrwx-wx</code></dd>
<dt><kbd>chmod 174</kbd></dt>
<dd><code>&#45;&#45;-xrwxr&#45;-</code></dd>
<dt><kbd>chmod 175</kbd></dt>
<dd><code>&#45;&#45;-xrwxr-x</code></dd>
<dt><kbd>chmod 176</kbd></dt>
<dd><code>&#45;&#45;-xrwxrw-</code></dd>
<dt><kbd>chmod 177</kbd></dt>
<dd><code>&#45;&#45;-xrwxrwx</code></dd>
<dt><kbd>chmod 200</kbd></dt>
<dd><code>&#45;-w&#45;&#45;-&#45;&#45;&#45;-</code></dd>
<dt><kbd>chmod 201</kbd></dt>
<dd><code>&#45;-w&#45;&#45;-&#45;&#45;-x</code></dd>
<dt><kbd>chmod 202</kbd></dt>
<dd><code>&#45;-w&#45;&#45;&#45;--w-</code></dd>
<dt><kbd>chmod 203</kbd></dt>
<dd><code>&#45;-w&#45;&#45;&#45;--wx</code></dd>
<dt><kbd>chmod 204</kbd></dt>
<dd><code>&#45;-w&#45;&#45;&#45;-r&#45;-</code></dd>
<dt><kbd>chmod 205</kbd></dt>
<dd><code>&#45;-w&#45;&#45;&#45;-r-x</code></dd>
<dt><kbd>chmod 206</kbd></dt>
<dd><code>&#45;-w&#45;&#45;&#45;-rw-</code></dd>
<dt><kbd>chmod 207</kbd></dt>
<dd><code>&#45;-w&#45;&#45;&#45;-rwx</code></dd>
<dt><kbd>chmod 210</kbd></dt>
<dd><code>&#45;-w&#45;&#45;-x&#45;&#45;-</code></dd>
<dt><kbd>chmod 211</kbd></dt>
<dd><code>&#45;-w&#45;&#45;-x&#45;-x</code></dd>
<dt><kbd>chmod 212</kbd></dt>
<dd><code>&#45;-w&#45;&#45;-x-w-</code></dd>
<dt><kbd>chmod 213</kbd></dt>
<dd><code>&#45;-w&#45;&#45;-x-wx</code></dd>
<dt><kbd>chmod 214</kbd></dt>
<dd><code>&#45;-w&#45;&#45;-xr&#45;-</code></dd>
<dt><kbd>chmod 215</kbd></dt>
<dd><code>&#45;-w&#45;&#45;-xr-x</code></dd>
<dt><kbd>chmod 216</kbd></dt>
<dd><code>&#45;-w&#45;&#45;-xrw-</code></dd>
<dt><kbd>chmod 217</kbd></dt>
<dd><code>&#45;-w&#45;&#45;-xrwx</code></dd>
<dt><kbd>chmod 220</kbd></dt>
<dd><code>&#45;-w&#45;-w&#45;&#45;&#45;-</code></dd>
<dt><kbd>chmod 221</kbd></dt>
<dd><code>&#45;-w&#45;-w&#45;&#45;-x</code></dd>
<dt><kbd>chmod 222</kbd></dt>
<dd><code>&#45;-w&#45;-w&#45;-w-</code></dd>
<dt><kbd>chmod 223</kbd></dt>
<dd><code>&#45;-w&#45;-w&#45;-wx</code></dd>
<dt><kbd>chmod 224</kbd></dt>
<dd><code>&#45;-w&#45;-w-r&#45;-</code></dd>
<dt><kbd>chmod 225</kbd></dt>
<dd><code>&#45;-w&#45;-w-r-x</code></dd>
<dt><kbd>chmod 226</kbd></dt>
<dd><code>&#45;-w&#45;-w-rw-</code></dd>
<dt><kbd>chmod 227</kbd></dt>
<dd><code>&#45;-w&#45;-w-rwx</code></dd>
<dt><kbd>chmod 230</kbd></dt>
<dd><code>&#45;-w&#45;-wx&#45;&#45;-</code></dd>
<dt><kbd>chmod 231</kbd></dt>
<dd><code>&#45;-w&#45;-wx&#45;-x</code></dd>
<dt><kbd>chmod 232</kbd></dt>
<dd><code>&#45;-w&#45;-wx-w-</code></dd>
<dt><kbd>chmod 233</kbd></dt>
<dd><code>&#45;-w&#45;-wx-wx</code></dd>
<dt><kbd>chmod 234</kbd></dt>
<dd><code>&#45;-w&#45;-wxr&#45;-</code></dd>
<dt><kbd>chmod 235</kbd></dt>
<dd><code>&#45;-w&#45;-wxr-x</code></dd>
<dt><kbd>chmod 236</kbd></dt>
<dd><code>&#45;-w&#45;-wxrw-</code></dd>
<dt><kbd>chmod 237</kbd></dt>
<dd><code>&#45;-w&#45;-wxrwx</code></dd>
<dt><kbd>chmod 240</kbd></dt>
<dd><code>&#45;-w-r&#45;&#45;&#45;--</code></dd>
<dt><kbd>chmod 241</kbd></dt>
<dd><code>&#45;-w-r&#45;&#45;&#45;-x</code></dd>
<dt><kbd>chmod 242</kbd></dt>
<dd><code>&#45;-w-r&#45;&#45;-w-</code></dd>
<dt><kbd>chmod 243</kbd></dt>
<dd><code>&#45;-w-r&#45;&#45;-wx</code></dd>
<dt><kbd>chmod 244</kbd></dt>
<dd><code>&#45;-w-r&#45;-r&#45;-</code></dd>
<dt><kbd>chmod 245</kbd></dt>
<dd><code>&#45;-w-r&#45;-r-x</code></dd>
<dt><kbd>chmod 246</kbd></dt>
<dd><code>&#45;-w-r&#45;-rw-</code></dd>
<dt><kbd>chmod 247</kbd></dt>
<dd><code>&#45;-w-r&#45;-rwx</code></dd>
<dt><kbd>chmod 250</kbd></dt>
<dd><code>&#45;-w-r-x&#45;&#45;-</code></dd>
<dt><kbd>chmod 251</kbd></dt>
<dd><code>&#45;-w-r-x&#45;-x</code></dd>
<dt><kbd>chmod 252</kbd></dt>
<dd><code>&#45;-w-r-x-w-</code></dd>
<dt><kbd>chmod 253</kbd></dt>
<dd><code>&#45;-w-r-x-wx</code></dd>
<dt><kbd>chmod 254</kbd></dt>
<dd><code>&#45;-w-r-xr&#45;-</code></dd>
<dt><kbd>chmod 255</kbd></dt>
<dd><code>&#45;-w-r-xr-x</code></dd>
<dt><kbd>chmod 256</kbd></dt>
<dd><code>&#45;-w-r-xrw-</code></dd>
<dt><kbd>chmod 257</kbd></dt>
<dd><code>&#45;-w-r-xrwx</code></dd>
<dt><kbd>chmod 260</kbd></dt>
<dd><code>&#45;-w-rw&#45;&#45;&#45;-</code></dd>
<dt><kbd>chmod 261</kbd></dt>
<dd><code>&#45;-w-rw&#45;&#45;-x</code></dd>
<dt><kbd>chmod 262</kbd></dt>
<dd><code>&#45;-w-rw&#45;-w-</code></dd>
<dt><kbd>chmod 263</kbd></dt>
<dd><code>&#45;-w-rw&#45;-wx</code></dd>
<dt><kbd>chmod 264</kbd></dt>
<dd><code>&#45;-w-rw-r&#45;-</code></dd>
<dt><kbd>chmod 265</kbd></dt>
<dd><code>&#45;-w-rw-r-x</code></dd>
<dt><kbd>chmod 266</kbd></dt>
<dd><code>&#45;-w-rw-rw-</code></dd>
<dt><kbd>chmod 267</kbd></dt>
<dd><code>&#45;-w-rw-rwx</code></dd>
<dt><kbd>chmod 270</kbd></dt>
<dd><code>&#45;-w-rwx&#45;&#45;-</code></dd>
<dt><kbd>chmod 271</kbd></dt>
<dd><code>&#45;-w-rwx&#45;-x</code></dd>
<dt><kbd>chmod 272</kbd></dt>
<dd><code>&#45;-w-rwx-w-</code></dd>
<dt><kbd>chmod 273</kbd></dt>
<dd><code>&#45;-w-rwx-wx</code></dd>
<dt><kbd>chmod 274</kbd></dt>
<dd><code>&#45;-w-rwxr&#45;-</code></dd>
<dt><kbd>chmod 275</kbd></dt>
<dd><code>&#45;-w-rwxr-x</code></dd>
<dt><kbd>chmod 276</kbd></dt>
<dd><code>&#45;-w-rwxrw-</code></dd>
<dt><kbd>chmod 277</kbd></dt>
<dd><code>&#45;-w-rwxrwx</code></dd>
<dt><kbd>chmod 300</kbd></dt>
<dd><code>&#45;-wx&#45;&#45;-&#45;&#45;-</code></dd>
<dt><kbd>chmod 301</kbd></dt>
<dd><code>&#45;-wx&#45;&#45;&#45;--x</code></dd>
<dt><kbd>chmod 302</kbd></dt>
<dd><code>&#45;-wx&#45;&#45;&#45;-w-</code></dd>
<dt><kbd>chmod 303</kbd></dt>
<dd><code>&#45;-wx&#45;&#45;&#45;-wx</code></dd>
<dt><kbd>chmod 304</kbd></dt>
<dd><code>&#45;-wx&#45;&#45;-r&#45;-</code></dd>
<dt><kbd>chmod 305</kbd></dt>
<dd><code>&#45;-wx&#45;&#45;-r-x</code></dd>
<dt><kbd>chmod 306</kbd></dt>
<dd><code>&#45;-wx&#45;&#45;-rw-</code></dd>
<dt><kbd>chmod 307</kbd></dt>
<dd><code>&#45;-wx&#45;&#45;-rwx</code></dd>
<dt><kbd>chmod 310</kbd></dt>
<dd><code>&#45;-wx&#45;-x&#45;&#45;-</code></dd>
<dt><kbd>chmod 311</kbd></dt>
<dd><code>&#45;-wx&#45;-x&#45;-x</code></dd>
<dt><kbd>chmod 312</kbd></dt>
<dd><code>&#45;-wx&#45;-x-w-</code></dd>
<dt><kbd>chmod 313</kbd></dt>
<dd><code>&#45;-wx&#45;-x-wx</code></dd>
<dt><kbd>chmod 314</kbd></dt>
<dd><code>&#45;-wx&#45;-xr&#45;-</code></dd>
<dt><kbd>chmod 315</kbd></dt>
<dd><code>&#45;-wx&#45;-xr-x</code></dd>
<dt><kbd>chmod 316</kbd></dt>
<dd><code>&#45;-wx&#45;-xrw-</code></dd>
<dt><kbd>chmod 317</kbd></dt>
<dd><code>&#45;-wx&#45;-xrwx</code></dd>
<dt><kbd>chmod 320</kbd></dt>
<dd><code>&#45;-wx-w&#45;&#45;&#45;-</code></dd>
<dt><kbd>chmod 321</kbd></dt>
<dd><code>&#45;-wx-w&#45;&#45;-x</code></dd>
<dt><kbd>chmod 322</kbd></dt>
<dd><code>&#45;-wx-w&#45;-w-</code></dd>
<dt><kbd>chmod 323</kbd></dt>
<dd><code>&#45;-wx-w&#45;-wx</code></dd>
<dt><kbd>chmod 324</kbd></dt>
<dd><code>&#45;-wx-w-r&#45;-</code></dd>
<dt><kbd>chmod 325</kbd></dt>
<dd><code>&#45;-wx-w-r-x</code></dd>
<dt><kbd>chmod 326</kbd></dt>
<dd><code>&#45;-wx-w-rw-</code></dd>
<dt><kbd>chmod 327</kbd></dt>
<dd><code>&#45;-wx-w-rwx</code></dd>
<dt><kbd>chmod 330</kbd></dt>
<dd><code>&#45;-wx-wx&#45;&#45;-</code></dd>
<dt><kbd>chmod 331</kbd></dt>
<dd><code>&#45;-wx-wx&#45;-x</code></dd>
<dt><kbd>chmod 332</kbd></dt>
<dd><code>&#45;-wx-wx-w-</code></dd>
<dt><kbd>chmod 333</kbd></dt>
<dd><code>&#45;-wx-wx-wx</code></dd>
<dt><kbd>chmod 334</kbd></dt>
<dd><code>&#45;-wx-wxr&#45;-</code></dd>
<dt><kbd>chmod 335</kbd></dt>
<dd><code>&#45;-wx-wxr-x</code></dd>
<dt><kbd>chmod 336</kbd></dt>
<dd><code>&#45;-wx-wxrw-</code></dd>
<dt><kbd>chmod 337</kbd></dt>
<dd><code>&#45;-wx-wxrwx</code></dd>
<dt><kbd>chmod 340</kbd></dt>
<dd><code>&#45;-wxr&#45;&#45;&#45;--</code></dd>
<dt><kbd>chmod 341</kbd></dt>
<dd><code>&#45;-wxr&#45;&#45;&#45;-x</code></dd>
<dt><kbd>chmod 342</kbd></dt>
<dd><code>&#45;-wxr&#45;&#45;-w-</code></dd>
<dt><kbd>chmod 343</kbd></dt>
<dd><code>&#45;-wxr&#45;&#45;-wx</code></dd>
<dt><kbd>chmod 344</kbd></dt>
<dd><code>&#45;-wxr&#45;-r&#45;-</code></dd>
<dt><kbd>chmod 345</kbd></dt>
<dd><code>&#45;-wxr&#45;-r-x</code></dd>
<dt><kbd>chmod 346</kbd></dt>
<dd><code>&#45;-wxr&#45;-rw-</code></dd>
<dt><kbd>chmod 347</kbd></dt>
<dd><code>&#45;-wxr&#45;-rwx</code></dd>
<dt><kbd>chmod 350</kbd></dt>
<dd><code>&#45;-wxr-x&#45;&#45;-</code></dd>
<dt><kbd>chmod 351</kbd></dt>
<dd><code>&#45;-wxr-x&#45;-x</code></dd>
<dt><kbd>chmod 352</kbd></dt>
<dd><code>&#45;-wxr-x-w-</code></dd>
<dt><kbd>chmod 353</kbd></dt>
<dd><code>&#45;-wxr-x-wx</code></dd>
<dt><kbd>chmod 354</kbd></dt>
<dd><code>&#45;-wxr-xr&#45;-</code></dd>
<dt><kbd>chmod 355</kbd></dt>
<dd><code>&#45;-wxr-xr-x</code></dd>
<dt><kbd>chmod 356</kbd></dt>
<dd><code>&#45;-wxr-xrw-</code></dd>
<dt><kbd>chmod 357</kbd></dt>
<dd><code>&#45;-wxr-xrwx</code></dd>
<dt><kbd>chmod 360</kbd></dt>
<dd><code>&#45;-wxrw&#45;&#45;&#45;-</code></dd>
<dt><kbd>chmod 361</kbd></dt>
<dd><code>&#45;-wxrw&#45;&#45;-x</code></dd>
<dt><kbd>chmod 362</kbd></dt>
<dd><code>&#45;-wxrw&#45;-w-</code></dd>
<dt><kbd>chmod 363</kbd></dt>
<dd><code>&#45;-wxrw&#45;-wx</code></dd>
<dt><kbd>chmod 364</kbd></dt>
<dd><code>&#45;-wxrw-r&#45;-</code></dd>
<dt><kbd>chmod 365</kbd></dt>
<dd><code>&#45;-wxrw-r-x</code></dd>
<dt><kbd>chmod 366</kbd></dt>
<dd><code>&#45;-wxrw-rw-</code></dd>
<dt><kbd>chmod 367</kbd></dt>
<dd><code>&#45;-wxrw-rwx</code></dd>
<dt><kbd>chmod 370</kbd></dt>
<dd><code>&#45;-wxrwx&#45;&#45;-</code></dd>
<dt><kbd>chmod 371</kbd></dt>
<dd><code>&#45;-wxrwx&#45;-x</code></dd>
<dt><kbd>chmod 372</kbd></dt>
<dd><code>&#45;-wxrwx-w-</code></dd>
<dt><kbd>chmod 373</kbd></dt>
<dd><code>&#45;-wxrwx-wx</code></dd>
<dt><kbd>chmod 374</kbd></dt>
<dd><code>&#45;-wxrwxr&#45;-</code></dd>
<dt><kbd>chmod 375</kbd></dt>
<dd><code>&#45;-wxrwxr-x</code></dd>
<dt><kbd>chmod 376</kbd></dt>
<dd><code>&#45;-wxrwxrw-</code></dd>
<dt><kbd>chmod 377</kbd></dt>
<dd><code>&#45;-wxrwxrwx</code></dd>
</dl>
<dl class="dlsm FL">
<dt><kbd>chmod 400</kbd></dt>
<dd><code>-r&#45;&#45;-&#45;&#45;&#45;--</code></dd>
<dt><kbd>chmod 401</kbd></dt>
<dd><code>-r&#45;&#45;-&#45;&#45;&#45;-x</code></dd>
<dt><kbd>chmod 402</kbd></dt>
<dd><code>-r&#45;&#45;-&#45;&#45;-w-</code></dd>
<dt><kbd>chmod 403</kbd></dt>
<dd><code>-r&#45;&#45;-&#45;&#45;-wx</code></dd>
<dt><kbd>chmod 404</kbd></dt>
<dd><code>-r&#45;&#45;&#45;--r&#45;-</code></dd>
<dt><kbd>chmod 405</kbd></dt>
<dd><code>-r&#45;&#45;&#45;--r-x</code></dd>
<dt><kbd>chmod 406</kbd></dt>
<dd><code>-r&#45;&#45;&#45;--rw-</code></dd>
<dt><kbd>chmod 407</kbd></dt>
<dd><code>-r&#45;&#45;&#45;--rwx</code></dd>
<dt><kbd>chmod 410</kbd></dt>
<dd><code>-r&#45;&#45;&#45;-x&#45;&#45;-</code></dd>
<dt><kbd>chmod 411</kbd></dt>
<dd><code>-r&#45;&#45;&#45;-x&#45;-x</code></dd>
<dt><kbd>chmod 412</kbd></dt>
<dd><code>-r&#45;&#45;&#45;-x-w-</code></dd>
<dt><kbd>chmod 413</kbd></dt>
<dd><code>-r&#45;&#45;&#45;-x-wx</code></dd>
<dt><kbd>chmod 414</kbd></dt>
<dd><code>-r&#45;&#45;&#45;-xr&#45;-</code></dd>
<dt><kbd>chmod 415</kbd></dt>
<dd><code>-r&#45;&#45;&#45;-xr-x</code></dd>
<dt><kbd>chmod 416</kbd></dt>
<dd><code>-r&#45;&#45;&#45;-xrw-</code></dd>
<dt><kbd>chmod 417</kbd></dt>
<dd><code>-r&#45;&#45;&#45;-xrwx</code></dd>
<dt><kbd>chmod 420</kbd></dt>
<dd><code>-r&#45;&#45;-w&#45;&#45;&#45;-</code></dd>
<dt><kbd>chmod 421</kbd></dt>
<dd><code>-r&#45;&#45;-w&#45;&#45;-x</code></dd>
<dt><kbd>chmod 422</kbd></dt>
<dd><code>-r&#45;&#45;-w&#45;-w-</code></dd>
<dt><kbd>chmod 423</kbd></dt>
<dd><code>-r&#45;&#45;-w&#45;-wx</code></dd>
<dt><kbd>chmod 424</kbd></dt>
<dd><code>-r&#45;&#45;-w-r&#45;-</code></dd>
<dt><kbd>chmod 425</kbd></dt>
<dd><code>-r&#45;&#45;-w-r-x</code></dd>
<dt><kbd>chmod 426</kbd></dt>
<dd><code>-r&#45;&#45;-w-rw-</code></dd>
<dt><kbd>chmod 427</kbd></dt>
<dd><code>-r&#45;&#45;-w-rwx</code></dd>
<dt><kbd>chmod 430</kbd></dt>
<dd><code>-r&#45;&#45;-wx&#45;&#45;-</code></dd>
<dt><kbd>chmod 431</kbd></dt>
<dd><code>-r&#45;&#45;-wx&#45;-x</code></dd>
<dt><kbd>chmod 432</kbd></dt>
<dd><code>-r&#45;&#45;-wx-w-</code></dd>
<dt><kbd>chmod 433</kbd></dt>
<dd><code>-r&#45;&#45;-wx-wx</code></dd>
<dt><kbd>chmod 434</kbd></dt>
<dd><code>-r&#45;&#45;-wxr&#45;-</code></dd>
<dt><kbd>chmod 435</kbd></dt>
<dd><code>-r&#45;&#45;-wxr-x</code></dd>
<dt><kbd>chmod 436</kbd></dt>
<dd><code>-r&#45;&#45;-wxrw-</code></dd>
<dt><kbd>chmod 437</kbd></dt>
<dd><code>-r&#45;&#45;-wxrwx</code></dd>
<dt><kbd>chmod 440</kbd></dt>
<dd><code>-r&#45;-r&#45;&#45;&#45;--</code></dd>
<dt><kbd>chmod 441</kbd></dt>
<dd><code>-r&#45;-r&#45;&#45;&#45;-x</code></dd>
<dt><kbd>chmod 442</kbd></dt>
<dd><code>-r&#45;-r&#45;&#45;-w-</code></dd>
<dt><kbd>chmod 443</kbd></dt>
<dd><code>-r&#45;-r&#45;&#45;-wx</code></dd>
<dt><kbd>chmod 444</kbd></dt>
<dd><code>-r&#45;-r&#45;-r&#45;-</code></dd>
<dt><kbd>chmod 445</kbd></dt>
<dd><code>-r&#45;-r&#45;-r-x</code></dd>
<dt><kbd>chmod 446</kbd></dt>
<dd><code>-r&#45;-r&#45;-rw-</code></dd>
<dt><kbd>chmod 447</kbd></dt>
<dd><code>-r&#45;-r&#45;-rwx</code></dd>
<dt><kbd>chmod 450</kbd></dt>
<dd><code>-r&#45;-r-x&#45;&#45;-</code></dd>
<dt><kbd>chmod 451</kbd></dt>
<dd><code>-r&#45;-r-x&#45;-x</code></dd>
<dt><kbd>chmod 452</kbd></dt>
<dd><code>-r&#45;-r-x-w-</code></dd>
<dt><kbd>chmod 453</kbd></dt>
<dd><code>-r&#45;-r-x-wx</code></dd>
<dt><kbd>chmod 454</kbd></dt>
<dd><code>-r&#45;-r-xr&#45;-</code></dd>
<dt><kbd>chmod 455</kbd></dt>
<dd><code>-r&#45;-r-xr-x</code></dd>
<dt><kbd>chmod 456</kbd></dt>
<dd><code>-r&#45;-r-xrw-</code></dd>
<dt><kbd>chmod 457</kbd></dt>
<dd><code>-r&#45;-r-xrwx</code></dd>
<dt><kbd>chmod 460</kbd></dt>
<dd><code>-r&#45;-rw&#45;&#45;&#45;-</code></dd>
<dt><kbd>chmod 461</kbd></dt>
<dd><code>-r&#45;-rw&#45;&#45;-x</code></dd>
<dt><kbd>chmod 462</kbd></dt>
<dd><code>-r&#45;-rw&#45;-w-</code></dd>
<dt><kbd>chmod 463</kbd></dt>
<dd><code>-r&#45;-rw&#45;-wx</code></dd>
<dt><kbd>chmod 464</kbd></dt>
<dd><code>-r&#45;-rw-r&#45;-</code></dd>
<dt><kbd>chmod 465</kbd></dt>
<dd><code>-r&#45;-rw-r-x</code></dd>
<dt><kbd>chmod 466</kbd></dt>
<dd><code>-r&#45;-rw-rw-</code></dd>
<dt><kbd>chmod 467</kbd></dt>
<dd><code>-r&#45;-rw-rwx</code></dd>
<dt><kbd>chmod 470</kbd></dt>
<dd><code>-r&#45;-rwx&#45;&#45;-</code></dd>
<dt><kbd>chmod 471</kbd></dt>
<dd><code>-r&#45;-rwx&#45;-x</code></dd>
<dt><kbd>chmod 472</kbd></dt>
<dd><code>-r&#45;-rwx-w-</code></dd>
<dt><kbd>chmod 473</kbd></dt>
<dd><code>-r&#45;-rwx-wx</code></dd>
<dt><kbd>chmod 474</kbd></dt>
<dd><code>-r&#45;-rwxr&#45;-</code></dd>
<dt><kbd>chmod 475</kbd></dt>
<dd><code>-r&#45;-rwxr-x</code></dd>
<dt><kbd>chmod 476</kbd></dt>
<dd><code>-r&#45;-rwxrw-</code></dd>
<dt><kbd>chmod 477</kbd></dt>
<dd><code>-r&#45;-rwxrwx</code></dd>
<dt><kbd>chmod 500</kbd></dt>
<dd><code>-r-x&#45;&#45;-&#45;&#45;-</code></dd>
<dt><kbd>chmod 501</kbd></dt>
<dd><code>-r-x&#45;&#45;&#45;--x</code></dd>
<dt><kbd>chmod 502</kbd></dt>
<dd><code>-r-x&#45;&#45;&#45;-w-</code></dd>
<dt><kbd>chmod 503</kbd></dt>
<dd><code>-r-x&#45;&#45;&#45;-wx</code></dd>
<dt><kbd>chmod 504</kbd></dt>
<dd><code>-r-x&#45;&#45;-r&#45;-</code></dd>
<dt><kbd>chmod 505</kbd></dt>
<dd><code>-r-x&#45;&#45;-r-x</code></dd>
<dt><kbd>chmod 506</kbd></dt>
<dd><code>-r-x&#45;&#45;-rw-</code></dd>
<dt><kbd>chmod 507</kbd></dt>
<dd><code>-r-x&#45;&#45;-rwx</code></dd>
<dt><kbd>chmod 510</kbd></dt>
<dd><code>-r-x&#45;-x&#45;&#45;-</code></dd>
<dt><kbd>chmod 511</kbd></dt>
<dd><code>-r-x&#45;-x&#45;-x</code></dd>
<dt><kbd>chmod 512</kbd></dt>
<dd><code>-r-x&#45;-x-w-</code></dd>
<dt><kbd>chmod 513</kbd></dt>
<dd><code>-r-x&#45;-x-wx</code></dd>
<dt><kbd>chmod 514</kbd></dt>
<dd><code>-r-x&#45;-xr&#45;-</code></dd>
<dt><kbd>chmod 515</kbd></dt>
<dd><code>-r-x&#45;-xr-x</code></dd>
<dt><kbd>chmod 516</kbd></dt>
<dd><code>-r-x&#45;-xrw-</code></dd>
<dt><kbd>chmod 517</kbd></dt>
<dd><code>-r-x&#45;-xrwx</code></dd>
<dt><kbd>chmod 520</kbd></dt>
<dd><code>-r-x-w&#45;&#45;&#45;-</code></dd>
<dt><kbd>chmod 521</kbd></dt>
<dd><code>-r-x-w&#45;&#45;-x</code></dd>
<dt><kbd>chmod 522</kbd></dt>
<dd><code>-r-x-w&#45;-w-</code></dd>
<dt><kbd>chmod 523</kbd></dt>
<dd><code>-r-x-w&#45;-wx</code></dd>
<dt><kbd>chmod 524</kbd></dt>
<dd><code>-r-x-w-r&#45;-</code></dd>
<dt><kbd>chmod 525</kbd></dt>
<dd><code>-r-x-w-r-x</code></dd>
<dt><kbd>chmod 526</kbd></dt>
<dd><code>-r-x-w-rw-</code></dd>
<dt><kbd>chmod 527</kbd></dt>
<dd><code>-r-x-w-rwx</code></dd>
<dt><kbd>chmod 530</kbd></dt>
<dd><code>-r-x-wx&#45;&#45;-</code></dd>
<dt><kbd>chmod 531</kbd></dt>
<dd><code>-r-x-wx&#45;-x</code></dd>
<dt><kbd>chmod 532</kbd></dt>
<dd><code>-r-x-wx-w-</code></dd>
<dt><kbd>chmod 533</kbd></dt>
<dd><code>-r-x-wx-wx</code></dd>
<dt><kbd>chmod 534</kbd></dt>
<dd><code>-r-x-wxr&#45;-</code></dd>
<dt><kbd>chmod 535</kbd></dt>
<dd><code>-r-x-wxr-x</code></dd>
<dt><kbd>chmod 536</kbd></dt>
<dd><code>-r-x-wxrw-</code></dd>
<dt><kbd>chmod 537</kbd></dt>
<dd><code>-r-x-wxrwx</code></dd>
<dt><kbd>chmod 540</kbd></dt>
<dd><code>-r-xr&#45;&#45;&#45;--</code></dd>
<dt><kbd>chmod 541</kbd></dt>
<dd><code>-r-xr&#45;&#45;&#45;-x</code></dd>
<dt><kbd>chmod 542</kbd></dt>
<dd><code>-r-xr&#45;&#45;-w-</code></dd>
<dt><kbd>chmod 543</kbd></dt>
<dd><code>-r-xr&#45;&#45;-wx</code></dd>
<dt><kbd>chmod 544</kbd></dt>
<dd><code>-r-xr&#45;-r&#45;-</code></dd>
<dt><kbd>chmod 545</kbd></dt>
<dd><code>-r-xr&#45;-r-x</code></dd>
<dt><kbd>chmod 546</kbd></dt>
<dd><code>-r-xr&#45;-rw-</code></dd>
<dt><kbd>chmod 547</kbd></dt>
<dd><code>-r-xr&#45;-rwx</code></dd>
<dt><kbd>chmod 550</kbd></dt>
<dd><code>-r-xr-x&#45;&#45;-</code></dd>
<dt><kbd>chmod 551</kbd></dt>
<dd><code>-r-xr-x&#45;-x</code></dd>
<dt><kbd>chmod 552</kbd></dt>
<dd><code>-r-xr-x-w-</code></dd>
<dt><kbd>chmod 553</kbd></dt>
<dd><code>-r-xr-x-wx</code></dd>
<dt><kbd>chmod 554</kbd></dt>
<dd><code>-r-xr-xr&#45;-</code></dd>
<dt><kbd>chmod 555</kbd></dt>
<dd><code>-r-xr-xr-x</code></dd>
<dt><kbd>chmod 556</kbd></dt>
<dd><code>-r-xr-xrw-</code></dd>
<dt><kbd>chmod 557</kbd></dt>
<dd><code>-r-xr-xrwx</code></dd>
<dt><kbd>chmod 560</kbd></dt>
<dd><code>-r-xrw&#45;&#45;&#45;-</code></dd>
<dt><kbd>chmod 561</kbd></dt>
<dd><code>-r-xrw&#45;&#45;-x</code></dd>
<dt><kbd>chmod 562</kbd></dt>
<dd><code>-r-xrw&#45;-w-</code></dd>
<dt><kbd>chmod 563</kbd></dt>
<dd><code>-r-xrw&#45;-wx</code></dd>
<dt><kbd>chmod 564</kbd></dt>
<dd><code>-r-xrw-r&#45;-</code></dd>
<dt><kbd>chmod 565</kbd></dt>
<dd><code>-r-xrw-r-x</code></dd>
<dt><kbd>chmod 566</kbd></dt>
<dd><code>-r-xrw-rw-</code></dd>
<dt><kbd>chmod 567</kbd></dt>
<dd><code>-r-xrw-rwx</code></dd>
<dt><kbd>chmod 570</kbd></dt>
<dd><code>-r-xrwx&#45;&#45;-</code></dd>
<dt><kbd>chmod 571</kbd></dt>
<dd><code>-r-xrwx&#45;-x</code></dd>
<dt><kbd>chmod 572</kbd></dt>
<dd><code>-r-xrwx-w-</code></dd>
<dt><kbd>chmod 573</kbd></dt>
<dd><code>-r-xrwx-wx</code></dd>
<dt><kbd>chmod 574</kbd></dt>
<dd><code>-r-xrwxr&#45;-</code></dd>
<dt><kbd>chmod 575</kbd></dt>
<dd><code>-r-xrwxr-x</code></dd>
<dt><kbd>chmod 576</kbd></dt>
<dd><code>-r-xrwxrw-</code></dd>
<dt><kbd>chmod 577</kbd></dt>
<dd><code>-r-xrwxrwx</code></dd>
<dt><kbd>chmod 600</kbd></dt>
<dd><code>-rw&#45;&#45;-&#45;&#45;&#45;-</code></dd>
<dt><kbd>chmod 601</kbd></dt>
<dd><code>-rw&#45;&#45;-&#45;&#45;-x</code></dd>
<dt><kbd>chmod 602</kbd></dt>
<dd><code>-rw&#45;&#45;&#45;--w-</code></dd>
<dt><kbd>chmod 603</kbd></dt>
<dd><code>-rw&#45;&#45;&#45;--wx</code></dd>
<dt><kbd>chmod 604</kbd></dt>
<dd><code>-rw&#45;&#45;&#45;-r&#45;-</code></dd>
<dt><kbd>chmod 605</kbd></dt>
<dd><code>-rw&#45;&#45;&#45;-r-x</code></dd>
<dt><kbd>chmod 606</kbd></dt>
<dd><code>-rw&#45;&#45;&#45;-rw-</code></dd>
<dt><kbd>chmod 607</kbd></dt>
<dd><code>-rw&#45;&#45;&#45;-rwx</code></dd>
<dt><kbd>chmod 610</kbd></dt>
<dd><code>-rw&#45;&#45;-x&#45;&#45;-</code></dd>
<dt><kbd>chmod 611</kbd></dt>
<dd><code>-rw&#45;&#45;-x&#45;-x</code></dd>
<dt><kbd>chmod 612</kbd></dt>
<dd><code>-rw&#45;&#45;-x-w-</code></dd>
<dt><kbd>chmod 613</kbd></dt>
<dd><code>-rw&#45;&#45;-x-wx</code></dd>
<dt><kbd>chmod 614</kbd></dt>
<dd><code>-rw&#45;&#45;-xr&#45;-</code></dd>
<dt><kbd>chmod 615</kbd></dt>
<dd><code>-rw&#45;&#45;-xr-x</code></dd>
<dt><kbd>chmod 616</kbd></dt>
<dd><code>-rw&#45;&#45;-xrw-</code></dd>
<dt><kbd>chmod 617</kbd></dt>
<dd><code>-rw&#45;&#45;-xrwx</code></dd>
<dt><kbd>chmod 620</kbd></dt>
<dd><code>-rw&#45;-w&#45;&#45;&#45;-</code></dd>
<dt><kbd>chmod 621</kbd></dt>
<dd><code>-rw&#45;-w&#45;&#45;-x</code></dd>
<dt><kbd>chmod 622</kbd></dt>
<dd><code>-rw&#45;-w&#45;-w-</code></dd>
<dt><kbd>chmod 623</kbd></dt>
<dd><code>-rw&#45;-w&#45;-wx</code></dd>
<dt><kbd>chmod 624</kbd></dt>
<dd><code>-rw&#45;-w-r&#45;-</code></dd>
<dt><kbd>chmod 625</kbd></dt>
<dd><code>-rw&#45;-w-r-x</code></dd>
<dt><kbd>chmod 626</kbd></dt>
<dd><code>-rw&#45;-w-rw-</code></dd>
<dt><kbd>chmod 627</kbd></dt>
<dd><code>-rw&#45;-w-rwx</code></dd>
<dt><kbd>chmod 630</kbd></dt>
<dd><code>-rw&#45;-wx&#45;&#45;-</code></dd>
<dt><kbd>chmod 631</kbd></dt>
<dd><code>-rw&#45;-wx&#45;-x</code></dd>
<dt><kbd>chmod 632</kbd></dt>
<dd><code>-rw&#45;-wx-w-</code></dd>
<dt><kbd>chmod 633</kbd></dt>
<dd><code>-rw&#45;-wx-wx</code></dd>
<dt><kbd>chmod 634</kbd></dt>
<dd><code>-rw&#45;-wxr&#45;-</code></dd>
<dt><kbd>chmod 635</kbd></dt>
<dd><code>-rw&#45;-wxr-x</code></dd>
<dt><kbd>chmod 636</kbd></dt>
<dd><code>-rw&#45;-wxrw-</code></dd>
<dt><kbd>chmod 637</kbd></dt>
<dd><code>-rw&#45;-wxrwx</code></dd>
<dt><kbd>chmod 640</kbd></dt>
<dd><code>-rw-r&#45;&#45;&#45;--</code></dd>
<dt><kbd>chmod 641</kbd></dt>
<dd><code>-rw-r&#45;&#45;&#45;-x</code></dd>
<dt><kbd>chmod 642</kbd></dt>
<dd><code>-rw-r&#45;&#45;-w-</code></dd>
<dt><kbd>chmod 643</kbd></dt>
<dd><code>-rw-r&#45;&#45;-wx</code></dd>
<dt><kbd>chmod 644</kbd></dt>
<dd><code>-rw-r&#45;-r&#45;-</code></dd>
<dt><kbd>chmod 645</kbd></dt>
<dd><code>-rw-r&#45;-r-x</code></dd>
<dt><kbd>chmod 646</kbd></dt>
<dd><code>-rw-r&#45;-rw-</code></dd>
<dt><kbd>chmod 647</kbd></dt>
<dd><code>-rw-r&#45;-rwx</code></dd>
<dt><kbd>chmod 650</kbd></dt>
<dd><code>-rw-r-x&#45;&#45;-</code></dd>
<dt><kbd>chmod 651</kbd></dt>
<dd><code>-rw-r-x&#45;-x</code></dd>
<dt><kbd>chmod 652</kbd></dt>
<dd><code>-rw-r-x-w-</code></dd>
<dt><kbd>chmod 653</kbd></dt>
<dd><code>-rw-r-x-wx</code></dd>
<dt><kbd>chmod 654</kbd></dt>
<dd><code>-rw-r-xr&#45;-</code></dd>
<dt><kbd>chmod 655</kbd></dt>
<dd><code>-rw-r-xr-x</code></dd>
<dt><kbd>chmod 656</kbd></dt>
<dd><code>-rw-r-xrw-</code></dd>
<dt><kbd>chmod 657</kbd></dt>
<dd><code>-rw-r-xrwx</code></dd>
<dt><kbd>chmod 660</kbd></dt>
<dd><code>-rw-rw&#45;&#45;&#45;-</code></dd>
<dt><kbd>chmod 661</kbd></dt>
<dd><code>-rw-rw&#45;&#45;-x</code></dd>
<dt><kbd>chmod 662</kbd></dt>
<dd><code>-rw-rw&#45;-w-</code></dd>
<dt><kbd>chmod 663</kbd></dt>
<dd><code>-rw-rw&#45;-wx</code></dd>
<dt><kbd>chmod 664</kbd></dt>
<dd><code>-rw-rw-r&#45;-</code></dd>
<dt><kbd>chmod 665</kbd></dt>
<dd><code>-rw-rw-r-x</code></dd>
<dt><kbd>chmod 666</kbd></dt>
<dd><code>-rw-rw-rw-</code></dd>
<dt><kbd>chmod 667</kbd></dt>
<dd><code>-rw-rw-rwx</code></dd>
<dt><kbd>chmod 670</kbd></dt>
<dd><code>-rw-rwx&#45;&#45;-</code></dd>
<dt><kbd>chmod 671</kbd></dt>
<dd><code>-rw-rwx&#45;-x</code></dd>
<dt><kbd>chmod 672</kbd></dt>
<dd><code>-rw-rwx-w-</code></dd>
<dt><kbd>chmod 673</kbd></dt>
<dd><code>-rw-rwx-wx</code></dd>
<dt><kbd>chmod 674</kbd></dt>
<dd><code>-rw-rwxr&#45;-</code></dd>
<dt><kbd>chmod 675</kbd></dt>
<dd><code>-rw-rwxr-x</code></dd>
<dt><kbd>chmod 676</kbd></dt>
<dd><code>-rw-rwxrw-</code></dd>
<dt><kbd>chmod 677</kbd></dt>
<dd><code>-rw-rwxrwx</code></dd>
<dt><kbd>chmod 700</kbd></dt>
<dd><code>-rwx&#45;&#45;-&#45;&#45;-</code></dd>
<dt><kbd>chmod 701</kbd></dt>
<dd><code>-rwx&#45;&#45;&#45;--x</code></dd>
<dt><kbd>chmod 702</kbd></dt>
<dd><code>-rwx&#45;&#45;&#45;-w-</code></dd>
<dt><kbd>chmod 703</kbd></dt>
<dd><code>-rwx&#45;&#45;&#45;-wx</code></dd>
<dt><kbd>chmod 704</kbd></dt>
<dd><code>-rwx&#45;&#45;-r&#45;-</code></dd>
<dt><kbd>chmod 705</kbd></dt>
<dd><code>-rwx&#45;&#45;-r-x</code></dd>
<dt><kbd>chmod 706</kbd></dt>
<dd><code>-rwx&#45;&#45;-rw-</code></dd>
<dt><kbd>chmod 707</kbd></dt>
<dd><code>-rwx&#45;&#45;-rwx</code></dd>
<dt><kbd>chmod 710</kbd></dt>
<dd><code>-rwx&#45;-x&#45;&#45;-</code></dd>
<dt><kbd>chmod 711</kbd></dt>
<dd><code>-rwx&#45;-x&#45;-x</code></dd>
<dt><kbd>chmod 712</kbd></dt>
<dd><code>-rwx&#45;-x-w-</code></dd>
<dt><kbd>chmod 713</kbd></dt>
<dd><code>-rwx&#45;-x-wx</code></dd>
<dt><kbd>chmod 714</kbd></dt>
<dd><code>-rwx&#45;-xr&#45;-</code></dd>
<dt><kbd>chmod 715</kbd></dt>
<dd><code>-rwx&#45;-xr-x</code></dd>
<dt><kbd>chmod 716</kbd></dt>
<dd><code>-rwx&#45;-xrw-</code></dd>
<dt><kbd>chmod 717</kbd></dt>
<dd><code>-rwx&#45;-xrwx</code></dd>
<dt><kbd>chmod 720</kbd></dt>
<dd><code>-rwx-w&#45;&#45;&#45;-</code></dd>
<dt><kbd>chmod 721</kbd></dt>
<dd><code>-rwx-w&#45;&#45;-x</code></dd>
<dt><kbd>chmod 722</kbd></dt>
<dd><code>-rwx-w&#45;-w-</code></dd>
<dt><kbd>chmod 723</kbd></dt>
<dd><code>-rwx-w&#45;-wx</code></dd>
<dt><kbd>chmod 724</kbd></dt>
<dd><code>-rwx-w-r&#45;-</code></dd>
<dt><kbd>chmod 725</kbd></dt>
<dd><code>-rwx-w-r-x</code></dd>
<dt><kbd>chmod 726</kbd></dt>
<dd><code>-rwx-w-rw-</code></dd>
<dt><kbd>chmod 727</kbd></dt>
<dd><code>-rwx-w-rwx</code></dd>
<dt><kbd>chmod 730</kbd></dt>
<dd><code>-rwx-wx&#45;&#45;-</code></dd>
<dt><kbd>chmod 731</kbd></dt>
<dd><code>-rwx-wx&#45;-x</code></dd>
<dt><kbd>chmod 732</kbd></dt>
<dd><code>-rwx-wx-w-</code></dd>
<dt><kbd>chmod 733</kbd></dt>
<dd><code>-rwx-wx-wx</code></dd>
<dt><kbd>chmod 734</kbd></dt>
<dd><code>-rwx-wxr&#45;-</code></dd>
<dt><kbd>chmod 735</kbd></dt>
<dd><code>-rwx-wxr-x</code></dd>
<dt><kbd>chmod 736</kbd></dt>
<dd><code>-rwx-wxrw-</code></dd>
<dt><kbd>chmod 737</kbd></dt>
<dd><code>-rwx-wxrwx</code></dd>
<dt><kbd>chmod 740</kbd></dt>
<dd><code>-rwxr&#45;&#45;&#45;--</code></dd>
<dt><kbd>chmod 741</kbd></dt>
<dd><code>-rwxr&#45;&#45;&#45;-x</code></dd>
<dt><kbd>chmod 742</kbd></dt>
<dd><code>-rwxr&#45;&#45;-w-</code></dd>
<dt><kbd>chmod 743</kbd></dt>
<dd><code>-rwxr&#45;&#45;-wx</code></dd>
<dt><kbd>chmod 744</kbd></dt>
<dd><code>-rwxr&#45;-r&#45;-</code></dd>
<dt><kbd>chmod 745</kbd></dt>
<dd><code>-rwxr&#45;-r-x</code></dd>
<dt><kbd>chmod 746</kbd></dt>
<dd><code>-rwxr&#45;-rw-</code></dd>
<dt><kbd>chmod 747</kbd></dt>
<dd><code>-rwxr&#45;-rwx</code></dd>
<dt><kbd>chmod 750</kbd></dt>
<dd><code>-rwxr-x&#45;&#45;-</code></dd>
<dt><kbd>chmod 751</kbd></dt>
<dd><code>-rwxr-x&#45;-x</code></dd>
<dt><kbd>chmod 752</kbd></dt>
<dd><code>-rwxr-x-w-</code></dd>
<dt><kbd>chmod 753</kbd></dt>
<dd><code>-rwxr-x-wx</code></dd>
<dt><kbd>chmod 754</kbd></dt>
<dd><code>-rwxr-xr&#45;-</code></dd>
<dt><kbd>chmod 755</kbd></dt>
<dd><code>-rwxr-xr-x</code></dd>
<dt><kbd>chmod 756</kbd></dt>
<dd><code>-rwxr-xrw-</code></dd>
<dt><kbd>chmod 757</kbd></dt>
<dd><code>-rwxr-xrwx</code></dd>
<dt><kbd>chmod 760</kbd></dt>
<dd><code>-rwxrw&#45;&#45;&#45;-</code></dd>
<dt><kbd>chmod 761</kbd></dt>
<dd><code>-rwxrw&#45;&#45;-x</code></dd>
<dt><kbd>chmod 762</kbd></dt>
<dd><code>-rwxrw&#45;-w-</code></dd>
<dt><kbd>chmod 763</kbd></dt>
<dd><code>-rwxrw&#45;-wx</code></dd>
<dt><kbd>chmod 764</kbd></dt>
<dd><code>-rwxrw-r&#45;-</code></dd>
<dt><kbd>chmod 765</kbd></dt>
<dd><code>-rwxrw-r-x</code></dd>
<dt><kbd>chmod 766</kbd></dt>
<dd><code>-rwxrw-rw-</code></dd>
<dt><kbd>chmod 767</kbd></dt>
<dd><code>-rwxrw-rwx</code></dd>
<dt><kbd>chmod 770</kbd></dt>
<dd><code>-rwxrwx&#45;&#45;-</code></dd>
<dt><kbd>chmod 771</kbd></dt>
<dd><code>-rwxrwx&#45;-x</code></dd>
<dt><kbd>chmod 772</kbd></dt>
<dd><code>-rwxrwx-w-</code></dd>
<dt><kbd>chmod 773</kbd></dt>
<dd><code>-rwxrwx-wx</code></dd>
<dt><kbd>chmod 774</kbd></dt>
<dd><code>-rwxrwxr&#45;-</code></dd>
<dt><kbd>chmod 775</kbd></dt>
<dd><code>-rwxrwxr-x</code></dd>
<dt><kbd>chmod 776</kbd></dt>
<dd><code>-rwxrwxrw-</code></dd>
<dt><kbd>chmod 777</kbd></dt>
<dd><code>-rwxrwxrwx</code></dd>
</dl>
<h2>Congratulations!</h2>
<p>Here's my custom stat function, which I am definately not finished with, so check back in a couple days and if you find any improvements please hook me up with a comment!</p>
<p>function askapache_stat( $filename ) {<br />
$p=@fileperms($filename);<br />
$s=@stat($filename);<br />
$str='';<br />
$t=decoct($s['mode'] &#038; 0170000);</p>
<p>switch (octdec($t)) {<br />
case 0140000: $str = 's'; $stat['type']='socket'; break;<br />
case 0120000: $str = 'l'; $stat['type']='link'; break;<br />
case 0100000: $str = '-'; $stat['type']='file'; break;<br />
case 0060000: $str = 'b'; $stat['type']='block'; break;<br />
case 0040000: $str = 'd'; $stat['type']='dir'; break;<br />
case 0020000: $str = 'c'; $stat['type']='char'; break;<br />
case 0010000: $str = 'p'; $stat['type']='fifo'; break;<br />
default: $str = 'u'; $stat['type']='unknown'; break;<br />
}</p>
<p>$stat['type_octal'] = sprintf("%07o", octdec($t));</p>
<p>$str .= (($p&#038;0x0100)?'r':'-').(($p&#038;0x0080)?'w':'-').(($p&#038;0x0040)?(($p&#038;0x0800)?'s':'x'):(($p&#038;0x0800)?'S':'-'));<br />
$str .= (($p&#038;0x0020)?'r':'-').(($p&#038;0x0010)?'w':'-').(($p&#038;0x0008)?(($p&#038;0x0400)?'s':'x'):(($p&#038;0x0400)?'S':'-'));<br />
$str .= (($p&#038;0x0004)?'r':'-').(($p&#038;0x0002)?'w':'-').(($p&#038;0x0001)?(($p&#038;0x0200)?'t':'x'):(($p&#038;0x0200)?'T':'-'));</p>
<p>$stat['default_umask']=sprintf("%04o",umask());<br />
$stat['perm_human']=$str;<br />
$stat['perm_octal1'] = sprintf( "%o", ( $s['mode'] &#038; 00777 ) );<br />
$stat['perm_octal2'] = sprintf("0%o", 0777 &#038; $p);<br />
$stat['perm_dec'] = sprintf("%04o", $p);<br />
$stat['perm_mode']=$s['mode'];	 // File mode.</p>
<p>$stat['file'] = @realpath($filename);<br />
$stat['basename'] = basename( $filename );</p>
<p>$stat['user_id'] = $s['uid'];<br />
$stat['group_id'] = $s['gid'];</p>
<p>$stat['device']=$s['dev'];			// Device<br />
$stat['device_number']=$s['rdev'];		// Device number, if device.<br />
$stat['inode']=$s['ino'];			// File serial number<br />
$stat['link_count']=$s['nlink'];		// link count<br />
if($stat['type']=='link')$stat['link_to']=@readlink( $filename );</p>
<p>$stat['size']=$s['size'];		// Size of file, in bytes.<br />
$stat['block_size']=$s['blksize'];	// Optimal block size for I/O.<br />
$stat['blocks']=$s['blocks'];	// Number 512-byte blocks allocated</p>
<p>$stat['time_access']=@date( 'Y M D H:i:s',$s['atime']);		// Time of last access.<br />
$stat['time_modified']=@date( 'Y M D H:i:s',$s['mtime']);		// Time of last modification<br />
$stat['time_created']=@date( 'Y M D H:i:s',$s['ctime']);		// Time of last status change</p>
<p>clearstatcache();<br />
return $stat;<br />
}</p>
<p>header('Content-Type: text/plain');<br />
$stat=askapache_stat(__FILE__);<br />
print_r($stat);
</pre>
<h3>Defining Permission Bits</h3>
<p>!defined('S_IFMT') &#038;&#038; define('S_IFMT', 0170000); //	mask for all types<br />
!defined('S_IFSOCK') &#038;&#038; define('S_IFSOCK', 0140000); // type: socket<br />
!defined('S_IFLNK') &#038;&#038; define('S_IFLNK', 0120000); // type:	symbolic link<br />
!defined('S_IFREG') &#038;&#038; define('S_IFREG', 0100000); // type:	regular file<br />
!defined('S_IFBLK') &#038;&#038; define('S_IFBLK', 0060000); // type:	block device<br />
!defined('S_IFDIR') &#038;&#038; define('S_IFDIR', 0040000); // type:	directory<br />
!defined('S_IFCHR') &#038;&#038; define('S_IFCHR', 0020000); // type:	character device<br />
!defined('S_IFIFO') &#038;&#038; define('S_IFIFO', 0010000); // type:	fifo</p>
<p>!defined('S_ISUID') &#038;&#038; define('S_ISUID', 0004000); // set-uid bit<br />
!defined('S_ISGID') &#038;&#038; define('S_ISGID', 0002000); // set-gid bit<br />
!defined('S_ISVTX') &#038;&#038; define('S_ISVTX', 0001000); // sticky bit<br />
!defined('S_IRWXU') &#038;&#038; define('S_IRWXU', 00700); //	mask for owner permissions<br />
!defined('S_IRUSR') &#038;&#038; define('S_IRUSR', 00400); //	owner: read permission<br />
!defined('S_IWUSR') &#038;&#038; define('S_IWUSR', 00200); //	owner: write permission<br />
!defined('S_IXUSR') &#038;&#038; define('S_IXUSR', 00100); //	owner: execute permission<br />
!defined('S_IRWXG') &#038;&#038; define('S_IRWXG', 00070); //	mask for group permissions<br />
!defined('S_IRGRP') &#038;&#038; define('S_IRGRP', 00040); //	group: read permission<br />
!defined('S_IWGRP') &#038;&#038; define('S_IWGRP', 00020); //	group: write permission<br />
!defined('S_IXGRP') &#038;&#038; define('S_IXGRP', 00010); //	group: execute permission<br />
!defined('S_IRWXO') &#038;&#038; define('S_IRWXO', 00007); //	mask for others permissions<br />
!defined('S_IROTH') &#038;&#038; define('S_IROTH', 00004); //	others:	read permission<br />
!defined('S_IWOTH') &#038;&#038; define('S_IWOTH', 00002); //	others:	write permission<br />
!defined('S_IXOTH') &#038;&#038; define('S_IXOTH', 00001); //	others:	execute permission</p>
<p>!defined('S_IRWXUGO') &#038;&#038; define('S_IRWXUGO', (S_IRWXU | S_IRWXG | S_IRWXO));<br />
!defined('S_IALLUGO') &#038;&#038; define('S_IALLUGO', (S_ISUID | S_ISGID | S_ISVTX | S_IRWXUGO));<br />
!defined('S_IRUGO') &#038;&#038; define('S_IRUGO', (S_IRUSR | S_IRGRP | S_IROTH));<br />
!defined('S_IWUGO') &#038;&#038; define('S_IWUGO', (S_IWUSR | S_IWGRP | S_IWOTH));<br />
!defined('S_IXUGO') &#038;&#038; define('S_IXUGO', (S_IXUSR | S_IXGRP | S_IXOTH));<br />
!defined('S_IRWUGO') &#038;&#038; define('S_IRWUGO', (S_IRUGO | S_IWUGO));
</pre>
<h2>How File Permissions Work</h2>
<p>When PHP is installed on your server by you or whoever runs the server, it uses the file permissions that are used by the Operating System running the server..  If you are smart or just lucky than you are running some type of BSD/Unix/Solaris/Linux/Sun based Operating system and PHP won't have any problems.  If you are running on a Locked, proprietary OS like Windows, PHP will still work but it has to use a lot of shortcuts and hacks to basically "Pretend" to act like the OS is BSD/Unix, and some key features just won't be available.</p>
<h2>The OS Permission Bits</h2>
<p>Here's the file permissions my Linux server uses, and which PHP automatically uses.  The code basically just defines the default permissions for files, and defines the file atributes for each file that you can access by using the stat function, which I've improved upon to make things easier.</p>
<p>Download: <a href='/wp-content/uploads/2008/11/stat.h' title="POSIX Standard: 5.6 File Characteristics">POSIX Standard: 5.6 File Characteristics<code>sys/stat.h</code></a></p>
<h3>Protection bits for File Owner</h3>
<pre class='brushbash'>
#define S_IRWXU 00700
#define S_IRUSR 00400
#define S_IWUSR 00200
#define S_IXUSR 00100
</pre>
<h3>Protection bits for File Group</h3>
<pre class='brushbash'>
#define S_IRWXG 00070
#define S_IRGRP 00040
#define S_IWGRP 00020
#define S_IXGRP 00010
</pre>
<h3>Protection bits for All Others</h3>
<pre class='brushbash'>
#define S_IRWXO 00007
#define S_IROTH 00004
#define S_IWOTH 00002
#define S_IXOTH 00001
</pre>
<h2>Some Example Permissions</h2>
<p><code>0477</code>  // owner has read only, other and group has rwx<br />
<code>0677</code>  // owner has rw only, other and group has rwx</p>
<p><code>0444</code>  // all have read only<br />
<code>0666</code>  // all have rw only</p>
<p><code>0400</code>  // owner has read only, group and others have no permission<br />
<code>0600</code> // owner has rw only, group and others have no permission</p>
<p><code>0470</code>  // owner has read only, group has rwx, others have no permission<br />
<code>0407</code>  // owner has read only, other has rwx, group has no permission</p>
<p><code>0670</code>  // owner has rw only, group has rwx, others have no permission<br />
<code>0607</code>  // owner has rw only, group has no permission and others have rwx</p>
<h2>What's a File</h2>
<p>A file is not merely its contents, a name, and a file type. A file also has an owner (a user ID), a group (a group ID), permissions (what the owner can do with the file, what people in the group can do, and what everyone else can do), various timestamps, and other information. Collectively, we call these a file's attributes.</p>
<h2>Structure of File Mode Bits</h2>
<p>The file mode bits have two parts: the file permission bits, which control ordinary access to the file, and special mode bits, which affect only some files.</p>
<p>There are three kinds of permissions that a user can have for a file:</p>
<ol>
<li>permission to read the file. For directories, this means permission to list the contents of the directory.</li>
<li>permission to write to (change) the file. For directories, this means permission to create and remove files in the directory.</li>
<li>permission to execute the file (run it as a program). For directories, this means permission to access files in the directory.</li>
</ol>
<p>There are three categories of users who may have different permissions to perform any of the above operations on a file:</p>
<ol>
<li>the file's owner.</li>
<li>other users who are in the file's group</li>
<li>everyone else.</li>
</ol>
<p>Files are given an owner and group when they are created. Usually the owner is the current user and the group is the group of the directory the file is in, but this varies with the operating system, the file system the file is created on, and the way the file is created. You can change the owner and group of a file by using the <strong>chown</strong> and <strong>chgrp</strong> commands.</p>
<p>In addition to the three sets of three permissions listed above, the file mode bits have three special components, which affect only executable files (programs) and, on most systems, directories:</p>
<ol>
<li>Set the process's effective user ID to that of the file upon execution (called the set-user-ID bit, or sometimes the setuid bit). For directories on a few systems, give files created in the directory the same owner as the directory, no matter who creates them, and set the set-user-ID bit of newly-created subdirectories.</li>
<li>Set the process's effective group ID to that of the file upon execution (called the set-group-ID bit, or sometimes the setgid bit). For directories on most systems, give files created in the directory the same group as the directory, no matter what group the user who creates them is in, and set the set-group-ID bit of newly-created subdirectories.</li>
<li>Prevent unprivileged users from removing or renaming a file in a directory unless they own the file or the directory; this is called the restricted deletion flag for the directory, and is commonly found on world-writable directories like /tmp.</li>
</ol>
<p>For regular files on some older systems, save the program's text image on the swap device so it will load more quickly when run; this is called the <code>sticky bit</code>.</p>
<h2>Setting Permissions</h2>
<p>The basic symbolic operations on a file's permissions are adding, removing, and setting the permission that certain users have to read, write, and execute or search the file. These operations have the following format:</p>
<p><code>users operation permissions</code></p>
<p>The spaces between the three parts above are shown for readability only; symbolic modes cannot contain spaces.  The users part tells which users' access to the file is changed. It consists of one or more of the following letters (or it can be empty). When more than one of these letters is given, the order that they are in does not matter.</p>
<ul>
<li><strong>u</strong> - the user who owns the file.</li>
<li><strong>g</strong> - other users who are in the file's group.</li>
<li><strong>o</strong> - all other users.</li>
<li><strong>a</strong> - all users; the same as ugo.</li>
</ul>
<p>The operation part tells how to change the affected users' access to the file, and is one of the following symbols:</p>
<ul>
<li><strong>+</strong> - to add the permissions to whatever permissions the users already have for the file.</li>
<li><strong>-</strong> - to remove the permissions from whatever permissions the users already have for the file.</li>
<li><strong>=</strong> - to make the permissions the only permissions that the users have for the file.</li>
</ul>
<p>The permissions part tells what kind of access to the file should be changed; it is normally zero or more of the following letters. As with the users part, the order does not matter when more than one letter is given. Omitting the permissions part is useful only with the = operation, where it gives the specified users no access at all to the file.</p>
<ul>
<li><strong>r</strong> - the permission the users have to read the file.</li>
<li><strong>w</strong> - the permission the users have to write to the file.</li>
<li><strong>x</strong> - the permission the users have to execute the file, or search it if it is a directory.</li>
</ul>
<p>For example, to give everyone permission to read and write a regular file, but not to execute it, use:</p>
<pre class='brushbash'>
a=rw
</pre>
<p>To remove write permission for all users other than the file's owner, use:</p>
<pre class='brushbash'>
go-w
</pre>
<p>The above command does not affect the access that the owner of the file has to it, nor does it affect whether other users can read or execute the file.</p>
<p>To give everyone except a file's owner no permission to do anything with that file, use the mode below. Other users could still remove the file, if they have write permission on the directory it is in.</p>
<pre class='brushbash'>
go=
</pre>
<p>Another way to specify the same thing is:</p>
<pre class='brushbash'>
og-rwx
</pre>
<h2>Copying Existing Permissions</h2>
<p>You can base a file's permissions on its existing permissions. To do this, instead of using a series of <strong>r, w, or x</strong> letters after the operator, you use the letter <strong>u, g, or o</strong>. For example, the mode</p>
<pre class='brushbash'>
o+g
</pre>
<p>adds the permissions for users who are in a file's group to the permissions that other users have for the file. Thus, if the file started out as mode 664 (rw-rw-r--), the above mode would change it to mode 666 (rw-rw-rw-). If the file had started out as mode 741 (rwxr----x), the above mode would change it to mode 745 (rwxr--r-x). The - and = operations work analogously.</p>
<h2>Umask and Protection</h2>
<p>If the users part of a symbolic mode is omitted, it defaults to a (affect all users), except that any permissions that are set in the system variable umask are not affected. The value of umask can be set using the umask command. Its default value varies from system to system.</p>
<p>Omitting the users part of a symbolic mode is generally not useful with operations other than +. It is useful with + because it allows you to use umask as an easily customizable protection against giving away more permission to files than you intended to.  As an example, if umask has the value 2, which removes write permission for users who are not in the file's group, then the mode:</p>
<pre class='brushbash'>
+w
</pre>
<p>adds permission to write to the file to its owner and to other users who are in the file's group, but not to other users. In contrast, the mode:</p>
<pre class='brushbash'>
a+w
</pre>
<p>ignores umask, and does give write permission for the file to all users.</p>
<h2>Directories, Set-User-ID and Set-Group-ID Bits</h2>
<p>On most systems, if a directory's set-group-ID bit is set, newly created subfiles inherit the same group as the directory, and newly created subdirectories inherit the set-group-ID bit of the parent directory. On a few systems, a directory's set-user-ID bit has a similar effect on the ownership of new subfiles and the set-user-ID bits of new subdirectories. These mechanisms let users share files more easily, by lessening the need to use chmod or chown to share new files.</p>
<p>These convenience mechanisms rely on the set-user-ID and set-group-ID bits of directories. If commands like chmod and mkdir routinely cleared these bits on directories, the mechanisms would be less convenient and it would be harder to share files. Therefore, a command like chmod does not affect the set-user-ID or set-group-ID bits of a directory unless the user specifically mentions them in a symbolic mode, or sets them in a numeric mode. For example, on systems that support set-group-ID inheritance:</p>
<pre class='brushbash'>
# These commands leave the set-user-ID and
# set-group-ID bits of the subdirectories alone,
# so that they retain their default values.
mkdir A B C
chmod 755 A
chmod 0755 B
chmod u=rwx,go=rx C
mkdir -m 755 D
mkdir -m 0755 E
mkdir -m u=rwx,go=rx F
</pre>
<p>If you want to try to set these bits, you must mention them explicitly in the symbolic or numeric modes, e.g.:</p>
<pre class='brushbash'>
# These commands try to set the set-user-ID
# and set-group-ID bits of the subdirectories.
mkdir G H
chmod 6755 G
chmod u=rwx,go=rx,a+s H
mkdir -m 6755 I
mkdir -m u=rwx,go=rx,a+s J
</pre>
<p>If you want to try to clear these bits, you must mention them explicitly in a symbolic mode, e.g.:</p>
<pre class='brushbash'>
# This command tries to clear the set-user-ID
# and set-group-ID bits of the directory D.
chmod a-s D
</pre>
<h2>Numeric Modes</h2>
<p>The permissions granted to the user, to other users in the file's group, and to other users not in the file's group each require three bits, which are represented as one octal digit. The three special mode bits also require one bit each, and they are as a group represented as another octal digit. Here is how the bits are arranged, starting with the lowest valued bit:</p>
<h3>Other users not in the file's group:</h3>
<pre class='brushbash'>
1 Execute/search
2 Write
4 Read
</pre>
<h3>Other users in the file's group:</h3>
<pre class='brushbash'>
10 Execute/search
20 Write
40 Read
</pre>
<h3>The file's owner:</h3>
<pre class='brushbash'>
100 Execute/search
200 Write
400 Read
</pre>
<h3>Special mode bits:</h3>
<pre class='brushbash'>
1000 Restricted deletion flag or sticky bit
2000 Set group ID on execution
4000 Set user ID on execution
</pre>
<p>For example, numeric <code>mode 4755</code> corresponds to symbolic mode <code>u=rwxs,go=rx</code>, and numeric m<code>ode 664</code> corresponds to symbolic mode <code>ug=rw,o=r</code>. Numeric <code>mode 0</code> corresponds to symbolic mode <code>a=</code>.</p>
<h2>Apache's Internal Bits (hex)</h2>
<pre>
#define APR_FPROT_USETID   0x8000 /* Set user id */
#define APR_FPROT_UREAD   0x0400 /* Read by user */
#define APR_FPROT_UWRITE   0x0200 /* Write by user */
#define APR_FPROT_UEXECUTE 0x0100 /* Execute by user */
&nbsp;
#define APR_FPROT_GSETID   0x4000 /* Set group id */
#define APR_FPROT_GREAD   0x0040 /* Read by group */
#define APR_FPROT_GWRITE   0x0020 /* Write by group */
#define APR_FPROT_GEXECUTE 0x0010 /* Execute by group */
&nbsp;
#define APR_FPROT_WSTICKY 0x2000 /* Sticky bit */
#define APR_FPROT_WREAD   0x0004 /* Read by others */
#define APR_FPROT_WWRITE 0x0002 /* Write by others */
#define APR_FPROT_WEXECUTE 0x0001 /* Execute by others */
&nbsp;
#define APR_FPROT_OS_DEFAULT  0x0FFF /* use OS&#039;s default permissions */
&nbsp;
/* additional permission flags for apr_file_copy  and apr_file_append */
#define APR_FPROT_FILE_SOURCE_PERMS 0x1000 /* Copy source file&#039;s permissions */
</pre>
<p>Download: <a href='/wp-content/uploads/2008/11/fileacc.c' title="A file to put ALL of the accessor functions for apr_file_t types"><code>httpd-2.2.10/srclib/apr/file_io/unix/fileacc.c</code></a> Here's some interesting bitmasking done by Apache that uses the defined bits set earlier by stat.h</p>
<pre>
apr_unix_perms2mode(perms){
 mode=0;
 if (perms &#038; APR_USETID) mode |= S_ISUID;
 if (perms &#038; APR_UREAD)  mode |= S_IRUSR;
 if (perms &#038; APR_UWRITE) mode |= S_IWUSR;
 if (perms &#038; APR_UEXECUTE) mode |= S_IXUSR;
&nbsp;
 if (perms &#038; APR_GSETID) mode |= S_ISGID;
 if (perms &#038; APR_GREAD)  mode |= S_IRGRP;
 if (perms &#038; APR_GWRITE) mode |= S_IWGRP;
 if (perms &#038; APR_GEXECUTE) mode |= S_IXGRP;
&nbsp;
 if (perms &#038; APR_WSTICKY) mode |= S_ISVTX;
 if (perms &#038; APR_WREAD)  mode |= S_IROTH;
 if (perms &#038; APR_WWRITE) mode |= S_IWOTH;
 if (perms &#038; APR_WEXECUTE) mode |= S_IXOTH;
 return mode;
}
&nbsp;
apr_unix_mode2perms(mode){
 perms = 0;
 if (mode &#038; S_ISUID)perms |= APR_USETID;
 if (mode &#038; S_IRUSR)perms |= APR_UREAD;
 if (mode &#038; S_IWUSR)perms |= APR_UWRITE;
 if (mode &#038; S_IXUSR)perms |= APR_UEXECUTE;
&nbsp;
 if (mode &#038; S_ISGID)perms |= APR_GSETID;
 if (mode &#038; S_IRGRP)perms |= APR_GREAD;
 if (mode &#038; S_IWGRP)perms |= APR_GWRITE;
 if (mode &#038; S_IXGRP)perms |= APR_GEXECUTE;
&nbsp;
 if (mode &#038; S_ISVTX)perms |= APR_WSTICKY;
 if (mode &#038; S_IROTH)perms |= APR_WREAD;
 if (mode &#038; S_IWOTH)perms |= APR_WWRITE;
 if (mode &#038; S_IXOTH)perms |= APR_WEXECUTE;
 return perms;
}
</pre>
<h2>umask</h2>
<pre>
umask(int mask){
 arg1;
 int oldumask;
 int arg_count = ZEND_NUM_ARGS();
 oldumask = umask(077);
&nbsp;
 if (BG(umask) == -1) BG(umask) = oldumask;
 if (arg_count == 0) umask(oldumask);
&nbsp;
 convert_to_long_ex(arg1);
 umask(Z_LVAL_PP(arg1));
 RETURN_LONG(oldumask);
}
</pre>
<h2>File Attributes</h2>
<p>Each file will have attributes based on the type of OS.. Using the stat command you can view them.</p>
<h3>Viewing stat results</h3>
<pre class='brushbash'>
* %a - Access rights in octal
* %A - Access rights in human readable form
* %b - Number of blocks allocated (see %B)
* %B - The size in bytes of each block reported by %b
* %d - Device number in decimal
* %D - Device number in hex
* %f - Raw mode in hex
* %F - File type
* %g - Group ID of owner
* %G - Group name of owner
* %h - Number of hard links
* %i - Inode number
* %n - File name
* %N - Quoted file name with dereference if symbolic link
* %o - I/O block size
* %s - Total size, in bytes
* %t - Major device type in hex
* %T - Minor device type in hex
* %u - User ID of owner
* %U - User name of owner
* %x - Time of last access
* %X - Time of last access as seconds since Epoch
* %y - Time of last modification
* %Y - Time of last modification as seconds since Epoch
* %z - Time of last change
* %Z - Time of last change as seconds since Epoch
</pre>
<h2>The OS Attribute Bits</h2>
<p>These defined values are what allows your operating system to determine the type of file being accessed. </p>
<pre>
#define S_IFMT   00170000  /* These bits determine file type. */
#define S_IFSOCK 0140000  /* Socket file */
#define S_IFLNK   0120000  /* Symbolic Link */
#define S_IFREG   0100000  /* Regular file */
#define S_IFDIR   0040000  /* Directory */
#define S_IFIFO  0010000   /* FIFO first-in-first-out file */
&nbsp;
/* Such devices can be read either a character at a time or a "block" (many characters) at a time,
hence we say there are block special files and character special files. */
#define S_IFBLK   0060000  /* Block device */
#define S_IFCHR  0020000  /* Character device */
</pre>
<h3>Special Permission Bits</h3>
<pre>
#define S_ISUID  0004000  /* Set user ID on execution.  */
#define S_ISGID  0002000  /* Set group ID on execution.  */
#define S_ISVTX  0001000 /* Save swapped text after use (sticky).  */
</pre>
<h3>Bitmasking to determine Filetype</h3>
<pre>
#define S_ISLNK(m) (((m) &#038; S_IFMT) == S_IFLNK)
#define S_ISREG(m) (((m) &#038; S_IFMT) == S_IFREG)
#define S_ISDIR(m) (((m) &#038; S_IFMT) == S_IFDIR)
#define S_ISCHR(m) (((m) &#038; S_IFMT) == S_IFCHR)
#define S_ISBLK(m) (((m) &#038; S_IFMT) == S_IFBLK)
#define S_ISFIFO(m) (((m) &#038; S_IFMT) == S_IFIFO)
#define S_ISSOCK(m) (((m) &#038; S_IFMT) == S_IFSOCK)
</pre>
<h3>Default Permission Masks</h3>
<pre>
#define S_IRWXUGO (S_IRWXU|S_IRWXG|S_IRWXO)
#define S_IALLUGO (S_ISUID|S_ISGID|S_ISVTX|S_IRWXUGO)
#define S_IRUGO  (S_IRUSR|S_IRGRP|S_IROTH)
#define S_IWUGO  (S_IWUSR|S_IWGRP|S_IWOTH)
#define S_IXUGO  (S_IXUSR|S_IXGRP|S_IXOTH)
</pre>
<p>Download: <a href='/wp-content/uploads/2008/11/filestat.c' title="handles file stat"><code>httpd-2.2.10/srclib/apr/file_io/unix/filestat.c</code></a>, this file shows a simple way to determine the type of file.</p>
<pre>
filetype_from_mode(mode){
 type;
 switch (mode &#038; S_IFMT) {
  case S_IFREG:  type = APR_REG;  break;
  case S_IFDIR:  type = APR_DIR;  break;
  case S_IFLNK:  type = APR_LNK;  break;
&nbsp;
  case S_IFCHR:  type = APR_CHR;  break;
  case S_IFBLK:  type = APR_BLK;  break;
  case S_IFFIFO: type = APR_PIPE; break;
  case S_IFSOCK: type = APR_SOCK; break;
  default: type = APR_UNKFILE;
 }
 return type;
}
</pre>
<h3>Apache Stat Bits</h3>
<pre>
#define APR_FINFO_LINK  0x00000001 /* Stat the link not the file itself if it is a link */
#define APR_FINFO_MTIME  0x00000010 /* Modification Time */
#define APR_FINFO_CTIME  0x00000020 /* Creation or inode-changed time */
#define APR_FINFO_ATIME  0x00000040 /* Access Time */
#define APR_FINFO_SIZE  0x00000100 /* Size of the file */
#define APR_FINFO_CSIZE  0x00000200 /* Storage size consumed by the file */
#define APR_FINFO_DEV  0x00001000 /* Device */
#define APR_FINFO_INODE  0x00002000 /* Inode */
#define APR_FINFO_NLINK  0x00004000 /* Number of links */
#define APR_FINFO_TYPE  0x00008000 /* Type */
#define APR_FINFO_USER  0x00010000 /* User */
#define APR_FINFO_GROUP  0x00020000 /* Group */
#define APR_FINFO_UPROT  0x00100000 /* User protection bits */
#define APR_FINFO_GPROT  0x00200000 /* Group protection bits */
#define APR_FINFO_WPROT  0x00400000 /* World protection bits */
#define APR_FINFO_ICASE  0x01000000 /* if dev is case insensitive */
#define APR_FINFO_NAME  0x02000000 /* name in proper case */
#define APR_FINFO_MIN  0x00008170 /* type, mtime, ctime, atime, size */
#define APR_FINFO_IDENT  0x00003000 /* dev and inode */
#define APR_FINFO_OWNER  0x00030000 /* user and group */
#define APR_FINFO_PROT  0x00700000 /* all protections */
#define APR_FINFO_NORM  0x0073b170 /* an atomic unix apr_stat() */
#define APR_FINFO_DIRENT 0x02000000 /* an atomic unix apr_dir_read() */
</pre>
<h3>The Apache file information structure.</h3>
<pre>
apr_uid_t user;  /* The user id that owns the file */
apr_gid_t group;  /* The group id that owns the file */
apr_ino_t inode; /* The inode of the file. */
apr_dev_t device; /* The id of the device the file is on. */
apr_int32_t nlink; /* The number of hard links to the file. */
apr_off_t size;  /* The size of the file */
apr_off_t csize; /* The storage size consumed by the file */
apr_time_t atime; /* The time the file was last accessed */
apr_time_t mtime; /* The time the file was last modified */
apr_time_t ctime; /* The time the file was created, or the inode was last changed */
const char *fname; /* The pathname of the file (possibly unrooted) */
const char *name; /* The file&#039;s name (no path) in filesystem case */
</pre>
<h3>File Time Attributes</h3>
<blockquote><h3><a href="http://php.net/manual/en/function.touch.php">touch</a></</p>
<p>If changing both the access and modification times to the current time, touch can change the timestamps for files that the user running it does not own but has write permission for. Otherwise, the user must own the files.</p>
<p>Although touch provides options for changing two of the times the times of last access and modification of a file, there is actually a third one as well: the inode change time. This is often referred to as a file's ctime. The inode change time represents the time when the file's meta-information last changed. One common example of this is when the permissions of a file change. Changing the permissions doesn't access the file, so the atime doesn't change, nor does it modify the file, so the mtime doesn't change. Yet, something about the file itself has changed, and this must be noted somewhere. This is the job of the ctime field. This is necessary, so that, for example, a backup program can make a fresh copy of the file, including the new permissions value. Another operation that modifies a file's ctime without affecting the others is renaming. In any case, it is not possible, in normal operations, for a user to change the ctime field to a user-specified value.</p>
</blockquote>
<p><a name="Shared_hosting_user_security"></a></p>
<h2>Shared hosting user security </h2>
<ul>
<li><a href="#Shared_hosting_user_security">Shared hosting user security</a></li>
<li><a href="#Apache_Security">Apache Security</a></li>
<li><a href="#Multiuser_security_setup_example">Multiuser security setup example</a></li>
<li><a href="#SSH_key_fingerprints">SSH key fingerprints</a></li>
<li><a href="#External_Links">External Links</a></li>
</ul>
<p>WebHost allows you to create multiple users per account. Each user can have domain assigned to its home home directory accessible via FTP or SSH/SCP. The problem with multiple users on the same account is that they share the same default unix group, and default permissions allow their files to be easily modified by the members of this group. Usually this doesn't pose a problem as each user is probably trusted by account owner to not to mess with others files, but if one of the users have their web application hacked then all other users on the same account will be in danger. </p>
<p>By default all files in your account are created with 644 privileges and directories are with 775. That means any user can read your files and any user from the same account can move and add files in your freshly made directories. Your home directory is different, though. By default it carries 751 attribute meaning that only members of your group can see your files, but can't add any new. These group access schemes are possible, because every user in your account has its primary/default group set to "pgxxxxxx", which is assigned to every new file you create by default. The normal way to secure users from web-intrusion is to assign a separate group to the web-server user, removing it from default group. This way, exploited scripts will not be able to traverse into home directories of other users on your account. To allow account users to update centralized web-site they could be added to web-site group explicitly. But this "normal way" doesn't work with DreamHost, because you can't delete web-user from the default group and unless you set access for every new file explicitly, it will be possible for an intruder to read it. </p>
<p>To make managing privileges easier in interactive sessions "umask 007" command can be specified in your .bash_profile - this makes all new files carry xx0 mask. You also need to control your scripts (web based or cron/shell) so that they set mask for critical files explicitly. To secure account users from access by means of hacked user script you would also like to define another group for every user in your account and change group ownership of the user's home directory to that group with "set gid" bit set (and optional umask 007 in .bash_profile). </p>
<p>Therefore, to secure your users from web-intrusion you need to: </p>
<ol>
<li>Add a separate user and group for every domain where apache will be running </li>
<li>Add a separate group for other user accounts </li>
<li>Change the default group for new files created by your users by changing the group of their home directory and setting "set gid" bit for it (it is impossible to do this with FTP accounts, therefore you will need to login in each account via SSH) </li>
<li>Add users who need access to web-site into the web-user group </li>
<li>Optionally set umask 007 in .bash_profile for every user to tweak default WebHost775/664 permissions to something like 770/660 for directories and files that are not meant to be read by Apache (660 could also be used for all web scripts including .php as they are not read by dhapache CGI, but merely executed) </li>
</ol>
<h2><a name="Apache_Security"></a>Apache Security </h2>
<p>All your web files that need to be read by Apache should be readable by everyone as Apache itself is run under dhapache user. However, executable scripts like .php are executed under your own user and do not have to be world readable as they are not actually read by Apache, but executed via <a href="http://en.wikipedia.org/wiki/suEXEC">suEXEC</a>. Quite the opposite - to prevent your code or database settings from being messed by any third-parties you SHOULD set permissions to these files explicitly to something like 640 or even 600 depending on who do you trust. </p>
<p><a name="Multiuser_security_setup_example"></a></p>
<h2>Multiuser security setup example </h2>
<p>For our example, we will create a <em>rainforce_www</em> user and a <em>aapp_www</em> group for serving web files with apache and setup a <em>rainforce</em> user with a 'aapp<em> group to manage mail and keep other files on DH privately. Since these records already exist, you will need to subsitute your own names.</em> </p>
<ul>
<li>Login to create the users <em>rainforce_www</em> and <em>rainforce</em> with shell access. </li>
<li>Create two groups - <em>aapp_www</em> and <em>aapp</em>. Note that users created in previous step are still members of the same default <em>pg</em>xxxxxx group. </li>
<li>Add <em>rainforce_www</em> to 'the 'aapp_www<em> group and </em>rainforce<em> to both the </em>aapp_www<em> and </em>aapp<em> groups</em> </li>
<li>Move your domain to <em>rainforce_www</em> account (mine is rainforce.org) </li>
<li>Now login to SSH with your <em>rainforce_www</em> user and change the default group for your home directory with "sgid" bit set to make all current and new files/directories created in this directory have the same <em>aapp_www</em> group. </li>
</ul>
<pre class='brushbash'>
 $ chgrp -R aapp_www .
 $ chmod 2751 .
 $ chmod 2771 rainforce.org
</pre>
<p>By setting 2771 the directory will be writable by the owner, the group and will be only executable by others. The contents of an executable only directory cannot be listed, but the files inside it can be read (if the permissions of the file allow it). It is important that the directory can be executable in order to allow static content (e.g. .html files) inside it to be read. Remember that directories you don't want anyone to have web access to, should be 0770 (writable by the owner and group, or 0750 writable by the owner and readable by group). Such strict permissions should by applied to password files, php include files or databases files (such as SQLite, BDB, etc). </p>
<ul>
<li>Do the same for <em>rainforce</em> user, but specify <em>aapp</em> group instead. </li>
</ul>
<pre class='brushbash'>
 $ chgrp -R aapp .
 $ chmod 2751 .
</pre>
<ul>
<li>Optionally modify umask in .bash_profile in user's home to 007 to make all files created by this user have 660 permissions set by default. If you want that newly created files by accessible by the web, you need to manually setup it's permissions to 664. </li>
</ul>
<p>Now I can login as the user "rainforce" and update the web-site in the ../rainforce_www/rainforce.org directory. There is one more setup needed. Because files copied from other accounts can have 644 permissions set instead of 664, you need a script which will update permissions to 664 or 660 to allow other group members modify such files. </p>
<h2><a name="SSH_key_fingerprints"></a>SSH key fingerprints </h2>
<p>Just gen your own I guess </p>
<h2>External Links </h2>
<ul>
<li><a href="http://oldfield.wattle.id.au/luv/permissions.html" title="http://oldfield.wattle.id.au/luv/permissions.html" rel="nofollow">Introduction to Unix file permissions</a> </li>
<li><a href="http://www.perlfect.com/articles/chmod.shtml" title="http://www.perlfect.com/articles/chmod.shtml" rel="nofollow">Understanding UNIX permission and chmod</a> </li>
</ul>
<p>Original Article from <a href="http://wiki.dreamhost.com/index.php?title=Security">DreamHost Wiki</a></p>
<p>Content is available under <a href="http://www.gnu.org/copyleft/fdl.html" class="external " title="http://www.gnu.org/copyleft/fdl.html" rel="nofollow">GNU Free Documentation License 1.2</a>.</p>
<h2>Example File Permission Bits</h2>
<h3><code>/usr/lib/w3m/cgi-bin/dirlist.cgi</code></h3>
<pre>
sub utype {
  local($_) = @_;
  local(%T) = (
    0010000, &#039;PIPE&#039;,
    0020000, &#039;CHR&#039;,
    0040000, &#039;DIR&#039;,
    0060000, &#039;BLK&#039;,
    0100000, &#039;FILE&#039;,
    0120000, &#039;LINK&#039;,
    0140000, &#039;SOCK&#039;,
  );
  return $T{($_ &#038; 0170000)} || &#039;FILE&#039;;
}
&nbsp;
sub umode {
  local($_) = @_;
  local(%T) = (
    0010000, &#039;p&#039;,
    0020000, &#039;c&#039;,
    0040000, &#039;d&#039;,
    0060000, &#039;b&#039;,
    0100000, &#039;-&#039;,
    0120000, &#039;l&#039;,
    0140000, &#039;s&#039;,
  );
&nbsp;
  return ($T{($_ &#038; 0170000)} || &#039;-&#039;)
     . (($_ &#038; 00400) ? &#039;r&#039; : &#039;-&#039;)
     . (($_ &#038; 00200) ? &#039;w&#039; : &#039;-&#039;)
     . (($_ &#038; 04000) ? &#039;s&#039; :
       (($_ &#038; 00100) ? &#039;x&#039; : &#039;-&#039;))
     . (($_ &#038; 00040) ? &#039;r&#039; : &#039;-&#039;)
     . (($_ &#038; 00020) ? &#039;w&#039; : &#039;-&#039;)
     . (($_ &#038; 02000) ? &#039;s&#039; :
       (($_ &#038; 00010) ? &#039;x&#039; : &#039;-&#039;))
     . (($_ &#038; 00004) ? &#039;r&#039; : &#039;-&#039;)
     . (($_ &#038; 00002) ? &#039;w&#039; : &#039;-&#039;)
     . (($_ &#038; 01000) ? &#039;t&#039; :
       (($_ &#038; 00001) ? &#039;x&#039; : &#039;-&#039;));
}
</pre>
<h3><code>/usr/lib/perl/5.8.4/linux/stat.ph</code></h3>
<pre>
        eval &#039;sub S_IFMT () {00170000;}&#039; unless defined(&#038;S_IFMT);
        eval &#039;sub S_IFSOCK () {0140000;}&#039; unless defined(&#038;S_IFSOCK);
        eval &#039;sub S_IFLNK () {0120000;}&#039; unless defined(&#038;S_IFLNK);
        eval &#039;sub S_IFREG () {0100000;}&#039; unless defined(&#038;S_IFREG);
        eval &#039;sub S_IFBLK () {0060000;}&#039; unless defined(&#038;S_IFBLK);
        eval &#039;sub S_IFDIR () {0040000;}&#039; unless defined(&#038;S_IFDIR);
        eval &#039;sub S_IFCHR () {0020000;}&#039; unless defined(&#038;S_IFCHR);
        eval &#039;sub S_IFIFO () {0010000;}&#039; unless defined(&#038;S_IFIFO);
        eval &#039;sub S_ISUID () {0004000;}&#039; unless defined(&#038;S_ISUID);
        eval &#039;sub S_ISGID () {0002000;}&#039; unless defined(&#038;S_ISGID);
        eval &#039;sub S_ISVTX () {0001000;}&#039; unless defined(&#038;S_ISVTX);
        eval &#039;sub S_ISLNK {
            local($m) = @_;
            eval q(((($m) &#038;  &#038;S_IFMT) ==  &#038;S_IFLNK));
        }&#039; unless defined(&#038;S_ISLNK);
        eval &#039;sub S_ISREG {
            local($m) = @_;
            eval q(((($m) &#038;  &#038;S_IFMT) ==  &#038;S_IFREG));
        }&#039; unless defined(&#038;S_ISREG);
        eval &#039;sub S_ISDIR {
            local($m) = @_;
            eval q(((($m) &#038;  &#038;S_IFMT) ==  &#038;S_IFDIR));
        }&#039; unless defined(&#038;S_ISDIR);
        eval &#039;sub S_ISCHR {
            local($m) = @_;
            eval q(((($m) &#038;  &#038;S_IFMT) ==  &#038;S_IFCHR));
        }&#039; unless defined(&#038;S_ISCHR);
        eval &#039;sub S_ISBLK {
            local($m) = @_;
            eval q(((($m) &#038;  &#038;S_IFMT) ==  &#038;S_IFBLK));
        }&#039; unless defined(&#038;S_ISBLK);
        eval &#039;sub S_ISFIFO {
            local($m) = @_;
            eval q(((($m) &#038;  &#038;S_IFMT) ==  &#038;S_IFIFO));
        }&#039; unless defined(&#038;S_ISFIFO);
        eval &#039;sub S_ISSOCK {
            local($m) = @_;
            eval q(((($m) &#038;  &#038;S_IFMT) ==  &#038;S_IFSOCK));
        }&#039; unless defined(&#038;S_ISSOCK);
        eval &#039;sub S_IRWXU () {00700;}&#039; unless defined(&#038;S_IRWXU);
        eval &#039;sub S_IRUSR () {00400;}&#039; unless defined(&#038;S_IRUSR);
        eval &#039;sub S_IWUSR () {00200;}&#039; unless defined(&#038;S_IWUSR);
        eval &#039;sub S_IXUSR () {00100;}&#039; unless defined(&#038;S_IXUSR);
        eval &#039;sub S_IRWXG () {00070;}&#039; unless defined(&#038;S_IRWXG);
        eval &#039;sub S_IRGRP () {00040;}&#039; unless defined(&#038;S_IRGRP);
        eval &#039;sub S_IWGRP () {00020;}&#039; unless defined(&#038;S_IWGRP);
        eval &#039;sub S_IXGRP () {00010;}&#039; unless defined(&#038;S_IXGRP);
        eval &#039;sub S_IRWXO () {00007;}&#039; unless defined(&#038;S_IRWXO);
        eval &#039;sub S_IROTH () {00004;}&#039; unless defined(&#038;S_IROTH);
        eval &#039;sub S_IWOTH () {00002;}&#039; unless defined(&#038;S_IWOTH);
        eval &#039;sub S_IXOTH () {00001;}&#039; unless defined(&#038;S_IXOTH);
    }
    if(defined(&#038;__KERNEL__)) {
        eval &#039;sub S_IRWXUGO () {( &#038;S_IRWXU| &#038;S_IRWXG| &#038;S_IRWXO);}&#039; unless defined(&#038;S_IRWXUGO);
        eval &#039;sub S_IALLUGO () {( &#038;S_ISUID| &#038;S_ISGID| &#038;S_ISVTX| &#038;S_IRWXUGO);}&#039; unless defined(&#038;S_IALLUGO);
        eval &#039;sub S_IRUGO () {( &#038;S_IRUSR| &#038;S_IRGRP| &#038;S_IROTH);}&#039; unless defined(&#038;S_IRUGO);
        eval &#039;sub S_IWUGO () {( &#038;S_IWUSR| &#038;S_IWGRP| &#038;S_IWOTH);}&#039; unless defined(&#038;S_IWUGO);
        eval &#039;sub S_IXUGO () {( &#038;S_IXUSR| &#038;S_IXGRP| &#038;S_IXOTH);}&#039; unless defined(&#038;S_IXUGO);
        require &#039;linux/types.ph&#039;;
        require &#039;linux/time.ph&#039;;
    }
&nbsp;
</pre>
<p><a href="http://ftp.mozilla.org/pub/mozilla.org/mozilla/releases/mozilla1.8a2/src/mozilla-source-1.8a2.tar.bz2 ">Mozilla-Source 1.8a2</a></p>
<pre>
/* notice that these valuse are octal. */
const PERM_IRWXU = 00700;  /* read, write, execute/search by owner */
const PERM_IRUSR = 00400;  /* read permission, owner */
const PERM_IWUSR = 00200;  /* write permission, owner */
const PERM_IXUSR = 00100;  /* execute/search permission, owner */
const PERM_IRWXG = 00070;  /* read, write, execute/search by group */
const PERM_IRGRP = 00040;  /* read permission, group */
const PERM_IWGRP = 00020;  /* write permission, group */
const PERM_IXGRP = 00010;  /* execute/search permission, group */
const PERM_IRWXO = 00007;  /* read, write, execute/search by others */
const PERM_IROTH = 00004;  /* read permission, others */
const PERM_IWOTH = 00002;  /* write permission, others */
const PERM_IXOTH = 00001;  /* execute/search permission, others */
&nbsp;
const MODE_RDONLY   = 0x01;
const MODE_WRONLY   = 0x02;
const MODE_RDWR     = 0x04;
const MODE_CREATE   = 0x08;
const MODE_APPEND   = 0x10;
const MODE_TRUNCATE = 0x20;
const MODE_SYNC     = 0x40;
const MODE_EXCL     = 0x80;
</pre>
<h3><code>/usr/include/libpng12/png.h</code></h3>
<pre>
/* Transform masks for the high-level interface */
#define PNG_TRANSFORM_IDENTITY       0x0000    /* read and write */
#define PNG_TRANSFORM_STRIP_16       0x0001    /* read only */
#define PNG_TRANSFORM_STRIP_ALPHA    0x0002    /* read only */
#define PNG_TRANSFORM_PACKING        0x0004    /* read and write */
#define PNG_TRANSFORM_PACKSWAP       0x0008    /* read and write */
#define PNG_TRANSFORM_EXPAND         0x0010    /* read only */
#define PNG_TRANSFORM_INVERT_MONO    0x0020    /* read and write */
#define PNG_TRANSFORM_SHIFT          0x0040    /* read and write */
#define PNG_TRANSFORM_BGR            0x0080    /* read and write */
#define PNG_TRANSFORM_SWAP_ALPHA     0x0100    /* read and write */
#define PNG_TRANSFORM_SWAP_ENDIAN    0x0200    /* read and write */
#define PNG_TRANSFORM_INVERT_ALPHA   0x0400    /* read and write */
#define PNG_TRANSFORM_STRIP_FILLER   0x0800    /* WRITE only */
</pre>
<h3><code>/usr/lib/python2.4/stat.py</code></h3>
<pre>
# Extract bits from the mode
&nbsp;
def S_IMODE(mode):
    return mode &#038; 07777
&nbsp;
def S_IFMT(mode):
    return mode &#038; 0170000
&nbsp;
# Constants used as S_IFMT() for various file types
# (not all are implemented on all systems)
&nbsp;
S_IFDIR  = 0040000
S_IFCHR  = 0020000
S_IFBLK  = 0060000
S_IFREG  = 0100000
S_IFIFO  = 0010000
S_IFLNK  = 0120000
S_IFSOCK = 0140000
&nbsp;
# Functions to test for each file type
&nbsp;
def S_ISDIR(mode):
    return S_IFMT(mode) == S_IFDIR
&nbsp;
def S_ISCHR(mode):
    return S_IFMT(mode) == S_IFCHR
&nbsp;
def S_ISBLK(mode):
    return S_IFMT(mode) == S_IFBLK
&nbsp;
def S_ISREG(mode):
    return S_IFMT(mode) == S_IFREG
&nbsp;
def S_ISFIFO(mode):
    return S_IFMT(mode) == S_IFIFO
&nbsp;
def S_ISLNK(mode):
    return S_IFMT(mode) == S_IFLNK
&nbsp;
def S_ISSOCK(mode):
    return S_IFMT(mode) == S_IFSOCK
&nbsp;
# Names for permission bits
&nbsp;
S_ISUID = 04000
S_ISGID = 02000
S_ENFMT = S_ISGID
S_ISVTX = 01000
S_IREAD = 00400
S_IWRITE = 00200
S_IEXEC = 00100
S_IRWXU = 00700
S_IRUSR = 00400
S_IWUSR = 00200
S_IXUSR = 00100
S_IRWXG = 00070
S_IRGRP = 00040
S_IWGRP = 00020
S_IXGRP = 00010
S_IRWXO = 00007
S_IROTH = 00004
S_IWOTH = 00002
S_IXOTH = 00001
</pre>
<h3><code>/usr/include/bits/stat.h</code></h3>
<pre>
/* Encoding of the file mode.  */
&nbsp;
#define __S_IFMT        0170000 /* These bits determine file type.  */
&nbsp;
/* File types.  */
#define __S_IFDIR       0040000 /* Directory.  */
#define __S_IFCHR       0020000 /* Character device.  */
#define __S_IFBLK       0060000 /* Block device.  */
#define __S_IFREG       0100000 /* Regular file.  */
#define __S_IFIFO       0010000 /* FIFO.  */
#define __S_IFLNK       0120000 /* Symbolic link.  */
#define __S_IFSOCK      0140000 /* Socket.  */
&nbsp;
/* POSIX.1b objects.  Note that these macros always evaluate to zero.  But
   they do it by enforcing the correct use of the macros.  */
#define __S_TYPEISMQ(buf)  ((buf)-&gt;st_mode - (buf)-&gt;st_mode)
#define __S_TYPEISSEM(buf) ((buf)-&gt;st_mode - (buf)-&gt;st_mode)
#define __S_TYPEISSHM(buf) ((buf)-&gt;st_mode - (buf)-&gt;st_mode)
&nbsp;
/* Protection bits.  */
&nbsp;
#define __S_ISUID       04000   /* Set user ID on execution.  */
#define __S_ISGID       02000   /* Set group ID on execution.  */
#define __S_ISVTX       01000   /* Save swapped text after use (sticky).  */
#define __S_IREAD       0400    /* Read by owner.  */
#define __S_IWRITE      0200    /* Write by owner.  */
#define __S_IEXEC       0100    /* Execute by owner.  */
</pre>
<h3><code>/usr/include/linux/nfs.h</code></h3>
<pre>
#define NFS_FIFO_DEV    (-1)
#define NFSMODE_FMT     0170000
#define NFSMODE_DIR     0040000
#define NFSMODE_CHR     0020000
#define NFSMODE_BLK     0060000
#define NFSMODE_REG     0100000
#define NFSMODE_LNK     0120000
#define NFSMODE_SOCK    0140000
#define NFSMODE_FIFO    0010000
</pre>
<h3><code>/usr/include/linux/nfs3.h</code></h3>
<pre>
#define NFS3_FIFO_DEV           (-1)
#define NFS3MODE_FMT            0170000
#define NFS3MODE_DIR            0040000
#define NFS3MODE_CHR            0020000
#define NFS3MODE_BLK            0060000
#define NFS3MODE_REG            0100000
#define NFS3MODE_LNK            0120000
#define NFS3MODE_SOCK           0140000
#define NFS3MODE_FIFO           0010000
&nbsp;
/* Flags for access() call */
#define NFS3_ACCESS_READ        0x0001
#define NFS3_ACCESS_LOOKUP      0x0002
#define NFS3_ACCESS_MODIFY      0x0004
#define NFS3_ACCESS_EXTEND      0x0008
#define NFS3_ACCESS_DELETE      0x0010
#define NFS3_ACCESS_EXECUTE     0x0020
#define NFS3_ACCESS_FULL        0x003f
</pre>
<h3><code>/usr/include/linux/stat.h</code></h3>
<pre>
#define S_IFMT  00170000
#define S_IFSOCK 0140000
#define S_IFLNK  0120000
#define S_IFREG  0100000
#define S_IFBLK  0060000
#define S_IFDIR  0040000
#define S_IFCHR  0020000
#define S_IFIFO  0010000
#define S_ISUID  0004000
#define S_ISGID  0002000
#define S_ISVTX  0001000
&nbsp;
#define S_ISLNK(m)      (((m) &#038; S_IFMT) == S_IFLNK)
#define S_ISREG(m)      (((m) &#038; S_IFMT) == S_IFREG)
#define S_ISDIR(m)      (((m) &#038; S_IFMT) == S_IFDIR)
#define S_ISCHR(m)      (((m) &#038; S_IFMT) == S_IFCHR)
#define S_ISBLK(m)      (((m) &#038; S_IFMT) == S_IFBLK)
#define S_ISFIFO(m)     (((m) &#038; S_IFMT) == S_IFIFO)
#define S_ISSOCK(m)     (((m) &#038; S_IFMT) == S_IFSOCK)
&nbsp;
#define S_IRWXU 00700
#define S_IRUSR 00400
#define S_IWUSR 00200
#define S_IXUSR 00100
&nbsp;
#define S_IRWXG 00070
#define S_IRGRP 00040
#define S_IWGRP 00020
#define S_IXGRP 00010
&nbsp;
#define S_IRWXO 00007
#define S_IROTH 00004
#define S_IWOTH 00002
#define S_IXOTH 00001
</pre>
<h2>Further File Permissions Reading</h2>
<h3>Related PHP Functions</h3>
<ul>
<li><a href="http://php.net/manual/en/function.fileperms.php">fileperms</a></li>
<li><a href="http://php.net/manual/en/function.stat.php">stat</a></li>
<li><a href="http://php.net/manual/en/function.chmod.php">chmod</a></li>
<li><a href="http://php.net/manual/en/function.clearstatcache.php">clearstatcache</a></li>
<li><a href="http://php.net/manual/en/function.chown.php">chown</a></li>
<li><a href="http://php.net/manual/en/function.chgrp.php">chgrp</a></li>
<li><a href="http://php.net/manual/en/function.lchown.php">lchown</a></li>
<li><a href="http://php.net/manual/en/function.lchgrp.php">lchgrp</a></li>
<li><a href="http://php.net/manual/en/function.touch.php">touch</a></li>
<li><a href="http://php.net/manual/en/function.lstat.php">lstat</a></li>
<li><a href="http://php.net/manual/en/function.fstat.php">filestat</a></li>
<li><a href="http://php.net/manual/en/function.fileatime.php">fileatime</a></li>
<li><a href="http://php.net/manual/en/function.filectime.php">filectime</a></li>
<li><a href="http://php.net/manual/en/function.filegroup.php">filegroup</a></li>
<li><a href="http://php.net/manual/en/function.fileinode.php">fileinode</a></li>
<li><a href="http://php.net/manual/en/function.filemtime.php">filemtime</a></li>
<li><a href="http://php.net/manual/en/function.fileowner.php">fileowner</a></li>
<li><a href="http://php.net/manual/en/function.filesize.php">filesize</a></li>
<li><a href="http://php.net/manual/en/function.filetype.php">filetype</a></li>
<li><a href="http://php.net/manual/en/function.is-writable.php">is_writable</a></li>
<li><a href="http://php.net/manual/en/function.is-readable.php">is_readable</a></li>
<li><a href="http://php.net/manual/en/function.is-executable.php">is_executable</a></li>
<li><a href="http://php.net/manual/en/function.is-file.php">is_file</a></li>
<li><a href="http://php.net/manual/en/function.is-dir.php">is_dir</a></li>
<li><a href="http://php.net/manual/en/function.is-link.php">is_link</a></li>
<li><a href="http://php.net/manual/en/function.file-exists.php">file_exists</a></li>
<li><a href="http://php.net/manual/en/function.disk-total-space.php">disk_total_space</a></li>
<li><a href="http://php.net/manual/en/function.disk-free-space.php">disk_free_space</a></li>
</ul>
<h3>Special file types</h3>
<ul>
<li><a href="http://www.gnu.org/software/coreutils/manual/coreutils.html#link-invocation">link invocation</a>:  Make a hard link via the link syscall</li>
<li><a href="http://www.gnu.org/software/coreutils/manual/coreutils.html#ln-invocation">ln invocation</a>: Make links between files</li>
<li><a href="http://www.gnu.org/software/coreutils/manual/coreutils.html#mkdir-invocation">mkdir invocation</a>: Make directories</li>
<li><a href="http://www.gnu.org/software/coreutils/manual/coreutils.html#mkfifo-invocation">mkfifo invocation</a>: Make FIFOs (named pipes)</li>
<li><a href="http://www.gnu.org/software/coreutils/manual/coreutils.html#mknod-invocation">mknod invocation</a>: Make block or character special files</li>
<li><a href="http://www.gnu.org/software/coreutils/manual/coreutils.html#readlink-invocation">readlink invocation</a>: Print the referent of a symbolic link</li>
<li><a href="http://www.gnu.org/software/coreutils/manual/coreutils.html#rmdir-invocation">rmdir invocation</a>: Remove empty directories</li>
<li><a href="http://www.gnu.org/software/coreutils/manual/coreutils.html#unlink-invocation">unlink invocation</a>: Remove files via unlink syscall</li>
</ul>
<h3>Changing file attributes</h3>
<ul>
<li><a href="http://www.gnu.org/software/coreutils/manual/coreutils.html#chown-invocation">chown invocation</a>: Change file owner and group</li>
<li><a href="http://www.gnu.org/software/coreutils/manual/coreutils.html#chgrp-invocation">chgrp invocation</a>: Change group ownership</li>
<li><a href="http://www.gnu.org/software/coreutils/manual/coreutils.html#chmod-invocation">chmod invocation</a>: Change access permissions</li>
<li><a href="http://www.gnu.org/software/coreutils/manual/coreutils.html#touch-invocation">touch invocation</a>: Change file timestamps</li>
</ul>
<p><a href="http://www.askapache.com/security/chmod-stat.html">Chmod, Umask, Stat, Fileperms, and File Permissions</a> originally appeared on AskApache.</p>]]></content:encoded>
			<wfw:commentRss>http://www.askapache.com/security/chmod-stat.html/feed/</wfw:commentRss>
		<slash:comments>5</slash:comments>
		</item>
		<item>
		<title>Optimize a Website for Speed, Security, and Easy Management</title>
		<link>http://www.askapache.com/htaccess/optimize-website-files-cache-security.html</link>
		<comments>http://www.askapache.com/htaccess/optimize-website-files-cache-security.html#comments</comments>
		<pubDate>Sun, 19 Oct 2008 00:45:26 +0000</pubDate>
		<dc:creator>AskApache</dc:creator>
				<category><![CDATA[Apache]]></category>
		<category><![CDATA[Cache]]></category>
		<category><![CDATA[DreamHost]]></category>
		<category><![CDATA[Featured]]></category>
		<category><![CDATA[Hacking]]></category>
		<category><![CDATA[Linux Unix BSD]]></category>
		<category><![CDATA[PHP]]></category>
		<category><![CDATA[SEO]]></category>
		<category><![CDATA[Security]]></category>
		<category><![CDATA[Shell Scripting]]></category>
		<category><![CDATA[Web Design]]></category>
		<category><![CDATA[Web Hosting]]></category>
		<category><![CDATA[Webmaster]]></category>
		<category><![CDATA[WordPress]]></category>
		<category><![CDATA[htaccess]]></category>
		<category><![CDATA[compression]]></category>
		<category><![CDATA[hosting]]></category>
		<category><![CDATA[optimization]]></category>
		<category><![CDATA[server]]></category>

		<guid isPermaLink="false">http://www.askapache.com/?p=1380</guid>
		<description><![CDATA[<p><a href="http://www.askapache.com/htaccess/optimize-website-files-cache-security.html" class="IFL hs hs37" title="Discover how to setup and manage a website from top to bottom for optimized speed, security, and simplicity"></a>Learn how to setup, configure, secure, optimize, and create a low-maintenance website the AskApache way.  I'm piecing together all the hacks, tricks, methods, and ideas discussed throughout this blog and all across Netdom and glueing them all together to show you how to have the most optimized, crazy fastest, and best website setup I can think of.<br class="C" /></p>]]></description>
			<content:encoded><![CDATA[<p><img class="IFL" src="/wp-content/uploads/2008/09/computerimg.jpg" alt="optimized server setup" title="optimized server setup"  />Over the past 10 or so years I&#8217;ve been directly or indirectly involved in configuring/administrating/hacking thousands of websites, and I realized today that I&#8217;ve actually learned quite a bit about how to really make them work hard for me, instead of the other way around. It came as a mild shock to think of where I was back then vs. now because the improvements and optimizations are hundreds of smaller improvements, but taken together, the  optimization hacks I&#8217;ve found through trial and error and much reading are as Donald would say, <strong>YOOUUGE</strong> compared to a basic website setup.<br class="C" /></p>
<p class="cnote">I use this awesome skeleton setup for all my high-paying clients <em>sorry poor people!</em> and also of course on this blog, which I use as a bleeding-edge dev server for my crazy testing.  So realize that I&#8217;m already past this setup and using it to do cooler stuff.  In order for you to use these more advanced ideas, you first need to get up to speed on what I&#8217;m doing so you know what I&#8217;m talking about.  This article tries to help you accomplish that&#8230; remains to be seen.</p>
<h2>An Optimized Website, The Real Deal</h2>
<p>This first article is to give you some ideas and get you thinking and reading before the first article in this series comes out.  This series details how to setup, configure, secure, optimize, and manage a website the best possible way I can come up with.  It pieces together all the AskApache hacks and tricks and uses methods and ideas discussed all over this blog and all over the net and glues them all together to show you how to have the most optimized, fastest, best website setup I can think of.</p>
<p>Knowing the why and how behind the operation of a Web Server allows us to optimize that operation.  For this example we will be creating the website <code>www.askapache.com</code>, which will be running WordPress and php.  We will also set up <code>static.askapache.com</code> to serve all of our sites uploads, images, css and javascript files, flash files, etc. with advanced caching and security using Apache Server .htaccess files.  So lets get started and take a look at this site structure for a moment.</p>
<pre class='brushbash'>
/home/askapache.com
|&#45;- /home/askapache.com/backups/
|&#45;- /home/askapache.com/htdocs/
|&#45;- /home/askapache.com/inc/
|&#45;- /home/askapache.com/logs/
|&#45;- /home/askapache.com/static/
|&#45;- /home/askapache.com/tmp/
|&#45;- /home/askapache.com/.htpasswd-basic
`&#45;- /home/askapache.com/.htpasswd-digest
</pre>
<ul>
<li><code>/backups/</code> &#8211; For <a href="http://www.askapache.com/wordpress/encrypted-wordpress-site-backups.html">encrypted backups of WordPress database and site files</a>. And any other backups.</li>
<li><code>/htdocs/</code> &#8211; The document root for <code>www.askapache.com</code></li>
<li><code>/inc/</code> &#8211; Folder to keep your php include files for extra security and easy management.</li>
<li><code>/logs/</code> &#8211; Save your php, apache, and other logs here or create symlinks to them.</li>
<li><code>/static/</code> &#8211; The document root for <code>static.askapache.com</code></li>
<li><code>/tmp/</code> &#8211; Only need this if your host doesn&#8217;t already have a /tmp folder</li>
</ul>
<hr class="HR0" />
<h2>Strong Security, Top to Bottom</h2>
<p><img class="IFL" src="/wp-content/uploads/2008/09/1023103_warning_icon_32.jpg" width="150" alt="" title="Site Security with Apache" />Simply by implementing correct access permissions, file permissions, password protection and segmenting various folders and services we are already ahead of the game.  I&#8217;ve always taken security extremely seriously, so you can benefit from alot of the simple solutions I&#8217;m recommending for a really locked down site.<br class="C" /></p>
<p>Indeed, security is a major part of every step of this setup process, as security concerns are what drives a lot of the motivations I have for coming up with this setup in the first place.  We will be doing very simple but very effective site security like the following items, which is a short list compared to everything we will be doing.</p>
<ul>
<li>Fixing file permissions automatically</li>
<li>Searching for modified files on the server</li>
<li>Encrypting your backups</li>
<li>Get alerted to breakin attempts</li>
<li>Block tons of bad clients</li>
<li>Disallowing cgi scripts or any other handlers, just serve files.</li>
<li>Configuring PHP</li>
<li>Password Protection for certain areas</li>
</ul>
<h3>Ready for Warfare?</h3>
<p>My past work for an Internet Service Provider, followed by 4 years of auditing the security of organizations external/internal networks has given me a fresh perspective on website security, and I think it allows me to see what would really be effective at preventing and killing attacks.  In fact just last night I was once again doing some research into some off-the-wall security topics, and I discovered a new defense method that I will be writing about very soon.  I believe that this new method,  could be quickly adopted and implemented by hosting providers and software developers, which would result in us finally taking the Internet back from all those zombies and robots.  This method will be discussed in great detail soon, and will be a core part of this site setups security and optimization.</p>
<hr class="HR0" />
<h2>Built to <span style="color:red">Bleed Speed</span></h2>
<p><img class="IFL" src="/wp-content/uploads/2008/09/speedontheroadimg.jpg" width="150" alt="" title="326255_speed_on_the_road" />Serve&#8217;s files as fast and efficiently as possible using advanced caching, HTTP Protocols, php/server configurations.<br class="C" /></p>
<p>Many of the articles and research on this blog is about improving the speed and efficiency of your website.  In fact that is why I am helping develop open-source software to block spammers from WordPress blogs&#8230; not because I&#8217;m bothered by the spam, but because they make the net slow!  So lets look at some of the ideas we&#8217;ll be implementing.</p>
<p>Many techniques I&#8217;ve been using and tweaking for several years, and recently many of them were included in the high-performance websites list.  Of course we will be taking a look at this list in practical terms, meaning almost all of it, the caching, compression, etc., will be automated in keeping with our &#8220;comfort&#8221; goal, which is to say we want to make the Web Developer and Server Admin&#8217;s lives as easy and comfy as possible.  After all, we do the work right?</p>
<ol>
<li>Reduce HTTP requests &#8211; <a href="http://www.askapache.com/htaccess/speed-up-sites-with-htaccess-caching.html" title="304 If Modified article">Reducing 304 requests with Cache-Control Headers</a></li>
<li>Use a customized php.ini &#8211; <a href="http://www.askapache.com/php/custom-phpini-tips-and-tricks.html">Creating and using a custom PHP.ini</a></li>
<li>Add an Expires header &#8211; <a href="http://www.askapache.com/htaccess/speed-up-your-site-with-caching-and-cache-control.html#caching-with-mod_expires" title="mod_expires Caching article">Caching with mod_expires on Apache</a></li>
<li>Gzip components</li>
<li>Make CSS and unobtrusive Javascript as external files not inline</li>
<li>Reduce DNS lookups &#8211; Use Static IP address, use a subdomain with static IP address for static content.</li>
<li>Minimize Javascript &#8211; Refactor the code, compress with <a href="http://dojotoolkit.org/docs/shrinksafe">dojo</a></li>
<li>Avoid external redirects &#8211; <a href="http://www.askapache.com/htaccess/mod_rewrite-tips-and-tricks.html" title="mod_rewrite internal redirection and rewrites">Use internal redirection with mod_rewrite</a>, <a href="http://www.askapache.com/htaccess/301-redirect-with-mod_rewrite-or-redirectmatch.html" title="301 Redirect with mod_rewrite or RedirectMatch">The correct way to redirect with 301</a></li>
<li>Turn off ETags &#8211; <a href="http://www.askapache.com/htaccess/using-http-headers-with-htaccess.html#prevent-caching-with-htaccess">Prevent Caching with htaccess</a></li>
<li>Make AJAX cacheable and small</li>
</ol>
<h3>AskApache.com, Fastest Site Ever!</h3>
<p>Ok it <em>might</em> not be the #1, but surely the top 10.. ;)</p>
<p>I&#8217;m very proud of the performance I am able to achieve on this site.  Very proud.  I started looking for ways to improve the wp-cache and wp-super-cache WordPress plugins, and came up with hacks for both of them.. but they still didn&#8217;t do what I wanted so I started from scratch and wrote my own caching plugin.</p>
<p>With much more advanced caching options and unquestionably higher performance and lower time usage on the machine.  I&#8217;m hesitant to release it to the public until I get faded on it.. I just really love it.. it has been running my site for several months now and I keep finding ways to improve it.. Stay tuned.</p>
<p>One feature it has is the ability to allow negotiation of a resource between apache and the client.  Think almost transparent mutli-lingual blogs, mutliple formats per document (look at the rdf for this page for an example*). But that plugin is the future and this is the present.. so back to it we go.</p>
<hr class="HR0" />
<h2>Pamper the Webmaster with Extreme Comfort</h2>
<p><img class="IFL" src="/wp-content/uploads/2008/09/wwwonthebeachimg.jpg" width="150" alt="" title="Low Maintenance Web Development" />This section alone would make this setup appealing.  I have developed all types of techniques and methods to make my life as easy as possible.  I could literally DIE right now and this blog would continue to run and operate for years on its own.   The general philosophy that I have used to get to where I can goto the beach with my laptop and do all this crazy stuff is <strong>the idea of perfection</strong>.  That may sound a little put-offish, but it basically means I will focus in on one very specific area for improvement or research and just get sick with it.  Most of this blogs articles are enlightening examples of this in action.  I will take a relatively unknown or unused piece of code or software and experiment with it until I feel I have it down, then I move on to the next item of never-ending research.  Mostly I think this is just plain habit from when I was studying security.  I&#8217;m much better at this then that :)</p>
<h3>Apache ErrorDocuments</h3>
<p>The <a href="http://www.askapache.com/htaccess/apache-status-code-headers-errordocument.html">57 HTTP Status Codes and Apache ErrorDocuments</a> article is a prime example.  I was simply searching for an authoritative list of HTTP status codes, an issue not many web people find worth their time, and that search led to some of the most useful stuff I&#8217;ve found about the Web</p>
<p>This &#8220;Comfort&#8221; article will include multi-language, intelligent, and optimized error documents for handling any type of HTTP error with class and allow us to stop spammers, save bandwidth, redirect correctly, etc..  You will probably be surprised at all the uses an Apache ErrorDocument can have.. It IS one of the foundations of the HTTP-based Net after all.<br class="C" /></p>
<h3>Emphasis on Easy Upgrades</h3>
<p>The whole setup is geared to make hassle-free WordPress/PHP/application upgrades possible by keeping different types of files in separate places, keeping backups, other misc tricks and since all of these files are in /home/askapache.com, your FTP connection can still access every file easily.  Sometimes security and optimizing your server can lead to it being more of a pain to do updates, backups, and general maintenance.  This article tries to overwhelm the balance with a trifecta of goals.</p>
<h3>Move to a new host? Ok!</h3>
<p>Comfort to me also means being able to pack up the whole website and database and move to another web host in under an hour.  I can move the whole AskApache site to one of several other hosting providers accounts I have in about 30minutes.  If this was a clients site or I was getting paid more, I&#8217;d also be focused on round-robin DNS technology, balance-load setups, and just go crazy making it fast.</p>
<h3>Staying Online, Improving Uptime</h3>
<p>Ever since I started sharing information and software to stop all these resource hogging zombies attacking everything I&#8217;ve been attacked several times.  Normally I get over 10K exploit attempts or requests per day, which I pretty much block 100%.  But a few times they&#8217;ve actually tried to DDOS me off the net in a distributed attack.  I have implemented several &#8220;poor mans&#8221; techniques to put up your best effort at surviving, which I did.  Basically you want to configure your server to KILL connections just as fast as possible and prevent your server resources from skyrocketing and surpassing your quotas.  A skilled attacker could easily shut you down even without the use of a widespread botnet if they are clever, which could be devastating to your small blog or site if it goes down at a crucial instant.</p>
<hr class="HR0" />
<h2>Organization with Templates and Systems</h2>
<p>I used to work with a guy who did alot of the coldfusion programming for us, and I used to cringe every  time I was called in to upgrade a site or do a re-design.  Files and folders EVERYWHERE!  Literally images in every folder, multiple index.html, index1.html, index-old.html, and on and on it went.. It would take me hours just to reverse-engineer the site enough so I could modify files on it without having some unkown consequence happen.</p>
<h3>Do You Have a Cluttered Desktop?</h3>
<p>Everyone has this problem, what I do all the time is just grab everything on my desktop and put it in a folder named with the date.  Then the process repeats itself and invariably a few months later I&#8217;m looking at a cluttered screen again.</p>
<p>This absolutely is the worst thing that can happen to a website, worst for security, comfort for webmaster, and speed.  So this setup addresses that issue completely heads on.  With all the different pages, tools, and resources available on this blog, I can almost promise you that my site has less files than yours.  No small feat to be sure, but worth every second I spent researching how to do it now that its on and popping.</p>
<hr class="HR0" />
<h2>What&#8217;s a Website really?</h2>
<p>All hosts are different, but any host worth their salt is running some kind of <a href="http://www.askapache.com/linux-unix/">BSD/Linux</a> operating system, and that is good news because those operating systems all use very similarly excellent file/folder structures with huge organization systems.  If your web hosting provider is running on a Windows based operating system or other locked/proprietary software than this article is not for you and I would recommend switching hosts to a BSD/Linux open-source operating system.</p>
<h3>Listening for Requests with Web Hosting and DNS</h3>
<p>First you set your website up so it can start serving.</p>
<ol>
<li>You buy your domain name, which just gives you the right to use it.</li>
<li>You pay your webhost for an account on their machine running a Server connected to the Net <em>via a fast connection link</em>.</li>
<li>You pay a DNS provider to redirect requests for your domain  name to be sent to your webhosts machine running the server.</li>
</ol>
<h2>Sub-Domain for Serving Assets</h2>
<p>This is a very cool method I&#8217;ve been using more and more frequently because it makes updates, upgrades, and changes so much easier to manage.  And segmenting various parts of the site is smart security, and even smarter in the way of speeding up a website and keeping your <strong>servers running mean and lean</strong>.</p>
<h2>Full Site Structure Expanded</h2>
<pre class='brushbash'>
/home/askapache.com
|&#45;- /home/askapache.com/backups/
|&#45;- /home/askapache.com/htdocs/
|   |&#45;- /home/askapache.com/htdocs/about/
|   |&#45;- /home/askapache.com/htdocs/admin/
|   |&#45;- /home/askapache.com/htdocs/cgi-bin/
|   |&#45;- /home/askapache.com/htdocs/.htaccess
|   |&#45;- /home/askapache.com/htdocs/index.php
|   `&#45;- /home/askapache.com/htdocs/robots.txt
|&#45;- /home/askapache.com/inc/
|   |&#45;- /home/askapache.com/inc/config.inc.php
|   `&#45;- /home/askapache.com/inc/settings.inc.php
|&#45;- /home/askapache.com/logs/
|   |&#45;- /home/askapache.com/logs/access.log
|   |&#45;- /home/askapache.com/logs/error.log
|   |&#45;- /home/askapache.com/logs/logins.log
|   |&#45;- /home/askapache.com/logs/modsec_audit.log
|   |&#45;- /home/askapache.com/logs/modsec_debug.log
|   `&#45;- /home/askapache.com/logs/php_error.log
|&#45;- /home/askapache.com/static/
|   |&#45;- /home/askapache.com/static/css/
|   |&#45;- /home/askapache.com/static/flv/
|   |&#45;- /home/askapache.com/static/img/
|   |&#45;- /home/askapache.com/static/js/
|   |&#45;- /home/askapache.com/static/mp3/
|   |&#45;- /home/askapache.com/static/pdf/
|   |&#45;- /home/askapache.com/static/swf/
|   |&#45;- /home/askapache.com/static/.htaccess
|   |&#45;- /home/askapache.com/static/index.html
|   `&#45;- /home/askapache.com/static/robots.txt
|&#45;- /home/askapache.com/tmp/
|&#45;- /home/askapache.com/.htpasswd-basic
`&#45;- /home/askapache.com/.htpasswd-digest
</pre>
<h2>Full Expanded Structure</h2>
<pre class='brushbash'>
/home/askapache.com
|&#45;- /home/askapache.com/backups/
|&#45;- /home/askapache.com/htdocs/
|   |&#45;- /home/askapache.com/htdocs/about/
|   |   `&#45;- /home/askapache.com/htdocs/about/index.html
|   |&#45;- /home/askapache.com/htdocs/admin/
|   |   |&#45;- /home/askapache.com/htdocs/admin/.htaccess
|   |   `&#45;- /home/askapache.com/htdocs/admin/index.html
|   |&#45;- /home/askapache.com/htdocs/cgi-bin/
|   |   |&#45;- /home/askapache.com/htdocs/cgi-bin/bin/
|   |   |   |&#45;- /home/askapache.com/htdocs/cgi-bin/bin/.htaccess
|   |   |   |&#45;- /home/askapache.com/htdocs/cgi-bin/bin/php.cgi*
|   |   |   |&#45;- /home/askapache.com/htdocs/cgi-bin/bin/php.ini
|   |   |   |&#45;- /home/askapache.com/htdocs/cgi-bin/bin/php4.cgi*
|   |   |   `&#45;- /home/askapache.com/htdocs/cgi-bin/bin/php5.cgi*
|   |   |&#45;- /home/askapache.com/htdocs/cgi-bin/private/
|   |   |   |&#45;- /home/askapache.com/htdocs/cgi-bin/private/.htaccess
|   |   |   |&#45;- /home/askapache.com/htdocs/cgi-bin/private/debug.php
|   |   |   `&#45;- /home/askapache.com/htdocs/cgi-bin/private/stats.php
|   |   |&#45;- /home/askapache.com/htdocs/cgi-bin/.htaccess
|   |   |&#45;- /home/askapache.com/htdocs/cgi-bin/login.php
|   |   |&#45;- /home/askapache.com/htdocs/cgi-bin/printenv.cgi*
&nbsp;
|   |   `&#45;- /home/askapache.com/htdocs/cgi-bin/redir.cgi*
|   |&#45;- /home/askapache.com/htdocs/.htaccess
|   |&#45;- /home/askapache.com/htdocs/index.php
|   `&#45;- /home/askapache.com/htdocs/robots.txt
|&#45;- /home/askapache.com/inc/
|   |&#45;- /home/askapache.com/inc/config.php
|   `&#45;- /home/askapache.com/inc/functions.php
|&#45;- /home/askapache.com/logs/
|   |&#45;- /home/askapache.com/logs/access.log
|   |&#45;- /home/askapache.com/logs/error.log
|   |&#45;- /home/askapache.com/logs/logins.log
|   |&#45;- /home/askapache.com/logs/modsec_audit.log
|   |&#45;- /home/askapache.com/logs/modsec_debug.log
|   `&#45;- /home/askapache.com/logs/php_error.log
|&#45;- /home/askapache.com/static/
|   |&#45;- /home/askapache.com/static/css/
|   |   `&#45;- /home/askapache.com/static/css/apache.css
|   |&#45;- /home/askapache.com/static/flv/
|   |   `&#45;- /home/askapache.com/static/flv/apache.flv
|   |&#45;- /home/askapache.com/static/img/
|   |   |&#45;- /home/askapache.com/static/img/apache.gif
|   |   |&#45;- /home/askapache.com/static/img/apache.jpg
|   |   `&#45;- /home/askapache.com/static/img/apache.png
|   |&#45;- /home/askapache.com/static/js/
|   |   `&#45;- /home/askapache.com/static/js/apache.js
|   |&#45;- /home/askapache.com/static/mp3/
|   |   `&#45;- /home/askapache.com/static/mp3/apache.mp3
|   |&#45;- /home/askapache.com/static/pdf/
|   |   `&#45;- /home/askapache.com/static/pdf/apache.pdf
|   |&#45;- /home/askapache.com/static/swf/
|   |   `&#45;- /home/askapache.com/static/swf/apache.swf
|   |&#45;- /home/askapache.com/static/.htaccess
|   |&#45;- /home/askapache.com/static/index.html
|   `&#45;- /home/askapache.com/static/robots.txt
|&#45;- /home/askapache.com/tmp/
|&#45;- /home/askapache.com/.htpasswd-basic
`&#45;- /home/askapache.com/.htpasswd-digest
</pre>
<p>The buzz about apache and open-source is very real, apache is becoming more of a discussed topic as people realize the power and importance of <q cite="LL Cool J">Doing it and Doing it and Doing it well.</q> &#8211;  <small><a href="http://www.webmonkey.com/blog/Jumpbox_Offers_an_Easier_Way_to_Install_Movable_Type">Movable Type Apache Installs made easy</a>, <a href="http://www.ubuntugeek.com/webalizer-apache-web-server-log-file-analysis-tool.html">Checking out Apache Web logs</a>, <a href="http://eventurebiz.com/blog/securing-securing-your-wordpress-blog-post-6-protecting-the-wp-configphp-file/">Securing Wordpress with .htaccess</a>, <a href="http://marketingdefined.com/blog/wordpress/using-wordpress-permalink-redirect-plugins-correctly/">WordPress Permalinks and .htaccess</a>, <a href="http://corpocrat.com/2008/09/19/install-apache-mod_substitute/">New search and replace module for apache!</a>, <a href="http://www.csskarma.com/blog/creating-an-htaccess-template/">creating an .htaccess template</a>, <a href="http://www.thelinuxblog.com/htaccess-allow-from/">.htaccess allow directive</a></small></p>
<p class="anote">Check back in a week for the first article, or better yet subscribe to my <a href="http://www.askapache.com/feed/">rss feed</a> or use the comment form below to get notified.</p>
<p><a href="http://www.askapache.com/htaccess/optimize-website-files-cache-security.html">Optimize a Website for Speed, Security, and Easy Management</a> originally appeared on AskApache.</p>]]></content:encoded>
			<wfw:commentRss>http://www.askapache.com/htaccess/optimize-website-files-cache-security.html/feed/</wfw:commentRss>
		<slash:comments>5</slash:comments>
		</item>
		<item>
		<title>SEO Secrets of AskApache Part 2</title>
		<link>http://www.askapache.com/seo/seo-advanced-pagerank-indexing.html</link>
		<comments>http://www.askapache.com/seo/seo-advanced-pagerank-indexing.html#comments</comments>
		<pubDate>Fri, 17 Oct 2008 21:44:22 +0000</pubDate>
		<dc:creator>AskApache</dc:creator>
				<category><![CDATA[Apache]]></category>
		<category><![CDATA[CSS]]></category>
		<category><![CDATA[Cache]]></category>
		<category><![CDATA[DreamHost]]></category>
		<category><![CDATA[Featured]]></category>
		<category><![CDATA[Google]]></category>
		<category><![CDATA[Hacking]]></category>
		<category><![CDATA[Javascript]]></category>
		<category><![CDATA[Linux Unix BSD]]></category>
		<category><![CDATA[Making Money]]></category>
		<category><![CDATA[PHP]]></category>
		<category><![CDATA[SEO]]></category>
		<category><![CDATA[Server Administration]]></category>
		<category><![CDATA[Web Design]]></category>
		<category><![CDATA[Web Hosting]]></category>
		<category><![CDATA[Webmaster]]></category>
		<category><![CDATA[XHTML]]></category>
		<category><![CDATA[htaccess]]></category>
		<category><![CDATA[mod_rewrite]]></category>
		<category><![CDATA[askapache]]></category>
		<category><![CDATA[Pagerank]]></category>
		<category><![CDATA[seo secrets]]></category>

		<guid isPermaLink="false">http://www.askapache.com/?p=567</guid>
		<description><![CDATA[<p><a rel="lb" class="IFL hs hs31" href='http://www.askapache.com/seo/seo-advanced-pagerank-indexing.html' title="Advanced SEO part 2: Search Engine Indexing and Pagerank Control"></a>This is part II of the <a href="http://www.askapache.com/seo/seo-secrets.html">Advanced SEO used on AskApache.com Series</a> and describes how to control which urls are indexed by Search Engines and how to move them higher up in Search Results.<br class="C" /></p>]]></description>
			<content:encoded><![CDATA[<p>In <a href="http://www.askapache.com/seo/seo-secrets.html">part I: SEO Secrets of AskApache.com</a> we talked about content and building a website to be your &#8220;SEO Base&#8221;.  This article discusses some advanced SEO concepts to get a site indexed, move your best pages higher in search results, and controlling the pagerank/seo-juice of your site.  but relatively easy ways to control and tweak WHAT urls on your site are indexed, and HOW.  I&#8217;ve heard some people refer to this as &#8220;controlling pagerank flow&#8221; and &#8220;controlling pagerank juice&#8221;, basically we want our best pages to rank higher in the search engine results.</p>
<p><strong>Big Picture:</strong> Going from no website to AskApache.com in less than a year can be accomplished by anyone with unique content and a resolve to avoid any shortcuts and take it one step at a time.</p>
<p class="cnote">We want what Google wants, to provide the most relevant content for someone who is doing a search.   Basically, you want every url on your site that has unique content to be included in the index.  In other words, <strong>you have to think like a search engine</strong>!</p>
<hr class="C" />
<h2>The Goal of Google</h2>
<p>Here&#8217;s what I mean, Google approaches search with the overwhelming goal of bringing the content to a searcher that is the most-likely to be what that searcher is searching for.  Another way of looking at it is something I read on Google&#8230;</p>
<blockquote cite="http://www.google.com/"><p>Google&#8217;s goal is to get you off of their site as fast as possible by providing you with exactly what you are looking for.</p></blockquote>
<hr class="C" />
<h2>Think about SEO like this</h2>
<p>If you search for <a href="http://www.google.com/search?aq=f&#038;complete=1&#038;hl=en&#038;q=htaccess+tutorial+for+seo&#038;btnG=Search" title="Google Search for htaccess tutorial for seo">htaccess tutorial for seo</a> on Google, would you be more likely to visit a <strong>tutorial about using .htaccess for seo</strong> or <strong>a category page for htaccess articles</strong>?  AskApache.com has both of those urls included in the index, but the article ranks higher than the category page, as it very well should.</p>
<hr class="C" />
<h2>Get Your URLS Indexed</h2>
<p>Before I explain how I am able to help Google and other search engines rank my article pages higher than my category pages, we need to get the urls in the index or nothing will show up.  There are many well-discussed methods for getting included in the index, so I&#8217;ll just list a few that I use.</p>
<ol>
<li>Provide High-Quality original content, people will link to it</li>
<li>Get a sitemap and use <a href="https://www.google.com/webmasters/tools/siteoverview?hl=en">Google Webmaster Tools</a></li>
<li>Publish an <a href="http://feeds.askapache.com/apache/htaccess">RSS/Atom Feed</a> and ping the net when you publish a post</li>
<li>Use robots.txt and robots meta tags</li>
</ol>
<h3>Your URLs in the index</h3>
<p>Here&#8217;s how to find out which of your pages are indexed.</p>
<ul>
<li>Indexed pages in your site: <a href="http://www.google.com/search?q=site:www.askapache.com">site:</a></li>
<li>Pages that link to your site&#8217;s front page: <a href="http://www.google.com/search?q=link:www.askapache.com">link:</a></li>
<li>The current cache of your site: <a href="http://www.google.com/search?q=cache:www.askapache.com">cache:</a></li>
<li>Information about your site: <a href="http://www.google.com/search?q=info:www.askapache.com">info:</a></li>
<li>Pages that are similar to your site: <a href="http://www.google.com/search?q=related:www.askapache.com">related:</a></li>
</ul>
<hr class="C" />
<h2>Break It Down</h2>
<p>Yo homeslice! I didn&#8217;t mean break dance..  I mean lets simplify AskApache in the context of getting our urls indexed high/low.  Here&#8217;s the stats:  <strong>1 Homepage, 206 Articles, 19 Pages, 31 Categories</strong></p>
<h3>1 Homepage</h3>
<p>This page is generally the highest ranking page in the index, it should contain links to your best urls, and provide easy navigation</p>
<h3>206 Specific Topic Articles</h3>
<p>These are the article&#8217;s (like this one) of AskApache.com and are the main source of search engine traffic.  You want each url (if its a good article) to be ranked as high as possible.  Some keys are to really make each article specific to a topic by using best-practice (X)HTML.</p>
<h3>19 Static Pages</h3>
<p>Most of these are pages like the online-tools hosted on this site, or other basic pages like about, contact us, site-map, etc..  Some of these you may want to rank very high ( like the /about/ page ) and some you may not want to even be included in the index.</p>
<h3>31 Specific Topic Category Pages</h3>
<p>These are tricky because they are generally just lists of articles from each category, which isn&#8217;t specific enough to get much seach-engine-traffic, but is very useful to site visitors.  I beefed up my category pages by adding additional information about the category topic in addition to excerpts of the articles.</p>
<hr class="C" />
<h2>Higher Pagerank = Higher Up in Search Results</h2>
<p>So Googlebot and other search engine robots have these crazy complicated algorithms (many patented) that SEO Industry types may get caught up in and try to technically analyze them.  I&#8217;m sure you&#8217;ve seen/read/heard the complicated advice that will always be pushed by many&#8230; advice like:</p>
<ul>
<li>analyze the number of words in the description/title/1st paragraph/etc.</li>
<li>Make sure your &#8220;keyword&#8221; is sprinkled throughout the text every 10-30th word..</li>
<li>Other equally unexciting technical analysis</li>
</ul>
<p>Now if you&#8217;ve had success with that then props to you, success is success, but I personally choose to completely ignore all that.  The number 1 thing that the top search engines advise is to design your page for a <strong>Human Visitor</strong>, not a computer.  The golden rule for me is how I would rank the page, not how some algorithm would.</p>
<hr class="C" />
<h2>Designing for a Human Visitor</h2>
<p>This is a major factor in your site being at the top vs. nowhere.   You design your HTML to be as minimal as possible (see source code for my homepage) and contain ONLY the neccessary elements.  Above all, use semantically sound XHTML markup.  (view source of <a href="http://www.w3.org/" title="World Wide Web Consortium"><acronym title="World Wide Web Consortium">W3C</acronym></a>)</p>
<h3>External CSS/Javascript</h3>
<p>Get your javascript and CSS out of your HTML and use external files (like this site) ALWAYS!  You should start with just the HTML, no css, no colors, no javascript, and THEN you add the .css and then you add the javascript.</p>
<h3>Site Accessibility</h3>
<p>Say your browser didn&#8217;t have a mouse, didn&#8217;t support images, css, javascript, or even colors!  Your HTML should be structured such that your page is still easily readable and easy to navigate.  You can use lynx, links, and many other terminal-based browsers to test for this&#8230; please see the <a href="http://www.w3.org/WAI/">Web Accessibility Initiative (<acronym title="Web Accessibility Initiative">WAI</acronym>)</a> for detailed info.</p>
<blockquote cite="http://en.wikipedia.org/wiki/Web_accessibility"><p><a href="http://en.wikipedia.org/wiki/Web_accessibility">Web accessibility</a> refers to the practice of making websites usable by people of all abilities and disabilities. When sites are correctly designed, developed and edited, all users can have equal access to information and functionality. For example, when a site is coded with semantically meaningful HTML, with textual equivalents provided for images and with links named meaningfully, this helps blind users using text-to-speech software and/or text-to-Braille hardware.</p>
</blockquote>
<hr class="C" />
<h2>Controlling a URL&#8217;s Pagerank</h2>
<p>A few tools and techniques are available for controlling the &#8220;juice&#8221; or &#8220;pagerank&#8221; of your urls.</p>
<ol>
<li>Robots.txt</li>
<li>Robots Meta Tags</li>
<li>Links</li>
</ol>
<h2>Robots.txt</h2>
<p>I&#8217;ve done quite a bit of research and experimentation with <a href="http://www.askapache.com/search/robots.txt">robots.txt files</a>, which is a file located in the root of your website at <a href="http://www.askapache.com/robots.txt">http://www.askapache.com/robots.txt</a> that is downloaded by all legitimate search engine spiders/bots and used as a Blacklist to prevent certain urls from being indexed.  Here are a few of the articles on this site, which you may skip if you like as they don&#8217;t illustrate the big-picture that I am going to discuss now.</p>
<ul>
<li><a href="http://www.askapache.com/seo/robotstxt-mattcutts-noindex.html">Control Flow of Pagerank with robots.txt and NoFollow, NoIndex</a></li>
<li><a href="http://www.askapache.com/seo/updated-robotstxt-for-wordpress.html">WordPress robots.txt</a></li>
<li><a href="http://www.askapache.com/seo/seo-with-robotstxt.html">SEO with Robots.txt</a></li>
<li><a href="http://www.askapache.com/google/adsense-robots.html">Google AdSense using robots.txt</a></li>
<li><a href="http://www.askapache.com/seo/wordpress-robotstxt-seo.html">WordPress robots.txt file optimized for SEO and Google</a></li>
</ul>
<h3>How To Use Robots.txt</h3>
<p>Even though robots.txt files are for whitelisting and blacklisting urls, I have found that they should only be used as an extreme form of blacklisting.  When you Disallow a url in your robots.txt file, that means most search engine bots won&#8217;t even LOOK at the url.  As you can see in the below example, I only disallow urls that shouldn&#8217;t ever be LOOKED at.  The real powertool is the <strong>robots meta tag</strong>.</p>
<pre>
User-agent: *
Disallow: /cgi-bin
Disallow: /wp-admin
Disallow: /wp-includes
Disallow: /wp-content
&nbsp;
Sitemap: http://www.askapache.com/sitemap.xml
</pre>
<hr class="C" />
<h2>Robots Meta Tag</h2>
<p>Ok I&#8217;m really trying to simplify, because what you should understand is the big-picture.  Every page can have a robots meta tag in the header, and this robots meta tag can tell the search-engine to index/not-index AND follow/not-follow.  Here are some examples:</p>
<pre>
&lt;meta name="robots" content="index" /&gt;
&lt;meta name="robots" content="noindex" /&gt;
&lt;meta name="robots" content="noindex,follow" /&gt;
&lt;meta name="robots" content="index,nofollow" /&gt;
</pre>
<h3>content=&#8221;index&#8221; / noindex</h3>
<p><code>index</code> means the search engine is free to index, archive, cache, and follow the page whereas <code>noindex</code> means DO NOT include this page in the search engine results.</p>
<h3>content=&#8221;follow&#8221; / nofollow</h3>
<p><code>follow</code> means the search engine is free to LOOK at the page and follow the links on the page whereas <code>nofollow</code> means DO NOT follow the links on the page.</p>
<hr class="C" />
<h3>WordPress Auto-Robots meta tag code</h3>
<p>Just add this to any plugin file and it will add the right robots meta tag to your site..  tweak to taste.</p>
<pre>
function askapache_robots_header(){
 global $wpdb;
&nbsp;
 $robot = &#039;&lt;meta name="robots" content="noindex,follow,nocache,noarchive" /&gt;&#039;;

 if ( is_paged() || is_search() || is_404() || is_author() || is_tag() )
   $robot = &#039;&lt;meta name="robots" content="noindex,follow" /&gt;&#039;;
 elseif ( is_home() || is_front_page() || is_single() )
   $robot = &#039;&lt;meta name="robots" content="follow,index" /&gt;&#039;;
 elseif ( is_category() || is_page() )
   $robot = &#039;&lt;meta name="robots" content="follow" /&gt;&#039;;
&nbsp;
 echo $robot . "\n";
}
add_action( &#039;wp_head&#039;, &#039;askapache_robots_header&#039; );
</pre>
<hr class="C" />
<h2>Links</h2>
<p>External and Internal Links are the crux of SEO.  It&#8217;s important to start FIRST on your Internal Links and linking structure&#8230; Once you are satisfied that the correct pages are indexed and ranked appropriately, then you can begin to look at external links.</p>
<ul>
<li>The fewer links on a page, generally the better.</li>
<li>If every page of your site points to the same url on your site, pagerank goes up from the number of internal links.</li>
<li>Use of the <code>rel</code>, <code>title</code>, <code>alt</code> attribute semantically is very helpful.  (next, prev, index)</li>
<li>Put your best links higher up in the XHTML, and put helpful/solid links at the end.</li>
<li>You can add <code>rel="nofollow"</code> to links that you dont want followed.</li>
</ul>
<p>The web has gotten to be so full of malicious/non-helpful SEO activity that I recommend developing your content NOT external links.  If you want to do this right and provide great content that makes search engine users happy and makes the web better, then explore this blog and develop content until the next article in this series, where I&#8217;ll show you how to <strong>make your site explode</strong>.</p>
<p class="anote">Stay tuned for Part III, which will dive deeper into the pipeworks of AskApache.com</p>
<p><a href="http://www.askapache.com/seo/seo-advanced-pagerank-indexing.html">SEO Secrets of AskApache Part 2</a> originally appeared on AskApache.</p>]]></content:encoded>
			<wfw:commentRss>http://www.askapache.com/seo/seo-advanced-pagerank-indexing.html/feed/</wfw:commentRss>
		<slash:comments>3</slash:comments>
		</item>
	</channel>
</rss>
