2007-06-03 17:10:45 +00:00
|
|
|
<?php
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Class to represent a file in the oldimage table
|
|
|
|
|
*
|
|
|
|
|
* @addtogroup FileRepo
|
|
|
|
|
*/
|
|
|
|
|
class OldLocalFile extends LocalFile {
|
|
|
|
|
var $requestedTime, $archive_name;
|
|
|
|
|
|
|
|
|
|
const CACHE_VERSION = 1;
|
|
|
|
|
const MAX_CACHE_ROWS = 20;
|
|
|
|
|
|
2007-06-05 18:44:49 +00:00
|
|
|
static function newFromTitle( $title, $repo, $time ) {
|
2007-06-03 17:10:45 +00:00
|
|
|
return new self( $title, $repo, $time, null );
|
|
|
|
|
}
|
|
|
|
|
|
2007-06-05 18:44:49 +00:00
|
|
|
static function newFromArchiveName( $title, $repo, $archiveName ) {
|
2007-06-03 17:10:45 +00:00
|
|
|
return new self( $title, $repo, null, $archiveName );
|
|
|
|
|
}
|
|
|
|
|
|
2007-06-05 18:44:49 +00:00
|
|
|
static function newFromRow( $row, $repo ) {
|
2007-06-03 17:10:45 +00:00
|
|
|
$title = Title::makeTitle( NS_IMAGE, $row->oi_name );
|
|
|
|
|
$file = new self( $title, $repo, null, $row->oi_archive_name );
|
|
|
|
|
$file->loadFromRow( $row, 'oi_' );
|
|
|
|
|
return $file;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @param Title $title
|
|
|
|
|
* @param FileRepo $repo
|
|
|
|
|
* @param string $time Timestamp or null to load by archive name
|
|
|
|
|
* @param string $archiveName Archive name or null to load by timestamp
|
|
|
|
|
*/
|
|
|
|
|
function __construct( $title, $repo, $time, $archiveName ) {
|
|
|
|
|
parent::__construct( $title, $repo );
|
|
|
|
|
$this->requestedTime = $time;
|
|
|
|
|
$this->archive_name = $archiveName;
|
|
|
|
|
if ( is_null( $time ) && is_null( $archiveName ) ) {
|
|
|
|
|
throw new MWException( __METHOD__.': must specify at least one of $time or $archiveName' );
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function getCacheKey() {
|
|
|
|
|
$hashedName = md5($this->getName());
|
|
|
|
|
return wfMemcKey( 'oldfile', $hashedName );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function getArchiveName() {
|
|
|
|
|
if ( !isset( $this->archive_name ) ) {
|
|
|
|
|
$this->load();
|
|
|
|
|
}
|
|
|
|
|
return $this->archive_name;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function isOld() {
|
|
|
|
|
return true;
|
|
|
|
|
}
|
2008-03-15 00:27:57 +00:00
|
|
|
|
|
|
|
|
function isVisible() {
|
|
|
|
|
return $this->exists() && !$this->isDeleted(File::DELETED_FILE);
|
|
|
|
|
}
|
2007-06-03 17:10:45 +00:00
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Try to load file metadata from memcached. Returns true on success.
|
|
|
|
|
*/
|
|
|
|
|
function loadFromCache() {
|
|
|
|
|
global $wgMemc;
|
|
|
|
|
wfProfileIn( __METHOD__ );
|
|
|
|
|
$this->dataLoaded = false;
|
|
|
|
|
$key = $this->getCacheKey();
|
|
|
|
|
if ( !$key ) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
$oldImages = $wgMemc->get( $key );
|
|
|
|
|
|
2007-07-22 14:45:12 +00:00
|
|
|
if ( isset( $oldImages['version'] ) && $oldImages['version'] == self::CACHE_VERSION ) {
|
2007-06-03 17:10:45 +00:00
|
|
|
unset( $oldImages['version'] );
|
|
|
|
|
$more = isset( $oldImages['more'] );
|
|
|
|
|
unset( $oldImages['more'] );
|
|
|
|
|
$found = false;
|
|
|
|
|
if ( is_null( $this->requestedTime ) ) {
|
|
|
|
|
foreach ( $oldImages as $timestamp => $info ) {
|
|
|
|
|
if ( $info['archive_name'] == $this->archive_name ) {
|
|
|
|
|
$found = true;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
krsort( $oldImages );
|
|
|
|
|
foreach ( $oldImages as $timestamp => $info ) {
|
|
|
|
|
if ( $timestamp <= $this->requestedTime ) {
|
|
|
|
|
$found = true;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if ( $found ) {
|
|
|
|
|
wfDebug( "Pulling file metadata from cache key {$key}[{$timestamp}]\n" );
|
|
|
|
|
$this->dataLoaded = true;
|
2007-07-22 14:45:12 +00:00
|
|
|
$this->fileExists = true;
|
|
|
|
|
foreach ( $info as $name => $value ) {
|
2007-06-03 17:10:45 +00:00
|
|
|
$this->$name = $value;
|
|
|
|
|
}
|
|
|
|
|
} elseif ( $more ) {
|
|
|
|
|
wfDebug( "Cache key was truncated, oldimage row might be found in the database\n" );
|
|
|
|
|
} else {
|
|
|
|
|
wfDebug( "Image did not exist at the specified time.\n" );
|
|
|
|
|
$this->fileExists = false;
|
|
|
|
|
$this->dataLoaded = true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ( $this->dataLoaded ) {
|
|
|
|
|
wfIncrStats( 'image_cache_hit' );
|
|
|
|
|
} else {
|
|
|
|
|
wfIncrStats( 'image_cache_miss' );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
wfProfileOut( __METHOD__ );
|
|
|
|
|
return $this->dataLoaded;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function saveToCache() {
|
Basic integrated audio/video support, with Ogg implementation.
* JavaScript video player based loosely on Greg Maxwell's player
* Image page text snippet customisation
* Abstraction of transform parameters in the parser. Introduced Linker::makeImageLink2().
* Made canRender(), mustRender() depend on file, not just on handler. Moved width=0, height=0 checking to ImageHandler::canRender(), since audio streams have width=height=0 but should be rendered.
Also:
* Automatic upgrade for oldimage rows on image page view, allows media handler selection based on oi_*_mime
* oi_*_mime unconditionally referenced, REQUIRES SCHEMA UPGRADE
* Don't destroy file info for missing files on upgrade
* Simple, centralised extension message file handling
* Made MessageCache::loadAllMessages non-static, optimised for repeated-call case due to abuse in User.php
* Support for lightweight parser output hooks, with callback whitelist for security
* Moved Linker::formatSize() to Language, to join the new formatTimePeriod() and formatBitrate()
* Introduced MagicWordArray, regex capture trick requires that magic word IDs DO NOT CONTAIN HYPHENS.
2007-08-15 10:50:09 +00:00
|
|
|
// If a timestamp was specified, cache the entire history of the image (up to MAX_CACHE_ROWS).
|
|
|
|
|
if ( is_null( $this->requestedTime ) ) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
2007-06-03 17:10:45 +00:00
|
|
|
// This is expensive, so we only do it if $wgMemc is real
|
|
|
|
|
global $wgMemc;
|
|
|
|
|
if ( $wgMemc instanceof FakeMemcachedClient ) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
$key = $this->getCacheKey();
|
|
|
|
|
if ( !$key ) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
wfProfileIn( __METHOD__ );
|
|
|
|
|
|
|
|
|
|
$dbr = $this->repo->getSlaveDB();
|
2007-07-22 14:45:12 +00:00
|
|
|
$res = $dbr->select( 'oldimage', $this->getCacheFields( 'oi_' ),
|
2007-06-03 17:10:45 +00:00
|
|
|
array( 'oi_name' => $this->getName() ), __METHOD__,
|
|
|
|
|
array(
|
|
|
|
|
'LIMIT' => self::MAX_CACHE_ROWS + 1,
|
|
|
|
|
'ORDER BY' => 'oi_timestamp DESC',
|
|
|
|
|
));
|
|
|
|
|
$cache = array( 'version' => self::CACHE_VERSION );
|
|
|
|
|
$numRows = $dbr->numRows( $res );
|
|
|
|
|
if ( $numRows > self::MAX_CACHE_ROWS ) {
|
|
|
|
|
$cache['more'] = true;
|
|
|
|
|
$numRows--;
|
|
|
|
|
}
|
|
|
|
|
for ( $i = 0; $i < $numRows; $i++ ) {
|
|
|
|
|
$row = $dbr->fetchObject( $res );
|
2007-07-22 14:45:12 +00:00
|
|
|
$decoded = $this->decodeRow( $row, 'oi_' );
|
|
|
|
|
$cache[$row->oi_timestamp] = $decoded;
|
2007-06-03 17:10:45 +00:00
|
|
|
}
|
|
|
|
|
$dbr->freeResult( $res );
|
|
|
|
|
$wgMemc->set( $key, $cache, 7*86400 /* 1 week */ );
|
|
|
|
|
wfProfileOut( __METHOD__ );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function loadFromDB() {
|
|
|
|
|
wfProfileIn( __METHOD__ );
|
Basic integrated audio/video support, with Ogg implementation.
* JavaScript video player based loosely on Greg Maxwell's player
* Image page text snippet customisation
* Abstraction of transform parameters in the parser. Introduced Linker::makeImageLink2().
* Made canRender(), mustRender() depend on file, not just on handler. Moved width=0, height=0 checking to ImageHandler::canRender(), since audio streams have width=height=0 but should be rendered.
Also:
* Automatic upgrade for oldimage rows on image page view, allows media handler selection based on oi_*_mime
* oi_*_mime unconditionally referenced, REQUIRES SCHEMA UPGRADE
* Don't destroy file info for missing files on upgrade
* Simple, centralised extension message file handling
* Made MessageCache::loadAllMessages non-static, optimised for repeated-call case due to abuse in User.php
* Support for lightweight parser output hooks, with callback whitelist for security
* Moved Linker::formatSize() to Language, to join the new formatTimePeriod() and formatBitrate()
* Introduced MagicWordArray, regex capture trick requires that magic word IDs DO NOT CONTAIN HYPHENS.
2007-08-15 10:50:09 +00:00
|
|
|
$this->dataLoaded = true;
|
2007-06-03 17:10:45 +00:00
|
|
|
$dbr = $this->repo->getSlaveDB();
|
|
|
|
|
$conds = array( 'oi_name' => $this->getName() );
|
|
|
|
|
if ( is_null( $this->requestedTime ) ) {
|
|
|
|
|
$conds['oi_archive_name'] = $this->archive_name;
|
|
|
|
|
} else {
|
2008-03-27 21:55:39 +00:00
|
|
|
$conds[] = 'oi_timestamp <= ' . $dbr->addQuotes( $dbr->timestamp( $this->requestedTime ) );
|
2007-06-03 17:10:45 +00:00
|
|
|
}
|
|
|
|
|
$row = $dbr->selectRow( 'oldimage', $this->getCacheFields( 'oi_' ),
|
|
|
|
|
$conds, __METHOD__, array( 'ORDER BY' => 'oi_timestamp DESC' ) );
|
|
|
|
|
if ( $row ) {
|
|
|
|
|
$this->loadFromRow( $row, 'oi_' );
|
|
|
|
|
} else {
|
|
|
|
|
$this->fileExists = false;
|
|
|
|
|
}
|
2007-07-22 14:45:12 +00:00
|
|
|
wfProfileOut( __METHOD__ );
|
2007-06-03 17:10:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function getCacheFields( $prefix = 'img_' ) {
|
|
|
|
|
$fields = parent::getCacheFields( $prefix );
|
|
|
|
|
$fields[] = $prefix . 'archive_name';
|
2008-03-15 00:27:57 +00:00
|
|
|
$fields[] = $prefix . 'deleted';
|
2007-06-03 17:10:45 +00:00
|
|
|
return $fields;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function getRel() {
|
|
|
|
|
return 'archive/' . $this->getHashPath() . $this->getArchiveName();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function getUrlRel() {
|
|
|
|
|
return 'archive/' . $this->getHashPath() . urlencode( $this->getArchiveName() );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function upgradeRow() {
|
|
|
|
|
wfProfileIn( __METHOD__ );
|
|
|
|
|
$this->loadFromFile();
|
Basic integrated audio/video support, with Ogg implementation.
* JavaScript video player based loosely on Greg Maxwell's player
* Image page text snippet customisation
* Abstraction of transform parameters in the parser. Introduced Linker::makeImageLink2().
* Made canRender(), mustRender() depend on file, not just on handler. Moved width=0, height=0 checking to ImageHandler::canRender(), since audio streams have width=height=0 but should be rendered.
Also:
* Automatic upgrade for oldimage rows on image page view, allows media handler selection based on oi_*_mime
* oi_*_mime unconditionally referenced, REQUIRES SCHEMA UPGRADE
* Don't destroy file info for missing files on upgrade
* Simple, centralised extension message file handling
* Made MessageCache::loadAllMessages non-static, optimised for repeated-call case due to abuse in User.php
* Support for lightweight parser output hooks, with callback whitelist for security
* Moved Linker::formatSize() to Language, to join the new formatTimePeriod() and formatBitrate()
* Introduced MagicWordArray, regex capture trick requires that magic word IDs DO NOT CONTAIN HYPHENS.
2007-08-15 10:50:09 +00:00
|
|
|
|
|
|
|
|
# Don't destroy file info of missing files
|
|
|
|
|
if ( !$this->fileExists ) {
|
|
|
|
|
wfDebug( __METHOD__.": file does not exist, aborting\n" );
|
|
|
|
|
wfProfileOut( __METHOD__ );
|
|
|
|
|
return;
|
|
|
|
|
}
|
2007-06-03 17:10:45 +00:00
|
|
|
|
|
|
|
|
$dbw = $this->repo->getMasterDB();
|
|
|
|
|
list( $major, $minor ) = self::splitMime( $this->mime );
|
|
|
|
|
|
|
|
|
|
wfDebug(__METHOD__.': upgrading '.$this->archive_name." to the current schema\n");
|
|
|
|
|
$dbw->update( 'oldimage',
|
|
|
|
|
array(
|
|
|
|
|
'oi_width' => $this->width,
|
|
|
|
|
'oi_height' => $this->height,
|
|
|
|
|
'oi_bits' => $this->bits,
|
Basic integrated audio/video support, with Ogg implementation.
* JavaScript video player based loosely on Greg Maxwell's player
* Image page text snippet customisation
* Abstraction of transform parameters in the parser. Introduced Linker::makeImageLink2().
* Made canRender(), mustRender() depend on file, not just on handler. Moved width=0, height=0 checking to ImageHandler::canRender(), since audio streams have width=height=0 but should be rendered.
Also:
* Automatic upgrade for oldimage rows on image page view, allows media handler selection based on oi_*_mime
* oi_*_mime unconditionally referenced, REQUIRES SCHEMA UPGRADE
* Don't destroy file info for missing files on upgrade
* Simple, centralised extension message file handling
* Made MessageCache::loadAllMessages non-static, optimised for repeated-call case due to abuse in User.php
* Support for lightweight parser output hooks, with callback whitelist for security
* Moved Linker::formatSize() to Language, to join the new formatTimePeriod() and formatBitrate()
* Introduced MagicWordArray, regex capture trick requires that magic word IDs DO NOT CONTAIN HYPHENS.
2007-08-15 10:50:09 +00:00
|
|
|
'oi_media_type' => $this->media_type,
|
|
|
|
|
'oi_major_mime' => $major,
|
|
|
|
|
'oi_minor_mime' => $minor,
|
|
|
|
|
'oi_metadata' => $this->metadata,
|
2007-07-22 14:45:12 +00:00
|
|
|
'oi_sha1' => $this->sha1,
|
|
|
|
|
), array(
|
|
|
|
|
'oi_name' => $this->getName(),
|
|
|
|
|
'oi_archive_name' => $this->archive_name ),
|
2007-06-03 17:10:45 +00:00
|
|
|
__METHOD__
|
|
|
|
|
);
|
|
|
|
|
wfProfileOut( __METHOD__ );
|
|
|
|
|
}
|
2008-03-15 00:27:57 +00:00
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* int $field one of DELETED_* bitfield constants
|
|
|
|
|
* for file or revision rows
|
|
|
|
|
* @return bool
|
|
|
|
|
*/
|
|
|
|
|
function isDeleted( $field ) {
|
|
|
|
|
return ($this->deleted & $field) == $field;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Determine if the current user is allowed to view a particular
|
|
|
|
|
* field of this FileStore image file, if it's marked as deleted.
|
|
|
|
|
* @param int $field
|
|
|
|
|
* @return bool
|
|
|
|
|
*/
|
|
|
|
|
function userCan( $field ) {
|
|
|
|
|
if( isset($this->deleted) && ($this->deleted & $field) == $field ) {
|
|
|
|
|
global $wgUser;
|
|
|
|
|
$permission = ( $this->deleted & File::DELETED_RESTRICTED ) == File::DELETED_RESTRICTED
|
|
|
|
|
? 'hiderevision'
|
|
|
|
|
: 'deleterevision';
|
|
|
|
|
wfDebug( "Checking for $permission due to $field match on $this->mDeleted\n" );
|
|
|
|
|
return $wgUser->isAllowed( $permission );
|
|
|
|
|
} else {
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
}
|
2007-06-03 17:10:45 +00:00
|
|
|
|
2008-03-15 00:27:57 +00:00
|
|
|
}
|
2007-06-29 01:19:14 +00:00
|
|
|
|
2007-10-01 19:50:25 +00:00
|
|
|
|