<?php /** * Version of FileJournal that logs to a DB table. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 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 General Public License for more details. * * You should have received a copy of the GNU General Public License along * with this program; if not, write to the Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. * http://www.gnu.org/copyleft/gpl.html * * @file * @ingroup FileJournal */ use MediaWiki\MediaWikiServices; use Wikimedia\Rdbms\DBError; use Wikimedia\Rdbms\IDatabase; use Wikimedia\Timestamp\ConvertibleTimestamp; /** * Version of FileJournal that logs to a DB table * @since 1.20 */ class DBFileJournal extends FileJournal { /** @var IDatabase */ protected $dbw; /** @var string */ protected $domain; /** * Construct a new instance from configuration. * * @param array $config Includes: * - domain: database domain ID of the wiki */ public function __construct( array $config ) { parent::__construct( $config ); $this->domain = $config['domain'] ?? $config['wiki']; // b/c } /** * @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 = ConvertibleTimestamp::time(); $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__ ); // XXX Should we do this deterministically so it's testable? Maybe look at the last two // digits of a hash of a bunch of the data? if ( mt_rand( 0, 99 ) == 0 ) { // occasionally delete old logs $this->purgeOldLogs(); // @codeCoverageIgnore } } 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|null $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( ConvertibleTimestamp::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->domain ); $this->dbw->clearFlag( DBO_TRX ); } return $this->dbw; } }