2001-11-10 21:49:43 +00:00
|
|
|
<?xml version="1.0" encoding="iso-8859-1"?>
|
2005-04-10 09:56:39 +00:00
|
|
|
<!-- $Revision: 1.42 $ -->
|
2001-01-09 12:38:23 +00:00
|
|
|
<chapter id="features.http-auth">
|
|
|
|
<title>HTTP authentication with PHP</title>
|
|
|
|
|
|
|
|
<simpara>
|
2004-05-20 22:40:42 +00:00
|
|
|
The <acronym>HTTP</acronym> Authentication hooks in PHP are only available when it is
|
2001-01-09 12:38:23 +00:00
|
|
|
running as an Apache module and is hence not available in the CGI version.
|
|
|
|
In an Apache module PHP script, it is possible to use the
|
2001-06-23 20:10:52 +00:00
|
|
|
<function>header</function> function to send an "Authentication Required"
|
2001-01-09 12:38:23 +00:00
|
|
|
message to the client browser causing it to pop up a Username/Password
|
|
|
|
input window. Once the user has filled in a username and a password,
|
2002-07-13 21:37:03 +00:00
|
|
|
the URL containing the PHP script will be called again with the
|
|
|
|
<link linkend="reserved.variables">predefined variables</link>
|
2004-05-24 21:20:47 +00:00
|
|
|
<varname>PHP_AUTH_USER</varname>, <varname>PHP_AUTH_PW</varname>,
|
|
|
|
and <varname>AUTH_TYPE</varname> set to the user name, password and
|
2002-07-13 21:37:03 +00:00
|
|
|
authentication type respectively. These predefined variables are found
|
|
|
|
in the <link linkend="reserved.variables.server">$_SERVER</link> and
|
2005-04-10 09:56:39 +00:00
|
|
|
<varname>$HTTP_SERVER_VARS</varname> arrays. Both "Basic" and "Digest"
|
|
|
|
(since PHP 5.1.0) authentication methods are supported. See the
|
|
|
|
<function>header</function> function for more information.
|
2002-07-13 21:37:03 +00:00
|
|
|
</simpara>
|
|
|
|
|
|
|
|
<note>
|
|
|
|
<title>PHP Version Note</title>
|
|
|
|
<para>
|
|
|
|
<link linkend="language.variables.superglobals">Autoglobals</link>,
|
|
|
|
such as <link linkend="reserved.variables.server">$_SERVER</link>, became
|
2004-03-06 02:40:23 +00:00
|
|
|
available in PHP <ulink url="&url.php.release4.1.0;">4.1.0</ulink>.
|
2002-07-13 21:37:03 +00:00
|
|
|
<varname>$HTTP_SERVER_VARS</varname> has been available since PHP 3.
|
|
|
|
</para>
|
|
|
|
</note>
|
|
|
|
|
2001-01-09 12:38:23 +00:00
|
|
|
<para>
|
|
|
|
An example script fragment which would force client authentication
|
2002-07-13 21:37:03 +00:00
|
|
|
on a page is as follows:
|
|
|
|
</para>
|
|
|
|
<para>
|
2001-01-09 12:38:23 +00:00
|
|
|
<example>
|
2005-04-10 09:56:39 +00:00
|
|
|
<title>Basic HTTP Authentication example</title>
|
2001-01-09 12:38:23 +00:00
|
|
|
<programlisting role="php">
|
2001-11-23 21:51:24 +00:00
|
|
|
<![CDATA[
|
|
|
|
<?php
|
2002-06-10 23:02:34 +00:00
|
|
|
if (!isset($_SERVER['PHP_AUTH_USER'])) {
|
2002-07-13 21:37:03 +00:00
|
|
|
header('WWW-Authenticate: Basic realm="My Realm"');
|
|
|
|
header('HTTP/1.0 401 Unauthorized');
|
2002-09-24 19:57:27 +00:00
|
|
|
echo 'Text to send if user hits Cancel button';
|
2001-01-09 12:38:23 +00:00
|
|
|
exit;
|
|
|
|
} else {
|
2002-06-10 23:02:34 +00:00
|
|
|
echo "<p>Hello {$_SERVER['PHP_AUTH_USER']}.</p>";
|
2002-07-02 06:19:44 +00:00
|
|
|
echo "<p>You entered {$_SERVER['PHP_AUTH_PW']} as your password.</p>";
|
2001-01-09 12:38:23 +00:00
|
|
|
}
|
2005-04-10 09:56:39 +00:00
|
|
|
?>
|
|
|
|
]]>
|
|
|
|
</programlisting>
|
|
|
|
</example>
|
|
|
|
</para>
|
|
|
|
|
|
|
|
<para>
|
|
|
|
<example>
|
|
|
|
<title>Digest HTTP Authentication example</title>
|
|
|
|
<para>
|
|
|
|
This example shows you how to implement a simple Digest HTTP
|
|
|
|
authentication script. For more information read the <ulink
|
|
|
|
url="&url.rfc;2617">RFC 2617</ulink>.
|
|
|
|
</para>
|
|
|
|
<programlisting role="php">
|
|
|
|
<![CDATA[
|
|
|
|
<?php
|
|
|
|
$realm = 'Restricted area';
|
|
|
|
|
|
|
|
//user => password
|
|
|
|
$users = array('admin' => 'mypass', 'guest' => 'guest');
|
|
|
|
|
|
|
|
|
|
|
|
if (!isset($_SERVER['PHP_AUTH_DIGEST'])) {
|
|
|
|
header('HTTP/1.1 401 Unauthorized');
|
|
|
|
header('WWW-Authenticate: Digest realm="'.$realm.
|
|
|
|
'" qop="auth" nonce="'.uniqid().'" opaque="'.md5($realm).'"');
|
|
|
|
|
|
|
|
die('Text to send if user hits Cancel button');
|
|
|
|
}
|
|
|
|
|
|
|
|
// analise the PHP_AUTH_DIGEST variable
|
|
|
|
preg_match('/username="(?P<username>.*)",\s*realm="(?P<realm>.*)",\s*nonce="(?P<nonce>.*)",\s*uri="(?P<uri>.*)",\s*response="(?P<response>.*)",\s*opaque="(?P<opaque>.*)",\s*qop=(?P<qop>.*),\s*nc=(?P<nc>.*),\s*cnonce="(?P<cnonce>.*)"/', $_SERVER['PHP_AUTH_DIGEST'], $digest);
|
|
|
|
|
|
|
|
if (!isset($users[$digest['username']]))
|
|
|
|
die('Username not valid!');
|
|
|
|
|
|
|
|
|
|
|
|
// generate the valid response
|
|
|
|
$A1 = md5($digest['username'] . ':' . $realm . ':' . $users[$digest['username']]);
|
|
|
|
$A2 = md5($_SERVER['REQUEST_METHOD'].':'.$digest['uri']);
|
|
|
|
$valid_response = md5($A1.':'.$digest['nonce'].':'.$digest['nc'].':'.$digest['cnonce'].':'.$digest['qop'].':'.$A2);
|
|
|
|
|
|
|
|
if ($digest['response'] != $valid_response)
|
|
|
|
die('Wrong Credentials!');
|
|
|
|
|
|
|
|
// ok, valid username & password
|
|
|
|
echo 'Your are logged in as: ' . $digest['username'];
|
|
|
|
|
2001-01-09 12:38:23 +00:00
|
|
|
?>
|
2001-11-23 21:51:24 +00:00
|
|
|
]]>
|
2001-01-09 12:38:23 +00:00
|
|
|
</programlisting>
|
2002-07-13 21:37:03 +00:00
|
|
|
</example>
|
|
|
|
</para>
|
2001-01-09 12:38:23 +00:00
|
|
|
|
2001-06-01 12:51:38 +00:00
|
|
|
<note>
|
2002-07-13 21:37:03 +00:00
|
|
|
<title>Compatibility Note</title>
|
2001-06-01 12:51:38 +00:00
|
|
|
<para>
|
|
|
|
Please be careful when coding the HTTP header lines. In order to guarantee maximum
|
|
|
|
compatibility with all clients, the keyword "Basic" should be written with an
|
|
|
|
uppercase "B", the realm string must be enclosed in double (not single) quotes,
|
2002-07-13 21:37:03 +00:00
|
|
|
and exactly one space should precede the <emphasis>401</emphasis> code in the
|
|
|
|
<emphasis>HTTP/1.0 401</emphasis> header line.
|
2001-06-01 12:51:38 +00:00
|
|
|
</para>
|
|
|
|
</note>
|
|
|
|
|
2001-01-09 12:38:23 +00:00
|
|
|
<para>
|
2004-05-24 21:20:47 +00:00
|
|
|
Instead of simply printing out <varname>PHP_AUTH_USER</varname>
|
|
|
|
and <varname>PHP_AUTH_PW</varname>, as done in the above example,
|
2002-07-13 21:37:03 +00:00
|
|
|
you may want to check the username and password for validity.
|
|
|
|
Perhaps by sending a query to a database, or by looking up the
|
|
|
|
user in a dbm file.
|
|
|
|
</para>
|
2001-01-09 12:38:23 +00:00
|
|
|
|
|
|
|
<para>
|
2001-06-19 22:18:05 +00:00
|
|
|
Watch out for buggy Internet Explorer browsers out there. They
|
2001-01-09 12:38:23 +00:00
|
|
|
seem very picky about the order of the headers. Sending the
|
|
|
|
<emphasis>WWW-Authenticate</emphasis> header before the
|
2002-02-08 13:40:44 +00:00
|
|
|
<literal>HTTP/1.0 401</literal> header seems to do the trick
|
2002-07-13 21:37:03 +00:00
|
|
|
for now.
|
|
|
|
</para>
|
2001-01-09 12:38:23 +00:00
|
|
|
|
|
|
|
<simpara>
|
2002-12-25 10:19:32 +00:00
|
|
|
As of PHP 4.3.0, in order to prevent someone from writing a script which
|
|
|
|
reveals the password for a page that was authenticated through a
|
|
|
|
traditional external mechanism, the PHP_AUTH variables will not be
|
2001-01-09 12:38:23 +00:00
|
|
|
set if external authentication is enabled for that particular
|
2003-02-04 13:05:06 +00:00
|
|
|
page and &safemode; is enabled. Regardless,
|
|
|
|
<varname>REMOTE_USER</varname> can be used
|
2002-12-25 10:19:32 +00:00
|
|
|
to identify the externally-authenticated user. So, you can use
|
2002-07-13 21:37:03 +00:00
|
|
|
<varname>$_SERVER['REMOTE_USER']</varname>.
|
|
|
|
</simpara>
|
2001-01-09 12:38:23 +00:00
|
|
|
|
2001-06-01 12:51:38 +00:00
|
|
|
<note>
|
|
|
|
<title>Configuration Note</title>
|
|
|
|
<para>
|
|
|
|
PHP uses the presence of an <literal>AuthType</literal> directive
|
|
|
|
to determine whether external authentication is in effect.
|
|
|
|
</para>
|
|
|
|
</note>
|
|
|
|
|
2001-01-09 12:38:23 +00:00
|
|
|
<simpara>
|
|
|
|
Note, however, that the above does not prevent someone who
|
|
|
|
controls a non-authenticated URL from stealing passwords from
|
2002-07-13 21:37:03 +00:00
|
|
|
authenticated URLs on the same server.
|
|
|
|
</simpara>
|
2001-01-09 12:38:23 +00:00
|
|
|
<simpara>
|
2001-01-12 23:25:52 +00:00
|
|
|
Both Netscape Navigator and Internet Explorer will clear the local browser
|
2001-01-09 12:38:23 +00:00
|
|
|
window's authentication cache for the realm upon receiving a
|
|
|
|
server response of 401. This can effectively "log out" a user,
|
|
|
|
forcing them to re-enter their username and password. Some people
|
2002-07-13 21:37:03 +00:00
|
|
|
use this to "time out" logins, or provide a "log-out" button.
|
|
|
|
</simpara>
|
|
|
|
<para>
|
2001-01-09 12:38:23 +00:00
|
|
|
<example>
|
2002-12-15 15:48:39 +00:00
|
|
|
<title>HTTP Authentication example forcing a new name/password</title>
|
|
|
|
<programlisting role="php">
|
2001-11-23 21:51:24 +00:00
|
|
|
<![CDATA[
|
|
|
|
<?php
|
2001-06-01 12:51:38 +00:00
|
|
|
function authenticate() {
|
2002-07-13 21:37:03 +00:00
|
|
|
header('WWW-Authenticate: Basic realm="Test Authentication System"');
|
|
|
|
header('HTTP/1.0 401 Unauthorized');
|
2001-11-23 21:51:24 +00:00
|
|
|
echo "You must enter a valid login ID and password to access this resource\n";
|
2001-06-01 12:51:38 +00:00
|
|
|
exit;
|
|
|
|
}
|
|
|
|
|
2003-01-19 09:46:52 +00:00
|
|
|
if (!isset($_SERVER['PHP_AUTH_USER']) ||
|
|
|
|
($_POST['SeenBefore'] == 1 && $_POST['OldAuth'] == $_SERVER['PHP_AUTH_USER'])) {
|
2001-06-01 12:51:38 +00:00
|
|
|
authenticate();
|
|
|
|
}
|
|
|
|
else {
|
2004-01-06 09:23:31 +00:00
|
|
|
echo "<p>Welcome: {$_SERVER['PHP_AUTH_USER']}<br />";
|
2002-07-02 17:27:25 +00:00
|
|
|
echo "Old: {$_REQUEST['OldAuth']}";
|
2004-01-06 09:23:31 +00:00
|
|
|
echo "<form action='{$_SERVER['PHP_SELF']}' METHOD='post'>\n";
|
|
|
|
echo "<input type='hidden' name='SeenBefore' value='1' />\n";
|
|
|
|
echo "<input type='hidden' name='OldAuth' value='{$_SERVER['PHP_AUTH_USER']}' />\n";
|
|
|
|
echo "<input type='submit' value='Re Authenticate' />\n";
|
2001-11-23 21:51:24 +00:00
|
|
|
echo "</form></p>\n";
|
2001-01-09 12:38:23 +00:00
|
|
|
}
|
|
|
|
?>
|
2001-11-23 21:51:24 +00:00
|
|
|
]]>
|
2002-07-13 21:37:03 +00:00
|
|
|
</programlisting>
|
|
|
|
</example>
|
|
|
|
</para>
|
2001-01-09 12:38:23 +00:00
|
|
|
<simpara>
|
|
|
|
This behavior is not required by the HTTP Basic authentication
|
|
|
|
standard, so you should never depend on this. Testing with Lynx
|
|
|
|
has shown that Lynx does not clear the authentication credentials
|
|
|
|
with a 401 server response, so pressing back and then forward
|
2001-01-12 23:25:52 +00:00
|
|
|
again will open the resource as long as the credential
|
|
|
|
requirements haven't changed. The user can press the
|
|
|
|
'_' key to clear their authentication information, however.
|
|
|
|
</simpara>
|
2001-01-09 12:38:23 +00:00
|
|
|
<simpara>
|
2003-06-05 17:36:29 +00:00
|
|
|
Also note that until PHP 4.3.3, HTTP Authentication did not work
|
|
|
|
using Microsoft's IIS server with the CGI version of PHP due to a
|
|
|
|
limitation of IIS. In order to get it to work in PHP 4.3.3+,
|
|
|
|
you must edit your IIS configuration "Directory Security". Click
|
|
|
|
on "Edit" and only check "Anonymous Access", all other fields
|
|
|
|
should be left unchecked.
|
2001-01-12 23:25:52 +00:00
|
|
|
</simpara>
|
2003-06-05 17:36:29 +00:00
|
|
|
<simpara>
|
2004-12-08 15:36:10 +00:00
|
|
|
Another limitation is if you're using the IIS module (ISAPI) and PHP 4, you
|
|
|
|
may not use the <literal>PHP_AUTH_*</literal> variables but instead, the
|
|
|
|
variable <literal>HTTP_AUTHORIZATION</literal> is available. For example,
|
|
|
|
consider the following code: <literal>list($user, $pw) = explode(':',
|
|
|
|
base64_decode(substr($_SERVER['HTTP_AUTHORIZATION'], 6)));</literal>
|
2003-06-05 17:36:29 +00:00
|
|
|
</simpara>
|
|
|
|
<note>
|
|
|
|
<title>IIS Note:</title>
|
|
|
|
<simpara>
|
|
|
|
For HTTP Authentication to work with IIS, the PHP directive
|
2003-06-05 17:39:38 +00:00
|
|
|
<link linkend="ini.cgi.rfc2616-headers">cgi.rfc2616_headers</link> must
|
2003-06-05 17:36:29 +00:00
|
|
|
be set to <literal>0</literal> (the default value).
|
|
|
|
</simpara>
|
|
|
|
</note>
|
2002-05-11 20:29:49 +00:00
|
|
|
<note>
|
|
|
|
<para>
|
2002-07-13 21:37:03 +00:00
|
|
|
If <link linkend="ini.safe-mode">safe mode</link> is enabled, the
|
2002-05-11 20:29:49 +00:00
|
|
|
uid of the script is added to the <literal>realm</literal> part of
|
|
|
|
the <literal>WWW-Authenticate</literal> header.
|
|
|
|
</para>
|
|
|
|
</note>
|
|
|
|
|
2001-01-09 12:38:23 +00:00
|
|
|
</chapter>
|
|
|
|
|
|
|
|
<!-- Keep this comment at the end of the file
|
|
|
|
Local variables:
|
|
|
|
mode: sgml
|
|
|
|
sgml-omittag:t
|
|
|
|
sgml-shorttag:t
|
|
|
|
sgml-minimize-attributes:nil
|
|
|
|
sgml-always-quote-attributes:t
|
|
|
|
sgml-indent-step:1
|
|
|
|
sgml-indent-data:t
|
2001-12-12 20:47:43 +00:00
|
|
|
indent-tabs-mode:nil
|
2001-01-09 12:38:23 +00:00
|
|
|
sgml-parent-document:nil
|
2001-09-21 22:47:49 +00:00
|
|
|
sgml-default-dtd-file:"../../manual.ced"
|
2001-01-09 12:38:23 +00:00
|
|
|
sgml-exposed-tags:nil
|
|
|
|
sgml-local-catalogs:nil
|
|
|
|
sgml-local-ecat-files:nil
|
|
|
|
End:
|
2001-09-21 22:47:49 +00:00
|
|
|
vim600: syn=xml fen fdm=syntax fdl=2 si
|
|
|
|
vim: et tw=78 syn=sgml
|
|
|
|
vi: ts=1 sw=1
|
2001-01-09 12:38:23 +00:00
|
|
|
-->
|