| 
									
										
										
										
											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
										 |  |  |     /** | 
					
						
							| 
									
										
										
										
											2009-12-14 16:36:01 -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. | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2009-12-15 12:38:15 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-14 16:36:01 -08:00
										 |  |  |     function __destruct() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $this->free(); | 
					
						
							| 
									
										
										
										
											2009-12-15 12:38:15 -05:00
										 |  |  |         if (method_exists('DB_DataObject', '__destruct')) { | 
					
						
							|  |  |  |             parent::__destruct(); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-12-14 16:36:01 -08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							|  |  |  |         // We don't have any local DBO refs, so clear these out.
 | 
					
						
							|  |  |  |         $this->_link_loaded = false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-22 15:08:44 -08:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * 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); | 
					
						
							| 
									
										
										
										
											2010-01-04 10:00:17 -10:00
										 |  |  |         if ($i !== false) { // false == cache miss
 | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |             return $i; | 
					
						
							|  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2010-01-01 10:57:22 -10:00
										 |  |  |             $i = DB_DataObject::factory($cls); | 
					
						
							|  |  |  |             if (empty($i)) { | 
					
						
							| 
									
										
										
										
											2010-01-02 16:28:46 -10:00
										 |  |  |                 return false; | 
					
						
							| 
									
										
										
										
											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-04 10:00:17 -10:00
										 |  |  |                 // save the fact that no such row exists
 | 
					
						
							|  |  |  |                 $c = self::memcache(); | 
					
						
							|  |  |  |                 if (!empty($c)) { | 
					
						
							|  |  |  |                     $ck = self::cachekey($cls, $k, $v); | 
					
						
							|  |  |  |                     $c->set($ck, null); | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2010-01-02 16:28:46 -10:00
										 |  |  |                 return false; | 
					
						
							| 
									
										
										
										
											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); | 
					
						
							| 
									
										
										
										
											2010-01-04 10:00:17 -10:00
										 |  |  |         if ($i !== false) { // false == cache miss
 | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |             return $i; | 
					
						
							|  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2010-01-04 10:00:17 -10:00
										 |  |  |             $i = DB_DataObject::factory($cls); | 
					
						
							|  |  |  |             if (empty($i)) { | 
					
						
							|  |  |  |                 return false; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |             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; | 
					
						
							| 
									
										
										
										
											2010-01-04 10:00:17 -10:00
										 |  |  |                 $c = self::memcache(); | 
					
						
							|  |  |  |                 if (!empty($c)) { | 
					
						
							|  |  |  |                     $ck = self::multicacheKey($cls, $kv); | 
					
						
							|  |  |  |                     $c->set($ck, 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() | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2010-01-05 15:05:53 -08:00
										 |  |  |         $this->decache(); // in case of cached negative lookups
 | 
					
						
							| 
									
										
										
										
											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 { | 
					
						
							| 
									
										
										
										
											2009-07-05 12:44:18 -04:00
										 |  |  |             return $c->get(Memcached_DataObject::cacheKey($cls, $k, $v)); | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2008-09-26 12:09:41 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-03 11:24:05 -10:00
										 |  |  |     function keyTypes() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         global $_DB_DATAOBJECT; | 
					
						
							|  |  |  |         if (!isset($_DB_DATAOBJECT['INI'][$this->_database][$this->__table."__keys"])) { | 
					
						
							|  |  |  |             $this->databaseStructure(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return $_DB_DATAOBJECT['INI'][$this->_database][$this->__table."__keys"]; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-23 14:33:23 -05:00
										 |  |  |     function encache() | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |         $c = $this->memcache(); | 
					
						
							| 
									
										
										
										
											2010-01-04 08:59:19 -10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |         if (!$c) { | 
					
						
							|  |  |  |             return false; | 
					
						
							| 
									
										
										
										
											2010-01-04 08:59:19 -10:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $keys = $this->_allCacheKeys(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         foreach ($keys as $key) { | 
					
						
							|  |  |  |             $c->set($key, $this); | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											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(); | 
					
						
							| 
									
										
										
										
											2010-01-04 08:59:19 -10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |         if (!$c) { | 
					
						
							|  |  |  |             return false; | 
					
						
							| 
									
										
										
										
											2010-01-04 08:59:19 -10:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $keys = $this->_allCacheKeys(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         foreach ($keys as $key) { | 
					
						
							|  |  |  |             $c->delete($key, $this); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function _allCacheKeys() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $ckeys = array(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $types = $this->keyTypes(); | 
					
						
							|  |  |  |         ksort($types); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $pkey = array(); | 
					
						
							|  |  |  |         $pval = array(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         foreach ($types as $key => $type) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             assert(!empty($key)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if ($type == 'U') { | 
					
						
							|  |  |  |                 if (empty($this->$key)) { | 
					
						
							|  |  |  |                     continue; | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2010-01-04 08:59:19 -10:00
										 |  |  |                 $ckeys[] = $this->cacheKey($this->tableName(), $key, $this->$key); | 
					
						
							|  |  |  |             } else if ($type == 'K' || $type == 'N') { | 
					
						
							|  |  |  |                 $pkey[] = $key; | 
					
						
							|  |  |  |                 $pval[] = $this->$key; | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 throw new Exception("Unknown key type $key => $type for " . $this->tableName()); | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-01-04 08:59:19 -10:00
										 |  |  | 
 | 
					
						
							|  |  |  |         assert(count($pkey) > 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // XXX: should work for both compound and scalar pkeys
 | 
					
						
							|  |  |  |         $pvals = implode(',', $pval); | 
					
						
							|  |  |  |         $pkeys = implode(',', $pkey); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $ckeys[] = $this->cacheKey($this->tableName(), $pkeys, $pvals); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $ckeys; | 
					
						
							| 
									
										
										
										
											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 multicache($cls, $kv) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2009-07-05 12:44:18 -04:00
										 |  |  |         ksort($kv); | 
					
						
							| 
									
										
										
										
											2010-01-04 10:00:17 -10:00
										 |  |  |         $c = self::memcache(); | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |         if (!$c) { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2010-01-04 10:00:17 -10:00
										 |  |  |             return $c->get(self::multicacheKey($cls, $kv)); | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2008-11-20 16:50:41 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-04 10:00:17 -10:00
										 |  |  |     static function multicacheKey($cls, $kv) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         ksort($kv); | 
					
						
							|  |  |  |         $pkeys = implode(',', array_keys($kv)); | 
					
						
							|  |  |  |         $pvals = implode(',', array_values($kv)); | 
					
						
							|  |  |  |         return self::cacheKey($cls, $pkeys, $pvals); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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); | 
					
						
							| 
									
										
										
										
											2010-01-04 10:00:17 -10:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if ($stored !== false) { | 
					
						
							| 
									
										
										
										
											2009-01-22 19:13:26 +01:00
										 |  |  |             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
										 |  |  | 
 | 
					
						
							|  |  |  |     // 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; | 
					
						
							|  |  |  |        } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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
										 |  |  | } |