ILIAS  trunk Revision v11.0_alpha-1702-gfd3ecb7f852
All Data Structures Namespaces Files Functions Variables Enumerations Enumerator Modules Pages
class.ilECSCmsCourseMemberCommandQueueHandler.php
Go to the documentation of this file.
1 <?php
2 
19 declare(strict_types=1);
26 {
27  protected ilLogger $log;
28 
30  private int $mid = 0;
31 
32  private ?ilECSnodeMappingSettings $mapping = null;
33 
37  public function __construct(ilECSSetting $server)
38  {
39  global $DIC;
40  $this->log = $DIC->logger()->wsrv();
41  $this->server = $server;
42  }
43 
47  public function getServer(): \ilECSSetting
48  {
49  return $this->server;
50  }
51 
55  public function getMid(): int
56  {
57  return $this->mid;
58  }
59 
63  public function checkAllocationActivation(ilECSSetting $server, $a_content_id): bool
64  {
65  try {
66  $crsm_reader = new ilECSCourseMemberConnector($server);
67  $details = $crsm_reader->getCourseMember($a_content_id, true);
68  $this->mid = $details->getMySender();
69 
70  // Check if import is enabled
71  $part = ilECSParticipantSetting::getInstance($this->getServer()->getServerId(), $this->getMid());
72  if (!$part->isImportEnabled()) {
73  $this->log->warning('Import disabled for mid ' . $this->getMid());
74  return false;
75  }
76  // Check course allocation setting
78  $this->getServer()->getServerId(),
79  $this->getMid()
80  );
81  return $this->mapping->isCourseAllocationEnabled();
82  } catch (ilECSConnectorException $e) {
83  $this->log->error('Reading course member details failed with message ' . $e->getMessage());
84  return false;
85  }
86  }
87 
88 
92  public function handleCreate(ilECSSetting $server, $a_content_id): bool
93  {
94  if (!$this->checkAllocationActivation($server, $a_content_id)) {
95  return true;
96  }
97  try {
98  //$course = $this->readCourse($server, $a_content_id);
99  $course_member = $this->readCourseMember($server, $a_content_id);
100  $this->doUpdate($a_content_id, $course_member);
101  return true;
102  } catch (ilECSConnectorException $e) {
103  $this->log->error('Course member creation failed with mesage ' . $e->getMessage());
104  return false;
105  }
106  return true;
107  }
108 
112  public function handleDelete(ilECSSetting $server, $a_content_id): bool
113  {
114  // nothing todo
115  return true;
116  }
117 
121  public function handleUpdate(ilECSSetting $server, $a_content_id): bool
122  {
123  if (!$this->checkAllocationActivation($server, $a_content_id)) {
124  return true;
125  }
126 
127  try {
128  $course_member = $this->readCourseMember($server, $a_content_id);
129  $this->doUpdate($a_content_id, $course_member);
130  return true;
131  } catch (ilECSConnectorException $e) {
132  $this->log->error('Course member update failed with mesage ' . $e->getMessage());
133  return false;
134  }
135  return true;
136  }
137 
138 
142  protected function doUpdate($a_content_id, $course_member): bool
143  {
144  $this->log->debug('Starting ecs member update');
145 
146  $course_id = (int) $course_member->lectureID;
147  if (!$course_id) {
148  $this->log->warning('Missing course id');
149  return false;
150  }
151  $this->log->debug('sid: ' . $this->getServer()->getServerId() . ' course_id: ' . $course_id . ' mid: ' . $this->mid);
152  //$crs_obj_id = ilECSImportManager::getInstance()->_lookupObjId($this->getServer()->getServerId(), $course_id, $this->mid);
153  $crs_obj_id = ilECSImportManager::getInstance()->lookupObjIdByContentId($this->getServer()->getServerId(), $this->mid, $course_id);
154 
155  if (!$crs_obj_id) {
156  $this->log->info('No main course created. Group scenario >= 3 ?');
157  }
158 
159  $course = $this->readCourse($course_member);
160  if (is_null($course)) {
161  $this->log->info('No course found, skip processing' . print_r($course_member, true));
162  return true;
163  }
164  // Lookup already imported users and update their status
165  $assignments = $this->readAssignments($course, $course_member);
166 
167  $this->log->debug('Parallel group assignments');
168  $this->log->dump($assignments, ilLogLevel::DEBUG);
169  $this->log->debug('------------------ End assignemtns');
170 
171  // iterate through all parallel groups
172  foreach ($assignments as $cms_id => $assigned) {
173  $sub_id = ($cms_id === $course_id) ? null : (string) $cms_id;
174 
175  $this->log->debug('sub id is ' . is_null($sub_id) ? "<null>" : $sub_id . ' for ' . $cms_id);
176 
177  $obj_id = ilECSImportManager::getInstance()->lookupObjIdByContentId(
178  $this->getServer()->getServerId(),
179  $this->getMid(),
180  $course_id,
181  $sub_id
182  );
183 
184  $this->refreshAssignmentStatus($course_member, $obj_id, $sub_id, $assigned);
185  }
186  return true;
187  }
188 
192  protected function readAssignments($course, $course_member): array
193  {
194  //TODO check if this switch is still needed
195  switch ((int) $course->groupScenario) {
197  $this->log->debug('Parallel group scenario one course.');
198  break;
199 
201  $this->log->debug('Parallel group scenario groups in courses.');
202  break;
203 
205  $this->log->debug('Parallel group scenario only courses.');
206  break;
207 
208  default:
209  $this->log->debug('Parallel group scenario undefined.');
210  break;
211  }
212 
213  $course_id = $course_member->lectureID;
214  $assigned = array();
215  foreach ((array) $course_member->members as $member) {
216  $assigned[$course_id][$member->personID] = array(
217  'id' => $member->personID,
218  'role' => $member->role
219  );
220  if ((int) $course->groupScenario === ilECSMappingUtils::PARALLEL_ONE_COURSE) {
221  $this->log->debug('Group scenarion "one course". Ignoring group assignments');
222  continue;
223  }
224 
225  foreach ((array) $member->groups as $pgroup) {
226  // the sequence number in the course ressource
227  $sequence_number = (int) $pgroup->num;
228  // find parallel group with by sequence number
229  $tmp_pgroup = $course->groups[$sequence_number];
230  if (is_object($tmp_pgroup) && $tmp_pgroup->id !== '') {
231  $this->log->debug('Found parallel group with id: ' . $tmp_pgroup->id . ': for sequence number: ' . $sequence_number);
232 
233  // @todo check hierarchy of roles
234  $assigned[$tmp_pgroup->id][$member->personID] = array(
235  'id' => $member->personID,
236  'role' => $pgroup->role
237  );
238  } else {
239  $this->log->warning('Cannot find parallel group with sequence id: ' . $sequence_number);
240  }
241  }
242  }
243  $this->log->debug('ECS member assignments ' . print_r($assigned, true));
244  return $assigned;
245  }
246 
247 
248 
252  protected function refreshAssignmentStatus(object $course_member, int $obj_id, ?string $sub_id, $assigned): bool
253  {
254  $this->log->debug('Currrent sub_id = ' . $sub_id . ', obj_id = ' . $obj_id);
255 
256  $type = ilObject::_lookupType($obj_id);
257  if ($type === 'crs') {
259  } elseif ($type === 'grp') {
261  } else {
262  $this->log->warning('Invalid object type given for obj_id: ' . $obj_id);
263  return false;
264  }
265 
266  $course_id = (int) $course_member->lectureID;
268  $course_id,
269  is_null($sub_id) ? $sub_id : (int) $sub_id,
270  $obj_id
271  );
272 
273  // Delete remote deleted
274  foreach ($usr_ids as $usr_id) {
275  if (!isset($assigned[$usr_id])) {
276  $ass = ilECSCourseMemberAssignment::lookupAssignment($course_id, is_null($sub_id) ? $sub_id : (int) $sub_id, $obj_id, $usr_id);
277  if ($ass instanceof ilECSCourseMemberAssignment) {
279  $this->mapping->getAuthMode(),
280  (string) $usr_id
281  );
282 
283  $this->log->debug('Local user assignment: ' . $usr_id . ' <-> ' . $login);
284 
285  if ($il_usr_id = ilObjUser::_lookupId($login)) {
286  // this removes also admin, tutor roles
287  $part->delete($il_usr_id);
288  $this->log->info('Deassigning user ' . $usr_id . ' ' . 'from course ' . ilObject::_lookupTitle($obj_id));
289  } else {
290  $this->log->notice('Deassigning unknown ILIAS user ' . $usr_id . ' ' . 'from course ' . ilObject::_lookupTitle($obj_id));
291  }
292 
293  $ass->delete();
294  }
295  }
296  }
297 
298  $this->log->debug('Handled assignmnent...');
299 
300  // Assign new participants
301  foreach ((array) $assigned as $person_id => $person) {
302  $role = $this->lookupRole((string) $person['role'], $type);
303  $role_info = ilECSMappingUtils::getRoleMappingInfo($role);
304 
305  $this->log->debug('Using role info...');
307  $this->mapping->getAuthMode(),
308  (string) $person_id
309  );
310  $this->log->info('Handling user ' . $person_id);
311 
312  if (in_array($person_id, $usr_ids, true)) {
313  if ($il_usr_id = ilObjUser::_lookupId($login)) {
314  $part->updateRoleAssignments($il_usr_id, array($part->getAutoGeneratedRoleId($role)));
315  } elseif ($role_info['create']) {
316  $this->createMember($person_id);
317  $this->log->info('Added new user ' . $person_id);
319  $this->mapping->getAuthMode(),
320  (string) $person_id
321  );
322  if ($role && $il_usr_id = ilObjUser::_lookupId($login)) {
323  $part->add($il_usr_id, $role);
324  $part->sendNotification(ilCourseMembershipMailNotification::TYPE_ADMISSION_MEMBER, $il_usr_id);
325  }
326  }
327  } else {
328  if ($il_usr_id = ilObjUser::_lookupId($login)) {
329  // user exists => assign to course/group
330  if ($role) {
331  // Assign user
332  $this->log->info('Assigning new user ' . $person_id . ' ' . 'to ' . ilObject::_lookupTitle($obj_id) . ' using role: ' . $role);
333  $part->add($il_usr_id, $role);
334  $part->sendNotification(ilCourseMembershipMailNotification::TYPE_ADMISSION_MEMBER, $il_usr_id);
335  }
336  } else {
337  // no local user exists
338  if ($role_info['create']) {
339  $this->createMember($person_id);
340  $this->log->info('Added new user ' . $person_id);
342  $this->mapping->getAuthMode(),
343  (string) $person_id
344  );
345  }
346  // Assign to role
347  if ($role && $il_usr_id = ilObjUser::_lookupId($login)) {
348  $part->add($il_usr_id, $role);
349  $part->sendNotification(ilCourseMembershipMailNotification::TYPE_ADMISSION_MEMBER, $il_usr_id);
350  }
351  }
352 
353  $assignment = new ilECSCourseMemberAssignment();
354  $assignment->setServer($this->getServer()->getServerId());
355  $assignment->setMid($this->mid);
356  $assignment->setCmsId($course_id);
357  $assignment->setCmsSubId((int) $sub_id);
358  $assignment->setObjId($obj_id);
359  $assignment->setUid((string) $person_id);
360  $assignment->save();
361  }
362  }
363  return true;
364  }
365 
369  protected function lookupRole(string $role_value, $a_obj_type): int
370  {
371  $role_mappings = $this->mapping->getRoleMappings();
372 
373  /* Zero is an allowed value */
374  if (!$role_value) {
375  $this->log->debug('No role assignment attribute: role');
376  }
377  foreach ($role_mappings as $name => $map) {
378  $this->log->debug('Role "name" is ' . $name);
379 
380  // map is a string of ids seperated by ","
381  $exploded_map = (array) explode(',', $map);
382  if (in_array($role_value, $exploded_map, true)) {
383  switch ($name) {
387  if ($a_obj_type === 'crs') {
388  $this->log->debug('Role: ' . $role_value . ' maps: ' . $map);
389  return $name;
390  }
391  break;
392 
395  if ($a_obj_type === 'grp') {
396  $this->log->debug('Role: ' . $role_value . ' maps: ' . $map);
397  return $name;
398  }
399  break;
400  }
401  }
402  }
403  $this->log->info('No role assignment mapping for role ' . $role_value);
404  return 0;
405  }
406 
410  private function createMember($a_person_id): void
411  {
412  if (!$this->mapping instanceof ilECSNodeMappingSettings) {
413  $this->log->warning('Node mapping settings not initialized.');
414  }
415  $auth_mode = $this->mapping->getAuthMode();
416 
417  if (
418  $this->mapping->getAuthMode() ===
420  ) {
421  $this->log->info('Not handling direct user creation for auth mode: ' . $auth_mode);
422  return;
423  }
424  if (strpos($auth_mode, 'ldap') !== 0) {
425  $this->log->info('Not handling direct user creation for auth mode: ' . $auth_mode);
426  return;
427  }
428 
429  try {
431  $server->doConnectionCheck();
432 
433  $query = new ilLDAPQuery($server);
434  $query->bind(ilLDAPQuery::LDAP_BIND_DEFAULT);
435 
436  $users = $query->fetchUser((string) $a_person_id);
437  if ($users) {
439 
440  $xml = new ilLDAPAttributeToUser($server);
441  $xml->setNewUserAuthMode($server->getAuthenticationMappingKey());
442  $xml->setUserData($users);
443  $xml->refresh();
444  }
445  } catch (ilLDAPQueryException $exc) {
446  $this->log->error($exc->getMessage());
447  }
448  }
449 
450 
454  private function readCourseMember(ilECSSetting $server, $a_content_id)
455  {
456  return (new ilECSCourseMemberConnector($server))->getCourseMember($a_content_id);
457  }
458 
462  private function readCourse($course_member)
463  {
464  $ecs_id = ilECSImportManager::getInstance()->lookupEContentIdByContentId(
465  $this->getServer()->getServerId(),
466  $this->getMid(),
467  $course_member->lectureID
468  );
469  if (0 === $ecs_id) {
470  return null;
471  }
472  return (new ilECSCourseConnector($this->getServer()))->getCourse($ecs_id);
473  }
474 }
handleCreate(ilECSSetting $server, $a_content_id)
Handle create.
static _getFirstActiveServer()
Get first active server.
static getInstance(int $a_server_id, int $mid)
Get instance by server id and mid.
static getRoleMappingInfo($a_role_type_info=0)
Get role mapping info.
static getInstanceByServerId(int $a_server_id)
Get instance by server id.
static getInstanceByServerMid(int $a_server_id, int $a_mid)
Get instance.
if(!file_exists('../ilias.ini.php'))
static _lookupId($a_user_str)
lookupRole(string $role_value, $a_obj_type)
Lookup local role by assignment.
static _checkExternalAuthAccount(string $a_auth, string $a_account, bool $tryFallback=true)
check whether external account and authentication method matches with a user
server()
description: > This example shows how a Progress Bar can be rendered and updated by the server...
Definition: server.php:43
checkAllocationActivation(ilECSSetting $server, $a_content_id)
Check if course allocation is activated for one recipient of the.
static _getAuthModeName($a_auth_key)
Interface for all command queue handler classes.
handleDelete(ilECSSetting $server, $a_content_id)
Handle delete.
while($session_entry=$r->fetchRow(ilDBConstants::FETCHMODE_ASSOC)) return null
static lookupAssignment(int $a_cms_id, ?int $a_cms_sub_id, int $a_obj_id, string $a_usr_id)
Lookup assignment of user.
static getInstance()
Get the singleton instance of this ilECSImportManager.
static _lookupTitle(int $obj_id)
static _getInstanceByObjId(int $a_obj_id)
readAssignments($course, $course_member)
Read assignments for all parallel groups.
global $DIC
Definition: shib_login.php:22
static lookupUserIds(int $a_cms_id, ?int $a_cms_sub_id, int $a_obj_id)
Lookup user ids.
Connector for course member ressource.
Storage of ecs course assignments.
static _getInstanceByObjId(int $a_obj_id)
Get singleton instance.
readCourseMember(ilECSSetting $server, $a_content_id)
Read course from ecs.
handleUpdate(ilECSSetting $server, $a_content_id)
Handle update.
Update/create ILIAS user account by given LDAP attributes according to user attribute mapping setting...
refreshAssignmentStatus(object $course_member, int $obj_id, ?string $sub_id, $assigned)
Refresh status of course member assignments.
static _lookupType(int $id, bool $reference=false)