|
<br> php3中没有session的功能,以下是一个php4的backport,能为php3提供session <BR><BR>file: sessions.php3<BR>----------------------------------------------------- <BR><? <BR>/********************************************************************* <BR>******** <BR>* <BR>* <BR>* Web Application Development with PHP <BR>* <BR>* by <BR>* <BR>* Tobias Ratschiller and Till Gerken <BR>* <BR>* <BR>* <BR>* Copyright (c) 2000, New Riders Publishing <BR>* <BR>* <BR>* <BR>********************************************************************* <BR>******** <BR>* <BR>* <BR>* $Title: PHP 3 implementation of PHP 4's session management API $ <BR>* <BR>* $Chapter: Web Application Concepts $ <BR>* <BR>* $Executable: false $ <BR>* <BR>* <BR>* <BR>* $Description: <BR>* <BR>* This is a backport of the PHP 4 session_* functions to native PHP - <BR>so * <BR>* that you can use the same session management functions under both <BR>* <BR>* versions of PHP. They're believed to be about 75% compatible at the <BR>* <BR>* moment, but it's already possible to use the most common stuff. $ <BR>* <BR>* <BR>* <BR>********************************************************************* <BR>********/ <BR><BR>/* <BR>* Differences from PHP 4: <BR>* - no URL rewriting (of course) <BR>* - options aren't specified in the php.ini but in the session c <BR>lass below <BR>* - auto_start doesn't work with user callbacks <BR>* - the session ID is produced by a different algorithm <BR>* - shared memory support is still missing <BR>* - <?=SID?> doesn't work - use <?print($SID);?> <BR>* - the WDDX serializer doesn't work yet. <BR>* - serializing objects is limited due to PHP 3's serializer() <BR>* <BR>* Notes: <BR>* The session class contains the configuration variables. Th <BR>is is the <BR>* only part of the code you should need to edit. <BR>* <BR>* To reproduce the module concept used in PHP 4's session li <BR>brary, we <BR>* use classes. An example class has been been provided: file <BR>s. You can <BR>* easily create your own classes, for example a class mysql <BR>to store <BR>* session data to MySQL. It needs to provide the following f <BR>unctions: <BR>* bool open(string save_path, string sess_name): <BR>* used on startup of a session to initialize variables o <BR>r memory <BR>* returns false on error or true on success <BR>* bool close: <BR>* used on shutdown of a session to unset variables or fr <BR>ee memory <BR>* mixed read(string sess_id): <BR>* reads the session data of the session identified with <BR>sess_id. <BR>* returns false on error or the serialized session data <BR><BR>* bool write(string sess_id, string val): <BR>* saves the session data of the session identified with <BR>sess_id <BR>* returns false on error or true on success <BR>* bool destroy(string sess_id): <BR>* destroy the session identified with sess_id <BR>* returns false on error or true on success <BR>* bool gc(int max_lifetime): <BR>* provides garbage collection to remove sessions older t <BR>han <BR>* time() - max_lifetime <BR>* returns false on error or true on success <BR>* <BR>* While it may be faster to provide your own class, the reco <BR>mmended way <BR>* to add storage modules is to use session_set_save_handler( <BR>), as this <BR>* is compatible to PHP 4. <BR>*/ <BR><BR>$SID = ""; <BR>class session <BR>{ <BR>// Public variables <BR>var $auto_start = true; <BR>var $save_path = "/tmp"; <BR>var $name = " HPSESSID"; <BR>var $save_handler = "files"; <BR>var $lifetime = 0; <BR>var $gc_probability = 1; <BR>var $gc_maxlifetime = 0; <BR>var $serialize_handler = "php"; <BR>var $extern_referer_check = false; <BR>var $use_cookies = true; <BR>var $ID; <BR><BR>// Private variables <BR>var $nr_open_sessions = 0; <BR>var $mod_name = ""; <BR>var $id; <BR>var $delimiter = "\n"; <BR>var $delimiter_value = "[==]"; <BR><BR>function session() <BR>{ <BR>$this->mod_name = $this->save_handler; <BR>} <BR>} <BR><BR>class user <BR>{ <BR>var $open_func; <BR>var $close_func; <BR>var $read_func; <BR>var $write_func; <BR>var $destroy_func; <BR>var $gc_func; <BR><BR>function open($save_path, $sess_name) <BR>{ <BR>$func = $this->open_func; <BR>if(function_exists($func)) <BR>{ <BR>return($func($save_path, $sess_name)); <BR>} <BR><BR>return(true); <BR>} <BR><BR>function close($save_path, $sess_name) <BR>{ <BR>$func = $this->close_func; <BR>if(function_exists($func)) <BR>{ <BR>return($func()); <BR>} <BR><BR>return(true); <BR>} <BR><BR>function read($sess_id) <BR>{ <BR>$func = $this->read_func; <BR><BR>return($func($sess_id)); <BR>} <BR><BR>function write($sess_id, $val) <BR>{ <BR>$func = $this->write_func; <BR><BR>return($func($sess_id, $val)); <BR>} <BR><BR>function destroy($sess_id) <BR>{ <BR>$func = $this->destroy_func; <BR>if(function_exists($func)) <BR>{ <BR>return($func($sess_id)); <BR>} <BR><BR>return(true); <BR>} <BR><BR>function gc($max_lifteime) <BR>{ <BR>$func = $this->gc_func; <BR>if(function_exists($func)) <BR>{ <BR>return($func($max_lifetime)); <BR>} <BR><BR>return(true); <BR>} <BR><BR>} <BR><BR>class files <BR>{ <BR>function open($save_path, $sess_name) <BR>{ <BR>return(true); <BR>} <BR><BR>function close() <BR>{ <BR>return(true); <BR>} <BR><BR>function read($sess_id) <BR>{ <BR>global $session; <BR><BR>// Open, read in, close file with session data <BR>$file = $session->save_path."/sess$sess_id"; <BR>if (!file_exists($file)) <BR>{ <BR>// Create it <BR>touch($file); <BR>} <BR>$fp = fopen($file, "r") or die("Could not open session file ($ <BR>file)."); <BR>$val = fread($fp, filesize($file)); <BR>fclose($fp); <BR><BR>return($val); <BR>} <BR><BR>function write($sess_id, $val) <BR>{ <BR>global $session; <BR><BR>// Open, write to, close file with session data <BR>$file = $session->save_path."/sess$sess_id"; <BR>$fp = fopen($file, "w") or die("Could not write session file ( <BR>$file)"); <BR>$val = fputs($fp, $val); <BR>fclose($fp); <BR><BR>return(true); <BR>} <BR><BR>function destroy($sess_id) <BR>{ <BR>global $session; <BR><BR>$file = $session->save_path."/sess$sess_id"; <BR>unlink($file); <BR><BR>return(true); <BR>} <BR><BR>function gc($max_lifetime) <BR>{ <BR>// We return true, since all cleanup should be handled by <BR>// an external entity (i.e. find -ctime x | xargs rm) <BR>return(true); <BR>} <BR>} <BR><BR><BR>function _session_create_id() <BR>{ <BR>return(md5(uniqid(microtime()))); <BR>} <BR><BR>function _php_encode() <BR>{ <BR>global $session; <BR><BR>$ret = ""; <BR>// Create a string containing the serialized variables <BR>for ($i=0; $i<count($session->vars); $i++) <BR>{ <BR>$ret .= $session->vars[$i].$session->delimiter_value.serialize <BR>($GLOBALS[$session->vars[$i}).$session->delimiter; <BR>} <BR><BR>return($ret); <BR>} <BR><BR>function _php_decode($data) <BR>{ <BR>global $session; <BR><BR>$data = trim($data); <BR>$vars = explode($session->delimiter, $data); <BR><BR>// Add the variables to the global namespace <BR>for ($i=0; $i<count($vars); $i++) <BR>{ <BR>$tmp = explode($session->delimiter_value, $vars[$i]); <BR>$name = trim($tmp[0]); <BR>$value = trim($tmp[1]); <BR>$GLOBALS[$name] = unserialize($value); <BR>} <BR>} <BR><BR>function _wddx_encode($data) <BR>{ <BR>global $session; <BR><BR>$ret = wddx_serialize_vars($session->vars); <BR>return($ret); <BR>} <BR><BR>function _wddx_decode($data) <BR>{ <BR>return(wddx_deserialize($data)); <BR>} <BR><BR>function session_name($name = "") <BR>{ <BR>global $session; <BR>if(empty($name)) <BR>{ <BR>return($session->name); <BR>} <BR>$session->name = $name; <BR>} <BR><BR>function session_set_save_handler($open, $close, $read, $write, $destr <BR>oy, $gc) <BR>{ <BR>global $session, $user; <BR><BR>$user = new user; <BR>$user->open_func = $open; <BR>$user->close_func = $close; <BR>$user->read_func = $read; <BR>$user->write_func = $write; <BR>$user->destroy_func = $destroy; <BR>$user->gc_func = $gc; <BR>$session->mod_name = "user"; <BR>} <BR><BR>function session_module_name($name = "") <BR>{ <BR>global $session; <BR><BR>if(empty($name)) <BR>{ <BR>return($session->mod_name); <BR>} <BR>$session->mod_name = $name; <BR>} <BR><BR>function session_save_path($path = "") <BR>{ <BR>global $session; <BR><BR>if(empty($path)) <BR>{ <BR>return($session->save_path); <BR>} <BR>$session->save_path = $path; <BR>} <BR><BR>function session_id($id = "") <BR>{ <BR>global $session; <BR><BR>if(empty($id)) <BR>{ <BR>return($session->id); <BR>} <BR>$session->id = $id; <BR>} <BR><BR>function session_register($var) <BR>{ <BR>global $session; <BR><BR>if ($session->nr_open_sessions == 0) <BR>{ <BR>session_start(); <BR>} <BR>$session->vars[] = trim($var); <BR>} <BR><BR>function session_unregister($var) <BR>{ <BR>global $session; <BR><BR>for ($i=0; $i<count($session->vars); $i++) <BR>{ <BR>if ($session->vars[$i] == trim($var)) <BR>{ <BR>unset($session->vars[$i]); <BR>break; <BR>} <BR>} <BR>} <BR><BR>function session_is_registered($var) <BR>{ <BR>global $session; <BR><BR>for ($i=0; $i<count($session->vars); $i++) <BR>{ <BR>if ($session->vars[$i] == trim($var)) <BR>{ <BR>return(true); <BR>} <BR>} <BR><BR>return(false); <BR>} <BR><BR>function session_encode() <BR>{ <BR>global $session; <BR><BR>$serializer = "_".$session->serialize_handler."_encode"; <BR>$ret = $serializer(); <BR><BR>return($ret); <BR>} <BR><BR>function session_decode($data) <BR>{ <BR>global $session; <BR><BR>$serializer = "_".$session->serialize_handler."_decode"; <BR>$ret = $serializer($data); <BR><BR>return($ret); <BR>} <BR><BR>function session_start() <BR>{ <BR>global $session, $SID, $HTTP_COOKIE_VARS; <BR><BR>// Define the global variable $SID? <BR>$define_sid = true; <BR><BR>// Check if session_start() has been called once already <BR>if ($session->nr_open_sessions != 0) <BR>{ <BR>return(false); <BR>} <BR><BR>// Try to get the session ID <BR>if (empty($GLOBALS[$session->name])) <BR>{ <BR>// Maybe it is encoded into the URL (form <session-name>=<sess <BR>ion-id>? <BR>eregi($session->name."=([^/]+)", $GLOBALS["REQUEST_URI"], $reg <BR>s); <BR>$regs[1] = trim($regs[1]); <BR>if (!empty($regs[1])) <BR>{ <BR>$session->id = $regs[1]; <BR>} <BR>} <BR>else <BR>{ <BR>$session->id = $GLOBALS[$session->name]; <BR>if (isset($HTTP_COOKIE_VARS[$session->name])) <BR>{ <BR>$define_sid = false; <BR>} <BR>} <BR><BR>// Check for external referer <BR>if ($session->extern_referer_check) <BR>{ <BR>$url = parse_url($GLOBALS["HTTP_REFERER"]); <BR>if(trim($url["host"]) != $GLOBALS["SERVER_NAME"]) <BR>{ <BR>unset($session->id); <BR>$define_sid = true; <BR>} <BR>} <BR><BR>$mod = $GLOBALS[$session->mod_name]; <BR><BR>// Do we have an existing session ID? <BR>if (!empty($session->id)) <BR>{ <BR>// Start session <BR>if (!$mod->open($session->save_path, $session->name)) <BR>{ <BR>die("Failed to initialize session module."); <BR>} <BR>// Read session data <BR>if ($val = $mod->read($session->id)) <BR>{ <BR>// Decode session data <BR>session_decode($val); <BR>} <BR>} <BR>else <BR>{ <BR>// Create new session ID <BR>$session->id = _session_create_id(); <BR><BR>// Store cookie with this session ID? <BR>if ($session>use_cookies) <BR>{ <BR>SetCookie($session->name, $session->id, $session->lifetime <BR>); <BR>} <BR>} <BR><BR>// Check if we should clean up (call the garbage collection routin <BR>es) <BR>if ($session->gc_probability > 0) <BR>{ <BR>srand(time()); <BR>$randmax = getrandmax(); <BR>$nrand = (int)(100 * rand() / $randmax); <BR>if($nrand < $session->gc_probability) <BR>{ <BR>$mod->gc($session->gc_maxlifetime); <BR>} <BR>} <BR><BR>if($define_sid) <BR>{ <BR>$SID = $session->name."=".$session->id; <BR>} <BR>$session->nr_open_sessions++; <BR><BR>return(true); <BR>} <BR><BR>function session_destroy() <BR>{ <BR>global $session; <BR><BR>if($session->nr_open_sessions == 0) <BR>{ <BR>return(false); <BR>} <BR>// Destroy session <BR>$mod = $GLOBALS[$session->mod_name]; <BR>if (!$mod->destroy($session->name)) <BR>{ <BR>return(false); <BR>} <BR>unset($session); <BR>$session = new session; <BR><BR>return(true); <BR>} <BR><BR>function session_close() <BR>{ <BR>global $session, $SID; <BR><BR>if($session->nr_open_sessions == 0) <BR>{ <BR>return(false); <BR>} <BR>// Encode session <BR>$val = session_encode(); <BR>$len = strlen($val); <BR><BR>// Save session <BR>$mod = $GLOBALS[$session->mod_name]; <BR>if (!$mod->write($session->id, $val)) <BR>{ <BR>die("Session could not be saved."); <BR>} <BR>// Close session <BR>if (function_exists($session->mod_name."->close") &&!$mod->close() <BR>) <BR>{ <BR>die("Session could not be closed."); <BR>} <BR>$SID = ""; <BR>$session->nr_open_sessions--; <BR><BR>return(true); <BR>} <BR><BR>$session = new session; <BR>$mod = $session->save_handler; <BR>$$mod = new $mod; <BR><BR>if ($session->auto_start) <BR>{ <BR>$ret = session_start() or die("Session could not be started."); <BR>} <BR>register_shutdown_function("session_close"); <BR><BR>/* <BR>* Basic Example <BR>* <BR>* This basic example shows the normal use. The code is the same <BR>as in <BR>* PHP 4, except for the require("sessions.php3"); <BR><BR>require("sessions.php3"); <BR>session_start(); <BR>print("Our session ID is: ".session_id()." <BR>"); <BR>print("The counter value is: $counter <BR>"); <BR>print("The foo value is: $foo <BR>"); <BR>$counter++; <BR>$foo = "Foobar=Fobar"; <BR>session_register("counter"); <BR>session_register("foo"); <BR><BR>* <BR>*/ <BR><BR>/* <BR>* User Callback Example <BR>* <BR>* This example uses callback functions. It's a slightly modified <BR>version <BR>* of Sascha Schumann's original test script for the callbacks. 1 <BR>00% <BR>* the same code as in PHP 4 (except for the require(), of course <BR>). <BR><BR>require("sessions.php3"); <BR>function my_open($save_path, $sess_name) <BR>{ <BR>echo $save_path." <BR>"; <BR>echo $sess_name." <BR>"; <BR>return true; <BR>} <BR><BR>function my_read($sess_id) <BR>{ <BR>echo $sess_id." <BR>"; <BR>return true; <BR>} <BR><BR>function my_write($sess_id, $val) <BR>{ <BR>echo $val." <BR>"; <BR>return true; <BR>} <BR><BR>$foo = 10; <BR>session_set_save_handler("my_open", "", "my_read", "my_write", "", "") <BR>; <BR>session_start(); <BR>session_register("foo"); <BR>echo "foo: $foo"; <BR><BR>* <BR>*/ <BR>?> <br><br> |
|