ILIAS  trunk Revision v11.0_alpha-1689-g66c127b4ae8
All Data Structures Namespaces Files Functions Variables Enumerations Enumerator Modules Pages
class.ilUtil.php
Go to the documentation of this file.
1 <?php
22 
36 class ilUtil
37 {
43  public static function getImageTagByType(string $a_type, string $a_path, bool $a_big = false): string
44  {
45  global $DIC;
46 
47  $lng = $DIC->language();
48 
49  $size = ($a_big)
50  ? "big"
51  : "small";
52 
53  $filename = ilObject::_getIcon(0, $size, $a_type);
54 
55  return "<img src=\"" . $filename . "\" alt=\"" . $lng->txt("obj_" . $a_type) . "\" title=\"" . $lng->txt(
56  "obj_" . $a_type
57  ) . "\" border=\"0\" vspace=\"0\"/>";
58  }
59 
66  public static function getImagePath(
67  string $image_name,
68  string $module_path = "", // this parameter only supports "" value
69  string $mode = "output", // this parameter does not control anything anymore
70  bool $offline = false
71  ): string {
72  if ($offline) {
73  return "./images/" . $image_name;
74  }
75 
76  global $DIC;
77  $styleDefinition = $DIC["styleDefinition"] ?? null;
78 
79  if ($module_path != "") {
80  throw new \LogicException(
81  "\$module_path only supports '' as value."
82  );
83  }
84 
85  $use_custom_skin = (ilStyleDefinition::getCurrentSkin() !== "default");
86 
87  if ($use_custom_skin) {
88  $filename =
89  "./Customizing/skin/"
92  . (!is_object($styleDefinition) ? "images" : $styleDefinition->getImageDirectory(ilStyleDefinition::getCurrentStyle())) . "/"
93  . $image_name;
94 
95  if (file_exists($filename)) {
96  return $filename;
97  }
98  }
99 
100  return "./assets/images/" . $image_name;
101  }
102 
109  public static function getHtmlPath(string $relative_path): string
110  {
111  if (substr($relative_path, 0, 2) == './') {
112  $relative_path = (substr($relative_path, 1));
113  }
114  if (substr($relative_path, 0, 1) != '/') {
115  $relative_path = '/' . $relative_path;
116  }
117  $htmlpath = ILIAS_HTTP_PATH . $relative_path;
118  return $htmlpath;
119  }
120 
131  public static function getStyleSheetLocation(
132  string $mode = "output",
133  string $a_css_name = ""
134  ): string {
135  $force_reload = ($mode !== "filesystem");
136  $use_default_style_file = ($a_css_name === "");
137  $use_custom_skin = (ilStyleDefinition::getCurrentSkin() !== "default");
138 
139  // use ilStyleDefinition instead of account to get the current style
140 
141  if ($use_default_style_file) {
142  $stylesheet_name = ilStyleDefinition::getCurrentStyle() . ".css";
143  } else {
144  $stylesheet_name = $a_css_name;
145  }
146 
147  if ($use_custom_skin) {
148  $filename =
149  "./Customizing/skin/"
152  . $stylesheet_name;
153  } else {
154  $filename = "./assets/css/" . $stylesheet_name;
155  }
156 
157  if (!$force_reload) {
158  return $filename;
159  }
160 
161  // add version as parameter to force reload for new releases
162 
163  // use version from template xml to force reload on changes
165  $skin_version = $skin->getVersion();
166  if ($skin_version !== "") {
167  $skin_version_appendix = str_replace(".", "-", $skin_version);
168  } else {
169  $skin_version_appendix = '0';
170  }
171 
172  return $filename . "?skin_version=" . $skin_version_appendix;
173  }
174 
180  public static function getNewContentStyleSheetLocation(string $mode = "output"): string
181  {
182 
183  // use ilStyleDefinition instead of account to get the current skin and style
184  if (ilStyleDefinition::getCurrentSkin() == "default") {
185  $in_style = "./templates/" . ilStyleDefinition::getCurrentSkin() . "/"
186  . ilStyleDefinition::getCurrentStyle() . "_cont.css";
187  } else {
188  $in_style = "./Customizing/global/skin/" . ilStyleDefinition::getCurrentSkin() . "/"
189  . ilStyleDefinition::getCurrentStyle() . "_cont.css";
190  }
191 
192  if (is_file("./" . $in_style)) {
193  return $in_style;
194  } else {
195  return "assets/css/delos_cont.css";
196  }
197  }
198 
205  public static function switchColor(int $a_num, string $a_css1, string $a_css2): string
206  {
207  if (!($a_num % 2)) {
208  return $a_css1;
209  } else {
210  return $a_css2;
211  }
212  }
213 
218  public static function makeClickable(string $a_text, bool $detectGotoLinks = false): string
219  {
220  global $DIC;
221 
222  $ret = $DIC->refinery()->string()->makeClickable()->transform($a_text);
223 
224  if ($detectGotoLinks) {
225  $goto = '<a[^>]*href="(' . str_replace('@', '\@', ILIAS_HTTP_PATH) . '/goto';
226  $regExp = $goto . '.php\?target=\w+_(\d+)[^"]*)"[^>]*>[^<]*</a>';
227  $ret = preg_replace_callback(
228  '@' . $regExp . '@i',
229  [self::class, 'replaceLinkProperties'],
230  $ret
231  );
232 
233  // Edited this regex to allow multiple links in $ret: .* to [^"><]*.
234  $regExp = $goto . '_[^"><]*[a-z0-9]+_([0-9]+)\.html)"[^>]*>[^<]*</a>';
235  $ret = preg_replace_callback(
236  '@' . $regExp . '@i',
237  [self::class, 'replaceLinkProperties'],
238  $ret
239  );
240  }
241 
242  return $ret;
243  }
244 
245  private static function replaceLinkProperties(array $matches): string
246  {
247  global $DIC;
248  $cache = $DIC['ilObjDataCache'];
249 
250  $link = $matches[0];
251  $ref_id = (int) $matches[2];
252  if ($ref_id > 0) {
253  $obj_id = $cache->lookupObjId($ref_id);
254  if ($obj_id > 0) {
255  $title = $cache->lookupTitle($obj_id);
256  $link = '<a href="' . $matches[1] . '" target="_self">' . $title . '</a>';
257  }
258  }
259 
260  return $link;
261  }
262 
272  public static function is_email(
273  string $a_email,
274  ?ilMailRfc822AddressParserFactory $mailAddressParserFactory = null
275  ): bool {
276  if ($mailAddressParserFactory === null) {
277  $mailAddressParserFactory = new ilMailRfc822AddressParserFactory();
278  }
279 
280  try {
281  $parser = $mailAddressParserFactory->getParser($a_email);
282  $addresses = $parser->parse();
283  return count($addresses) == 1 && $addresses[0]->getHost() != ilMail::ILIAS_HOST;
284  } catch (ilException $e) {
285  return false;
286  }
287  }
288 
292  public static function isLogin(string $a_login): bool
293  {
294  if (empty($a_login)) {
295  return false;
296  }
297 
298  if (strlen($a_login) < 3) {
299  return false;
300  }
301 
302  // FIXME - If ILIAS is configured to use RFC 822
303  // compliant mail addresses we should not
304  // allow the @ character.
305  if (!preg_match("/^[A-Za-z0-9_\.\+\*\@!\$\%\~\-]+$/", $a_login)) {
306  return false;
307  }
308 
309  return true;
310  }
311 
318  public static function img(
319  string $a_src,
320  ?string $a_alt = null,
321  $a_width = "",
322  $a_height = "",
323  $a_border = 0,
324  $a_id = "",
325  $a_class = ""
326  ) {
327  $img = '<img src="' . $a_src . '"';
328  if (!is_null($a_alt)) {
329  $img .= ' alt="' . htmlspecialchars($a_alt) . '"';
330  }
331  if ($a_width != "") {
332  $img .= ' width="' . htmlspecialchars($a_width) . '"';
333  }
334  if ($a_height != "") {
335  $img .= ' height="' . htmlspecialchars($a_height) . '"';
336  }
337  if ($a_class != "") {
338  $img .= ' class="' . $a_class . '"';
339  }
340  if ($a_id != "") {
341  $img .= ' id="' . $a_id . '"';
342  }
343  $img .= ' />';
344 
345  return $img;
346  }
347 
351  public static function deliverData(
352  string $a_data,
353  string $a_filename,
354  string $mime = "application/octet-stream"
355  ): void {
356  global $DIC;
357  $delivery = new Delivery(
358  Delivery::DIRECT_PHP_OUTPUT,
359  $DIC->http()
360  );
361  $delivery->setMimeType($mime);
362  $delivery->setSendMimeType(true);
363  $delivery->setDisposition(Delivery::DISP_ATTACHMENT);
364  $delivery->setDownloadFileName($a_filename);
365  $delivery->setConvertFileNameToAsci(true);
366  $repsonse = $DIC->http()->response()->withBody(Streams::ofString($a_data));
367  $DIC->http()->saveResponse($repsonse);
368  $delivery->deliver();
369  }
370 
374  public static function appendUrlParameterString(string $a_url, string $a_par, bool $xml_style = false): string
375  {
376  $amp = $xml_style
377  ? "&amp;"
378  : "&";
379 
380  $url = (is_int(strpos($a_url, "?")))
381  ? $a_url . $amp . $a_par
382  : $a_url . "?" . $a_par;
383 
384  return $url;
385  }
386 
390  public static function stripSlashes(string $a_str, bool $a_strip_html = true, string $a_allow = ""): string
391  {
392  if (ini_get("magic_quotes_gpc")) {
393  $a_str = stripslashes($a_str);
394  }
395 
396  return ilUtil::secureString($a_str, $a_strip_html, $a_allow);
397  }
398 
402  public static function stripOnlySlashes(string $a_str): string
403  {
404  if (ini_get("magic_quotes_gpc")) {
405  $a_str = stripslashes($a_str);
406  }
407 
408  return $a_str;
409  }
410 
414  public static function secureString(string $a_str, bool $a_strip_html = true, string $a_allow = ""): string
415  {
416  // check whether all allowed tags can be made secure
417  $only_secure = true;
418  $allow_tags = explode(">", $a_allow);
419  $sec_tags = ilUtil::getSecureTags();
420  $allow_array = [];
421  foreach ($allow_tags as $allow) {
422  if ($allow != "") {
423  $allow = str_replace("<", "", $allow);
424 
425  if (!in_array($allow, $sec_tags)) {
426  $only_secure = false;
427  }
428  $allow_array[] = $allow;
429  }
430  }
431 
432  // default behaviour: allow only secure tags 1:1
433  if (($only_secure || $a_allow == "") && $a_strip_html) {
434  if ($a_allow === "") {
435  $allow_array = ["b",
436  "i",
437  "strong",
438  "em",
439  "code",
440  "cite",
441  "gap",
442  "sub",
443  "sup",
444  "pre",
445  "strike",
446  "bdo"
447  ];
448  }
449 
450  // this currently removes parts of strings like "a <= b"
451  // because "a <= b" is treated like "<spam onclick='hurt()'>ss</spam>"
452  $a_str = ilUtil::maskSecureTags($a_str, $allow_array);
453  $a_str = strip_tags($a_str); // strip all other tags
454  $a_str = ilUtil::unmaskSecureTags($a_str, $allow_array);
455 
456  // a possible solution could be something like:
457  // $a_str = str_replace("<", "&lt;", $a_str);
458  // $a_str = str_replace(">", "&gt;", $a_str);
459  // $a_str = ilUtil::unmaskSecureTags($a_str, $allow_array);
460  //
461  // output would be ok then, but input fields would show
462  // "a &lt;= b" for input "a <= b" if data is brought back to a form
463  } else {
464  // only for scripts, that need to allow more/other tags and parameters
465  if ($a_strip_html) {
466  $a_str = ilUtil::stripScriptHTML($a_str, $a_allow);
467  }
468  }
469 
470  return $a_str;
471  }
472 
473  public static function getSecureTags(): array
474  {
475  return ["strong",
476  "em",
477  "u",
478  "strike",
479  "ol",
480  "li",
481  "ul",
482  "p",
483  "div",
484  "i",
485  "b",
486  "code",
487  "sup",
488  "sub",
489  "pre",
490  "gap",
491  "a",
492  "img",
493  "bdo"
494  ];
495  }
496 
497  private static function maskSecureTags(string $a_str, array $allow_array): string
498  {
499  foreach ($allow_array as $t) {
500  switch ($t) {
501  case "a":
502  $a_str = ilUtil::maskAttributeTag($a_str, "a", "href");
503  break;
504 
505  case "img":
506  $a_str = ilUtil::maskAttributeTag($a_str, "img", "src");
507  break;
508 
509  case "p":
510  case "div":
511  $a_str = ilUtil::maskTag($a_str, $t, [
512  ["param" => "align", "value" => "left"],
513  ["param" => "align", "value" => "center"],
514  ["param" => "align", "value" => "justify"],
515  ["param" => "align", "value" => "right"]
516  ]);
517  break;
518 
519  default:
520  $a_str = ilUtil::maskTag($a_str, $t);
521  break;
522  }
523  }
524 
525  return $a_str;
526  }
527 
528  private static function unmaskSecureTags(string $a_str, array $allow_array): string
529  {
530  foreach ($allow_array as $t) {
531  switch ($t) {
532  case "a":
533  $a_str = ilUtil::unmaskAttributeTag($a_str, "a", "href");
534  break;
535 
536  case "img":
537  $a_str = ilUtil::unmaskAttributeTag($a_str, "img", "src");
538  break;
539 
540  case "p":
541  case "div":
542  $a_str = ilUtil::unmaskTag($a_str, $t, [
543  ["param" => "align", "value" => "left"],
544  ["param" => "align", "value" => "center"],
545  ["param" => "align", "value" => "justify"],
546  ["param" => "align", "value" => "right"]
547  ]);
548  break;
549 
550  default:
551  $a_str = ilUtil::unmaskTag($a_str, $t);
552  break;
553  }
554  }
555 
556  return $a_str;
557  }
558 
562  public static function securePlainString(string $a_str): string
563  {
564  if (ini_get("magic_quotes_gpc")) {
565  return stripslashes($a_str);
566  } else {
567  return $a_str;
568  }
569  }
570 
587  public static function htmlencodePlainString(
588  string $a_str,
589  bool $a_make_links_clickable,
590  bool $a_detect_goto_links = false
591  ): string {
592  $encoded = "";
593 
594  if ($a_make_links_clickable) {
595  // Find text sequences in the plain text string which match
596  // the URI syntax rules, and pass them to ilUtil::makeClickable.
597  // Encode all other text sequences in the plain text string using
598  // htmlspecialchars and nl2br.
599  // The following expressions matches URI's as specified in RFC 2396.
600  //
601  // The expression matches URI's, which start with some well known
602  // schemes, like "http:", or with "www.". This must be followed
603  // by at least one of the following RFC 2396 expressions:
604  // - alphanum: [a-zA-Z0-9]
605  // - reserved: [;\/?:|&=+$,]
606  // - mark: [\\-_.!~*\'()]
607  // - escaped: %[0-9a-fA-F]{2}
608  // - fragment delimiter: #
609  // - uric_no_slash: [;?:@&=+$,]
610  $matches = [];
611  $numberOfMatches = preg_match_all(
612  '/(?:(?:http|https|ftp|ftps|mailto):|www\.)(?:[a-zA-Z0-9]|[;\/?:|&=+$,]|[\\-_.!~*\'()]|%[0-9a-fA-F]{2}|#|[;?:@&=+$,])+/',
613  $a_str,
614  $matches,
615  PREG_OFFSET_CAPTURE
616  );
617  $pos1 = 0;
618  $encoded = "";
619 
620  foreach ($matches[0] as $match) {
621  $matched_text = $match[0];
622  $pos2 = $match[1];
623 
624  // encode plain text
625  $encoded .= nl2br(htmlspecialchars(substr($a_str, $pos1, $pos2 - $pos1)));
626 
627  // encode URI
628  $encoded .= ilUtil::makeClickable($matched_text, $a_detect_goto_links);
629 
630 
631  $pos1 = $pos2 + strlen($matched_text);
632  }
633  if ($pos1 < strlen($a_str)) {
634  $encoded .= nl2br(htmlspecialchars(substr($a_str, $pos1)));
635  }
636  } else {
637  $encoded = nl2br(htmlspecialchars($a_str));
638  }
639  return $encoded;
640  }
641 
642  private static function maskAttributeTag(string $a_str, string $tag, string $tag_att): string
643  {
644  global $DIC;
645 
646  $ilLog = $DIC["ilLog"];
647 
648  $ws = "[\s]*";
649  $att = $ws . "[^>]*" . $ws;
650 
651  while (preg_match(
652  '/<(' . $tag . $att . '(' . $tag_att . $ws . '="' . $ws . '(([$@!*()~;,_0-9A-z\/:=%.&#?+\-])*)")' . $att . ')>/i',
653  $a_str,
654  $found
655  )) {
656  $old_str = $a_str;
657  $a_str = preg_replace(
658  "/<" . preg_quote($found[1], "/") . ">/i",
659  '&lt;' . $tag . ' ' . $tag_att . $tag_att . '="' . $found[3] . '"&gt;',
660  $a_str
661  );
662  if ($old_str == $a_str) {
663  $ilLog->write(
664  "ilUtil::maskA-" . htmlentities($old_str) . " == " .
665  htmlentities($a_str)
666  );
667  return $a_str;
668  }
669  }
670  $a_str = str_ireplace(
671  "</$tag>",
672  "&lt;/$tag&gt;",
673  $a_str
674  );
675  return $a_str;
676  }
677 
678  private static function unmaskAttributeTag(string $a_str, string $tag, string $tag_att): string
679  {
680  global $DIC;
681 
682  $ilLog = $DIC["ilLog"];
683 
684  while (preg_match(
685  '/&lt;(' . $tag . ' ' . $tag_att . $tag_att . '="(([$@!*()~;,_0-9A-z\/:=%.&#?+\-])*)")&gt;/i',
686  $a_str,
687  $found
688  )) {
689  $old_str = $a_str;
690  $a_str = preg_replace(
691  "/&lt;" . preg_quote($found[1], "/") . "&gt;/i",
692  '<' . $tag . ' ' . $tag_att . '="' . ilUtil::secureLink($found[2]) . '">',
693  $a_str
694  );
695  if ($old_str == $a_str) {
696  $ilLog->write(
697  "ilUtil::unmaskA-" . htmlentities($old_str) . " == " .
698  htmlentities($a_str)
699  );
700  return $a_str;
701  }
702  }
703  $a_str = str_replace('&lt;/' . $tag . '&gt;', '</' . $tag . '>', $a_str);
704  return $a_str;
705  }
706 
707  public static function maskTag(string $a_str, string $tag, array $fix_param = []): string
708  {
709  $a_str = str_replace(
710  ["<$tag>", "<" . strtoupper($tag) . ">"],
711  "&lt;" . $tag . "&gt;",
712  $a_str
713  );
714  $a_str = str_replace(
715  ["</$tag>", "</" . strtoupper($tag) . ">"],
716  "&lt;/" . $tag . "&gt;",
717  $a_str
718  );
719 
720  foreach ($fix_param as $p) {
721  $k = $p["param"];
722  $v = $p["value"];
723  $a_str = str_replace(
724  "<$tag $k=\"$v\">",
725  "&lt;" . "$tag $k=\"$v\"" . "&gt;",
726  $a_str
727  );
728  }
729 
730  return $a_str;
731  }
732 
733  private static function unmaskTag(string $a_str, string $tag, array $fix_param = []): string
734  {
735  $a_str = str_replace("&lt;" . $tag . "&gt;", "<" . $tag . ">", $a_str);
736  $a_str = str_replace("&lt;/" . $tag . "&gt;", "</" . $tag . ">", $a_str);
737 
738  foreach ($fix_param as $p) {
739  $k = $p["param"];
740  $v = $p["value"];
741  $a_str = str_replace(
742  "&lt;$tag $k=\"$v\"&gt;",
743  "<" . "$tag $k=\"$v\"" . ">",
744  $a_str
745  );
746  }
747  return $a_str;
748  }
749 
753  public static function secureLink(string $a_str): string
754  {
755  $a_str = str_ireplace("javascript", "jvscrpt", $a_str);
756  $a_str = str_ireplace(["%00",
757  "%0a",
758  "%0d",
759  "%1a",
760  "&#00;",
761  "&#x00;",
762  "&#0;",
763  "&#x0;",
764  "&#x0a;",
765  "&#x0d;",
766  "&#10;",
767  "&#13;"
768  ], "-", $a_str);
769  return $a_str;
770  }
771 
775  public static function stripScriptHTML(string $a_str, string $a_allow = "", bool $a_rm_js = true): string
776  {
777  $negativestr = "a,abbr,acronym,address,applet,area,base,basefont," .
778  "big,blockquote,body,br,button,caption,center,cite,code,col," .
779  "colgroup,dd,del,dfn,dir,div,dl,dt,em,fieldset,font,form,frame," .
780  "frameset,h1,h2,h3,h4,h5,h6,head,hr,html,i,iframe,img,input,ins,isindex,kbd," .
781  "label,legend,li,link,map,menu,meta,noframes,noscript,object,ol," .
782  "optgroup,option,p,param,q,s,samp,script,select,small,span," .
783  "strike,strong,style,sub,sup,table,tbody,td,textarea,tfoot,th,thead," .
784  "title,tr,tt,u,ul,var";
785  $a_allow = strtolower($a_allow);
786  $negatives = explode(",", $negativestr);
787  $outer_old_str = "";
788  while ($outer_old_str != $a_str) {
789  $outer_old_str = $a_str;
790  foreach ($negatives as $item) {
791  $pos = strpos($a_allow, "<$item>");
792 
793  // remove complete tag, if not allowed
794  if ($pos === false) {
795  $old_str = "";
796  while ($old_str != $a_str) {
797  $old_str = $a_str;
798  $a_str = preg_replace("/<\/?\s*$item(\/?)\s*>/i", "", $a_str);
799  $a_str = preg_replace("/<\/?\s*$item(\/?)\s+([^>]*)>/i", "", $a_str);
800  }
801  }
802  }
803  }
804 
805  if ($a_rm_js) {
806  // remove all attributes if an "on..." attribute is given
807  $a_str = preg_replace("/<\s*\w*(\/?)(\s+[^>]*)?(\s+on[^>]*)>/i", "", $a_str);
808 
809  // remove all attributes if a "javascript" is within tag
810  $a_str = preg_replace("/<\s*\w*(\/?)\s+[^>]*javascript[^>]*>/i", "", $a_str);
811 
812  // remove all attributes if an "expression" is within tag
813  // (IE allows something like <b style='width:expression(alert(1))'>test</b>)
814  $a_str = preg_replace("/<\s*\w*(\/?)\s+[^>]*expression[^>]*>/i", "", $a_str);
815  }
816 
817  return $a_str;
818  }
819 
823  public static function secureUrl(string $url): string
824  {
825  // check if url is valid (absolute or relative)
826  if (filter_var($url, FILTER_VALIDATE_URL) === false &&
827  filter_var("http://" . $url, FILTER_VALIDATE_URL) === false &&
828  filter_var("http:" . $url, FILTER_VALIDATE_URL) === false &&
829  filter_var("http://de.de" . $url, FILTER_VALIDATE_URL) === false &&
830  filter_var("http://de.de/" . $url, FILTER_VALIDATE_URL) === false) {
831  return "";
832  }
833  if (trim(strtolower(parse_url($url, PHP_URL_SCHEME) ?? '')) === "javascript") {
834  return "";
835  }
836 
837  return htmlspecialchars($url, ENT_QUOTES);
838  }
839 
843  public static function extractParameterString(string $a_parstr): array
844  {
845  // parse parameters in array
846  $par = [];
847  $ok = true;
848  while (($spos = strpos($a_parstr, "=")) && $ok) {
849  // extract parameter
850  $cpar = substr($a_parstr, 0, $spos);
851  $a_parstr = substr($a_parstr, $spos, strlen($a_parstr) - $spos);
852  while (substr($cpar, 0, 1) == "," || substr($cpar, 0, 1) == " " || substr($cpar, 0, 1) == chr(13) || substr(
853  $cpar,
854  0,
855  1
856  ) == chr(10)) {
857  $cpar = substr($cpar, 1, strlen($cpar) - 1);
858  }
859  while (substr($cpar, strlen($cpar) - 1, 1) == " " || substr($cpar, strlen($cpar) - 1, 1) == chr(
860  13
861  ) || substr($cpar, strlen($cpar) - 1, 1) == chr(10)) {
862  $cpar = substr($cpar, 0, strlen($cpar) - 1);
863  }
864 
865  // parameter name should only
866  $cpar_old = "";
867  while ($cpar != $cpar_old) {
868  $cpar_old = $cpar;
869  $cpar = preg_replace("/[^a-zA-Z0-9_]/i", "", $cpar);
870  }
871 
872  // extract value
873  if ($cpar != "") {
874  if ($spos = strpos($a_parstr, "\"")) {
875  $a_parstr = substr($a_parstr, $spos + 1, strlen($a_parstr) - $spos);
876  $spos = strpos($a_parstr, "\"");
877  if (is_int($spos)) {
878  $cval = substr($a_parstr, 0, $spos);
879  $par[$cpar] = $cval;
880  $a_parstr = substr($a_parstr, $spos + 1, strlen($a_parstr) - $spos - 1);
881  } else {
882  $ok = false;
883  }
884  } else {
885  $ok = false;
886  }
887  }
888  }
889 
890  return $ok ? $par : [];
891  }
892 
896  public static function yn2tf(string $a_yn): bool
897  {
898  if (strtolower($a_yn) === "y") {
899  return true;
900  } else {
901  return false;
902  }
903  }
904 
908  public static function tf2yn(bool $a_tf): string
909  {
910  if ($a_tf) {
911  return "y";
912  } else {
913  return "n";
914  }
915  }
916 
922  public static function deducibleSize(string $a_mime): bool
923  {
924  if (($a_mime == "image/gif") || ($a_mime == "image/jpeg") ||
925  ($a_mime == "image/png") || ($a_mime == "application/x-shockwave-flash") ||
926  ($a_mime == "image/tiff") || ($a_mime == "image/x-ms-bmp") ||
927  ($a_mime == "image/psd") || ($a_mime == "image/iff")) {
928  return true;
929  } else {
930  return false;
931  }
932  }
933 
937  public static function redirect(string $a_script): void
938  {
939  global $DIC;
940 
941  if (!isset($DIC['ilCtrl']) || !$DIC['ilCtrl'] instanceof ilCtrl) {
942  (new InitCtrlService())->init($DIC);
943  }
944  $DIC->ctrl()->redirectToURL($a_script);
945  }
946 
954  public static function insertInstIntoID(string $a_value): string
955  {
956  if (substr($a_value, 0, 4) == "il__") {
957  $a_value = "il_" . IL_INST_ID . "_" . substr($a_value, 4, strlen($a_value) - 4);
958  }
959 
960  return $a_value;
961  }
962 
974  public static function groupNameExists(string $a_group_name, ?int $a_id = null): bool
975  {
976  global $DIC;
977 
978  $ilDB = $DIC->database();
979 
980  $ilErr = null;
981  if (isset($DIC["ilErr"])) {
982  $ilErr = $DIC["ilErr"];
983  }
984 
985  if (empty($a_group_name)) {
986  $message = __METHOD__ . ": No groupname given!";
987  $ilErr->raiseError($message, $ilErr->WARNING);
988  }
989 
990  $clause = ($a_id !== null) ? " AND obj_id != " . $ilDB->quote($a_id) . " " : "";
991 
992  $q = "SELECT obj_id FROM object_data " .
993  "WHERE title = " . $ilDB->quote($a_group_name, "text") . " " .
994  "AND type = " . $ilDB->quote("grp", "text") .
995  $clause;
996 
997  $r = $ilDB->query($q);
998 
999  return $r->numRows() > 0;
1000  }
1001 
1005  public static function isWindows(): bool
1006  {
1007  return (strtolower(substr(php_uname(), 0, 3)) === "win");
1008  }
1009 
1015  public static function now(): string
1016  {
1017  return date("Y-m-d H:i:s");
1018  }
1019 
1048  public static function _getObjectsByOperations(
1049  $a_obj_type,
1050  string $a_operation,
1051  int $a_usr_id = 0,
1052  int $limit = 0
1053  ): array {
1054  global $DIC;
1055 
1056  $ilDB = $DIC->database();
1057  $rbacreview = $DIC->rbac()->review();
1058  $ilAccess = $DIC->access();
1059  $ilUser = $DIC->user();
1060  $ilSetting = $DIC->settings();
1061  $tree = $DIC->repositoryTree();
1062 
1063  if (!is_array($a_obj_type)) {
1064  $where = "WHERE type = " . $ilDB->quote($a_obj_type, "text") . " ";
1065  } else {
1066  $where = "WHERE " . $ilDB->in("type", $a_obj_type, false, "text") . " ";
1067  }
1068 
1069  // limit number of results default is search result limit
1070  if (!$limit) {
1071  $limit = (int) $ilSetting->get('search_max_hits', "100");
1072  }
1073  if ($limit == -1) {
1074  $limit = 10000;
1075  }
1076 
1077  // default to logged in usr
1078  $a_usr_id = $a_usr_id ?: $ilUser->getId();
1079  $a_roles = $rbacreview->assignedRoles($a_usr_id);
1080 
1081  // Since no rbac_pa entries are available for the system role. This function returns !all! ref_ids in the case the user
1082  // is assigned to the system role
1083  if ($rbacreview->isAssigned($a_usr_id, SYSTEM_ROLE_ID)) {
1084  $query = "SELECT ref_id FROM object_reference obr LEFT JOIN object_data obd ON obr.obj_id = obd.obj_id " .
1085  "LEFT JOIN tree ON obr.ref_id = tree.child " .
1086  $where .
1087  "AND tree = 1";
1088 
1089  $res = $ilDB->query($query);
1090  $counter = 0;
1091  $ref_ids = [];
1092  while ($row = $ilDB->fetchObject($res)) {
1093  // Filter recovery folder
1094  if ($tree->isGrandChild(RECOVERY_FOLDER_ID, $row->ref_id)) {
1095  continue;
1096  }
1097 
1098  if ($counter++ >= $limit) {
1099  break;
1100  }
1101 
1102  $ref_ids[] = $row->ref_id;
1103  }
1104  return $ref_ids;
1105  } // End Administrators
1106 
1107  // Check ownership if it is not asked for edit_permission or a create permission
1108  if ($a_operation == 'edit_permissions' or strpos($a_operation, 'create') !== false) {
1109  $check_owner = ") ";
1110  } else {
1111  $check_owner = "OR owner = " . $ilDB->quote($a_usr_id, "integer") . ") ";
1112  }
1113 
1114  $ops_ids = ilRbacReview::_getOperationIdsByName([$a_operation]);
1115  $ops_id = $ops_ids[0];
1116 
1117  $and = "AND ((" . $ilDB->in("rol_id", $a_roles, false, "integer") . " ";
1118 
1119  $query = "SELECT DISTINCT(obr.ref_id),obr.obj_id,type FROM object_reference obr " .
1120  "JOIN object_data obd ON obd.obj_id = obr.obj_id " .
1121  "LEFT JOIN rbac_pa ON obr.ref_id = rbac_pa.ref_id " .
1122  $where .
1123  $and .
1124  "AND (" . $ilDB->like("ops_id", "text", "%i:" . $ops_id . "%") . " " .
1125  "OR " . $ilDB->like("ops_id", "text", "%:\"" . $ops_id . "\";%") . ")) " .
1126  $check_owner;
1127 
1128  $res = $ilDB->query($query);
1129  $counter = 0;
1130  $ref_ids = [];
1131  while ($row = $res->fetchRow(ilDBConstants::FETCHMODE_OBJECT)) {
1132  if ($counter >= $limit) {
1133  break;
1134  }
1135 
1136  // Filter objects in recovery folder
1137  if ($tree->isGrandChild(RECOVERY_FOLDER_ID, $row->ref_id)) {
1138  continue;
1139  }
1140 
1141  // Check deleted, hierarchical access ...
1142  if ($ilAccess->checkAccessOfUser($a_usr_id, $a_operation, '', $row->ref_id, $row->type, $row->obj_id)) {
1143  $counter++;
1144  $ref_ids[] = $row->ref_id;
1145  }
1146  }
1147  return $ref_ids ?: [];
1148  }
1149 
1158  public static function isHTML(string $a_text): bool
1159  {
1160  if (strlen(strip_tags($a_text)) < strlen($a_text)) {
1161  return true;
1162  }
1163 
1164  return false;
1165  }
1166 
1173  public static function __extractRefId(string $role_title): ?int
1174  {
1175  $test_str = explode('_', $role_title);
1176  $prefix = $test_str[0] ?? '';
1177 
1178  if ($prefix === 'il') {
1179  $ref_id = $test_str[3] ?? null;
1180  return is_numeric($ref_id) ? (int) $ref_id : null;
1181  }
1182  return null;
1183  }
1184 
1193  public static function __extractId(string $ilias_id, int $inst_id): ?int
1194  {
1195  $test_str = explode('_', $ilias_id);
1196 
1197  $parsed_inst_id = (int) ($test_str[1] ?? 0);
1198  $prefix = $test_str[0] ?? '';
1199 
1200  if ($prefix === 'il' && $parsed_inst_id === $inst_id && count($test_str) === 4) {
1201  return is_numeric($test_str[3]) ? (int) $test_str[3] : null;
1202  }
1203  return null;
1204  }
1205 
1212  public static function _sortIds(array $a_ids, string $a_table, string $a_field, string $a_id_name): array
1213  {
1214  global $DIC;
1215 
1216  $ilDB = $DIC->database();
1217 
1218  if (!$a_ids) {
1219  return [];
1220  }
1221 
1222  // use database to sort user array
1223  $where = "WHERE " . $a_id_name . " IN (";
1224  $where .= implode(",", ilArrayUtil::quoteArray($a_ids));
1225  $where .= ") ";
1226 
1227  $query = "SELECT " . $a_id_name . " FROM " . $a_table . " " .
1228  $where .
1229  "ORDER BY " . $a_field;
1230 
1231  $res = $ilDB->query($query);
1232  $ids = [];
1233  while ($row = $res->fetchRow(ilDBConstants::FETCHMODE_OBJECT)) {
1234  $ids[] = $row->$a_id_name;
1235  }
1236  return $ids;
1237  }
1238 
1246  public static function getSystemMessageHTML(string $a_txt, string $a_type = "info")
1247  {
1248  global $DIC;
1249 
1250  $box_factory = $DIC->ui()->factory()->messageBox();
1251  switch ($a_type) {
1252  case 'info':
1253  $box = $box_factory->info($a_txt);
1254  break;
1255  case 'success':
1256  $box = $box_factory->success($a_txt);
1257  break;
1258  case 'question':
1259  $box = $box_factory->confirmation($a_txt);
1260  break;
1261  case 'failure':
1262  $box = $box_factory->failure($a_txt);
1263  break;
1264  default:
1265  throw new InvalidArgumentException();
1266  }
1267 
1268  return $DIC->ui()->renderer()->render($box);
1269  }
1270 
1274  public static function setCookie(
1275  string $a_cookie_name,
1276  string $a_cookie_value = '',
1277  bool $a_also_set_super_global = true,
1278  bool $a_set_cookie_invalid = false
1279  ): void {
1280  global $DIC;
1281 
1282  $http = $DIC->http();
1283  $cookie_jar = $http->cookieJar();
1284 
1285  $cookie_factory = new CookieFactoryImpl();
1286 
1287  $cookie_expire = null;
1288  if (defined('IL_COOKIE_EXPIRE') && is_numeric(IL_COOKIE_EXPIRE) && IL_COOKIE_EXPIRE > 0) {
1289  $cookie_expire = (int) IL_COOKIE_EXPIRE;
1290  }
1291 
1292  $expires = null;
1293  if ($a_set_cookie_invalid) {
1294  $expires = time() - 10;
1295  } elseif ($cookie_expire > 0) {
1296  $expires = time() + $cookie_expire;
1297  }
1298 
1299  $cookie = $cookie_factory->create($a_cookie_name, $a_cookie_value)
1300  ->withExpires($expires)
1301  ->withSecure(defined('IL_COOKIE_SECURE') ? IL_COOKIE_SECURE : false)
1302  ->withPath(defined('IL_COOKIE_PATH') ? IL_COOKIE_PATH : '')
1303  ->withDomain(defined('IL_COOKIE_DOMAIN') ? IL_COOKIE_DOMAIN : '')
1304  ->withHttpOnly(defined('IL_COOKIE_HTTPONLY') ? IL_COOKIE_HTTPONLY : false);
1305 
1306 
1307  if (
1308  defined('IL_COOKIE_SECURE') && IL_COOKIE_SECURE &&
1309  (!isset(session_get_cookie_params()['samesite']) || strtolower(session_get_cookie_params()['samesite']) !== 'strict')
1310  ) {
1311  $cookie = $cookie->withSamesite(Cookie::SAMESITE_LAX);
1312  }
1313  $jar = $cookie_jar->with($cookie);
1314  $response = $jar->renderIntoResponseHeader($http->response());
1315  $http->saveResponse($response);
1316  }
1317 
1321  public static function _getHttpPath(): string
1322  {
1323  global $DIC;
1324 
1325  $ilIliasIniFile = $DIC["ilIliasIniFile"];
1326 
1327  if ((isset($_SERVER['SHELL']) && $_SERVER['SHELL']) || PHP_SAPI === 'cli' ||
1328  // fallback for windows systems, useful in crons
1329  (class_exists("ilContext") && !ilContext::usesHTTP())) {
1330  return $ilIliasIniFile->readVariable('server', 'http_path');
1331  } else {
1332  return ILIAS_HTTP_PATH;
1333  }
1334  }
1335 
1350  public static function parseImportId(string $a_import_id): array
1351  {
1352  $exploded = explode('_', $a_import_id);
1353 
1354  $parsed['orig'] = $a_import_id;
1355  if ($exploded[0] == 'il') {
1356  $parsed['prefix'] = $exploded[0];
1357  }
1358  if (is_numeric($exploded[1])) {
1359  $parsed['inst_id'] = (int) $exploded[1];
1360  }
1361  $parsed['type'] = $exploded[2];
1362 
1363  if (is_numeric($exploded[3])) {
1364  $parsed['id'] = (int) $exploded[3];
1365  }
1366  return $parsed;
1367  }
1368 
1378  protected static function fmtFloat(
1379  float $a_float,
1380  int $a_decimals = 0,
1381  ?string $a_dec_point = null,
1382  ?string $a_thousands_sep = null,
1383  bool $a_suppress_dot_zero = false
1384  ): string {
1385  global $DIC;
1386 
1387  $lng = $DIC->language();
1388 
1389  if ($a_dec_point === null) {
1390  $a_dec_point = ".";
1391  }
1392  if ($a_dec_point === '-lang_sep_decimal-') {
1393  $a_dec_point = ".";
1394  }
1395 
1396  if ($a_thousands_sep === null) {
1397  $a_thousands_sep = $lng->txt('lang_sep_thousand');
1398  }
1399  if ($a_thousands_sep === '-lang_sep_thousand-') {
1400  $a_thousands_sep = ",";
1401  }
1402 
1403  $txt = number_format($a_float, $a_decimals, $a_dec_point, $a_thousands_sep);
1404 
1405  // remove trailing ".0"
1406  if (($a_suppress_dot_zero == 0 || $a_decimals == 0)
1407  && substr($txt, -2) == $a_dec_point . '0'
1408  ) {
1409  $txt = substr($txt, 0, strlen($txt) - 2);
1410  }
1411  if ($a_float == 0 and $txt == "") {
1412  $txt = "0";
1413  }
1414 
1415  return $txt;
1416  }
1417 
1434  public static function formatSize(int $size, string $a_mode = 'short', ?ilLanguage $a_lng = null): string
1435  {
1436  global $DIC;
1437 
1438  $lng = $DIC->language();
1439  if ($a_lng == null) {
1440  $a_lng = $lng;
1441  }
1442 
1443  $mag = 1024;
1444 
1445  if ($size >= $mag * $mag * $mag) {
1446  $scaled_size = $size / $mag / $mag / $mag;
1447  $scaled_unit = 'lang_size_gb';
1448  } else {
1449  if ($size >= $mag * $mag) {
1450  $scaled_size = $size / $mag / $mag;
1451  $scaled_unit = 'lang_size_mb';
1452  } else {
1453  if ($size >= $mag) {
1454  $scaled_size = $size / $mag;
1455  $scaled_unit = 'lang_size_kb';
1456  } else {
1457  $scaled_size = $size;
1458  $scaled_unit = 'lang_size_bytes';
1459  }
1460  }
1461  }
1462 
1463  $result = self::fmtFloat(
1464  $scaled_size,
1465  ($scaled_unit
1466  == 'lang_size_bytes') ? 0 : 1,
1467  $a_lng->txt('lang_sep_decimal'),
1468  $a_lng->txt('lang_sep_thousand'),
1469  true
1470  )
1471  . ' ' . $a_lng->txt($scaled_unit);
1472  if ($a_mode == 'long' && $size > $mag) {
1473  $result .= ' (' . self::fmtFloat(
1474  $size,
1475  0,
1476  $a_lng->txt('lang_sep_decimal'),
1477  $a_lng->txt('lang_sep_thousand')
1478  ) . ' '
1479  . $a_lng->txt('lang_size_bytes') . ')';
1480  }
1481 
1482  return $result;
1483  }
1484 }
setMimeType(string $mime_type)
Definition: Delivery.php:282
static getStyleSheetLocation(string $mode="output", string $a_css_name="")
get full style sheet file name (path inclusive) of current user
$res
Definition: ltiservices.php:66
static appendUrlParameterString(string $a_url, string $a_par, bool $xml_style=false)
static getSystemMessageHTML(string $a_txt, string $a_type="info")
Get HTML for a system message.
const IL_INST_ID
Definition: constants.php:40
static quoteArray(array $a_array)
Quotes all members of an array for usage in DB query statement.
static _getIcon(int $obj_id=0, string $size="big", string $type="", bool $offline=false)
Get icon for repository item.
Class InitCtrlService wraps the initialization of ilCtrl.
const ILIAS_HOST
static usesHTTP()
Uses HTTP aka browser.
static isHTML(string $a_text)
Checks if a given string contains HTML or not.
static switchColor(int $a_num, string $a_css1, string $a_css2)
switches style sheets for each even $a_num (used for changing colors of different result rows) ...
const SYSTEM_ROLE_ID
Definition: constants.php:29
static secureString(string $a_str, bool $a_strip_html=true, string $a_allow="")
static stripSlashes(string $a_str, bool $a_strip_html=true, string $a_allow="")
static stripScriptHTML(string $a_str, string $a_allow="", bool $a_rm_js=true)
$response
Definition: xapitoken.php:93
static getImageTagByType(string $a_type, string $a_path, bool $a_big=false)
Builds an html image tag.
static maskTag(string $a_str, string $tag, array $fix_param=[])
static maskSecureTags(string $a_str, array $allow_array)
static replaceLinkProperties(array $matches)
static _getOperationIdsByName(array $operations)
get ops_id&#39;s by name.
static getSecureTags()
$url
Definition: shib_logout.php:66
static tf2yn(bool $a_tf)
$http
Definition: deliver.php:30
static deliverData(string $a_data, string $a_filename, string $mime="application/octet-stream")
static secureLink(string $a_str)
static secureUrl(string $url)
static img(string $a_src, ?string $a_alt=null, $a_width="", $a_height="", $a_border=0, $a_id="", $a_class="")
Build img tag.
static now()
Return current timestamp in Y-m-d H:i:s format.
$ilErr
Definition: raiseError.php:33
static isWindows()
static getNewContentStyleSheetLocation(string $mode="output")
get full style sheet file name (path inclusive) of current user
static htmlencodePlainString(string $a_str, bool $a_make_links_clickable, bool $a_detect_goto_links=false)
Encodes a plain text string into HTML for display in a browser.
static is_email(string $a_email, ?ilMailRfc822AddressParserFactory $mailAddressParserFactory=null)
This preg-based function checks whether an e-mail address is formally valid.
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
Class ilMailRfc822AddressParserFactory.
$ilIliasIniFile
Definition: server.php:35
static unmaskAttributeTag(string $a_str, string $tag, string $tag_att)
$ref_id
Definition: ltiauth.php:65
static securePlainString(string $a_str)
static getCurrentSkin()
get the current skin use always this function instead of getting the account&#39;s skin the current skin ...
static setCookie(string $a_cookie_name, string $a_cookie_value='', bool $a_also_set_super_global=true, bool $a_set_cookie_invalid=false)
$_SERVER['HTTP_HOST']
Definition: raiseError.php:26
static formatSize(int $size, string $a_mode='short', ?ilLanguage $a_lng=null)
Returns the specified file size value in a human friendly form.
static isLogin(string $a_login)
global $DIC
Definition: shib_login.php:22
static getImagePath(string $image_name, string $module_path="", string $mode="output", bool $offline=false)
get image path (for images located in a template directory)
static fmtFloat(float $a_float, int $a_decimals=0, ?string $a_dec_point=null, ?string $a_thousands_sep=null, bool $a_suppress_dot_zero=false)
format a float
static unmaskSecureTags(string $a_str, array $allow_array)
static getHtmlPath(string $relative_path)
get url of path
static _getObjectsByOperations( $a_obj_type, string $a_operation, int $a_usr_id=0, int $limit=0)
Get all objects of a specific type and check access This function is not recursive, instead it parses the serialized rbac_pa entries.
$txt
Definition: error.php:31
static parseImportId(string $a_import_id)
Parse an ilias import id Typically of type il_[IL_INST_ID]_[OBJ_TYPE]_[OBJ_ID] returns array( &#39;orig&#39; ...
$filename
Definition: buildRTE.php:78
static _sortIds(array $a_ids, string $a_table, string $a_field, string $a_id_name)
Function that sorts ids by a given table field using WHERE IN E.g: __sort(array(6,7),&#39;usr_data&#39;,&#39;lastname&#39;,&#39;usr_id&#39;) => sorts by lastname.
static redirect(string $a_script)
static maskAttributeTag(string $a_str, string $tag, string $tag_att)
Class Delivery.
Definition: Delivery.php:39
static deducibleSize(string $a_mime)
checks if mime type is provided by getimagesize()
static _getHttpPath()
static extractParameterString(string $a_parstr)
global $ilSetting
Definition: privfeed.php:31
static __extractId(string $ilias_id, int $inst_id)
extract ref id from role title, e.g.
global $lng
Definition: privfeed.php:31
const RECOVERY_FOLDER_ID
Definition: constants.php:37
static groupNameExists(string $a_group_name, ?int $a_id=null)
checks if group name already exists.
$q
Definition: shib_logout.php:21
$message
Definition: xapiexit.php:31
static yn2tf(string $a_yn)
static getCurrentStyle()
get the current style or sub style use always this function instead of getting the account&#39;s style th...
static unmaskTag(string $a_str, string $tag, array $fix_param=[])
static insertInstIntoID(string $a_value)
inserts installation id into ILIAS id
static makeClickable(string $a_text, bool $detectGotoLinks=false)
const IL_COOKIE_PATH(isset($_GET['client_id']))
Definition: index.php:47
static __extractRefId(string $role_title)
extract ref id from role title, e.g.
static stripOnlySlashes(string $a_str)
$r