addError('setCredentials: Missing field ' . $field, true); return false; } } } if (empty($data['baseUrl'])) { $this->addError("No url is given", true); return false; } $this->username = $data['username']; if (!empty($data['role'])) { $this->username .= "\t" . $data['role']; } $this->password = $data['password']; $this->open = $data['open'] !== 'CourseService'; $url = preg_replace('#(/+qisserver(/+services\d+(/+OpenCourseService)?)?)?\W*$#i', '', $data['baseUrl']); if ($this->open) { $this->location = $url . "/qisserver/services2/OpenCourseService"; } else { $this->location = $url . "/qisserver/services2/CourseService"; } $this->verifyHostname = $data['verifyHostname']; $this->verifyCert = $data['verifyCert']; return true; } public function getCredentialDefinitions() { return [ new BackendProperty('baseUrl', 'string'), new BackendProperty('username', 'string'), new BackendProperty('role', 'string'), new BackendProperty('password', 'password'), new BackendProperty('open', ['OpenCourseService', 'CourseService'], 'OpenCourseService'), new BackendProperty('verifyCert', 'bool', true), new BackendProperty('verifyHostname', 'bool', true) ]; } public function checkConnection() { if (empty($this->location)) { $this->addError("Credentials are not set", true); return false; } return $this->findUnit(123456789, date('Y-m-d'), true) !== false; } /** * @param int $roomId his in one room id to get * @param bool $connectionCheckOnly true will only check if no soapError is returned, return value will be empty * @return array|bool if successful an array with the event ids that take place in the room */ public function findUnit($roomId, $day, $connectionCheckOnly = false) { $doc = new DOMDocument('1.0', 'utf-8'); $doc->formatOutput = true; $envelope = $doc->createElementNS('http://schemas.xmlsoap.org/soap/envelope/', 'SOAP-ENV:Envelope'); $doc->appendChild($envelope); if ($this->open) { $envelope->setAttributeNS('http://www.w3.org/2000/xmlns/', 'xmlns:ns1', 'http://www.his.de/ws/OpenCourseService'); } else { $envelope->setAttributeNS('http://www.w3.org/2000/xmlns/', 'xmlns:ns1', 'http://www.his.de/ws/CourseService'); } $header = $this->getHeader($doc); $envelope->appendChild($header); //Body of the request $body = $doc->createElement('SOAP-ENV:Body'); $envelope->appendChild($body); $findUnit = $doc->createElement('ns1:findUnit'); $body->appendChild($findUnit); $findUnit->appendChild($doc->createElement('ns1:individualDatesExecutionDate', $day)); $findUnit->appendChild($doc->createElement('ns1:roomId', $roomId)); $soap_request = $doc->saveXML(); $response1 = $this->postToServer($soap_request, "findUnit"); if ($response1 === false) { $this->addError('Could not fetch room ' . $roomId, true); return false; } $response2 = $this->xmlStringToArray($response1, $err); if (!is_array($response2)) { $this->addError("Parsing room $roomId: $err", false); return false; } if (!isset($response2['soapenvBody'])) { $this->addError('Backend reply is missing element soapenvBody', true); return false; } if (isset($response2['soapenvBody']['soapenvFault'])) { $this->addError('SOAP-Fault (' . $response2['soapenvBody']['soapenvFault']['faultcode'] . ") " . $response2['soapenvBody']['soapenvFault']['faultstring'], true); return false; } // We only need to check if the connection is working (URL ok, credentials ok, ..) so bail out early if ($connectionCheckOnly) { return array(); } if ($this->open) { $path = '/soapenvBody/hisfindUnitResponse/hisunits'; $subpath = '/hisunit/hisid'; } else { $path = '/soapenvBody/hisfindUnitResponse/hisunitIds'; $subpath = '/hisid'; } $idSubDoc = $this->getArrayPath($response2, $path); if ($idSubDoc === false) { $this->addError('Cannot find ' . $path, false); //@file_put_contents('/tmp/findUnit-1.' . $roomId . '.' . microtime(true), print_r($response2, true)); return false; } if (empty($idSubDoc)) return $idSubDoc; $idList = $this->getArrayPath($idSubDoc, $subpath); if ($idList === false) { $this->addError('Cannot find ' . $subpath . ' after ' . $path, false); @file_put_contents('/tmp/bwlp-findUnit-2.' . $roomId . '.' . microtime(true), print_r($idSubDoc, true)); } return $idList; } /** * @param $doc DOMDocument * @return DOMElement */ private function getHeader($doc) { $header = $doc->createElement('SOAP-ENV:Header'); $security = $doc->createElementNS('http://docs.oasis-open.org/wss/2004/01/oasis-200401-wss-wssecurity-secext-1.0.xsd', 'ns2:Security'); $mustunderstand = $doc->createAttribute('SOAP-ENV:mustUnderstand'); $mustunderstand->value = 1; $security->appendChild($mustunderstand); $header->appendChild($security); $token = $doc->createElement('ns2:UsernameToken'); $security->appendChild($token); $user = $doc->createElement('ns2:Username', $this->username); $token->appendChild($user); $pass = $doc->createElement('ns2:Password', $this->password); $type = $doc->createAttribute('Type'); $type->value = 'http://docs.oasis-open.org/wss/2004/01/oasis-200401-wss-username-token-profile-1.0#PasswordText'; $pass->appendChild($type); $token->appendChild($pass); return $header; } /** * @param $request string with xml SOAP request * @param $action string with the name of the SOAP action * @return bool|string if successful the answer xml from the SOAP server */ private function postToServer($request, $action) { $header = array( 'Content-type: text/xml;charset="utf-8"', 'SOAPAction: "' . $action . '"', ); if ($this->curlHandle === false) { $this->curlHandle = curl_init(); } $options = array( CURLOPT_RETURNTRANSFER => true, CURLOPT_FOLLOWLOCATION => true, CURLOPT_SSL_VERIFYHOST => $this->verifyHostname ? 2 : 0, CURLOPT_SSL_VERIFYPEER => $this->verifyCert ? 1 : 0, CURLOPT_URL => $this->location, CURLOPT_POSTFIELDS => $request, CURLOPT_HTTPHEADER => $header, CURLOPT_TIMEOUT => 15, CURLOPT_CONNECTTIMEOUT => 3, ); curl_setopt_array($this->curlHandle, $options); $output = curl_exec($this->curlHandle); if ($output === false) { $this->addError('Curl error: ' . curl_error($this->curlHandle), false); } return $output; } public function getCacheTime() { return 30 * 60; } public function getRefreshTime() { return 60 * 60; } public function getDisplayName() { return "HisInOne"; } public function fetchSchedulesInternal($requestedRoomIds) { if (empty($requestedRoomIds)) { return array(); } $currentWeek = $this->getCurrentWeekDates(); $tTables = []; //get all eventIDs in a given room $eventIds = []; foreach ($requestedRoomIds as $roomId) { $ok = false; foreach ($currentWeek as $day) { $roomEventIds = $this->findUnit($roomId, $day, false); if ($roomEventIds === false) continue; $ok = true; $eventIds = array_merge($eventIds, $roomEventIds); } if ($ok) { $tTables[$roomId] = []; } } $eventIds = array_unique($eventIds); if (empty($eventIds)) { return $tTables; } $eventDetails = []; //get all information on each event foreach ($eventIds as $eventId) { $event = $this->readUnit(intval($eventId)); if ($event === false) continue; $eventDetails = array_merge($eventDetails, $event); } $name = false; $now = time(); foreach ($eventDetails as $event) { foreach (array('/hisdefaulttext', '/hisshorttext', '/hisshortcomment') as $path) { $name = $this->getArrayPath($event, $path); if (!empty($name) && !empty($name[0])) break; $name = false; } if ($name === false) { $name = ['???']; } $planElements = $this->getArrayPath($event, '/hisplanelements/hisplanelement'); if ($planElements === false) { $this->addError('Cannot find ./hisplanelements/hisplanelement', false); //error_log('Cannot find ./hisplanelements/hisplanelement'); //error_log(print_r($event, true)); continue; } foreach ($planElements as $planElement) { if (empty($planElement['hisplannedDates'])) continue; // Do not use -- is set improperly for some courses :-( /* $checkDate = $this->getArrayPath($planElement, '/hisplannedDates/hisplannedDate/hisenddate'); if (!empty($checkDate) && strtotime($checkDate[0]) + 86400 < $now) continue; // Course ended $checkDate = $this->getArrayPath($planElement, '/hisplannedDates/hisplannedDate/hisstartdate'); if (!empty($checkDate) && strtotime($checkDate[0]) - 86400 > $now) continue; // Course didn't start yet */ $cancelled = $this->getArrayPath($planElement, '/hiscancelled'); $cancelled = $cancelled !== false && is_array($cancelled) && ($cancelled[0] > 0 || strtolower($cancelled[0]) === 'true'); $unitPlannedDates = $this->getArrayPath($planElement, '/hisplannedDates/hisplannedDate/hisindividualDates/hisindividualDate'); if ($unitPlannedDates === false) { $this->addError('Cannot find ./hisplannedDates/hisplannedDate/hisindividualDates/hisindividualDate', false); //error_log('Cannot find ./hisplannedDates/hisplannedDate/hisindividualDates/hisindividualDate'); //error_log(print_r($planElement, true)); continue; } $localName = $this->getArrayPath($planElement, '/hisdefaulttext'); if ($localName === false || empty($localName[0])) { $localName = $name; } foreach ($unitPlannedDates as $plannedDate) { $eventRoomId = $this->getArrayPath($plannedDate, '/hisroomId')[0]; $eventDate = $this->getArrayPath($plannedDate, '/hisexecutiondate')[0]; if (in_array($eventRoomId, $requestedRoomIds) && in_array($eventDate, $currentWeek)) { $startTime = $this->getArrayPath($plannedDate, '/hisstarttime')[0]; $endTime = $this->getArrayPath($plannedDate, '/hisendtime')[0]; $tTables[$eventRoomId][] = array( 'title' => $localName[0], 'start' => $eventDate . "T" . $startTime, 'end' => $eventDate . "T" . $endTime, 'cancelled' => $cancelled, ); } } } } return $tTables; } /** * @param $unit int ID of the subject in HisInOne database * @return bool|array false if there was an error otherwise an array with the information about the subject */ public function readUnit($unit) { $doc = new DOMDocument('1.0', 'utf-8'); $doc->formatOutput = true; $envelope = $doc->createElementNS('http://schemas.xmlsoap.org/soap/envelope/', 'SOAP-ENV:Envelope'); $doc->appendChild($envelope); if ($this->open) { $envelope->setAttributeNS('http://www.w3.org/2000/xmlns/', 'xmlns:ns1', 'http://www.his.de/ws/OpenCourseService'); } else { $envelope->setAttributeNS('http://www.w3.org/2000/xmlns/', 'xmlns:ns1', 'http://www.his.de/ws/CourseService'); } $header = $this->getHeader($doc); $envelope->appendChild($header); //body of the request $body = $doc->createElement('SOAP-ENV:Body'); $envelope->appendChild($body); $readUnit = $doc->createElement('ns1:readUnit'); $body->appendChild($readUnit); $readUnit->appendChild($doc->createElement('ns1:unitId', $unit)); $soap_request = $doc->saveXML(); $response1 = $this->postToServer($soap_request, "readUnit"); if ($response1 === false) { return false; } $response2 = $this->xmlStringToArray($response1, $err); if ($response2 === false) { $this->addError("Cannot parse unit $unit as XML: $err", false); return false; } if (!isset($response2['soapenvBody'])) { $this->addError('Backend reply is missing element soapenvBody', true); return false; } if (isset($response2['soapenvBody']['soapenvFault'])) { $this->addError('SOAP-Fault (' . $response2['soapenvBody']['soapenvFault']['faultcode'] . ") " . $response2['soapenvBody']['soapenvFault']['faultstring'], true); return false; } return $this->getArrayPath($response2, '/soapenvBody/hisreadUnitResponse/hisunit'); } /** * @return array with days of the current week in datetime format */ private function getCurrentWeekDates() { $returnValue = array(); $startDate = time(); for ($i = 0; $i < 7; $i++) { $returnValue[] = date('Y-m-d', strtotime("+{$i} day 12:00", $startDate)); } return $returnValue; } public function __destruct() { if ($this->curlHandle !== false) { curl_close($this->curlHandle); } } }