wiki = $config['wiki']; } /** * @see FileJournal::logChangeBatch() * @param array $entries * @param string $batchId * @return StatusValue */ protected function doLogChangeBatch( array $entries, $batchId ) { $status = StatusValue::newGood(); try { $dbw = $this->getMasterDB(); } catch ( DBError $e ) { $status->fatal( 'filejournal-fail-dbconnect', $this->backend ); return $status; } $now = wfTimestamp( TS_UNIX ); $data = []; foreach ( $entries as $entry ) { $data[] = [ 'fj_batch_uuid' => $batchId, 'fj_backend' => $this->backend, 'fj_op' => $entry['op'], 'fj_path' => $entry['path'], 'fj_new_sha1' => $entry['newSha1'], 'fj_timestamp' => $dbw->timestamp( $now ) ]; } try { $dbw->insert( 'filejournal', $data, __METHOD__ ); if ( mt_rand( 0, 99 ) == 0 ) { $this->purgeOldLogs(); // occasionally delete old logs } } catch ( DBError $e ) { $status->fatal( 'filejournal-fail-dbquery', $this->backend ); return $status; } return $status; } /** * @see FileJournal::doGetCurrentPosition() * @return bool|mixed The value from the field, or false on failure. */ protected function doGetCurrentPosition() { $dbw = $this->getMasterDB(); return $dbw->selectField( 'filejournal', 'MAX(fj_id)', [ 'fj_backend' => $this->backend ], __METHOD__ ); } /** * @see FileJournal::doGetPositionAtTime() * @param int|string $time Timestamp * @return bool|mixed The value from the field, or false on failure. */ protected function doGetPositionAtTime( $time ) { $dbw = $this->getMasterDB(); $encTimestamp = $dbw->addQuotes( $dbw->timestamp( $time ) ); return $dbw->selectField( 'filejournal', 'fj_id', [ 'fj_backend' => $this->backend, "fj_timestamp <= $encTimestamp" ], __METHOD__, [ 'ORDER BY' => 'fj_timestamp DESC' ] ); } /** * @see FileJournal::doGetChangeEntries() * @param int $start * @param int $limit * @return array */ protected function doGetChangeEntries( $start, $limit ) { $dbw = $this->getMasterDB(); $res = $dbw->select( 'filejournal', '*', [ 'fj_backend' => $this->backend, 'fj_id >= ' . $dbw->addQuotes( (int)$start ) ], // $start may be 0 __METHOD__, array_merge( [ 'ORDER BY' => 'fj_id ASC' ], $limit ? [ 'LIMIT' => $limit ] : [] ) ); $entries = []; foreach ( $res as $row ) { $item = []; foreach ( (array)$row as $key => $value ) { $item[substr( $key, 3 )] = $value; // "fj_op" => "op" } $entries[] = $item; } return $entries; } /** * @see FileJournal::purgeOldLogs() * @return StatusValue * @throws DBError */ protected function doPurgeOldLogs() { $status = StatusValue::newGood(); if ( $this->ttlDays <= 0 ) { return $status; // nothing to do } $dbw = $this->getMasterDB(); $dbCutoff = $dbw->timestamp( time() - 86400 * $this->ttlDays ); $dbw->delete( 'filejournal', [ 'fj_timestamp < ' . $dbw->addQuotes( $dbCutoff ) ], __METHOD__ ); return $status; } /** * Get a master connection to the logging DB * * @return IDatabase * @throws DBError */ protected function getMasterDB() { if ( !$this->dbw ) { // Get a separate connection in autocommit mode $lb = MediaWikiServices::getInstance()->getDBLoadBalancerFactory()->newMainLB(); $this->dbw = $lb->getConnection( DB_MASTER, [], $this->wiki ); $this->dbw->clearFlag( DBO_TRX ); } return $this->dbw; } }