[ Index ]

PHP Cross Reference of DokuWiki

title

Body

[close]

/lib/plugins/popularity/ -> helper.php (source)

   1  <?php
   2  
   3  use dokuwiki\Extension\AuthPlugin;
   4  use dokuwiki\HTTP\DokuHTTPClient;
   5  use dokuwiki\Extension\Event;
   6  
   7  /**
   8   * Popularity Feedback Plugin
   9   *
  10   * @license    GPL 2 (http://www.gnu.org/licenses/gpl.html)
  11   */
  12  class helper_plugin_popularity extends Dokuwiki_Plugin
  13  {
  14      /**
  15       * The url where the data should be sent
  16       */
  17      public $submitUrl = 'https://update.dokuwiki.org/popularity.php';
  18  
  19      /**
  20       * Name of the file which determine if the the autosubmit is enabled,
  21       * and when it was submited for the last time
  22       */
  23      public $autosubmitFile;
  24  
  25      /**
  26       * File where the last error which happened when we tried to autosubmit, will be log
  27       */
  28      public $autosubmitErrorFile;
  29  
  30      /**
  31       * Name of the file which determine when the popularity data was manually
  32       * submitted for the last time
  33       * (If this file doesn't exist, the data has never been sent)
  34       */
  35      public $popularityLastSubmitFile;
  36  
  37      /**
  38       * helper_plugin_popularity constructor.
  39       */
  40      public function __construct()
  41      {
  42          global $conf;
  43          $this->autosubmitFile = $conf['cachedir'] . '/autosubmit.txt';
  44          $this->autosubmitErrorFile = $conf['cachedir'] . '/autosubmitError.txt';
  45          $this->popularityLastSubmitFile = $conf['cachedir'] . '/lastSubmitTime.txt';
  46      }
  47  
  48      /**
  49       * Check if autosubmit is enabled
  50       *
  51       * @return boolean TRUE if we should send data once a month, FALSE otherwise
  52       */
  53      public function isAutoSubmitEnabled()
  54      {
  55          return file_exists($this->autosubmitFile);
  56      }
  57  
  58      /**
  59       * Send the data, to the submit url
  60       *
  61       * @param string $data The popularity data
  62       * @return string An empty string if everything worked fine, a string describing the error otherwise
  63       */
  64      public function sendData($data)
  65      {
  66          $error = '';
  67          $httpClient = new DokuHTTPClient();
  68          $status = $httpClient->sendRequest($this->submitUrl, ['data' => $data], 'POST');
  69          if (! $status) {
  70              $error = $httpClient->error;
  71          }
  72          return $error;
  73      }
  74  
  75      /**
  76       * Compute the last time the data was sent. If it has never been sent, we return 0.
  77       *
  78       * @return int
  79       */
  80      public function lastSentTime()
  81      {
  82          $manualSubmission = @filemtime($this->popularityLastSubmitFile);
  83          $autoSubmission   = @filemtime($this->autosubmitFile);
  84  
  85          return max((int) $manualSubmission, (int) $autoSubmission);
  86      }
  87  
  88      /**
  89       * Gather all information
  90       *
  91       * @return string The popularity data as a string
  92       */
  93      public function gatherAsString()
  94      {
  95          $data = $this->gather();
  96          $string = '';
  97          foreach ($data as $key => $val) {
  98              if (is_array($val)) foreach ($val as $v) {
  99                  $string .=  hsc($key) . "\t" . hsc($v) . "\n";
 100              } else {
 101                  $string .= hsc($key) . "\t" . hsc($val) . "\n";
 102              }
 103          }
 104          return $string;
 105      }
 106  
 107      /**
 108       * Initialize an empty list to be used in file traversing
 109       *
 110       * @return array
 111       * @see searchCountCallback
 112       */
 113      protected function initEmptySearchList()
 114      {
 115          return $list = array_fill_keys([
 116              'file_count',
 117              'file_size',
 118              'file_max',
 119              'file_min',
 120              'dir_count',
 121              'dir_nest',
 122              'file_oldest'
 123          ], 0);
 124      }
 125  
 126      /**
 127       * Gather all information
 128       *
 129       * @return array The popularity data as an array
 130       */
 131      protected function gather()
 132      {
 133          global $conf;
 134          /** @var $auth DokuWiki_Auth_Plugin */
 135          global $auth;
 136          $data = [];
 137          $phptime = ini_get('max_execution_time');
 138          @set_time_limit(0);
 139          $pluginInfo = $this->getInfo();
 140  
 141          // version
 142          $data['anon_id'] = md5(auth_cookiesalt());
 143          $data['version'] = getVersion();
 144          $data['popversion'] = $pluginInfo['date'];
 145          $data['language'] = $conf['lang'];
 146          $data['now']      = time();
 147          $data['popauto']  = (int) $this->isAutoSubmitEnabled();
 148  
 149          // some config values
 150          $data['conf_useacl']   = $conf['useacl'];
 151          $data['conf_authtype'] = $conf['authtype'];
 152          $data['conf_template'] = $conf['template'];
 153  
 154          // number and size of pages
 155          $list = $this->initEmptySearchList();
 156          search($list, $conf['datadir'], [$this, 'searchCountCallback'], ['all' => false], '');
 157          $data['page_count']    = $list['file_count'];
 158          $data['page_size']     = $list['file_size'];
 159          $data['page_biggest']  = $list['file_max'];
 160          $data['page_smallest'] = $list['file_min'];
 161          $data['page_nscount']  = $list['dir_count'];
 162          $data['page_nsnest']   = $list['dir_nest'];
 163          if ($list['file_count']) $data['page_avg'] = $list['file_size'] / $list['file_count'];
 164          $data['page_oldest']   = $list['file_oldest'];
 165          unset($list);
 166  
 167          // number and size of media
 168          $list = $this->initEmptySearchList();
 169          search($list, $conf['mediadir'], [$this, 'searchCountCallback'], ['all' => true]);
 170          $data['media_count']    = $list['file_count'];
 171          $data['media_size']     = $list['file_size'];
 172          $data['media_biggest']  = $list['file_max'];
 173          $data['media_smallest'] = $list['file_min'];
 174          $data['media_nscount']  = $list['dir_count'];
 175          $data['media_nsnest']   = $list['dir_nest'];
 176          if ($list['file_count']) $data['media_avg'] = $list['file_size'] / $list['file_count'];
 177          unset($list);
 178  
 179          // number and size of cache
 180          $list = $this->initEmptySearchList();
 181          search($list, $conf['cachedir'], [$this, 'searchCountCallback'], ['all' => true]);
 182          $data['cache_count']    = $list['file_count'];
 183          $data['cache_size']     = $list['file_size'];
 184          $data['cache_biggest']  = $list['file_max'];
 185          $data['cache_smallest'] = $list['file_min'];
 186          if ($list['file_count']) $data['cache_avg'] = $list['file_size'] / $list['file_count'];
 187          unset($list);
 188  
 189          // number and size of index
 190          $list = $this->initEmptySearchList();
 191          search($list, $conf['indexdir'], [$this, 'searchCountCallback'], ['all' => true]);
 192          $data['index_count']    = $list['file_count'];
 193          $data['index_size']     = $list['file_size'];
 194          $data['index_biggest']  = $list['file_max'];
 195          $data['index_smallest'] = $list['file_min'];
 196          if ($list['file_count']) $data['index_avg'] = $list['file_size'] / $list['file_count'];
 197          unset($list);
 198  
 199          // number and size of meta
 200          $list = $this->initEmptySearchList();
 201          search($list, $conf['metadir'], [$this, 'searchCountCallback'], ['all' => true]);
 202          $data['meta_count']    = $list['file_count'];
 203          $data['meta_size']     = $list['file_size'];
 204          $data['meta_biggest']  = $list['file_max'];
 205          $data['meta_smallest'] = $list['file_min'];
 206          if ($list['file_count']) $data['meta_avg'] = $list['file_size'] / $list['file_count'];
 207          unset($list);
 208  
 209          // number and size of attic
 210          $list = $this->initEmptySearchList();
 211          search($list, $conf['olddir'], [$this, 'searchCountCallback'], ['all' => true]);
 212          $data['attic_count']    = $list['file_count'];
 213          $data['attic_size']     = $list['file_size'];
 214          $data['attic_biggest']  = $list['file_max'];
 215          $data['attic_smallest'] = $list['file_min'];
 216          if ($list['file_count']) $data['attic_avg'] = $list['file_size'] / $list['file_count'];
 217          $data['attic_oldest']   = $list['file_oldest'];
 218          unset($list);
 219  
 220          // user count
 221          if ($auth instanceof AuthPlugin && $auth->canDo('getUserCount')) {
 222              $data['user_count'] = $auth->getUserCount();
 223          }
 224  
 225          // calculate edits per day
 226          $list = (array) @file($conf['metadir'] . '/_dokuwiki.changes');
 227          $count = count($list);
 228          if ($count > 2) {
 229              $first = (int) substr(array_shift($list), 0, 10);
 230              $last  = (int) substr(array_pop($list), 0, 10);
 231              $dur = ($last - $first) / (60 * 60 * 24); // number of days in the changelog
 232              $data['edits_per_day'] = $count / $dur;
 233          }
 234          unset($list);
 235  
 236          // plugins
 237          $data['plugin'] = plugin_list();
 238  
 239          // pcre info
 240          if (defined('PCRE_VERSION')) $data['pcre_version'] = PCRE_VERSION;
 241          $data['pcre_backtrack'] = ini_get('pcre.backtrack_limit');
 242          $data['pcre_recursion'] = ini_get('pcre.recursion_limit');
 243  
 244          // php info
 245          $data['os'] = PHP_OS;
 246          $data['webserver'] = $_SERVER['SERVER_SOFTWARE'];
 247          $data['php_version'] = phpversion();
 248          $data['php_sapi'] = PHP_SAPI;
 249          $data['php_memory'] = php_to_byte(ini_get('memory_limit'));
 250          $data['php_exectime'] = $phptime;
 251          $data['php_extension'] = get_loaded_extensions();
 252  
 253          // plugin usage data
 254          $this->addPluginUsageData($data);
 255  
 256          return $data;
 257      }
 258  
 259      /**
 260       * Triggers event to let plugins add their own data
 261       *
 262       * @param $data
 263       */
 264      protected function addPluginUsageData(&$data)
 265      {
 266          $pluginsData = [];
 267          Event::createAndTrigger('PLUGIN_POPULARITY_DATA_SETUP', $pluginsData);
 268          foreach ($pluginsData as $plugin => $d) {
 269              if (is_array($d)) {
 270                  foreach ($d as $key => $value) {
 271                      $data['plugin_' . $plugin . '_' . $key] = $value;
 272                  }
 273              } else {
 274                  $data['plugin_' . $plugin] = $d;
 275              }
 276          }
 277      }
 278  
 279      /**
 280       * Callback to search and count the content of directories in DokuWiki
 281       *
 282       * @param array &$data  Reference to the result data structure
 283       * @param string $base  Base usually $conf['datadir']
 284       * @param string $file  current file or directory relative to $base
 285       * @param string $type  Type either 'd' for directory or 'f' for file
 286       * @param int    $lvl   Current recursion depht
 287       * @param array  $opts  option array as given to search()
 288       * @return bool
 289       */
 290      public function searchCountCallback(&$data, $base, $file, $type, $lvl, $opts)
 291      {
 292          // traverse
 293          if ($type == 'd') {
 294              if ($data['dir_nest'] < $lvl) $data['dir_nest'] = $lvl;
 295              $data['dir_count']++;
 296              return true;
 297          }
 298  
 299          //only search txt files if 'all' option not set
 300          if ($opts['all'] || str_ends_with($file, '.txt')) {
 301              $size = filesize($base . '/' . $file);
 302              $date = filemtime($base . '/' . $file);
 303              $data['file_count']++;
 304              $data['file_size'] += $size;
 305              if (!isset($data['file_min']) || $data['file_min'] > $size) $data['file_min'] = $size;
 306              if ($data['file_max'] < $size) $data['file_max'] = $size;
 307              if (!isset($data['file_oldest']) || $data['file_oldest'] > $date) $data['file_oldest'] = $date;
 308          }
 309  
 310          return false;
 311      }
 312  }