ILIAS  trunk Revision v11.0_alpha-1715-g7fc467680fb
All Data Structures Namespaces Files Functions Variables Enumerations Enumerator Modules Pages
class.ilWACSignedPath.php
Go to the documentation of this file.
1 <?php
2 
19 // declare(strict_types=1);
24 
32 {
33  public const WAC_TOKEN_ID = 'il_wac_token';
34  public const WAC_TIMESTAMP_ID = 'il_wac_ts';
35  public const WAC_TTL_ID = 'il_wac_ttl';
36  public const TS_SUFFIX = 'ts';
37  public const TTL_SUFFIX = 'ttl';
38  public const MAX_LIFETIME = 600;
39 
40  protected ?ilWACPath $path_object = null;
42  protected int $type = PathType::FILE;
43  protected static int $token_max_lifetime_in_seconds = 5;
44  protected static int $cookie_max_lifetime_in_seconds = 300;
45  protected bool $checked = false;
46 
50  public function __construct(ilWACPath $ilWACPath, private GlobalHttpState $httpService, private CookieFactory $cookieFactory)
51  {
52  $this->setPathObject($ilWACPath);
53  }
54 
58  public function getSignedPath(): string
59  {
60  if ($this->getType() !== PathType::FILE) {
62  }
63  if ($this->getPathObject()->getOriginalRequest() === '' || $this->getPathObject()->getOriginalRequest() === '0') {
64  return '';
65  }
66  if (!$this->getPathObject()->fileExists()) {
67  // return $this->getPathObject()->getOriginalRequest();
68  }
69 
70  if (strpos($this->getPathObject()->getPath(), '?')) {
71  $path = $this->getPathObject()->getPath() . '&' . self::WAC_TOKEN_ID . '='
72  . $this->getTokenInstance()->getToken();
73  } else {
74  $path = $this->getPathObject()->getPath() . '?' . self::WAC_TOKEN_ID . '='
75  . $this->getTokenInstance()->getToken();
76  }
77 
78  $path .= '&' . self::WAC_TTL_ID . '=' . $this->getTokenInstance()->getTTL();
79 
80  return $path . '&' . self::WAC_TIMESTAMP_ID . '='
81  . $this->getTokenInstance()->getTimestamp();
82  }
83 
84  public function isFolderSigned(): bool
85  {
86  $this->httpService->cookieJar();
87 
88  $this->setType(PathType::FOLDER);
89  $plain_token = $this->buildTokenInstance();
90  $name = $plain_token->getHashedId();
91 
92  // Token
93  $default_token = '';
94  $token_cookie_value = $this->httpService->request()->getCookieParams()[$name] ?? $default_token;
95  // Timestamp
96  $default_timestamp = 0;
97  $timestamp_cookie_value = $this->httpService->request()->getCookieParams()[$name . self::TS_SUFFIX] ?? $default_timestamp;
98  $timestamp_cookie_value = (int) $timestamp_cookie_value;
99  // TTL
100  $default_ttl = 0;
101  $ttl_cookie_value = $this->httpService->request()->getCookieParams()[$name . self::TTL_SUFFIX] ?? $default_ttl;
102  $ttl_cookie_value = (int) $ttl_cookie_value;
103 
104  $this->getPathObject()->setToken($token_cookie_value);
105  $this->getPathObject()->setTimestamp($timestamp_cookie_value);
106  $this->getPathObject()->setTTL($ttl_cookie_value);
107  $this->buildAndSetTokenInstance();
108 
109  return $this->getPathObject()->hasToken();
110  }
111 
115  public function isFolderTokenValid(): bool
116  {
117  if (!$this->isFolderSigned()) {
118  return false;
119  }
120 
121  return $this->checkToken();
122  }
123 
124  protected function saveFolderToken(): void
125  {
126  $ttl = $this->getPathObject()->getTTL();
127  $cookie_lifetime = $ttl !== 0 ? $ttl : self::getCookieMaxLifetimeInSeconds();
128  $id = $this->getTokenInstance()->getHashedId();
129  $expire = time() + $cookie_lifetime + 3600;
130  $secure = true;
131  $domain = null;
132  $http_only = true;
133  $path = '/';
134 
135  $tokenCookie = $this->cookieFactory->create($id, $this->getTokenInstance()->getToken())
136  ->withExpires($expire)
137  ->withPath($path)
138  ->withSecure($secure)
139  ->withDomain($domain)
140  ->withHttpOnly($http_only);
141 
142  $timestampCookie = $this->cookieFactory->create($id . self::TS_SUFFIX, time())
143  ->withExpires($expire)
144  ->withPath($path)
145  ->withDomain($domain)
146  ->withSecure($secure)
147  ->withHttpOnly($http_only);
148 
149  $ttlCookie = $this->cookieFactory->create($id . self::TTL_SUFFIX, $cookie_lifetime)
150  ->withExpires($expire)
151  ->withPath($path)
152  ->withDomain($domain)
153  ->withSecure($secure)
154  ->withHttpOnly($http_only);
155 
156  $jar = $this->httpService->cookieJar()->with($tokenCookie)
157  ->with($timestampCookie)
158  ->with($ttlCookie);
159 
160  // FIX: currently the cookies are never stored, we must use setcookie
161  foreach ($jar->getAll() as $cookie) {
162  setcookie(
163  $cookie->getName(),
164  (string) $cookie->getValue(),
165  ['expires' => $cookie->getExpires(), 'path' => $cookie->getPath() ?? '/', 'domain' => $cookie->getDomain() ?? $_SERVER['REQUEST_URI'], 'secure' => $cookie->getSecure(), 'httponly' => $cookie->getHttpOnly()]
166  );
167  }
168  }
169 
170  public function revalidatingFolderToken(): bool
171  {
172  if ($this->getType() !== PathType::FOLDER) {
173  return false;
174  }
175  $this->buildAndSetTokenInstance(time(), $this->getPathObject()->getTTL());
176  $this->getPathObject()->setTTL($this->getTokenInstance()->getTTL());
177  $this->getPathObject()->setTimestamp($this->getTokenInstance()->getTimestamp());
178  $this->getPathObject()->setToken($this->getTokenInstance()->getToken());
179 
180  $this->saveFolderToken();
181 
182  return true;
183  }
184 
185  public function isSignedPath(): bool
186  {
187  return ($this->getPathObject()->hasToken() && $this->getPathObject()->hasTimestamp()
188  && $this->getPathObject()->hasTTL());
189  }
190 
194  public function isSignedPathValid(): bool
195  {
196  $this->buildAndSetTokenInstance($this->getPathObject()->getTimestamp(), $this->getPathObject()->getTTL());
197 
198  return $this->checkToken();
199  }
200 
206  public static function signFile(string $path_to_file): string
207  {
208  global $DIC;
209  if ($path_to_file === '' || $path_to_file === '0') {
210  return '';
211  }
212  $ilWACPath = new ilWACPath($path_to_file);
213  if ($ilWACPath->getClient() === '' || $ilWACPath->getClient() === '0') {
214  return $path_to_file;
215  }
216  $obj = new self($ilWACPath, $DIC->http(), new CookieFactoryImpl());
217  $obj->setType(PathType::FILE);
218  $obj->buildAndSetTokenInstance(time(), self::getTokenMaxLifetimeInSeconds());
219 
220  return $obj->getSignedPath();
221  }
222 
223  public static function signFolderOfStartFile(string $start_file_path): void
224  {
225  global $DIC;
226  $obj = new self(new ilWACPath($start_file_path), $DIC->http(), new CookieFactoryImpl());
227  $obj->setType(PathType::FOLDER);
228  $obj->buildAndSetTokenInstance(time(), self::getCookieMaxLifetimeInSeconds());
229  $obj->saveFolderToken();
230  }
231 
232  public function getTokenInstance(): ?\ilWACToken
233  {
234  return $this->token_instance;
235  }
236 
237  public function setTokenInstance(ilWACToken $token_instance): void
238  {
239  $this->token_instance = $token_instance;
240  }
241 
242  public function getType(): int
243  {
244  return $this->type;
245  }
246 
247  public function setType(int $type): void
248  {
249  $this->type = $type;
250  }
251 
252  public function getPathObject(): ?\ilWACPath
253  {
254  return $this->path_object;
255  }
256 
257  public function setPathObject(ilWACPath $path_object): void
258  {
259  $this->path_object = $path_object;
260  }
261 
265  protected function checkToken(): bool
266  {
267  $request_token_string = $this->getPathObject()->getToken();
268  $request_ttl = $this->getPathObject()->getTTL();
269  $request_timestamp = $this->getPathObject()->getTimestamp();
270  $current_timestamp = time();
271 
272  $timestamp_valid = ($current_timestamp < ($request_timestamp + $request_ttl));
273 
274  if (!$timestamp_valid) {
275  $this->setChecked(true);
276 
277  return false;
278  }
279 
280  $simulated_token = $this->buildTokenInstance($request_timestamp, $request_ttl);
281  $simulated_token_string = $simulated_token->getToken();
282  $token_valid = ($simulated_token_string === $request_token_string);
283 
284  if (!$token_valid) {
285  $this->setChecked(true);
286 
287  return false;
288  }
289 
290  return true;
291  }
292 
297  protected function buildTokenInstance(int $timestamp = 0, int $ttl = 0): \ilWACToken
298  {
299  if ($this->getType() === 0) {
301  }
302 
303  $path = match ($this->getType()) {
304  PathType::FOLDER => $this->getPathObject()->getSecurePath(),
305  default => $this->getPathObject()->getPathWithoutQuery(),
306  };
307 
308  $client = $this->getPathObject()->getClient();
309  $timestamp = $timestamp !== 0 ? $timestamp : $this->getPathObject()->getTimestamp();
310  $ttl = $ttl !== 0 ? $ttl : $this->getPathObject()->getTTL();
311 
312  return new ilWACToken($path, $client, $timestamp, $ttl);
313  }
314 
319  public function buildAndSetTokenInstance(int $timestamp = 0, int $ttl = 0): void
320  {
321  $this->setTokenInstance($this->buildTokenInstance($timestamp, $ttl));
322  }
323 
324  public static function getTokenMaxLifetimeInSeconds(): int
325  {
326  return self::$token_max_lifetime_in_seconds;
327  }
328 
333  public static function setTokenMaxLifetimeInSeconds(int $token_max_lifetime_in_seconds): void
334  {
335  if ($token_max_lifetime_in_seconds > self::MAX_LIFETIME) {
337  }
338  self::$token_max_lifetime_in_seconds = $token_max_lifetime_in_seconds;
339  }
340 
341  public static function getCookieMaxLifetimeInSeconds(): int
342  {
343  return self::$cookie_max_lifetime_in_seconds;
344  }
345 
351  public static function setCookieMaxLifetimeInSeconds(int $cookie_max_lifetime_in_seconds): void
352  {
353  if ($cookie_max_lifetime_in_seconds > self::MAX_LIFETIME) {
355  }
356  self::$cookie_max_lifetime_in_seconds = $cookie_max_lifetime_in_seconds;
357  }
358 
359  protected function getRelevantLifeTime(): int
360  {
361  $request_ttl = $this->getPathObject()->getTTL();
362  if ($request_ttl > 0) {
363  return $request_ttl;
364  }
365  $life_time = match ($this->getType()) {
366  PathType::FOLDER => self::getCookieMaxLifetimeInSeconds(),
367  PathType::FILE => self::getTokenMaxLifetimeInSeconds(),
368  default => 0,
369  };
370 
371  return $life_time;
372  }
373 
374  public function isChecked(): bool
375  {
376  return $this->checked;
377  }
378 
379  public function setChecked(bool $checked): void
380  {
381  $this->checked = $checked;
382  }
383 }
static getCookieMaxLifetimeInSeconds()
setTokenInstance(ilWACToken $token_instance)
buildAndSetTokenInstance(int $timestamp=0, int $ttl=0)
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
static setTokenMaxLifetimeInSeconds(int $token_max_lifetime_in_seconds)
static setCookieMaxLifetimeInSeconds(int $cookie_max_lifetime_in_seconds)
static int $token_max_lifetime_in_seconds
$path
Definition: ltiservices.php:29
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
$client
static signFolderOfStartFile(string $start_file_path)
setPathObject(ilWACPath $path_object)
$_SERVER['HTTP_HOST']
Definition: raiseError.php:26
__construct(ilWACPath $ilWACPath, private GlobalHttpState $httpService, private CookieFactory $cookieFactory)
ilWACSignedPath constructor.
global $DIC
Definition: shib_login.php:22
buildTokenInstance(int $timestamp=0, int $ttl=0)
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
This file is part of ILIAS, a powerful learning management system published by ILIAS open source e-Le...
foreach($mandatory_scripts as $file) $timestamp
Definition: buildRTE.php:70
$id
plugin.php for ilComponentBuildPluginInfoObjectiveTest::testAddPlugins
Definition: plugin.php:23
static int $cookie_max_lifetime_in_seconds
static signFile(string $path_to_file)
static getTokenMaxLifetimeInSeconds()
setChecked(bool $checked)