| 
									
										
										
										
											2008-09-26 12:09:41 -04:00
										 |  |  | <?php | 
					
						
							|  |  |  | /* | 
					
						
							| 
									
										
										
										
											2009-08-25 18:14:12 -04:00
										 |  |  |  * StatusNet - the distributed open-source microblogging tool | 
					
						
							| 
									
										
										
										
											2009-08-25 18:12:20 -04:00
										 |  |  |  * Copyright (C) 2008, 2009, StatusNet, Inc. | 
					
						
							| 
									
										
										
										
											2008-09-26 12:09:41 -04:00
										 |  |  |  * | 
					
						
							|  |  |  |  * This program is free software: you can redistribute it and/or modify | 
					
						
							|  |  |  |  * it under the terms of the GNU Affero General Public License as published by | 
					
						
							|  |  |  |  * the Free Software Foundation, either version 3 of the License, or | 
					
						
							|  |  |  |  * (at your option) any later version. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							|  |  |  |  * GNU Affero General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU Affero General Public License | 
					
						
							|  |  |  |  * along with this program.  If not, see <http://www.gnu.org/licenses/>. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-26 10:41:36 -04:00
										 |  |  | if (!defined('STATUSNET') && !defined('LACONICA')) { exit(1); } | 
					
						
							| 
									
										
										
										
											2008-09-26 12:09:41 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-22 19:13:26 +01:00
										 |  |  | class Memcached_DataObject extends DB_DataObject | 
					
						
							| 
									
										
										
										
											2008-09-26 12:09:41 -04:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-01-04 14:37:39 -08:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Destructor to free global memory resources associated with | 
					
						
							|  |  |  |      * this data object when it's unset or goes out of scope. | 
					
						
							|  |  |  |      * DB_DataObject doesn't do this yet by itself. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function __destruct() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $this->free(); | 
					
						
							|  |  |  |         if (method_exists('DB_DataObject', '__destruct')) { | 
					
						
							|  |  |  |             parent::__destruct(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-04 14:38:56 -08:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Magic function called at serialize() time. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * We use this to drop a couple process-specific references | 
					
						
							|  |  |  |      * from DB_DataObject which can cause trouble in future | 
					
						
							|  |  |  |      * processes. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return array of variable names to include in serialization. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     function __sleep() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $vars = array_keys(get_object_vars($this)); | 
					
						
							|  |  |  |         $skip = array('_DB_resultid', '_link_loaded'); | 
					
						
							|  |  |  |         return array_diff($vars, $skip); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Magic function called at unserialize() time. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * Clean out some process-specific variables which might | 
					
						
							|  |  |  |      * be floating around from a previous process's cached | 
					
						
							|  |  |  |      * objects. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * Old cached objects may still have them. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     function __wakeup() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         // Refers to global state info from a previous process.
 | 
					
						
							|  |  |  |         // Clear this out so we don't accidentally break global
 | 
					
						
							|  |  |  |         // state in *this* process.
 | 
					
						
							|  |  |  |         $this->_DB_resultid = null; | 
					
						
							| 
									
										
										
										
											2010-01-10 12:31:43 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-04 14:38:56 -08:00
										 |  |  |         // We don't have any local DBO refs, so clear these out.
 | 
					
						
							|  |  |  |         $this->_link_loaded = false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Wrapper for DB_DataObject's static lookup using memcached | 
					
						
							|  |  |  |      * as backing instead of an in-process cache array. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param string $cls classname of object type to load | 
					
						
							|  |  |  |      * @param mixed $k key field name, or value for primary key | 
					
						
							|  |  |  |      * @param mixed $v key field value, or leave out for primary key lookup | 
					
						
							|  |  |  |      * @return mixed Memcached_DataObject subtype or false | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2008-12-23 14:33:23 -05:00
										 |  |  |     function &staticGet($cls, $k, $v=null) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |         if (is_null($v)) { | 
					
						
							|  |  |  |             $v = $k; | 
					
						
							|  |  |  |             # XXX: HACK!
 | 
					
						
							|  |  |  |             $i = new $cls; | 
					
						
							|  |  |  |             $keys = $i->keys(); | 
					
						
							|  |  |  |             $k = $keys[0]; | 
					
						
							|  |  |  |             unset($i); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-07-05 12:44:18 -04:00
										 |  |  |         $i = Memcached_DataObject::getcached($cls, $k, $v); | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |         if ($i) { | 
					
						
							|  |  |  |             return $i; | 
					
						
							|  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2010-01-01 10:57:22 -10:00
										 |  |  |             $i = DB_DataObject::factory($cls); | 
					
						
							|  |  |  |             if (empty($i)) { | 
					
						
							| 
									
										
										
										
											2010-01-12 12:20:45 -08:00
										 |  |  |                 $i = false; | 
					
						
							|  |  |  |                 return $i; | 
					
						
							| 
									
										
										
										
											2010-01-01 10:57:22 -10:00
										 |  |  |             } | 
					
						
							|  |  |  |             $result = $i->get($k, $v); | 
					
						
							|  |  |  |             if ($result) { | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |                 $i->encache(); | 
					
						
							| 
									
										
										
										
											2010-01-01 10:57:22 -10:00
										 |  |  |                 return $i; | 
					
						
							|  |  |  |             } else { | 
					
						
							| 
									
										
										
										
											2010-01-12 12:20:45 -08:00
										 |  |  |                 $i = false; | 
					
						
							|  |  |  |                 return $i; | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2008-10-02 10:47:15 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-23 14:33:23 -05:00
										 |  |  |     function &pkeyGet($cls, $kv) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2009-07-05 12:44:18 -04:00
										 |  |  |         $i = Memcached_DataObject::multicache($cls, $kv); | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |         if ($i) { | 
					
						
							|  |  |  |             return $i; | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             $i = new $cls(); | 
					
						
							|  |  |  |             foreach ($kv as $k => $v) { | 
					
						
							|  |  |  |                 $i->$k = $v; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if ($i->find(true)) { | 
					
						
							|  |  |  |                 $i->encache(); | 
					
						
							|  |  |  |             } else { | 
					
						
							| 
									
										
										
										
											2008-12-23 14:21:29 -05:00
										 |  |  |                 $i = null; | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2008-12-10 18:35:03 -05:00
										 |  |  |             return $i; | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2008-10-02 10:47:15 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-23 14:33:23 -05:00
										 |  |  |     function insert() | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |         $result = parent::insert(); | 
					
						
							|  |  |  |         return $result; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-01-22 19:13:26 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-23 14:33:23 -05:00
										 |  |  |     function update($orig=null) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |         if (is_object($orig) && $orig instanceof Memcached_DataObject) { | 
					
						
							|  |  |  |             $orig->decache(); # might be different keys
 | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         $result = parent::update($orig); | 
					
						
							|  |  |  |         if ($result) { | 
					
						
							|  |  |  |             $this->encache(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return $result; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-01-22 19:13:26 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-23 14:33:23 -05:00
										 |  |  |     function delete() | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |         $this->decache(); # while we still have the values!
 | 
					
						
							|  |  |  |         return parent::delete(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-01-22 19:13:26 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |     static function memcache() { | 
					
						
							|  |  |  |         return common_memcache(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-01-22 19:13:26 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |     static function cacheKey($cls, $k, $v) { | 
					
						
							| 
									
										
										
										
											2010-01-02 16:21:19 -10:00
										 |  |  |         if (is_object($cls) || is_object($k) || is_object($v)) { | 
					
						
							| 
									
										
										
										
											2009-12-11 14:19:18 -08:00
										 |  |  |             $e = new Exception(); | 
					
						
							|  |  |  |             common_log(LOG_ERR, __METHOD__ . ' object in param: ' . | 
					
						
							|  |  |  |                 str_replace("\n", " ", $e->getTraceAsString())); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |         return common_cache_key(strtolower($cls).':'.$k.':'.$v); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-01-22 19:13:26 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |     static function getcached($cls, $k, $v) { | 
					
						
							| 
									
										
										
										
											2009-07-05 12:44:18 -04:00
										 |  |  |         $c = Memcached_DataObject::memcache(); | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |         if (!$c) { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2010-01-10 12:31:43 -08:00
										 |  |  |             $obj = $c->get(Memcached_DataObject::cacheKey($cls, $k, $v)); | 
					
						
							|  |  |  |             if (0 == strcasecmp($cls, 'User')) { | 
					
						
							|  |  |  |                 // Special case for User
 | 
					
						
							| 
									
										
										
										
											2010-01-11 13:24:52 -08:00
										 |  |  |                 if (is_object($obj) && is_object($obj->id)) { | 
					
						
							| 
									
										
										
										
											2010-01-10 12:31:43 -08:00
										 |  |  |                     common_log(LOG_ERR, "User " . $obj->nickname . " was cached with User as ID; deleting"); | 
					
						
							|  |  |  |                     $c->delete(Memcached_DataObject::cacheKey($cls, $k, $v)); | 
					
						
							|  |  |  |                     return false; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             return $obj; | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2008-09-26 12:09:41 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-23 14:33:23 -05:00
										 |  |  |     function keyTypes() | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |         global $_DB_DATAOBJECT; | 
					
						
							| 
									
										
										
										
											2008-09-26 12:09:41 -04:00
										 |  |  |         if (!isset($_DB_DATAOBJECT['INI'][$this->_database][$this->__table."__keys"])) { | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |             $this->databaseStructure(); | 
					
						
							| 
									
										
										
										
											2008-09-26 12:09:41 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |         return $_DB_DATAOBJECT['INI'][$this->_database][$this->__table."__keys"]; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-01-22 19:13:26 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-23 14:33:23 -05:00
										 |  |  |     function encache() | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |         $c = $this->memcache(); | 
					
						
							|  |  |  |         if (!$c) { | 
					
						
							|  |  |  |             return false; | 
					
						
							| 
									
										
										
										
											2010-01-10 12:31:43 -08:00
										 |  |  |         } else if ($this->tableName() == 'user' && is_object($this->id)) { | 
					
						
							|  |  |  |             // Special case for User bug
 | 
					
						
							|  |  |  |             $e = new Exception(); | 
					
						
							|  |  |  |             common_log(LOG_ERR, __METHOD__ . ' caching user with User object as ID ' . | 
					
						
							|  |  |  |                        str_replace("\n", " ", $e->getTraceAsString())); | 
					
						
							|  |  |  |             return false; | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |         } else { | 
					
						
							|  |  |  |             $pkey = array(); | 
					
						
							| 
									
										
										
										
											2009-01-22 19:13:26 +01:00
										 |  |  |             $pval = array(); | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |             $types = $this->keyTypes(); | 
					
						
							|  |  |  |             ksort($types); | 
					
						
							|  |  |  |             foreach ($types as $key => $type) { | 
					
						
							|  |  |  |                 if ($type == 'K') { | 
					
						
							|  |  |  |                     $pkey[] = $key; | 
					
						
							|  |  |  |                     $pval[] = $this->$key; | 
					
						
							|  |  |  |                 } else { | 
					
						
							|  |  |  |                     $c->set($this->cacheKey($this->tableName(), $key, $this->$key), $this); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             # XXX: should work for both compound and scalar pkeys
 | 
					
						
							|  |  |  |             $pvals = implode(',', $pval); | 
					
						
							|  |  |  |             $pkeys = implode(',', $pkey); | 
					
						
							|  |  |  |             $c->set($this->cacheKey($this->tableName(), $pkeys, $pvals), $this); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-01-22 19:13:26 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-23 14:33:23 -05:00
										 |  |  |     function decache() | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |         $c = $this->memcache(); | 
					
						
							|  |  |  |         if (!$c) { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             $pkey = array(); | 
					
						
							| 
									
										
										
										
											2009-01-22 19:13:26 +01:00
										 |  |  |             $pval = array(); | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |             $types = $this->keyTypes(); | 
					
						
							|  |  |  |             ksort($types); | 
					
						
							|  |  |  |             foreach ($types as $key => $type) { | 
					
						
							|  |  |  |                 if ($type == 'K') { | 
					
						
							|  |  |  |                     $pkey[] = $key; | 
					
						
							|  |  |  |                     $pval[] = $this->$key; | 
					
						
							|  |  |  |                 } else { | 
					
						
							|  |  |  |                     $c->delete($this->cacheKey($this->tableName(), $key, $this->$key)); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             # should work for both compound and scalar pkeys
 | 
					
						
							|  |  |  |             # XXX: comma works for now but may not be safe separator for future keys
 | 
					
						
							|  |  |  |             $pvals = implode(',', $pval); | 
					
						
							|  |  |  |             $pkeys = implode(',', $pkey); | 
					
						
							|  |  |  |             $c->delete($this->cacheKey($this->tableName(), $pkeys, $pvals)); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2008-10-02 10:47:15 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-23 14:33:23 -05:00
										 |  |  |     function multicache($cls, $kv) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2009-07-05 12:44:18 -04:00
										 |  |  |         ksort($kv); | 
					
						
							|  |  |  |         $c = Memcached_DataObject::memcache(); | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |         if (!$c) { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2009-07-05 12:44:18 -04:00
										 |  |  |             $pkeys = implode(',', array_keys($kv)); | 
					
						
							|  |  |  |             $pvals = implode(',', array_values($kv)); | 
					
						
							|  |  |  |             return $c->get(Memcached_DataObject::cacheKey($cls, $pkeys, $pvals)); | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2008-11-20 16:50:41 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-23 14:33:23 -05:00
										 |  |  |     function getSearchEngine($table) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2008-11-23 15:16:16 -05:00
										 |  |  |         require_once INSTALLDIR.'/lib/search_engines.php'; | 
					
						
							| 
									
										
										
										
											2008-11-20 16:50:41 -05:00
										 |  |  |         static $search_engine; | 
					
						
							|  |  |  |         if (!isset($search_engine)) { | 
					
						
							| 
									
										
										
										
											2009-11-03 16:57:39 -08:00
										 |  |  |             if (Event::handle('GetSearchEngine', array($this, $table, &$search_engine))) { | 
					
						
							|  |  |  |                 if ('mysql' === common_config('db', 'type')) { | 
					
						
							|  |  |  |                     $type = common_config('search', 'type'); | 
					
						
							|  |  |  |                     if ($type == 'like') { | 
					
						
							|  |  |  |                         $search_engine = new MySQLLikeSearch($this, $table); | 
					
						
							|  |  |  |                     } else if ($type == 'fulltext') { | 
					
						
							|  |  |  |                         $search_engine = new MySQLSearch($this, $table); | 
					
						
							| 
									
										
										
										
											2008-11-23 15:16:16 -05:00
										 |  |  |                     } else { | 
					
						
							| 
									
										
										
										
											2009-11-03 16:57:39 -08:00
										 |  |  |                         throw new ServerException('Unknown search type: ' . $type); | 
					
						
							| 
									
										
										
										
											2008-11-23 15:16:16 -05:00
										 |  |  |                     } | 
					
						
							| 
									
										
										
										
											2009-11-03 16:57:39 -08:00
										 |  |  |                 } else { | 
					
						
							|  |  |  |                     $search_engine = new PGSearch($this, $table); | 
					
						
							| 
									
										
										
										
											2008-11-20 16:50:41 -05:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2009-11-03 16:57:39 -08:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2008-11-20 16:50:41 -05:00
										 |  |  |         } | 
					
						
							|  |  |  |         return $search_engine; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-01-22 19:13:26 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     static function cachedQuery($cls, $qry, $expiry=3600) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2009-07-05 12:44:18 -04:00
										 |  |  |         $c = Memcached_DataObject::memcache(); | 
					
						
							| 
									
										
										
										
											2009-01-22 19:13:26 +01:00
										 |  |  |         if (!$c) { | 
					
						
							|  |  |  |             $inst = new $cls(); | 
					
						
							|  |  |  |             $inst->query($qry); | 
					
						
							| 
									
										
										
										
											2009-01-22 19:54:05 +00:00
										 |  |  |             return $inst; | 
					
						
							| 
									
										
										
										
											2009-01-22 19:13:26 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |         $key_part = common_keyize($cls).':'.md5($qry); | 
					
						
							|  |  |  |         $ckey = common_cache_key($key_part); | 
					
						
							|  |  |  |         $stored = $c->get($ckey); | 
					
						
							|  |  |  |         if ($stored) { | 
					
						
							|  |  |  |             return new ArrayWrapper($stored); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $inst = new $cls(); | 
					
						
							| 
									
										
										
										
											2009-01-22 20:51:05 +00:00
										 |  |  |         $inst->query($qry); | 
					
						
							| 
									
										
										
										
											2009-01-22 19:13:26 +01:00
										 |  |  |         $cached = array(); | 
					
						
							|  |  |  |         while ($inst->fetch()) { | 
					
						
							|  |  |  |             $cached[] = clone($inst); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         $inst->free(); | 
					
						
							|  |  |  |         $c->set($ckey, $cached, MEMCACHE_COMPRESSED, $expiry); | 
					
						
							| 
									
										
										
										
											2009-01-22 20:16:19 +00:00
										 |  |  |         return new ArrayWrapper($cached); | 
					
						
							| 
									
										
										
										
											2009-01-22 19:13:26 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-04-26 13:16:59 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-21 11:07:52 -08:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * sends query to database - this is the private one that must work  | 
					
						
							|  |  |  |      *   - internal functions use this rather than $this->query() | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * Overridden to do logging. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param  string  $string | 
					
						
							|  |  |  |      * @access private | 
					
						
							|  |  |  |      * @return mixed none or PEAR_Error | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     function _query($string) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $start = microtime(true); | 
					
						
							|  |  |  |         $result = parent::_query($string); | 
					
						
							|  |  |  |         $delta = microtime(true) - $start; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $limit = common_config('db', 'log_slow_queries'); | 
					
						
							|  |  |  |         if (($limit > 0 && $delta >= $limit) || common_config('db', 'log_queries')) { | 
					
						
							|  |  |  |             $clean = $this->sanitizeQuery($string); | 
					
						
							|  |  |  |             common_log(LOG_DEBUG, sprintf("DB query (%0.3fs): %s", $delta, $clean)); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return $result; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Sanitize a query for logging
 | 
					
						
							|  |  |  |     // @fixme don't trim spaces in string literals
 | 
					
						
							|  |  |  |     function sanitizeQuery($string) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $string = preg_replace('/\s+/', ' ', $string); | 
					
						
							|  |  |  |         $string = trim($string); | 
					
						
							|  |  |  |         return $string; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-26 13:16:59 -04:00
										 |  |  |     // We overload so that 'SET NAMES "utf8"' is called for
 | 
					
						
							|  |  |  |     // each connection
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function _connect() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         global $_DB_DATAOBJECT; | 
					
						
							| 
									
										
										
										
											2009-07-27 13:51:40 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |         $sum = $this->_getDbDsnMD5(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!empty($_DB_DATAOBJECT['CONNECTIONS'][$sum]) && | 
					
						
							|  |  |  |             !PEAR::isError($_DB_DATAOBJECT['CONNECTIONS'][$sum])) { | 
					
						
							|  |  |  |             $exists = true; | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             $exists = false; | 
					
						
							|  |  |  |        } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-12 19:57:15 -08:00
										 |  |  |         // @fixme horrible evil hack!
 | 
					
						
							|  |  |  |         //
 | 
					
						
							|  |  |  |         // In multisite configuration we don't want to keep around a separate
 | 
					
						
							|  |  |  |         // connection for every database; we could end up with thousands of
 | 
					
						
							|  |  |  |         // connections open per thread. In an ideal world we might keep
 | 
					
						
							|  |  |  |         // a connection per server and select different databases, but that'd
 | 
					
						
							|  |  |  |         // be reliant on having the same db username/pass as well.
 | 
					
						
							|  |  |  |         //
 | 
					
						
							|  |  |  |         // MySQL connections are cheap enough we're going to try just
 | 
					
						
							|  |  |  |         // closing out the old connection and reopening when we encounter
 | 
					
						
							|  |  |  |         // a new DSN.
 | 
					
						
							|  |  |  |         //
 | 
					
						
							|  |  |  |         // WARNING WARNING if we end up actually using multiple DBs at a time
 | 
					
						
							|  |  |  |         // we'll need some fancier logic here.
 | 
					
						
							|  |  |  |         if (!$exists && !empty($_DB_DATAOBJECT['CONNECTIONS'])) { | 
					
						
							|  |  |  |             foreach ($_DB_DATAOBJECT['CONNECTIONS'] as $index => $conn) { | 
					
						
							|  |  |  |                 if (!empty($conn)) { | 
					
						
							|  |  |  |                     $conn->disconnect(); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 unset($_DB_DATAOBJECT['CONNECTIONS'][$index]); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-01-22 10:46:11 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-26 13:16:59 -04:00
										 |  |  |         $result = parent::_connect(); | 
					
						
							| 
									
										
										
										
											2009-07-27 13:51:40 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if ($result && !$exists) { | 
					
						
							| 
									
										
										
										
											2009-04-26 13:16:59 -04:00
										 |  |  |             $DB = &$_DB_DATAOBJECT['CONNECTIONS'][$this->_database_dsn_md5]; | 
					
						
							| 
									
										
										
										
											2009-05-30 13:59:57 -04:00
										 |  |  |             if (common_config('db', 'type') == 'mysql' && | 
					
						
							|  |  |  |                 common_config('db', 'utf8')) { | 
					
						
							|  |  |  |                 $conn = $DB->connection; | 
					
						
							| 
									
										
										
										
											2009-06-18 19:19:19 +00:00
										 |  |  |                 if (!empty($conn)) { | 
					
						
							|  |  |  |                     if ($DB instanceof DB_mysqli) { | 
					
						
							|  |  |  |                         mysqli_set_charset($conn, 'utf8'); | 
					
						
							|  |  |  |                     } else if ($DB instanceof DB_mysql) { | 
					
						
							|  |  |  |                         mysql_set_charset('utf8', $conn); | 
					
						
							|  |  |  |                     } | 
					
						
							| 
									
										
										
										
											2009-05-30 13:59:57 -04:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2009-05-27 14:57:45 -04:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2009-04-26 13:16:59 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-07-27 13:51:40 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-26 13:16:59 -04:00
										 |  |  |         return $result; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-06-18 19:19:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-27 13:51:40 -04:00
										 |  |  |     // XXX: largely cadged from DB_DataObject
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function _getDbDsnMD5() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if ($this->_database_dsn_md5) { | 
					
						
							|  |  |  |             return $this->_database_dsn_md5; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $dsn = $this->_getDbDsn(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (is_string($dsn)) { | 
					
						
							|  |  |  |             $sum = md5($dsn); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             /// support array based dsn's
 | 
					
						
							|  |  |  |             $sum = md5(serialize($dsn)); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $sum; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function _getDbDsn() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         global $_DB_DATAOBJECT; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (empty($_DB_DATAOBJECT['CONFIG'])) { | 
					
						
							|  |  |  |             DB_DataObject::_loadConfig(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $options = &$_DB_DATAOBJECT['CONFIG']; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // if the databse dsn dis defined in the object..
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $dsn = isset($this->_database_dsn) ? $this->_database_dsn : null; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!$dsn) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (!$this->_database) { | 
					
						
							|  |  |  |                 $this->_database = isset($options["table_{$this->__table}"]) ? $options["table_{$this->__table}"] : null; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if ($this->_database && !empty($options["database_{$this->_database}"]))  { | 
					
						
							|  |  |  |                 $dsn = $options["database_{$this->_database}"]; | 
					
						
							|  |  |  |             } else if (!empty($options['database'])) { | 
					
						
							|  |  |  |                 $dsn = $options['database']; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!$dsn) { | 
					
						
							|  |  |  |             throw new Exception("No database name / dsn found anywhere"); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $dsn; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2008-09-26 12:09:41 -04:00
										 |  |  | } |