2004-02-18 02:15:00 +00:00
|
|
|
<?php
|
2004-09-02 23:28:24 +00:00
|
|
|
/**
|
|
|
|
|
* Blocks and bans object
|
|
|
|
|
*/
|
2004-02-14 12:37:25 +00:00
|
|
|
|
2004-09-02 23:28:24 +00:00
|
|
|
/**
|
|
|
|
|
* Some globals
|
|
|
|
|
*/
|
2004-08-15 07:23:39 +00:00
|
|
|
define ( 'EB_KEEP_EXPIRED', 1 );
|
|
|
|
|
define ( 'EB_FOR_UPDATE', 2 );
|
|
|
|
|
|
2004-09-02 23:28:24 +00:00
|
|
|
/**
|
|
|
|
|
* The block class
|
|
|
|
|
* All the functions in this class assume the object is either explicitly
|
|
|
|
|
* loaded or filled. It is not load-on-demand. There are no accessors.
|
|
|
|
|
*
|
|
|
|
|
* To use delete(), you only need to fill $mAddress
|
|
|
|
|
* Globals used: $wgBlockCache, $wgAutoblockExpiry
|
|
|
|
|
* @todo This could be used everywhere, but it isn't.
|
|
|
|
|
*/
|
2003-09-01 13:13:56 +00:00
|
|
|
class Block
|
|
|
|
|
{
|
2004-02-14 12:37:25 +00:00
|
|
|
/* public*/ var $mAddress, $mUser, $mBy, $mReason, $mTimestamp, $mAuto, $mId, $mExpiry;
|
2004-08-15 07:23:39 +00:00
|
|
|
/* private */ var $mNetworkBits, $mIntegerAddr, $mForUpdate;
|
2003-09-01 13:13:56 +00:00
|
|
|
|
2004-06-08 23:56:09 +00:00
|
|
|
function Block( $address = '', $user = '', $by = 0, $reason = '',
|
|
|
|
|
$timestamp = '' , $auto = 0, $expiry = '' )
|
2003-09-01 13:13:56 +00:00
|
|
|
{
|
|
|
|
|
$this->mAddress = $address;
|
|
|
|
|
$this->mUser = $user;
|
|
|
|
|
$this->mBy = $by;
|
|
|
|
|
$this->mReason = $reason;
|
|
|
|
|
$this->mTimestamp = $timestamp;
|
2003-09-07 13:56:25 +00:00
|
|
|
$this->mAuto = $auto;
|
2004-02-14 12:37:25 +00:00
|
|
|
$this->mExpiry = $expiry;
|
2004-08-15 07:23:39 +00:00
|
|
|
|
|
|
|
|
$this->mForUpdate = false;
|
2004-02-14 12:37:25 +00:00
|
|
|
$this->initialiseRange();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*static*/ function newFromDB( $address, $user = 0, $killExpired = true )
|
|
|
|
|
{
|
2003-09-01 13:13:56 +00:00
|
|
|
$ban = new Block();
|
|
|
|
|
$ban->load( $address, $user, $killExpired );
|
|
|
|
|
return $ban;
|
|
|
|
|
}
|
2004-02-14 12:37:25 +00:00
|
|
|
|
|
|
|
|
function clear()
|
|
|
|
|
{
|
2004-06-08 23:56:09 +00:00
|
|
|
$mAddress = $mReason = $mTimestamp = '';
|
2003-09-01 13:13:56 +00:00
|
|
|
$mUser = $mBy = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
# Get a ban from the DB, with either the given address or the given username
|
2004-08-22 17:24:50 +00:00
|
|
|
function load( $address = '', $user = 0, $killExpired = true )
|
2004-02-14 12:37:25 +00:00
|
|
|
{
|
2004-07-10 03:09:26 +00:00
|
|
|
$fname = 'Block::load';
|
|
|
|
|
|
2003-09-01 13:13:56 +00:00
|
|
|
$ret = false;
|
|
|
|
|
$killed = false;
|
2004-08-15 07:23:39 +00:00
|
|
|
if ( $this->forUpdate() ) {
|
|
|
|
|
$db =& wfGetDB( DB_MASTER );
|
|
|
|
|
$options = 'FOR UPDATE';
|
|
|
|
|
} else {
|
|
|
|
|
$db =& wfGetDB( DB_SLAVE );
|
|
|
|
|
$options = '';
|
|
|
|
|
}
|
|
|
|
|
$ipblocks = $db->tableName( 'ipblocks' );
|
2004-07-10 03:09:26 +00:00
|
|
|
|
2004-08-22 17:24:50 +00:00
|
|
|
if ( 0 == $user && $address=='' ) {
|
2004-08-15 07:23:39 +00:00
|
|
|
$sql = "SELECT * from $ipblocks $options";
|
2004-06-11 15:16:00 +00:00
|
|
|
} elseif ($address=="") {
|
2004-08-15 07:23:39 +00:00
|
|
|
$sql = "SELECT * FROM $ipblocks WHERE ipb_user={$user} $options";
|
2004-06-11 15:16:00 +00:00
|
|
|
} elseif ($user=="") {
|
2004-08-15 07:23:39 +00:00
|
|
|
$sql = "SELECT * FROM $ipblocks WHERE ipb_address='" . $db->strencode( $address ) . "' $options";
|
2003-09-01 13:13:56 +00:00
|
|
|
} else {
|
2004-08-15 07:23:39 +00:00
|
|
|
$sql = "SELECT * FROM $ipblocks WHERE (ipb_address='" . $db->strencode( $address ) .
|
|
|
|
|
"' OR ipb_user={$user}) $options";
|
2003-09-01 13:13:56 +00:00
|
|
|
}
|
|
|
|
|
|
2004-08-15 07:23:39 +00:00
|
|
|
$res = $db->query( $sql, $fname );
|
|
|
|
|
if ( 0 == $db->numRows( $res ) ) {
|
2003-09-01 13:13:56 +00:00
|
|
|
# User is not blocked
|
|
|
|
|
$this->clear();
|
|
|
|
|
} else {
|
|
|
|
|
# Get first block
|
2004-08-15 07:23:39 +00:00
|
|
|
$row = $db->fetchObject( $res );
|
2003-09-01 13:13:56 +00:00
|
|
|
$this->initFromRow( $row );
|
|
|
|
|
|
|
|
|
|
if ( $killExpired ) {
|
|
|
|
|
# If requested, delete expired rows
|
|
|
|
|
do {
|
|
|
|
|
$killed = $this->deleteIfExpired();
|
2003-09-07 13:56:25 +00:00
|
|
|
if ( $killed ) {
|
2004-08-15 07:23:39 +00:00
|
|
|
$row = $db->fetchObject( $res );
|
2003-09-07 13:56:25 +00:00
|
|
|
if ( $row ) {
|
|
|
|
|
$this->initFromRow( $row );
|
|
|
|
|
}
|
|
|
|
|
}
|
2003-09-01 13:13:56 +00:00
|
|
|
} while ( $killed && $row );
|
|
|
|
|
|
|
|
|
|
# If there were any left after the killing finished, return true
|
2003-09-07 13:56:25 +00:00
|
|
|
if ( !$row ) {
|
2003-09-01 13:13:56 +00:00
|
|
|
$ret = false;
|
|
|
|
|
$this->clear();
|
|
|
|
|
} else {
|
|
|
|
|
$ret = true;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
$ret = true;
|
|
|
|
|
}
|
|
|
|
|
}
|
2004-08-15 07:23:39 +00:00
|
|
|
$db->freeResult( $res );
|
2003-09-01 13:13:56 +00:00
|
|
|
return $ret;
|
|
|
|
|
}
|
|
|
|
|
|
2004-02-14 12:37:25 +00:00
|
|
|
function initFromRow( $row )
|
|
|
|
|
{
|
2003-09-01 13:13:56 +00:00
|
|
|
$this->mAddress = $row->ipb_address;
|
|
|
|
|
$this->mReason = $row->ipb_reason;
|
|
|
|
|
$this->mTimestamp = $row->ipb_timestamp;
|
|
|
|
|
$this->mUser = $row->ipb_user;
|
|
|
|
|
$this->mBy = $row->ipb_by;
|
2003-09-07 13:56:25 +00:00
|
|
|
$this->mAuto = $row->ipb_auto;
|
|
|
|
|
$this->mId = $row->ipb_id;
|
2004-02-14 12:37:25 +00:00
|
|
|
$this->mExpiry = $row->ipb_expiry;
|
|
|
|
|
|
|
|
|
|
$this->initialiseRange();
|
2003-09-01 13:13:56 +00:00
|
|
|
}
|
|
|
|
|
|
2004-02-14 12:37:25 +00:00
|
|
|
function initialiseRange()
|
|
|
|
|
{
|
|
|
|
|
if ( $this->mUser == 0 ) {
|
2004-06-08 23:56:09 +00:00
|
|
|
$rangeParts = explode( '/', $this->mAddress );
|
2004-02-14 12:37:25 +00:00
|
|
|
if ( count( $rangeParts ) == 2 ) {
|
|
|
|
|
$this->mNetworkBits = $rangeParts[1];
|
|
|
|
|
} else {
|
|
|
|
|
$this->mNetworkBits = 32;
|
|
|
|
|
}
|
|
|
|
|
$this->mIntegerAddr = ip2long( $rangeParts[0] );
|
|
|
|
|
} else {
|
|
|
|
|
$this->mNetworkBits = false;
|
|
|
|
|
$this->mIntegerAddr = false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2003-09-01 13:13:56 +00:00
|
|
|
# Callback with a Block object for every block
|
2004-08-15 07:23:39 +00:00
|
|
|
/*static*/ function enumBlocks( $callback, $tag, $flags = 0 )
|
2004-02-14 12:37:25 +00:00
|
|
|
{
|
2003-09-01 13:13:56 +00:00
|
|
|
$block = new Block();
|
2004-08-15 07:23:39 +00:00
|
|
|
if ( $flags & EB_FOR_UPDATE ) {
|
|
|
|
|
$db =& wfGetDB( DB_MASTER );
|
|
|
|
|
$options = 'FOR UPDATE';
|
|
|
|
|
$block->forUpdate( true );
|
|
|
|
|
} else {
|
|
|
|
|
$db =& wfGetDB( DB_SLAVE );
|
|
|
|
|
$options = '';
|
|
|
|
|
}
|
|
|
|
|
$ipblocks = $db->tableName( 'ipblocks' );
|
|
|
|
|
|
|
|
|
|
$sql = "SELECT * FROM $ipblocks ORDER BY ipb_timestamp DESC $options";
|
|
|
|
|
$res = $db->query( $sql, 'Block::enumBans' );
|
2003-09-01 13:13:56 +00:00
|
|
|
|
2004-08-15 07:23:39 +00:00
|
|
|
while ( $row = $db->fetchObject( $res ) ) {
|
2003-09-01 13:13:56 +00:00
|
|
|
$block->initFromRow( $row );
|
2004-08-15 07:23:39 +00:00
|
|
|
if ( !( $flags & EB_KEEP_EXPIRED ) ) {
|
2003-09-01 13:13:56 +00:00
|
|
|
if ( !$block->deleteIfExpired() ) {
|
|
|
|
|
$callback( $block, $tag );
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
$callback( $block, $tag );
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
wfFreeResult( $res );
|
|
|
|
|
}
|
|
|
|
|
|
2004-02-14 12:37:25 +00:00
|
|
|
function delete()
|
|
|
|
|
{
|
2004-06-08 23:56:09 +00:00
|
|
|
$fname = 'Block::delete';
|
2004-07-18 08:48:43 +00:00
|
|
|
$dbw =& wfGetDB( DB_MASTER );
|
2004-07-10 03:09:26 +00:00
|
|
|
|
2004-08-22 17:24:50 +00:00
|
|
|
if ( $this->mAddress == '' ) {
|
2004-07-10 03:09:26 +00:00
|
|
|
$condition = array( 'ipb_id' => $this->mId );
|
2003-09-07 13:56:25 +00:00
|
|
|
} else {
|
2004-07-10 03:09:26 +00:00
|
|
|
$condition = array( 'ipb_address' => $this->mAddress );
|
2003-09-07 13:56:25 +00:00
|
|
|
}
|
2004-07-10 03:09:26 +00:00
|
|
|
$dbw->delete( 'ipblocks', $condition, $fname );
|
2004-02-14 12:37:25 +00:00
|
|
|
$this->clearCache();
|
2003-09-01 13:13:56 +00:00
|
|
|
}
|
|
|
|
|
|
2004-02-14 12:37:25 +00:00
|
|
|
function insert()
|
|
|
|
|
{
|
2004-07-18 08:48:43 +00:00
|
|
|
$dbw =& wfGetDB( DB_MASTER );
|
2004-07-10 03:09:26 +00:00
|
|
|
$dbw->insertArray( 'ipblocks',
|
|
|
|
|
array(
|
|
|
|
|
'ipb_address' => $this->mAddress,
|
|
|
|
|
'ipb_user' => $this->mUser,
|
|
|
|
|
'ipb_by' => $this->mBy,
|
|
|
|
|
'ipb_reason' => $this->mReason,
|
|
|
|
|
'ipb_timestamp' => $this->mTimestamp,
|
|
|
|
|
'ipb_auto' => $this->mAuto,
|
|
|
|
|
'ipb_expiry' => $this->mExpiry,
|
|
|
|
|
), 'Block::insert'
|
|
|
|
|
);
|
2004-02-14 12:37:25 +00:00
|
|
|
|
|
|
|
|
$this->clearCache();
|
2003-09-01 13:13:56 +00:00
|
|
|
}
|
|
|
|
|
|
2004-02-14 12:37:25 +00:00
|
|
|
function deleteIfExpired()
|
|
|
|
|
{
|
2003-09-01 13:13:56 +00:00
|
|
|
if ( $this->isExpired() ) {
|
|
|
|
|
$this->delete();
|
|
|
|
|
return true;
|
|
|
|
|
} else {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2004-02-14 12:37:25 +00:00
|
|
|
function isExpired()
|
|
|
|
|
{
|
2004-02-16 00:05:25 +00:00
|
|
|
if ( !$this->mExpiry ) {
|
|
|
|
|
return false;
|
|
|
|
|
} else {
|
|
|
|
|
return wfTimestampNow() > $this->mExpiry;
|
|
|
|
|
}
|
2003-09-01 13:13:56 +00:00
|
|
|
}
|
|
|
|
|
|
2004-02-14 12:37:25 +00:00
|
|
|
function isValid()
|
|
|
|
|
{
|
2004-06-08 23:56:09 +00:00
|
|
|
return $this->mAddress != '';
|
2003-09-01 13:13:56 +00:00
|
|
|
}
|
|
|
|
|
|
2004-02-27 08:25:56 +00:00
|
|
|
function updateTimestamp()
|
|
|
|
|
{
|
|
|
|
|
if ( $this->mAuto ) {
|
|
|
|
|
$this->mTimestamp = wfTimestampNow();
|
|
|
|
|
$this->mExpiry = Block::getAutoblockExpiry( $this->mTimestamp );
|
|
|
|
|
|
2004-07-18 08:48:43 +00:00
|
|
|
$dbw =& wfGetDB( DB_MASTER );
|
2004-07-10 03:09:26 +00:00
|
|
|
$dbw->updateArray( 'ipblocks',
|
|
|
|
|
array( /* SET */
|
|
|
|
|
'ipb_timestamp' => $this->mTimestamp,
|
|
|
|
|
'ipb_expiry' => $this->mExpiry,
|
|
|
|
|
), array( /* WHERE */
|
|
|
|
|
'ipb_address' => $this->mAddress
|
|
|
|
|
), 'Block::updateTimestamp'
|
|
|
|
|
);
|
2004-02-27 08:25:56 +00:00
|
|
|
|
|
|
|
|
$this->clearCache();
|
|
|
|
|
}
|
2004-02-14 12:37:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* private */ function clearCache()
|
|
|
|
|
{
|
|
|
|
|
global $wgBlockCache;
|
|
|
|
|
if ( is_object( $wgBlockCache ) ) {
|
2004-08-15 07:23:39 +00:00
|
|
|
$wgBlockCache->loadFromDB();
|
2004-02-14 12:37:25 +00:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function getIntegerAddr()
|
|
|
|
|
{
|
|
|
|
|
return $this->mIntegerAddr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function getNetworkBits()
|
|
|
|
|
{
|
|
|
|
|
return $this->mNetworkBits;
|
|
|
|
|
}
|
|
|
|
|
|
2004-08-15 07:23:39 +00:00
|
|
|
function forUpdate( $x = NULL ) {
|
|
|
|
|
return wfSetVar( $this->mForUpdate, $x );
|
|
|
|
|
}
|
|
|
|
|
|
2004-02-14 12:37:25 +00:00
|
|
|
/* static */ function getAutoblockExpiry( $timestamp )
|
|
|
|
|
{
|
|
|
|
|
global $wgAutoblockExpiry;
|
|
|
|
|
return wfUnix2Timestamp( wfTimestamp2Unix( $timestamp ) + $wgAutoblockExpiry );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* static */ function normaliseRange( $range )
|
|
|
|
|
{
|
2004-06-08 23:56:09 +00:00
|
|
|
$parts = explode( '/', $range );
|
2004-02-14 12:37:25 +00:00
|
|
|
if ( count( $parts ) == 2 ) {
|
|
|
|
|
$shift = 32 - $parts[1];
|
|
|
|
|
$ipint = ip2long( $parts[0] );
|
|
|
|
|
$ipint = $ipint >> $shift << $shift;
|
|
|
|
|
$newip = long2ip( $ipint );
|
|
|
|
|
$range = "$newip/{$parts[1]}";
|
|
|
|
|
}
|
|
|
|
|
return $range;
|
2003-09-01 13:13:56 +00:00
|
|
|
}
|
2003-09-07 13:56:25 +00:00
|
|
|
|
2003-09-01 13:13:56 +00:00
|
|
|
}
|
|
|
|
|
?>
|