ILIAS  trunk Revision v11.0_alpha-1689-g66c127b4ae8
All Data Structures Namespaces Files Functions Variables Enumerations Enumerator Modules Pages
class.ilSetupLanguage.php
Go to the documentation of this file.
1 <?php
2 
19 declare(strict_types=1);
20 
43 {
44  public array $text;
45  public string $lang_default = "en";
46  public string $lang_path;
47  public string $lang_key;
48  public string $separator = "#:#";
49  public string $comment_separator = "###";
50  protected ilDBInterface $db;
51 
52  public function __construct(string $a_lang_key)
53  {
54  $this->lang_key = $a_lang_key ?: $this->lang_default;
55  $il_absolute_path = realpath(__DIR__ . "/../../../../../");
56  $this->lang_path = $il_absolute_path . "/lang";
57  $this->cust_lang_path = $il_absolute_path . "/Customizing/global/lang";
58  }
59 
67  public function txt(string $a_topic, string $a_default_lang_fallback_mod = ''): string
68  {
69  global $log;
70 
71  if (empty($a_topic)) {
72  return "";
73  }
74 
75  $translation = $this->text[$a_topic] ?? '';
76 
77  //get position of the comment_separator
78  $pos = strpos($translation, $this->comment_separator);
79 
80  if ($pos !== false) {
81  // remove comment
82  $translation = substr($translation, 0, $pos);
83  }
84 
85  if ($translation === "") {
86  $log->writeLanguageLog($a_topic, $this->lang_key);
87  return "-" . $a_topic . "-";
88  }
89 
90  return $translation;
91  }
92 
100  public function installLanguages(array $a_lang_keys, array $a_local_keys)
101  {
102  global $ilDB;
103 
104  if (empty($a_lang_keys)) {
105  $a_lang_keys = array();
106  }
107 
108  if (empty($a_local_keys)) {
109  $a_local_keys = array();
110  }
111 
112  $err_lang = array();
113 
114  $db_langs = $this->getAvailableLanguages();
115 
116  foreach ($a_lang_keys as $lang_key) {
117  if ($this->checkLanguage($lang_key)) {
118  $this->flushLanguage($lang_key, "keep_local");
119  $this->insertLanguage($lang_key);
120 
121  if (in_array($lang_key, $a_local_keys, true) && is_dir($this->cust_lang_path)) {
122  if ($this->checkLanguage($lang_key, "local")) {
123  $this->insertLanguage($lang_key, "local");
124  } else {
125  $err_lang[] = $lang_key;
126  }
127  }
128 
129  // register language first time install
130  if (!array_key_exists($lang_key, $db_langs)) {
131  if (in_array($lang_key, $a_local_keys, true)) {
132  $itype = "installed_local";
133  } else {
134  $itype = "installed";
135  }
136  $lid = $ilDB->nextId("object_data");
137  $query = "INSERT INTO object_data " .
138  "(obj_id,type,title,description,owner,create_date,last_update) " .
139  "VALUES " .
140  "(" .
141  $ilDB->quote($lid, "integer") . "," .
142  $ilDB->quote("lng", "text") . "," .
143  $ilDB->quote($lang_key, "text") . "," .
144  $ilDB->quote($itype, "text") . "," .
145  $ilDB->quote("-1", "integer") . "," .
146  $ilDB->now() . "," .
147  $ilDB->now() .
148  ")";
149  $ilDB->manipulate($query);
150  }
151  } else {
152  $err_lang[] = $lang_key;
153  }
154  }
155 
156  foreach ($db_langs as $key => $val) {
157  if (!in_array($key, $err_lang, true)) {
158  if (in_array($key, $a_lang_keys, true)) {
159  if (in_array($key, $a_local_keys, true)) {
160  $ld = "installed_local";
161  } else {
162  $ld = "installed";
163  }
164  $query = "UPDATE object_data SET " .
165  "description = " . $ilDB->quote($ld, "text") . ", " .
166  "last_update = " . $ilDB->now() . " " .
167  "WHERE obj_id = " . $ilDB->quote($val["obj_id"], "integer") . " " .
168  "AND type = " . $ilDB->quote("lng", "text");
169  $ilDB->manipulate($query);
170  } else {
171  $this->flushLanguage($key, "all");
172 
173  if (strpos($val["status"], "installed") === 0) {
174  $query = "UPDATE object_data SET " .
175  "description = " . $ilDB->quote("not_installed", "text") . ", " .
176  "last_update = " . $ilDB->now() . " " .
177  "WHERE obj_id = " . $ilDB->quote($val["obj_id"], "integer") . " " .
178  "AND type = " . $ilDB->quote("lng", "text");
179  $ilDB->manipulate($query);
180  }
181  }
182  }
183  }
184 
185  return ($err_lang) ?: true;
186  }
187 
188 
189 
193  public function getInstalledLanguages(): array
194  {
195  global $ilDB;
196 
197  $arr = [];
198  if ($ilDB instanceof ilDBInterface) {
199  $query = "SELECT * FROM object_data " .
200  "WHERE type = " . $ilDB->quote("lng", "text") . " " .
201  "AND " . $ilDB->like("description", "text", "installed%");
202  $r = $ilDB->query($query);
203 
204  while ($row = $ilDB->fetchObject($r)) {
205  $arr[] = $row->title;
206  }
207  }
208  return $arr;
209  }
210 
214  public function getInstalledLocalLanguages(): array
215  {
216  global $ilDB;
217 
218  $arr = [];
219  if ($ilDB instanceof ilDBInterface) {
220  $query = "SELECT * FROM object_data " .
221  "WHERE type = " . $ilDB->quote("lng", "text") . " " .
222  "AND description = " . $ilDB->quote("installed_local", "text");
223  $r = $ilDB->query($query);
224 
225  while ($row = $ilDB->fetchObject($r)) {
226  $arr[] = $row->title;
227  }
228  }
229  return $arr;
230  }
231 
235  protected function getAvailableLanguages(): array
236  {
237  global $ilDB;
238 
239  $arr = array();
240 
241  $query = "SELECT * FROM object_data " .
242  "WHERE type = " . $ilDB->quote("lng", "text");
243  $r = $ilDB->query($query);
244 
245  while ($row = $ilDB->fetchObject($r)) {
246  $arr[$row->title]["obj_id"] = $row->obj_id;
247  $arr[$row->title]["status"] = $row->description;
248  }
249 
250  return $arr;
251  }
252 
264  protected function checkLanguage(string $a_lang_key, string $scope = ""): bool
265  {
266  $scopeExtension = "";
267  if (!empty($scope)) {
268  if ($scope === "global") {
269  $scope = "";
270  } else {
271  $scopeExtension = "." . $scope;
272  }
273  }
274 
276  if ($scope === "local") {
278  }
279 
280  $tmpPath = getcwd();
281  chdir($path);
282 
283  // compute lang-file name format
284  $lang_file = "ilias_" . $a_lang_key . ".lang" . $scopeExtension;
285 
286  // file check
287  if (!is_file($lang_file)) {
288  chdir($tmpPath);
289  return false;
290  }
291 
292  // header check
293  if (!$content = $this->cut_header(file($lang_file))) {
294  chdir($tmpPath);
295  return false;
296  }
297 
298  // check (counting) elements of each lang-entry
299  foreach ($content as $key => $val) {
300  $separated = explode($this->separator, trim($val));
301  $num = count($separated);
302 
303  if ($num !== 3) {
304  chdir($tmpPath);
305  return false;
306  }
307  }
308 
309  chdir($tmpPath);
310 
311  // no error occured
312  return true;
313  }
314 
325  protected function cut_header(array $content)
326  {
327  foreach ($content as $key => $val) {
328  if (trim($val) === "<!-- language file start -->") {
329  return array_slice($content, $key + 1);
330  }
331  }
332  return false;
333  }
334 
340  protected function flushLanguage(string $a_lang_key, string $a_mode = "all"): void
341  {
342  global $ilDB;
343 
344  self::_deleteLangData($a_lang_key, ($a_mode === "keep_local"));
345 
346  if ($a_mode === "all") {
347  $ilDB->manipulate("DELETE FROM lng_modules WHERE lang_key = " .
348  $ilDB->quote($a_lang_key, "text"));
349  }
350  }
351 
357  public static function _deleteLangData(string $a_lang_key, bool $a_keep_local_change): void
358  {
359  global $ilDB;
360 
361  if (!$a_keep_local_change) {
362  $ilDB->manipulate("DELETE FROM lng_data WHERE lang_key = " .
363  $ilDB->quote($a_lang_key, "text"));
364  } else {
365  $ilDB->manipulate("DELETE FROM lng_data WHERE lang_key = " .
366  $ilDB->quote($a_lang_key, "text") .
367  " AND local_change IS NULL");
368  }
369  }
370 
378  public function getLocalChanges(string $a_lang_key, string $a_min_date = "", string $a_max_date = ""): array
379  {
380  global $ilDB;
381 
382  if ($a_min_date === "") {
383  $a_min_date = "1980-01-01 00:00:00";
384  }
385  if ($a_max_date === "") {
386  $a_max_date = "2200-01-01 00:00:00";
387  }
388 
389  $q = sprintf(
390  "SELECT * FROM lng_data WHERE lang_key = %s " .
391  "AND local_change >= %s AND local_change <= %s",
392  $ilDB->quote($a_lang_key, "text"),
393  $ilDB->quote($a_min_date, "timestamp"),
394  $ilDB->quote($a_max_date, "timestamp")
395  );
396  $result = $ilDB->query($q);
397 
398  $changes = array();
399  while ($row = $result->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) {
400  $changes[$row["module"]][$row["identifier"]] = $row["value"];
401  }
402  return $changes;
403  }
404 
405 
406  //TODO: remove redundant checks here!
413  protected function insertLanguage(string $lang_key, string $scope = ""): void
414  {
415  global $ilDB;
416 
417  $lang_array = array();
418 
419  $scopeExtension = "";
420  if (!empty($scope)) {
421  if ($scope === "global") {
422  $scope = "";
423  } else {
424  $scopeExtension = "." . $scope;
425  }
426  }
427 
429  if ($scope === "local") {
431  }
432 
433  $tmpPath = getcwd();
434  chdir($path);
435 
436  $lang_file = "ilias_" . $lang_key . ".lang" . $scopeExtension;
437  $change_date = null;
438 
439  if (is_file($lang_file)) {
440  // initialize the array for updating lng_modules below
441  $lang_array = [];
442  $lang_array["common"] = [];
443 
444  // remove header first
445  if ($content = $this->cut_header(file($lang_file))) {
446  // get the local changes from the database
447  if (empty($scope)) {
448  $local_changes = $this->getLocalChanges($lang_key);
449  } elseif ($scope === "local") {
450  // set the change date to import time for a local file
451  // get the modification date of the local file
452  // get the newer local changes for a local file
453  $change_date = date("Y-m-d H:i:s", time());
454  $min_date = date("Y-m-d H:i:s", filemtime($lang_file));
455  $local_changes = $this->getLocalChanges($lang_key, $min_date);
456  }
457 
458  $query_check = false;
459  $query = "INSERT INTO lng_data (module,identifier,lang_key,value,local_change,remarks) VALUES ";
460  foreach ($content as $key => $val) {
461  // split the line of the language file
462  // [0]: module
463  // [1]: identifier
464  // [2]: value
465  // [3]: comment (optional)
466  $separated = explode($this->separator, trim($val));
467 
468  //get position of the comment_separator
469  $pos = strpos($separated[2], $this->comment_separator);
470 
471  if ($pos !== false) {
472  //cut comment of
473  $separated[2] = substr($separated[2], 0, $pos);
474  }
475 
476  // check if the value has a local change
477  if (isset($local_changes[$separated[0]])) {
478  $local_value = $local_changes[$separated[0]][$separated[1]] ?? "";
479  } else {
480  $local_value = "";
481  }
482 
483  if (empty($scope)) {
484  if ($local_value !== "" && $local_value !== $separated[2]) {
485  // keep the locally changed value
486  $lang_array[$separated[0]][$separated[1]] = $local_value;
487  continue;
488  }
489  } elseif ($scope === "local") {
490  if ($local_value !== "") {
491  // keep a locally changed value that is newer than the local file
492  $lang_array[$separated[0]][$separated[1]] = $local_value;
493  continue;
494  }
495  }
496 
497  $query .= sprintf(
498  "(%s,%s,%s,%s,%s,%s),",
499  $ilDB->quote($separated[0], "text"),
500  $ilDB->quote($separated[1], "text"),
501  $ilDB->quote($lang_key, "text"),
502  $ilDB->quote($separated[2], "text"),
503  $ilDB->quote($change_date, "timestamp"),
504  $ilDB->quote($separated[3] ?? null, "text")
505  );
506  $query_check = true;
507  $lang_array[$separated[0]][$separated[1]] = $separated[2];
508  }
509  $query = rtrim($query, ",") . " ON DUPLICATE KEY UPDATE value=VALUES(value),remarks=VALUES(remarks);";
510  if ($query_check) {
511  $ilDB->manipulate($query);
512  }
513  }
514 
515  $query = "INSERT INTO lng_modules (module, lang_key, lang_array) VALUES ";
516  $modules_to_delete = [];
517  foreach ($lang_array as $module => $lang_arr) {
518  if ($scope === "local") {
519  $q = "SELECT * FROM lng_modules WHERE " .
520  " lang_key = " . $ilDB->quote($lang_key, "text") .
521  " AND module = " . $ilDB->quote($module, "text");
522  $set = $ilDB->query($q);
523  $row = $ilDB->fetchAssoc($set);
524  $arr2 = unserialize($row["lang_array"], ["allowed_classes" => false]);
525  if (is_array($arr2)) {
526  $lang_arr = array_merge($arr2, $lang_arr);
527  }
528  }
529  $query .= sprintf(
530  "(%s,%s,%s),",
531  $ilDB->quote($module, "text"),
532  $ilDB->quote($lang_key, "text"),
533  $ilDB->quote(serialize($lang_arr), "clob"),
534  );
535  $modules_to_delete[] = $module;
536  }
537 
538  $inModulesToDelete = $ilDB->in('module', $modules_to_delete, false, 'text');
539  $ilDB->manipulate(sprintf(
540  "DELETE FROM lng_modules WHERE lang_key = %s AND $inModulesToDelete",
541  $ilDB->quote($lang_key, "text")
542  ));
543 
544  $query = rtrim($query, ",") . ";";
545  $ilDB->manipulate($query);
546  }
547 
548  chdir($tmpPath);
549  }
550 
555  public function getLocalLanguages(): array
556  {
557  $local_langs = array();
558  if (is_dir($this->cust_lang_path)) {
559  $d = dir($this->cust_lang_path);
560  $tmpPath = getcwd();
561  chdir($this->cust_lang_path);
562 
563  // get available .lang.local files
564  while ($entry = $d->read()) {
565  if (is_file($entry) && (preg_match("~(^ilias_.{2}\.lang.local$)~", $entry))) {
566  $lang_key = substr($entry, 6, 2);
567  $local_langs[] = $lang_key;
568  }
569  }
570 
571  chdir($tmpPath);
572  }
573 
574  return $local_langs;
575  }
576 
580  public function getInstallableLanguages(): array
581  {
582  $d = dir($this->lang_path);
583  $tmpPath = getcwd();
584  chdir($this->lang_path);
585 
586  $installableLanguages = [];
587  // get available lang-files
588  while ($entry = $d->read()) {
589  if (is_file($entry) && (preg_match("~(^ilias_.{2}\.lang$)~", $entry))) {
590  $lang_key = substr($entry, 6, 2);
591  $installableLanguages[] = $lang_key;
592  }
593  }
594 
595  chdir($tmpPath);
596 
597  return $installableLanguages;
598  }
599 
605  public function setDbHandler(ilDBInterface $a_db_handler): bool
606  {
607  $this->db = &$a_db_handler;
608  return true;
609  }
610 
611  public function loadLanguageModule(string $a_module): void
612  {
613  }
614 }
$scope
Definition: ltiregstart.php:47
getInstallableLanguages()
Return installable languages.
flushLanguage(string $a_lang_key, string $a_mode="all")
remove language data from database $a_lang_key language key $a_mode "all" or "keep_local" ...
getLocalChanges(string $a_lang_key, string $a_min_date="", string $a_max_date="")
get locally changed language entries $a_lang_key language key $a_min_date minimum change date "yyyy-m...
__construct(string $a_lang_key)
txt(string $a_topic, string $a_default_lang_fallback_mod='')
gets the text for a given topic
language handling for setup
$path
Definition: ltiservices.php:29
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
cut_header(array $content)
Remove *.lang header information from &#39;$content&#39;.
checkLanguage(string $a_lang_key, string $scope="")
validate the logical structure of a lang-file
installLanguages(array $a_lang_keys, array $a_local_keys)
install languages
getInstalledLanguages()
get already installed languages (in db)
getInstalledLocalLanguages()
get already installed local languages (in db)
loadLanguageModule(string $a_module)
$q
Definition: shib_logout.php:21
getAvailableLanguages()
get already registered languages (in db)
getLocalLanguages()
Searches for the existence of *.lang.local files.
string $cust_lang_path
insertLanguage(string $lang_key, string $scope="")
insert language data from file in database
setDbHandler(ilDBInterface $a_db_handler)
set db handler object object db handler Return true on success
static _deleteLangData(string $a_lang_key, bool $a_keep_local_change)
Delete languge data.
$r