MediaWiki REL1_31
Pingback.php
Go to the documentation of this file.
1<?php
23use Psr\Log\LoggerInterface;
25
31class Pingback {
32
38 const SCHEMA_REV = 15781718;
39
41 protected $logger;
42
44 protected $config;
45
47 protected $key;
48
50 protected $id;
51
56 public function __construct( Config $config = null, LoggerInterface $logger = null ) {
57 $this->config = $config ?: RequestContext::getMain()->getConfig();
58 $this->logger = $logger ?: LoggerFactory::getInstance( __CLASS__ );
59 $this->key = 'Pingback-' . $this->config->get( 'Version' );
60 }
61
66 private function shouldSend() {
67 return $this->config->get( 'Pingback' ) && !$this->checkIfSent();
68 }
69
74 private function checkIfSent() {
76 $timestamp = $dbr->selectField(
77 'updatelog',
78 'ul_value',
79 [ 'ul_key' => $this->key ],
80 __METHOD__
81 );
82 if ( $timestamp === false ) {
83 return false;
84 }
85 // send heartbeat ping if last ping was over a month ago
86 if ( time() - (int)$timestamp > 60 * 60 * 24 * 30 ) {
87 return false;
88 }
89 return true;
90 }
91
96 private function markSent() {
97 $dbw = wfGetDB( DB_MASTER );
98 $timestamp = time();
99 return $dbw->upsert(
100 'updatelog',
101 [ 'ul_key' => $this->key, 'ul_value' => $timestamp ],
102 [ 'ul_key' ],
103 [ 'ul_value' => $timestamp ],
104 __METHOD__
105 );
106 }
107
116 private function acquireLock() {
117 $cache = ObjectCache::getLocalClusterInstance();
118 if ( !$cache->add( $this->key, 1, 60 * 60 ) ) {
119 return false; // throttled
120 }
121
122 $dbw = wfGetDB( DB_MASTER );
123 if ( !$dbw->lock( $this->key, __METHOD__, 0 ) ) {
124 return false; // already in progress
125 }
126
127 return true;
128 }
129
142 public function getSystemInfo() {
143 $event = [
144 'database' => $this->config->get( 'DBtype' ),
145 'MediaWiki' => $this->config->get( 'Version' ),
146 'PHP' => PHP_VERSION,
147 'OS' => PHP_OS . ' ' . php_uname( 'r' ),
148 'arch' => PHP_INT_SIZE === 8 ? 64 : 32,
149 'machine' => php_uname( 'm' ),
150 ];
151
152 if ( isset( $_SERVER['SERVER_SOFTWARE'] ) ) {
153 $event['serverSoftware'] = $_SERVER['SERVER_SOFTWARE'];
154 }
155
156 $limit = ini_get( 'memory_limit' );
157 if ( $limit && $limit != -1 ) {
158 $event['memoryLimit'] = $limit;
159 }
160
161 return $event;
162 }
163
169 private function getData() {
170 return [
171 'schema' => 'MediaWikiPingback',
172 'revision' => self::SCHEMA_REV,
173 'wiki' => $this->getOrCreatePingbackId(),
174 'event' => $this->getSystemInfo(),
175 ];
176 }
177
186 private function getOrCreatePingbackId() {
187 if ( !$this->id ) {
188 $id = wfGetDB( DB_REPLICA )->selectField(
189 'updatelog', 'ul_value', [ 'ul_key' => 'PingBack' ] );
190
191 if ( $id == false ) {
193 $dbw = wfGetDB( DB_MASTER );
194 $dbw->insert(
195 'updatelog',
196 [ 'ul_key' => 'PingBack', 'ul_value' => $id ],
197 __METHOD__,
198 'IGNORE'
199 );
200
201 if ( !$dbw->affectedRows() ) {
202 $id = $dbw->selectField(
203 'updatelog', 'ul_value', [ 'ul_key' => 'PingBack' ] );
204 }
205 }
206
207 $this->id = $id;
208 }
209
210 return $this->id;
211 }
212
228 private function postPingback( array $data ) {
229 $json = FormatJson::encode( $data );
230 $queryString = rawurlencode( str_replace( ' ', '\u0020', $json ) ) . ';';
231 $url = 'https://www.mediawiki.org/beacon/event?' . $queryString;
232 return Http::post( $url ) !== false;
233 }
234
250 public function sendPingback() {
251 if ( !$this->acquireLock() ) {
252 $this->logger->debug( __METHOD__ . ": couldn't acquire lock" );
253 return false;
254 }
255
256 $data = $this->getData();
257 if ( !$this->postPingback( $data ) ) {
258 $this->logger->warning( __METHOD__ . ": failed to send pingback; check 'http' log" );
259 return false;
260 }
261
262 $this->markSent();
263 $this->logger->debug( __METHOD__ . ": pingback sent OK ({$this->key})" );
264 return true;
265 }
266
271 public static function schedulePingback() {
272 DeferredUpdates::addCallableUpdate( function () {
273 $instance = new Pingback;
274 if ( $instance->shouldSend() ) {
275 $instance->sendPingback();
276 }
277 } );
278 }
279}
wfGetDB( $db, $groups=[], $wiki=false)
Get a Database object.
static post( $url, $options=[], $caller=__METHOD__)
Simple wrapper for Http::request( 'POST' )
Definition Http.php:121
static generateHex( $chars, $forceStrong=false)
Generate a run of (ideally) cryptographically random data and return it in hexadecimal string format.
PSR-3 logger instance factory.
Send information about this MediaWiki instance to MediaWiki.org.
Definition Pingback.php:31
acquireLock()
Acquire lock for sending a pingback.
Definition Pingback.php:116
getOrCreatePingbackId()
Get a unique, stable identifier for this wiki.
Definition Pingback.php:186
sendPingback()
Send information about this MediaWiki instance to MediaWiki.org.
Definition Pingback.php:250
LoggerInterface $logger
Definition Pingback.php:41
shouldSend()
Should a pingback be sent?
Definition Pingback.php:66
getData()
Get the EventLogging packet to be sent to the server.
Definition Pingback.php:169
postPingback(array $data)
Serialize pingback data and send it to MediaWiki.org via a POST to its event beacon endpoint.
Definition Pingback.php:228
string $id
Randomly-generated identifier for this wiki.
Definition Pingback.php:50
Config $config
Definition Pingback.php:44
string $key
updatelog key (also used as cache/db lock key)
Definition Pingback.php:47
markSent()
Record the fact that we have sent a pingback for this MediaWiki version, to ensure we don't submit da...
Definition Pingback.php:96
__construct(Config $config=null, LoggerInterface $logger=null)
Definition Pingback.php:56
static schedulePingback()
Schedule a deferred callable that will check if a pingback should be sent and (if so) proceed to send...
Definition Pingback.php:271
checkIfSent()
Has a pingback been sent in the last month for this MediaWiki version?
Definition Pingback.php:74
getSystemInfo()
Collect basic data about this MediaWiki installation and return it as an associative array conforming...
Definition Pingback.php:142
static getMain()
Get the RequestContext object associated with the main request.
design txt This is a brief overview of the new design More thorough and up to date information is available on the documentation wiki at etc Handles the details of getting and saving to the user table of the and dealing with sessions and cookies OutputPage Encapsulates the entire HTML page that will be sent in response to any server request It is used by calling its functions to add in any and then calling but I prefer the flexibility This should also do the output encoding The system allocates a global one in $wgOut Title Represents the title of an and does all the work of translating among various forms such as plain database key
Definition design.txt:26
Interface for configuration instances.
Definition Config.php:28
$cache
Definition mcc.php:33
const DB_REPLICA
Definition defines.php:25
const DB_MASTER
Definition defines.php:29