From 3bb52cc804d1e4e8fffaa43bf08e7df54d8175cd Mon Sep 17 00:00:00 2001 From: Aaron Schulz Date: Wed, 22 May 2013 17:23:21 -0700 Subject: [PATCH] database: Improve DatabaseMysql::masterPosWait() performance by caching the position * This is useful for things like loads of refreshLinks jobs that call wfGetLB()->waitFor(). Change-Id: I05a564a3b73999517fec2476ed946e092c16c3d4 --- includes/db/DatabaseMysqlBase.php | 45 ++++++++++++++++++++++++++++++--------- 1 file changed, 35 insertions(+), 10 deletions(-) diff --git a/includes/db/DatabaseMysqlBase.php b/includes/db/DatabaseMysqlBase.php index 84930c834da..cae133b9aa1 100644 --- a/includes/db/DatabaseMysqlBase.php +++ b/includes/db/DatabaseMysqlBase.php @@ -30,6 +30,8 @@ * @see Database */ abstract class DatabaseMysqlBase extends DatabaseBase { + /** @var MysqlMasterPos */ + protected $lastKnownSlavePos; /** * @return string @@ -578,23 +580,24 @@ abstract class DatabaseMysqlBase extends DatabaseBase { /** * Wait for the slave to catch up to a given master position. + * @TODO: return values for this and base class are rubbish * * @param $pos DBMasterPos object * @param $timeout Integer: the maximum number of seconds to wait for synchronisation * @return bool|string */ function masterPosWait( DBMasterPos $pos, $timeout ) { - $fname = __METHOD__; - wfProfileIn( $fname ); + if ( $this->lastKnownSlavePos && $this->lastKnownSlavePos->hasReached( $pos ) ) { + return '0'; // http://dev.mysql.com/doc/refman/5.0/en/miscellaneous-functions.html + } + wfProfileIn( __METHOD__ ); # Commit any open transactions - if ( $this->mTrxLevel ) { - $this->commit( $fname ); - } + $this->commit( __METHOD__, 'flush' ); if ( !is_null( $this->mFakeSlaveLag ) ) { $status = parent::masterPosWait( $pos, $timeout ); - wfProfileOut( $fname ); + wfProfileOut( __METHOD__ ); return $status; } @@ -604,12 +607,16 @@ abstract class DatabaseMysqlBase extends DatabaseBase { $sql = "SELECT MASTER_POS_WAIT($encFile, $encPos, $timeout)"; $res = $this->doQuery( $sql ); + $status = false; if ( $res && $row = $this->fetchRow( $res ) ) { - wfProfileOut( $fname ); - return $row[0]; + $status = $row[0]; // can be NULL, -1, or 0+ per the MySQL manual + if ( ctype_digit( $status ) ) { // success + $this->lastKnownSlavePos = $pos; + } } - wfProfileOut( $fname ); - return false; + + wfProfileOut( __METHOD__ ); + return $status; } /** @@ -1068,6 +1075,24 @@ class MySQLMasterPos implements DBMasterPos { } function __toString() { + // e.g db1034-bin.000976/843431247 return "{$this->file}/{$this->pos}"; } + + /** + * @return array|false (int, int) + */ + protected function getCoordinates() { + $m = array(); + if ( preg_match( '!\.(\d+)/(\d+)$!', (string)$this, $m ) ) { + return array( (int)$m[1], (int)$m[2] ); + } + return false; + } + + function hasReached( MySQLMasterPos $pos ) { + $thisPos = $this->getCoordinates(); + $thatPos = $pos->getCoordinates(); + return ( $thisPos && $thatPos && $thisPos >= $thatPos ); + } } -- 2.11.4.GIT