ILIAS  trunk Revision v11.0_alpha-1689-g66c127b4ae8
All Data Structures Namespaces Files Functions Variables Enumerations Enumerator Modules Pages
class.ilSessionMaxIdleIsSetObjective.php
Go to the documentation of this file.
1 <?php
2 
19 declare(strict_types=1);
20 
21 use ILIAS\Setup;
22 
23 class ilSessionMaxIdleIsSetObjective implements Setup\Objective
24 {
25  public function __construct(
26  protected Setup\Config $config
27  ) {
28  }
29 
30  public function getHash(): string
31  {
32  return hash('sha256', self::class);
33  }
34 
35  public function getLabel(): string
36  {
37  return "Ensures 'session_max_idle' is set properly";
38  }
39 
40  public function isNotable(): bool
41  {
42  return true;
43  }
44 
45  public function getPreconditions(Setup\Environment $environment): array
46  {
47  $http_config = $environment->getConfigFor('http');
48 
49  return [
53  new ilHttpConfigStoredObjective($http_config)
54  ];
55  }
56 
57  public function achieve(Setup\Environment $environment): Setup\Environment
58  {
60  $client_ini = $environment->getResource(Setup\Environment::RESOURCE_CLIENT_INI);
62  $ini = $environment->getResource(Setup\Environment::RESOURCE_ILIAS_INI);
64  $io = $environment->getResource(Setup\Environment::RESOURCE_ADMIN_INTERACTION);
65  $factory = $environment->getResource(Setup\Environment::RESOURCE_SETTINGS_FACTORY);
67  $settings = $factory->settingsFor('common');
68 
69  $session_max_idle = $this->config->getSessionMaxIdle();
70 
71  $url = $ini->readVariable('server', 'http_path');
72  $filename = uniqid((string) mt_rand(), true) . '.php';
73  $url .= '/' . $filename;
74  $token = bin2hex(random_bytes(32));
76 
77  try {
78  $curl = null;
80  $curl = $this->getCurlConnection($settings, $url, $token);
81  $result = $curl->exec();
82  } else {
83  $result = $this->getPHPIniValuesByFileGetContents($url, $token);
84  }
85  } catch (Throwable $e) {
86  $io->inform(
87  "An error occurred while trying to determine the values for 'session.cookie_lifetime' and" . PHP_EOL .
88  "'session.gc_maxlifetime' in your php.ini: {$e->getMessage()}" . PHP_EOL .
89  'You can IGNORE the the error if you are sure these settings comply with our expection to' . PHP_EOL .
90  'ensure a proper session handling.'
91  );
92 
93  $client_ini->setVariable('session', 'expire', (string) $session_max_idle);
94 
95  return $environment;
96  } finally {
97  if (!is_null($curl)) {
98  $curl->close();
99  }
100  unlink("public/$filename");
101  }
102 
103  if ($result === '') {
104  $message =
105  "ILIAS could not determine the value for 'session.cookie_lifetime' and 'session.gc_maxlifetime'" . PHP_EOL .
106  'in your php.ini to check whether it complies with our expection to ensure a proper session handling.' . PHP_EOL .
107  'Do you like to continue, anyway?';
108 
109  if (!$io->confirmOrDeny($message)) {
110  throw new Setup\NoConfirmationException($message);
111  }
112  }
113 
114  [$cookie_lifetime, $gc_maxlifetime] = explode('&', $result);
115 
116  if ($cookie_lifetime != 0) {
117  $message =
118  "The value 'session.cookie_lifetime' in your php.ini does not correspond" . PHP_EOL .
119  "to the value '0' recommended by ILIAS. Do you want to continue anyway?";
120 
121  if (!$io->confirmOrDeny($message)) {
122  throw new Setup\NoConfirmationException($message);
123  }
124  }
125 
126  if ($gc_maxlifetime <= $session_max_idle) {
127  $message =
128  "The value 'session.gc_maxlifetime' in your php.ini is smaller or equal than" . PHP_EOL .
129  "'session_max_idle' in your ILIAS-Config. ILIAS recommends a bigger value." . PHP_EOL .
130  'Do you want to continue anyway?';
131 
132  if (!$io->confirmOrDeny($message)) {
133  throw new Setup\NoConfirmationException($message);
134  }
135  }
136 
137  $client_ini->setVariable('session', 'expire', (string) $session_max_idle);
138 
139  return $environment;
140  }
141 
142  public function isApplicable(Setup\Environment $environment): bool
143  {
144  $factory = $environment->getResource(Setup\Environment::RESOURCE_SETTINGS_FACTORY);
146  $settings = $factory->settingsFor('common');
148  $ini = $environment->getResource(Setup\Environment::RESOURCE_ILIAS_INI);
150  $io = $environment->getResource(Setup\Environment::RESOURCE_ADMIN_INTERACTION);
151 
152  $url = $ini->readVariable('server', 'http_path');
153 
155  try {
156  $curl = $this->getCurlConnection($settings, $url);
157  $curl->exec();
158  $result = $curl->getInfo(CURLINFO_HTTP_CODE);
159  if ($result !== 200) {
160  throw new \Exception();
161  }
162  } catch (\Exception $e) {
163  $this->infoNoConnection($io);
164  return false;
165  } finally {
166  $curl->close();
167  }
168  } else {
169  try {
171  } catch (Exception $e) {
172  $this->infoNoConnection($io);
173  return false;
174  }
175  }
176 
177  return true;
178  }
179 
180  private function generateServerInfoFile(string $filename, string $token): void
181  {
182  $content = <<<TEXT
183 <?php
184 if (!isset(\$_GET['token'])) {
185  return "";
186 }
187 
188 if (\$_GET['token'] !== "$token") {
189  return "";
190 }
191 
192 \$scl = ini_get('session.cookie_lifetime');
193 \$smlt = ini_get('session.gc_maxlifetime');
194 
195 echo \$scl . "&" . \$smlt;
196 TEXT;
197 
198  file_put_contents("public/$filename", $content);
199  }
200 
204  private function getCurlConnection(ilSetting $settings, string $url, ?string $token = null): ilCurlConnection
205  {
206  if (!is_null($token)) {
207  $url = $url . "?token=" . $token;
208  }
209 
210  $curl = new ilCurlConnection(
211  $url,
212  new ilProxySettings($settings)
213  );
214  $curl->init();
215  $curl->setOpt(CURLOPT_SSL_VERIFYPEER, 0);
216  $curl->setOpt(CURLOPT_SSL_VERIFYHOST, 0);
217  $curl->setOpt(CURLOPT_RETURNTRANSFER, 1);
218  $curl->setOpt(CURLOPT_FOLLOWLOCATION, 1);
219  $curl->setOpt(CURLOPT_MAXREDIRS, 1);
220 
221  return $curl;
222  }
223 
227  private function getPHPIniValuesByFileGetContents(string $url, ?string $token = null): string
228  {
229  set_error_handler(static function (int $severity, string $message, string $file, int $line): never {
230  throw new ErrorException($message, $severity, $severity, $file, $line);
231  });
232 
233  if (!is_null($token)) {
234  $url = $url . "?token=" . $token;
235  }
236 
237  try {
238  return file_get_contents($url);
239  } catch (ErrorException $e) {
240  restore_error_handler();
241  throw $e;
242  }
243  }
244 
245  private function infoNoConnection(Setup\CLI\IOWrapper $io): void
246  {
247  $message =
248  "ilSessionMaxIdleIsSetObjective:\n" .
249  "Cannot establish proper connection to webserver.\n" .
250  "In the event of an installation the value for session expire\n" .
251  "will be the default value.\n" .
252  "In the event of an update, the current value for session expire\n" .
253  "is retained."
254  ;
255 
256  $io->inform($message);
257  }
258 }
getPHPIniValuesByFileGetContents(string $url, ?string $token=null)
$_GET["client_id"]
Definition: webdav.php:30
class ilProxySettings
$url
Definition: shib_logout.php:66
getCurlConnection(ilSetting $settings, string $url, ?string $token=null)
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
getPreconditions(Setup\Environment $environment)
$token
Definition: xapitoken.php:70
$filename
Definition: buildRTE.php:78
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
An environment holds resources to be used in the setup process.
Definition: Environment.php:27
$message
Definition: xapiexit.php:31
A configuration for the setup.
Definition: Config.php:26
generateServerInfoFile(string $filename, string $token)
$ini
Definition: raiseError.php:20