12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033 |
- <?php
- /* See license terms in /license.txt */
- //use Chamilo\UserBundle\Entity\User;
- use Chamilo\CoreBundle\Component\Utils\ChamiloApi;
- /**
- * Class Event
- * Functions of this library are used to record informations when some kind
- * of event occur. Each event has his own types of informations then each event
- * use its own function.
- */
- class Event
- {
- /**
- * @author Sebastien Piraux <piraux_seb@hotmail.com>
- * @desc Record information for open event (when homepage is opened)
- */
- public static function event_open()
- {
- global $_configuration;
- global $TABLETRACK_OPEN;
- // @getHostByAddr($_SERVER['REMOTE_ADDR']) : will provide host and country information
- // $_SERVER['HTTP_USER_AGENT'] : will provide browser and os information
- // $_SERVER['HTTP_REFERER'] : provide information about refering url
- if (isset($_SERVER['HTT_REFERER'])) {
- $referer = Database::escape_string($_SERVER['HTTP_REFERER']);
- } else {
- $referer = '';
- }
- // record informations only if user comes from another site
- //if(!eregi($_configuration['root_web'],$referer))
- $pos = strpos($referer, $_configuration['root_web']);
- if ($pos === false && $referer != '') {
- $ip = api_get_real_ip();
- $remhost = @ getHostByAddr($ip);
- if ($remhost == $ip) {
- $remhost = "Unknown";
- } // don't change this
- $reallyNow = api_get_utc_datetime();
- $params = [
- 'open_remote_host' => $remhost,
- 'open_agent' => $_SERVER['HTTP_USER_AGENT'],
- 'open_referer' => $referer,
- 'open_date' => $reallyNow,
- ];
- Database::insert($TABLETRACK_OPEN, $params);
- }
- return 1;
- }
- /**
- * @author Sebastien Piraux <piraux_seb@hotmail.com> old code
- * @author Julio Montoya 2013
- * @desc Record information for login event when an user identifies himself with username & password
- */
- public static function event_login($userId)
- {
- $userInfo = api_get_user_info($userId);
- $userId = intval($userId);
- if (empty($userInfo)) {
- return false;
- }
- $TABLETRACK_LOGIN = Database::get_main_table(TABLE_STATISTIC_TRACK_E_LOGIN);
- $reallyNow = api_get_utc_datetime();
- $sql = "INSERT INTO ".$TABLETRACK_LOGIN." (login_user_id, user_ip, login_date, logout_date) VALUES
- ('".$userId."',
- '".Database::escape_string(api_get_real_ip())."',
- '".$reallyNow."',
- '".$reallyNow."'
- )";
- Database::query($sql);
- // Auto subscribe
- $user_status = $userInfo['status'] == SESSIONADMIN ? 'sessionadmin' : $userInfo['status'] == COURSEMANAGER ? 'teacher' : $userInfo['status'] == DRH ? 'DRH' : 'student';
- $autoSubscribe = api_get_setting($user_status.'_autosubscribe');
- if ($autoSubscribe) {
- $autoSubscribe = explode('|', $autoSubscribe);
- foreach ($autoSubscribe as $code) {
- if (CourseManager::course_exists($code)) {
- CourseManager::subscribe_user($userId, $code);
- }
- }
- }
- }
- /**
- * @param tool name of the tool (name in mainDb.accueil table)
- * @author Sebastien Piraux <piraux_seb@hotmail.com>
- * @desc Record information for access event for courses
- */
- public static function accessCourse()
- {
- $TABLETRACK_ACCESS = Database::get_main_table(TABLE_STATISTIC_TRACK_E_ACCESS);
- //for "what's new" notification
- $TABLETRACK_LASTACCESS = Database::get_main_table(TABLE_STATISTIC_TRACK_E_LASTACCESS);
- $id_session = api_get_session_id();
- $now = api_get_utc_datetime();
- $courseId = api_get_course_int_id();
- $user_id = api_get_user_id();
- $ip = api_get_real_ip();
- if ($user_id) {
- $user_id = "'".$user_id."'";
- } else {
- $user_id = "0"; // no one
- }
- $sql = "INSERT INTO ".$TABLETRACK_ACCESS." (user_ip, access_user_id, c_id, access_date, access_session_id) VALUES
- ('".$ip."', ".$user_id.", '".$courseId."', '".$now."','".$id_session."')";
- Database::query($sql);
- // added for "what's new" notification
- $sql = "UPDATE $TABLETRACK_LASTACCESS SET access_date = '$now'
- WHERE access_user_id = $user_id AND c_id = '$courseId' AND access_tool IS NULL AND access_session_id=".$id_session;
- $result = Database::query($sql);
- if (Database::affected_rows($result) == 0) {
- $sql = "INSERT INTO $TABLETRACK_LASTACCESS (access_user_id, c_id, access_date, access_session_id)
- VALUES (".$user_id.", '".$courseId."', '$now', '".$id_session."')";
- Database::query($sql);
- }
- return 1;
- }
- /**
- * @param tool name of the tool (name in mainDb.accueil table)
- * @author Sebastien Piraux <piraux_seb@hotmail.com>
- * @desc Record information for access event for tools
- *
- * $tool can take this values :
- * Links, Calendar, Document, Announcements,
- * Group, Video, Works, Users, Exercices, Course Desc
- * ...
- * Values can be added if new modules are created (15char max)
- * I encourage to use $nameTool as $tool when calling this function
- *
- * Functionality for "what's new" notification is added by Toon Van Hoecke
- */
- public static function event_access_tool($tool, $id_session = 0)
- {
- if (empty($tool)) {
- return false;
- }
- $TABLETRACK_ACCESS = Database::get_main_table(TABLE_STATISTIC_TRACK_E_ACCESS);
- //for "what's new" notification
- $TABLETRACK_LASTACCESS = Database::get_main_table(TABLE_STATISTIC_TRACK_E_LASTACCESS);
- $_course = api_get_course_info();
- $courseId = api_get_course_int_id();
- $id_session = api_get_session_id();
- $tool = Database::escape_string($tool);
- $reallyNow = api_get_utc_datetime();
- $user_id = api_get_user_id();
- // record information
- // only if user comes from the course $_cid
- //if( eregi($_configuration['root_web'].$_cid,$_SERVER['HTTP_REFERER'] ) )
- //$pos = strpos($_SERVER['HTTP_REFERER'],$_configuration['root_web'].$_cid);
- $coursePath = isset($_course['path']) ? $_course['path'] : null;
- $pos = isset($_SERVER['HTTP_REFERER']) ? strpos(strtolower($_SERVER['HTTP_REFERER']), strtolower(api_get_path(WEB_COURSE_PATH).$coursePath)) : false;
- // added for "what's new" notification
- $pos2 = isset($_SERVER['HTTP_REFERER']) ? strpos(strtolower($_SERVER['HTTP_REFERER']), strtolower(api_get_path(WEB_PATH)."index")) : false;
- // end "what's new" notification
- if ($pos !== false || $pos2 !== false) {
- $params = [
- 'access_user_id' => $user_id,
- 'c_id' => $courseId,
- 'access_tool' => $tool,
- 'access_date' => $reallyNow,
- 'access_session_id' => $id_session,
- 'user_ip' => api_get_real_ip()
- ];
- Database::insert($TABLETRACK_ACCESS, $params);
- }
- // "what's new" notification
- $sql = "UPDATE $TABLETRACK_LASTACCESS
- SET access_date = '$reallyNow'
- WHERE access_user_id = ".$user_id." AND c_id = '".$courseId."' AND access_tool = '".$tool."' AND access_session_id=".$id_session;
- $result = Database::query($sql);
- if (Database::affected_rows($result) == 0) {
- $sql = "INSERT INTO $TABLETRACK_LASTACCESS (access_user_id, c_id, access_tool, access_date, access_session_id)
- VALUES (".$user_id.", '".$courseId."' , '$tool', '$reallyNow', $id_session)";
- Database::query($sql);
- }
- return 1;
- }
- /**
- * @param doc_id id of document (id in mainDb.document table)
- * @author Sebastien Piraux <piraux_seb@hotmail.com>
- * @desc Record information for download event
- * (when an user click to d/l a document)
- * it will be used in a redirection page
- * bug fixed: Roan Embrechts
- * Roan:
- * The user id is put in single quotes,
- * (why? perhaps to prevent sql insertion hacks?)
- * and later again.
- * Doing this twice causes an error, I remove one of them.
- */
- public static function event_download($doc_url)
- {
- $tbl_stats_downloads = Database::get_main_table(TABLE_STATISTIC_TRACK_E_DOWNLOADS);
- $doc_url = Database::escape_string($doc_url);
- $reallyNow = api_get_utc_datetime();
- $user_id = "'".api_get_user_id()."'";
- $_cid = api_get_course_int_id();
- $sql = "INSERT INTO $tbl_stats_downloads (
- down_user_id,
- c_id,
- down_doc_path,
- down_date,
- down_session_id
- )
- VALUES (
- ".$user_id.",
- '".$_cid."',
- '".$doc_url."',
- '".$reallyNow."',
- '".api_get_session_id()."'
- )";
- Database::query($sql);
- return 1;
- }
- /**
- * @param doc_id id of document (id in mainDb.document table)
- * @author Sebastien Piraux <piraux_seb@hotmail.com>
- * @desc Record information for upload event
- * used in the works tool to record informations when
- * an user upload 1 work
- */
- public static function event_upload($doc_id)
- {
- $TABLETRACK_UPLOADS = Database::get_main_table(TABLE_STATISTIC_TRACK_E_UPLOADS);
- $courseId = api_get_course_int_id();
- $reallyNow = api_get_utc_datetime();
- $user_id = api_get_user_id();
- $doc_id = intval($doc_id);
- $sql = "INSERT INTO ".$TABLETRACK_UPLOADS."
- ( upload_user_id,
- c_id,
- upload_cours_id,
- upload_work_id,
- upload_date,
- upload_session_id
- )
- VALUES (
- ".$user_id.",
- '".$courseId."',
- '',
- '".$doc_id."',
- '".$reallyNow."',
- '".api_get_session_id()."'
- )";
- Database::query($sql);
- return 1;
- }
- /**
- * @param link_id (id in coursDb liens table)
- * @author Sebastien Piraux <piraux_seb@hotmail.com>
- * @desc Record information for link event (when an user click on an added link)
- * it will be used in a redirection page
- */
- public static function event_link($link_id)
- {
- $TABLETRACK_LINKS = Database::get_main_table(TABLE_STATISTIC_TRACK_E_LINKS);
- $reallyNow = api_get_utc_datetime();
- $user_id = api_get_user_id();
- $sql = "INSERT INTO ".$TABLETRACK_LINKS."
- ( links_user_id,
- c_id,
- links_link_id,
- links_date,
- links_session_id
- ) VALUES (
- ".$user_id.",
- '".api_get_course_int_id()."',
- '".Database::escape_string($link_id)."',
- '".$reallyNow."',
- '".api_get_session_id()."'
- )";
- Database::query($sql);
- return 1;
- }
- /**
- * Update the TRACK_E_EXERCICES exercises
- *
- * @param int exeid id of the attempt
- * @param int exo_id exercise id
- * @param mixed result score
- * @param int weighting ( higher score )
- * @param int duration ( duration of the attempt in seconds )
- * @param int session_id
- * @param int learnpath_id (id of the learnpath)
- * @param int learnpath_item_id (id of the learnpath_item)
- *
- * @author Sebastien Piraux <piraux_seb@hotmail.com>
- * @author Julio Montoya Armas <gugli100@gmail.com> Reworked 2010
- * @desc Record result of user when an exercise was done
- */
- public static function update_event_exercise(
- $exeid,
- $exo_id,
- $score,
- $weighting,
- $session_id,
- $learnpath_id = 0,
- $learnpath_item_id = 0,
- $learnpath_item_view_id = 0,
- $duration = 0,
- $question_list = array(),
- $status = '',
- $remind_list = array(),
- $end_date = null
- ) {
- global $debug;
- if ($debug) error_log('Called to update_event_exercice');
- if ($debug) error_log('duration:'.$duration);
- if ($exeid != '') {
- /*
- * Code commented due BT#8423 do not change the score to 0.
- *
- * Validation in case of fraud with actived control time
- if (!ExerciseLib::exercise_time_control_is_valid($exo_id, $learnpath_id, $learnpath_item_id)) {
- $score = 0;
- }
- */
- if (!isset($status) || empty($status)) {
- $status = '';
- } else {
- $status = Database::escape_string($status);
- }
- $TABLETRACK_EXERCICES = Database::get_main_table(TABLE_STATISTIC_TRACK_E_EXERCISES);
- if (!empty($question_list)) {
- $question_list = array_map('intval', $question_list);
- }
- if (!empty($remind_list)) {
- $remind_list = array_map('intval', $remind_list);
- $remind_list = array_filter($remind_list);
- $remind_list = implode(",", $remind_list);
- } else {
- $remind_list = '';
- }
- if (empty($end_date)) {
- $end_date = api_get_utc_datetime();
- }
- $sql = "UPDATE $TABLETRACK_EXERCICES SET
- exe_exo_id = '".Database::escape_string($exo_id)."',
- exe_result = '".Database::escape_string($score)."',
- exe_weighting = '".Database::escape_string($weighting)."',
- session_id = '".Database::escape_string($session_id)."',
- orig_lp_id = '".Database::escape_string($learnpath_id)."',
- orig_lp_item_id = '".Database::escape_string($learnpath_item_id)."',
- orig_lp_item_view_id = '".Database::escape_string($learnpath_item_view_id)."',
- exe_duration = '".Database::escape_string($duration)."',
- exe_date = '".$end_date."',
- status = '".$status."',
- questions_to_check = '".$remind_list."',
- data_tracking = '".implode(',', $question_list)."',
- user_ip = '" . Database::escape_string(api_get_real_ip())."'
- WHERE exe_id = '".Database::escape_string($exeid)."'";
- $res = Database::query($sql);
- if ($debug) error_log('update_event_exercice called');
- if ($debug) error_log("$sql");
- //Deleting control time session track
- //ExerciseLib::exercise_time_control_delete($exo_id);
- return $res;
- } else {
- return false;
- }
- }
- /**
- * Record an event for this attempt at answering an exercise
- * @param float Score achieved
- * @param string Answer given
- * @param integer Question ID
- * @param integer Exercise attempt ID a.k.a exe_id (from track_e_exercise)
- * @param integer Position
- * @param integer Exercise ID (from c_quiz)
- * @param bool update results?
- * @param $fileName string Filename (for audio answers - using nanogong)
- * @param integer User ID The user who's going to get this score. Default value of null means "get from context".
- * @param integer Course ID (from the "id" column of course table). Default value of null means "get from context".
- * @param integer Session ID (from the session table). Default value of null means "get from context".
- * @param integer Learnpath ID (from c_lp table). Default value of null means "get from context".
- * @param integer Learnpath item ID (from the c_lp_item table). Default value of null means "get from context".
- * @return boolean Result of the insert query
- */
- public static function saveQuestionAttempt(
- $score,
- $answer,
- $question_id,
- $exe_id,
- $position,
- $exercise_id = 0,
- $updateResults = false,
- $fileName = null,
- $user_id = null,
- $course_id = null,
- $session_id = null,
- $learnpath_id = null,
- $learnpath_item_id = null
- ) {
- global $debug;
- $question_id = Database::escape_string($question_id);
- $exe_id = Database::escape_string($exe_id);
- $position = Database::escape_string($position);
- $now = api_get_utc_datetime();
- // check user_id or get from context
- if (empty($user_id)) {
- $user_id = api_get_user_id();
- // anonymous
- if (empty($user_id)) {
- $user_id = api_get_anonymous_id();
- }
- }
- // check course_id or get from context
- if (empty($course_id) or intval($course_id) != $course_id) {
- $course_id = api_get_course_int_id();
- }
- // check session_id or get from context
- if (empty($session_id)) {
- $session_id = api_get_session_id();
- }
- // check learnpath_id or get from context
- if (empty($learnpath_id)) {
- global $learnpath_id;
- }
- // check learnpath_item_id or get from context
- if (empty($learnpath_item_id)) {
- global $learnpath_item_id;
- }
- $TBL_TRACK_ATTEMPT = Database::get_main_table(TABLE_STATISTIC_TRACK_E_ATTEMPT);
- if ($debug) {
- error_log("----- entering saveQuestionAttempt() function ------");
- error_log("answer: $answer");
- error_log("score: $score");
- error_log("question_id : $question_id");
- error_log("position: $position");
- }
- //Validation in case of fraud with active control time
- if (!ExerciseLib::exercise_time_control_is_valid($exercise_id, $learnpath_id, $learnpath_item_id)) {
- if ($debug) {
- error_log("exercise_time_control_is_valid is false");
- }
- $score = 0;
- $answer = 0;
- }
- $session_id = api_get_session_id();
- if (!empty($question_id) && !empty($exe_id) && !empty($user_id)) {
- if (is_null($answer)) {
- $answer = '';
- }
- $attempt = array(
- 'user_id' => $user_id,
- 'question_id' => $question_id,
- 'answer' => $answer,
- 'marks' => $score,
- 'c_id' => $course_id,
- 'session_id' => $session_id,
- 'position' => $position,
- 'tms' => $now,
- 'filename' => !empty($fileName) ? basename($fileName) : $fileName,
- 'teacher_comment' => ''
- );
- // Check if attempt exists.
- $sql = "SELECT exe_id FROM $TBL_TRACK_ATTEMPT
- WHERE
- c_id = $course_id AND
- session_id = $session_id AND
- exe_id = $exe_id AND
- user_id = $user_id AND
- question_id = $question_id AND
- position = $position";
- $result = Database::query($sql);
- if (Database::num_rows($result)) {
- if ($debug) {
- error_log("Attempt already exist: exe_id: $exe_id - user_id:$user_id - question_id:$question_id");
- }
- if ($updateResults == false) {
- //The attempt already exist do not update use update_event_exercise() instead
- return false;
- }
- } else {
- $attempt['exe_id'] = $exe_id;
- }
- if ($debug) {
- error_log("updateResults : $updateResults");
- error_log("Saving question attempt: ");
- error_log($sql);
- }
- $recording_table = Database::get_main_table(TABLE_STATISTIC_TRACK_E_ATTEMPT_RECORDING);
- if ($updateResults == false) {
- $attempt_id = Database::insert($TBL_TRACK_ATTEMPT, $attempt);
- if ($debug) {
- error_log("Insert attempt with id #$attempt_id");
- }
- if (defined('ENABLED_LIVE_EXERCISE_TRACKING')) {
- if ($debug) {
- error_log("Saving e attempt recording ");
- }
- $attempt_recording = array(
- 'exe_id' => $attempt_id,
- 'question_id' => $question_id,
- 'marks' => $score,
- 'insert_date' => $now,
- 'author' => '',
- 'session_id' => $session_id,
- );
- Database::insert($recording_table, $attempt_recording);
- }
- } else {
- Database::update(
- $TBL_TRACK_ATTEMPT,
- $attempt,
- array('exe_id = ? AND question_id = ? AND user_id = ? ' => array($exe_id, $question_id, $user_id))
- );
- if (defined('ENABLED_LIVE_EXERCISE_TRACKING')) {
- $attempt_recording = array(
- 'exe_id' => $exe_id,
- 'question_id' => $question_id,
- 'marks' => $score,
- 'insert_date' => $now,
- 'author' => '',
- 'session_id' => $session_id,
- );
- Database::update(
- $recording_table,
- $attempt_recording,
- array('exe_id = ? AND question_id = ? AND session_id = ? ' => array($exe_id, $question_id, $session_id))
- );
- }
- $attempt_id = $exe_id;
- }
- return $attempt_id;
- } else {
- return false;
- }
- }
- /**
- * Record an hotspot spot for this attempt at answering an hotspot question
- * @param int $exeId
- * @param int $questionId Question ID
- * @param int $answerId Answer ID
- * @param int $correct
- * @param string $coords Coordinates of this point (e.g. 123;324)
- * @param bool $updateResults
- * @param int $exerciseId
- *
- * @return bool Result of the insert query
- * @uses Course code and user_id from global scope $_cid and $_user
- */
- public static function saveExerciseAttemptHotspot(
- $exeId,
- $questionId,
- $answerId,
- $correct,
- $coords,
- $updateResults = false,
- $exerciseId = 0
- ) {
- global $safe_lp_id, $safe_lp_item_id;
- if ($updateResults == false) {
- // Validation in case of fraud with activated control time
- if (!ExerciseLib::exercise_time_control_is_valid($exerciseId, $safe_lp_id, $safe_lp_item_id)) {
- $correct = 0;
- }
- }
- if (empty($exeId)) {
- return false;
- }
- $table = Database::get_main_table(TABLE_STATISTIC_TRACK_E_HOTSPOT);
- if ($updateResults) {
- $params = array(
- 'hotspot_correct' => $correct,
- 'hotspot_coordinate' => $coords
- );
- Database::update(
- $table,
- $params,
- array(
- 'hotspot_user_id = ? AND hotspot_exe_id = ? AND hotspot_question_id = ? AND hotspot_answer_id = ? ' => array(
- api_get_user_id(),
- $exeId,
- $questionId,
- $answerId
- )
- )
- );
- } else {
- return Database::insert(
- $table,
- [
- 'hotspot_course_code' => api_get_course_id(),
- 'hotspot_user_id' => api_get_user_id(),
- 'c_id' => api_get_course_int_id(),
- 'hotspot_exe_id' => $exeId,
- 'hotspot_question_id' => $questionId,
- 'hotspot_answer_id' => $answerId,
- 'hotspot_correct' => $correct,
- 'hotspot_coordinate' => $coords
- ]
- );
- }
- }
- /**
- * Records information for common (or admin) events (in the track_e_default table)
- * @author Yannick Warnier <yannick.warnier@beeznest.com>
- * @param string $event_type Type of event
- * @param string $event_value_type Type of value
- * @param string $event_value Value
- * @param string $datetime Datetime (UTC) (defaults to null)
- * @param int $user_id User ID (defaults to null)
- * @param int $course_id Course ID (defaults to null)
- * @param int $sessionId Session ID
- * @return bool
- * @assert ('','','') === false
- */
- public static function addEvent(
- $event_type,
- $event_value_type,
- $event_value,
- $datetime = null,
- $user_id = null,
- $course_id = null,
- $sessionId = 0
- ) {
- $table = Database::get_main_table(TABLE_STATISTIC_TRACK_E_DEFAULT);
- if (empty($event_type)) {
- return false;
- }
- $event_type = Database::escape_string($event_type);
- $event_value_type = Database::escape_string($event_value_type);
- if (!empty($course_id)) {
- $course_id = intval($course_id);
- } else {
- $course_id = api_get_course_int_id();
- }
- if (!empty($sessionId)) {
- $sessionId = intval($sessionId);
- } else {
- $sessionId = api_get_session_id();
- }
- //Clean the user_info
- if ($event_value_type == LOG_USER_OBJECT) {
- if (is_array($event_value)) {
- unset($event_value['complete_name']);
- unset($event_value['complete_name_with_username']);
- unset($event_value['firstName']);
- unset($event_value['lastName']);
- unset($event_value['avatar_small']);
- unset($event_value['avatar']);
- unset($event_value['mail']);
- unset($event_value['password']);
- unset($event_value['last_login']);
- unset($event_value['picture_uri']);
- $event_value = serialize($event_value);
- }
- }
- // If event is an array then the $event_value_type should finish with
- // the suffix _array for example LOG_WORK_DATA = work_data_array
- if (is_array($event_value)) {
- $event_value = serialize($event_value);
- }
- $event_value = Database::escape_string($event_value);
- $sessionId = empty($sessionId) ? api_get_session_id() : intval($sessionId);
- if (!isset($datetime)) {
- $datetime = api_get_utc_datetime();
- }
- $datetime = Database::escape_string($datetime);
- if (!isset($user_id)) {
- $user_id = api_get_user_id();
- }
- $params = array(
- 'default_user_id' => $user_id,
- 'c_id' => $course_id,
- 'default_date' => $datetime,
- 'default_event_type' => $event_type,
- 'default_value_type' => $event_value_type,
- 'default_value' => $event_value,
- 'session_id' => $sessionId
- );
- Database::insert($table, $params);
- return true;
- }
- /**
- * Get every email stored in the database
- *
- * @return array
- * @assert () !== false
- */
- public static function get_all_event_types()
- {
- global $event_config;
- $sql = 'SELECT etm.id, event_type_name, activated, language_id, message, subject, dokeos_folder
- FROM '.Database::get_main_table(TABLE_EVENT_EMAIL_TEMPLATE).' etm
- INNER JOIN '.Database::get_main_table(TABLE_MAIN_LANGUAGE).' l
- ON etm.language_id = l.id';
- $events_types = Database::store_result(Database::query($sql), 'ASSOC');
- $to_return = array();
- foreach ($events_types as $et) {
- $et['nameLangVar'] = $event_config[$et["event_type_name"]]["name_lang_var"];
- $et['descLangVar'] = $event_config[$et["event_type_name"]]["desc_lang_var"];
- $to_return[] = $et;
- }
- return $to_return;
- }
- /**
- * Get the users related to one event
- *
- * @param string $event_name
- */
- public static function get_event_users($event_name)
- {
- $event_name = Database::escape_string($event_name);
- $sql = 'SELECT user.user_id, user.firstname, user.lastname
- FROM '.Database::get_main_table(TABLE_MAIN_USER).' user
- JOIN '.Database::get_main_table(TABLE_EVENT_TYPE_REL_USER).' relUser
- ON relUser.user_id = user.user_id
- WHERE user.status <> '.ANONYMOUS.' AND relUser.event_type_name = "'.$event_name.'"';
- $user_list = Database::store_result(Database::query($sql), 'ASSOC');
- return json_encode($user_list);
- }
- /**
- * @param int $user_id
- * @param string $event_type
- * @return array|bool
- */
- public static function get_events_by_user_and_type($user_id, $event_type)
- {
- $TABLETRACK_DEFAULT = Database::get_main_table(TABLE_STATISTIC_TRACK_E_DEFAULT);
- $user_id = intval($user_id);
- $event_type = Database::escape_string($event_type);
- $sql = "SELECT * FROM $TABLETRACK_DEFAULT
- WHERE default_value_type = 'user_id' AND
- default_value = $user_id AND
- default_event_type = '$event_type'
- ORDER BY default_date ";
- $result = Database::query($sql);
- if ($result) {
- return Database::store_result($result, 'ASSOC');
- }
- return false;
- }
- /**
- * Save the new message for one event and for one language
- *
- * @param string $event_name
- * @param array $users
- * @param string $message
- * @param string $subject
- * @param string $event_message_language
- * @param int $activated
- */
- public static function save_event_type_message($event_name, $users, $message, $subject, $event_message_language, $activated)
- {
- $event_name = Database::escape_string($event_name);
- $activated = intval($activated);
- $event_message_language = Database::escape_string($event_message_language);
- // Deletes then re-adds the users linked to the event
- $sql = 'DELETE FROM '.Database::get_main_table(TABLE_EVENT_TYPE_REL_USER).' WHERE event_type_name = "'.$event_name.'" ';
- Database::query($sql);
- foreach ($users as $user) {
- $sql = 'INSERT INTO '.Database::get_main_table(TABLE_EVENT_TYPE_REL_USER).' (user_id,event_type_name)
- VALUES('.intval($user).',"'.$event_name.'")';
- Database::query($sql);
- }
- $language_id = api_get_language_id($event_message_language);
- // check if this template in this language already exists or not
- $sql = 'SELECT COUNT(id) as total FROM '.Database::get_main_table(TABLE_EVENT_EMAIL_TEMPLATE).'
- WHERE event_type_name = "'.$event_name.'" AND language_id = '.$language_id;
- $sql = Database::store_result(Database::query($sql), 'ASSOC');
- // if already exists, we update
- if ($sql[0]["total"] > 0) {
- $sql = 'UPDATE '.Database::get_main_table(TABLE_EVENT_EMAIL_TEMPLATE).'
- SET message = "'.Database::escape_string($message).'",
- subject = "'.Database::escape_string($subject).'",
- activated = '.$activated.'
- WHERE event_type_name = "'.$event_name.'" AND language_id = (SELECT id FROM '.Database::get_main_table(TABLE_MAIN_LANGUAGE).'
- WHERE dokeos_folder = "'.$event_message_language.'")';
- Database::query($sql);
- } else { // else we create a new record
- // gets the language_-_id
- $lang_id = '(SELECT id FROM '.Database::get_main_table(TABLE_MAIN_LANGUAGE).'
- WHERE dokeos_folder = "'.$event_message_language.'")';
- $lang_id = Database::store_result(Database::query($lang_id), 'ASSOC');
- if (!empty($lang_id[0]["id"])) {
- $sql = 'INSERT INTO '.Database::get_main_table(TABLE_EVENT_EMAIL_TEMPLATE).' (event_type_name, language_id, message, subject, activated)
- VALUES("'.$event_name.'", '.$lang_id[0]["id"].', "'.Database::escape_string($message).'", "'.Database::escape_string($subject).'", '.$activated.')';
- Database::query($sql);
- }
- }
- // set activated at every save
- $sql = 'UPDATE '.Database::get_main_table(TABLE_EVENT_EMAIL_TEMPLATE).'
- SET activated = '.$activated.'
- WHERE event_type_name = "'.$event_name.'"';
- Database::query($sql);
- }
- /**
- * Gets the last attempt of an exercise based in the exe_id
- * @param int $exe_id
- * @return mixed
- */
- public static function getLastAttemptDateOfExercise($exe_id)
- {
- $exe_id = intval($exe_id);
- $track_attempts = Database::get_main_table(TABLE_STATISTIC_TRACK_E_ATTEMPT);
- $sql = 'SELECT max(tms) as last_attempt_date
- FROM '.$track_attempts.'
- WHERE exe_id='.$exe_id;
- $rs_last_attempt = Database::query($sql);
- $row_last_attempt = Database::fetch_array($rs_last_attempt);
- $last_attempt_date = $row_last_attempt['last_attempt_date']; //Get the date of last attempt
- return $last_attempt_date;
- }
- /**
- * Gets the last attempt of an exercise based in the exe_id
- * @param int $exe_id
- * @return mixed
- */
- public static function getLatestQuestionIdFromAttempt($exe_id)
- {
- $exe_id = intval($exe_id);
- $track_attempts = Database::get_main_table(TABLE_STATISTIC_TRACK_E_ATTEMPT);
- $sql = 'SELECT question_id FROM '.$track_attempts.'
- WHERE exe_id='.$exe_id.'
- ORDER BY tms DESC
- LIMIT 1';
- $result = Database::query($sql);
- if (Database::num_rows($result)) {
- $row = Database::fetch_array($result);
- return $row['question_id'];
- } else {
- return false;
- }
- }
- /**
- * Gets how many attempts exists by user, exercise, learning path
- * @param int user id
- * @param int exercise id
- * @param int lp id
- * @param int lp item id
- * @param int lp item view id
- */
- public static function get_attempt_count($user_id, $exerciseId, $lp_id, $lp_item_id, $lp_item_view_id)
- {
- $stat_table = Database::get_main_table(TABLE_STATISTIC_TRACK_E_EXERCISES);
- $user_id = intval($user_id);
- $exerciseId = intval($exerciseId);
- $lp_id = intval($lp_id);
- $lp_item_id = intval($lp_item_id);
- $lp_item_view_id = intval($lp_item_view_id);
- $sql = "SELECT count(*) as count
- FROM $stat_table
- WHERE
- exe_exo_id = $exerciseId AND
- exe_user_id = $user_id AND
- status != 'incomplete' AND
- orig_lp_id = $lp_id AND
- orig_lp_item_id = $lp_item_id AND
- orig_lp_item_view_id = $lp_item_view_id AND
- c_id = '".api_get_course_int_id()."' AND
- session_id = '".api_get_session_id()."'";
- $query = Database::query($sql);
- if (Database::num_rows($query) > 0) {
- $attempt = Database::fetch_array($query, 'ASSOC');
- return $attempt['count'];
- } else {
- return 0;
- }
- }
- /**
- * @param $user_id
- * @param $exerciseId
- * @param $lp_id
- * @param $lp_item_id
- * @return int
- */
- public static function get_attempt_count_not_finished($user_id, $exerciseId, $lp_id, $lp_item_id)
- {
- $stat_table = Database::get_main_table(TABLE_STATISTIC_TRACK_E_EXERCISES);
- $user_id = intval($user_id);
- $exerciseId = intval($exerciseId);
- $lp_id = intval($lp_id);
- $lp_item_id = intval($lp_item_id);
- //$lp_item_view_id = intval($lp_item_view_id);
- $sql = "SELECT count(*) as count
- FROM $stat_table
- WHERE
- exe_exo_id = $exerciseId AND
- exe_user_id = $user_id AND
- status != 'incomplete' AND
- orig_lp_id = $lp_id AND
- orig_lp_item_id = $lp_item_id AND
- c_id = '".api_get_course_int_id()."' AND
- session_id = '".api_get_session_id()."'";
- $query = Database::query($sql);
- if (Database::num_rows($query) > 0) {
- $attempt = Database::fetch_array($query, 'ASSOC');
- return $attempt['count'];
- } else {
- return 0;
- }
- }
- /**
- * @param int $user_id
- * @param int $lp_id
- * @param array $course
- * @param int $session_id
- */
- public static function delete_student_lp_events($user_id, $lp_id, $course, $session_id)
- {
- $lp_view_table = Database::get_course_table(TABLE_LP_VIEW);
- $lp_item_view_table = Database::get_course_table(TABLE_LP_ITEM_VIEW);
- $lpInteraction = Database::get_course_table(TABLE_LP_IV_INTERACTION);
- $lpObjective = Database::get_course_table(TABLE_LP_IV_OBJECTIVE);
- $course_id = $course['real_id'];
- if (empty($course_id)) {
- $course_id = api_get_course_int_id();
- }
- $track_e_exercises = Database::get_main_table(
- TABLE_STATISTIC_TRACK_E_EXERCISES
- );
- $track_attempts = Database::get_main_table(
- TABLE_STATISTIC_TRACK_E_ATTEMPT
- );
- $recording_table = Database::get_main_table(
- TABLE_STATISTIC_TRACK_E_ATTEMPT_RECORDING
- );
- $user_id = intval($user_id);
- $lp_id = intval($lp_id);
- $session_id = intval($session_id);
- //Make sure we have the exact lp_view_id
- $sql = "SELECT id FROM $lp_view_table
- WHERE
- c_id = $course_id AND
- user_id = $user_id AND
- lp_id = $lp_id AND
- session_id = $session_id ";
- $result = Database::query($sql);
- if (Database::num_rows($result)) {
- $view = Database::fetch_array($result, 'ASSOC');
- $lp_view_id = $view['id'];
- $sql = "DELETE FROM $lp_item_view_table
- WHERE c_id = $course_id AND lp_view_id = $lp_view_id ";
- Database::query($sql);
- $sql = "DELETE FROM $lpInteraction
- WHERE c_id = $course_id AND lp_iv_id = $lp_view_id ";
- Database::query($sql);
- $sql = "DELETE FROM $lpObjective
- WHERE c_id = $course_id AND lp_iv_id = $lp_view_id ";
- Database::query($sql);
- }
- $sql = "DELETE FROM $lp_view_table
- WHERE
- c_id = $course_id AND
- user_id = $user_id AND
- lp_id= $lp_id AND
- session_id = $session_id
- ";
- Database::query($sql);
- $sql = "SELECT exe_id FROM $track_e_exercises
- WHERE exe_user_id = $user_id AND
- session_id = $session_id AND
- c_id = $course_id AND
- orig_lp_id = $lp_id";
- $result = Database::query($sql);
- $exe_list = array();
- while ($row = Database::fetch_array($result, 'ASSOC')) {
- $exe_list[] = $row['exe_id'];
- }
- if (!empty($exe_list) && is_array($exe_list) && count($exe_list) > 0) {
- $sql = "DELETE FROM $track_e_exercises
- WHERE exe_id IN (".implode(',', $exe_list).")";
- Database::query($sql);
- $sql = "DELETE FROM $track_attempts
- WHERE exe_id IN (".implode(',', $exe_list).")";
- Database::query($sql);
- $sql = "DELETE FROM $recording_table
- WHERE exe_id IN (".implode(',', $exe_list).")";
- Database::query($sql);
- }
- self::addEvent(
- LOG_LP_ATTEMPT_DELETE,
- LOG_LP_ID,
- $lp_id,
- null,
- null,
- $course_id,
- $session_id
- );
- }
- /**
- * Delete all exercise attempts (included in LP or not)
- *
- * @param int user id
- * @param int exercise id
- * @param int $course_id
- * @param int session id
- */
- public static function delete_all_incomplete_attempts($user_id, $exercise_id, $course_id, $session_id = 0)
- {
- $track_e_exercises = Database::get_main_table(TABLE_STATISTIC_TRACK_E_EXERCISES);
- $user_id = intval($user_id);
- $exercise_id = intval($exercise_id);
- $course_id = intval($course_id);
- $session_id = intval($session_id);
- if (!empty($user_id) && !empty($exercise_id) && !empty($course_code)) {
- $sql = "DELETE FROM $track_e_exercises
- WHERE
- exe_user_id = $user_id AND
- exe_exo_id = $exercise_id AND
- c_id = '$course_id' AND
- session_id = $session_id AND
- status = 'incomplete' ";
- Database::query($sql);
- self::addEvent(
- LOG_EXERCISE_RESULT_DELETE,
- LOG_EXERCISE_AND_USER_ID,
- $exercise_id.'-'.$user_id,
- null,
- null,
- $course_id,
- $session_id
- );
- }
- }
- /**
- * Gets all exercise results (NO Exercises in LPs ) from a given exercise id, course, session
- * @param int exercise id
- * @param int $courseId
- * @param int session id
- * @return array with the results
- *
- */
- public static function get_all_exercise_results(
- $exercise_id,
- $courseId,
- $session_id = 0,
- $load_question_list = true,
- $user_id = null
- ) {
- $TABLETRACK_EXERCICES = Database::get_main_table(TABLE_STATISTIC_TRACK_E_EXERCISES);
- $TBL_TRACK_ATTEMPT = Database::get_main_table(TABLE_STATISTIC_TRACK_E_ATTEMPT);
- $courseId = intval($courseId);
- $exercise_id = intval($exercise_id);
- $session_id = intval($session_id);
- $user_condition = null;
- if (!empty($user_id)) {
- $user_id = intval($user_id);
- $user_condition = "AND exe_user_id = $user_id ";
- }
- $sql = "SELECT * FROM $TABLETRACK_EXERCICES
- WHERE
- status = '' AND
- c_id = '$courseId' AND
- exe_exo_id = '$exercise_id' AND
- session_id = $session_id AND
- orig_lp_id =0 AND
- orig_lp_item_id = 0
- $user_condition
- ORDER BY exe_id";
- $res = Database::query($sql);
- $list = array();
- while ($row = Database::fetch_array($res, 'ASSOC')) {
- $list[$row['exe_id']] = $row;
- if ($load_question_list) {
- $sql = "SELECT * FROM $TBL_TRACK_ATTEMPT
- WHERE exe_id = {$row['exe_id']}";
- $res_question = Database::query($sql);
- while ($row_q = Database::fetch_array($res_question, 'ASSOC')) {
- $list[$row['exe_id']]['question_list'][$row_q['question_id']] = $row_q;
- }
- }
- }
- return $list;
- }
- /**
- * Gets all exercise results (NO Exercises in LPs ) from a given exercise id, course, session
- * @param int $courseId
- * @param int session id
- * @return array with the results
- *
- */
- public static function get_all_exercise_results_by_course($courseId, $session_id = 0, $get_count = true)
- {
- $table_track_exercises = Database::get_main_table(TABLE_STATISTIC_TRACK_E_EXERCISES);
- $table_track_attempt = Database::get_main_table(TABLE_STATISTIC_TRACK_E_ATTEMPT);
- $courseId = intval($courseId);
- $session_id = intval($session_id);
- $select = '*';
- if ($get_count) {
- $select = 'count(*) as count';
- }
- $sql = "SELECT $select FROM $table_track_exercises
- WHERE status = '' AND
- c_id = '$courseId' AND
- session_id = $session_id AND
- orig_lp_id = 0 AND
- orig_lp_item_id = 0
- ORDER BY exe_id";
- $res = Database::query($sql);
- if ($get_count) {
- $row = Database::fetch_array($res, 'ASSOC');
- return $row['count'];
- } else {
- $list = array();
- while ($row = Database::fetch_array($res, 'ASSOC')) {
- $list[$row['exe_id']] = $row;
- }
- return $list;
- }
- }
- /**
- * Gets all exercise results (NO Exercises in LPs) from a given exercise id, course, session
- * @param int exercise id
- * @param int $courseId
- * @param int session id
- * @return array with the results
- *
- */
- public static function get_all_exercise_results_by_user($user_id, $courseId, $session_id = 0)
- {
- $table_track_exercises = Database::get_main_table(TABLE_STATISTIC_TRACK_E_EXERCISES);
- $table_track_attempt = Database::get_main_table(TABLE_STATISTIC_TRACK_E_ATTEMPT);
- $courseId = intval($courseId);
- $session_id = intval($session_id);
- $user_id = intval($user_id);
- $sql = "SELECT * FROM $table_track_exercises
- WHERE
- status = '' AND
- exe_user_id = $user_id AND
- c_id = '$courseId' AND
- session_id = $session_id AND
- orig_lp_id = 0 AND
- orig_lp_item_id = 0
- ORDER by exe_id";
- $res = Database::query($sql);
- $list = array();
- while ($row = Database::fetch_array($res, 'ASSOC')) {
- $list[$row['exe_id']] = $row;
- $sql = "SELECT * FROM $table_track_attempt
- WHERE exe_id = {$row['exe_id']}";
- $res_question = Database::query($sql);
- while ($row_q = Database::fetch_array($res_question, 'ASSOC')) {
- $list[$row['exe_id']]['question_list'][$row_q['question_id']] = $row_q;
- }
- }
- return $list;
- }
- /**
- * Gets exercise results (NO Exercises in LPs) from a given exercise id, course, session
- * @param int $exe_id exercise id
- * @param string $status
- * @return array with the results
- *
- */
- public static function get_exercise_results_by_attempt($exe_id, $status = null)
- {
- $table_track_exercises = Database::get_main_table(TABLE_STATISTIC_TRACK_E_EXERCISES);
- $table_track_attempt = Database::get_main_table(TABLE_STATISTIC_TRACK_E_ATTEMPT);
- $table_track_attempt_recording = Database::get_main_table(TABLE_STATISTIC_TRACK_E_ATTEMPT_RECORDING);
- $exe_id = intval($exe_id);
- $status = Database::escape_string($status);
- $sql = "SELECT * FROM $table_track_exercises
- WHERE status = '".$status."' AND exe_id = $exe_id";
- $res = Database::query($sql);
- $list = array();
- if (Database::num_rows($res)) {
- $row = Database::fetch_array($res, 'ASSOC');
- //Checking if this attempt was revised by a teacher
- $sql_revised = 'SELECT exe_id FROM '.$table_track_attempt_recording.'
- WHERE author != "" AND exe_id = '.$exe_id.'
- LIMIT 1';
- $res_revised = Database::query($sql_revised);
- $row['attempt_revised'] = 0;
- if (Database::num_rows($res_revised) > 0) {
- $row['attempt_revised'] = 1;
- }
- $list[$exe_id] = $row;
- $sql = "SELECT * FROM $table_track_attempt
- WHERE exe_id = $exe_id
- ORDER BY tms ASC";
- $res_question = Database::query($sql);
- while ($row_q = Database::fetch_array($res_question, 'ASSOC')) {
- $list[$exe_id]['question_list'][$row_q['question_id']] = $row_q;
- }
- }
- return $list;
- }
- /**
- * Gets exercise results (NO Exercises in LPs) from a given user, exercise id, course, session, lp_id, lp_item_id
- * @param int user id
- * @param int exercise id
- * @param string course code
- * @param int session id
- * @param int lp id
- * @param int lp item id
- * @param string order asc or desc
- * @return array with the results
- *
- */
- public static function getExerciseResultsByUser(
- $user_id,
- $exercise_id,
- $courseId,
- $session_id = 0,
- $lp_id = 0,
- $lp_item_id = 0,
- $order = null
- ) {
- $table_track_exercises = Database::get_main_table(TABLE_STATISTIC_TRACK_E_EXERCISES);
- $table_track_attempt = Database::get_main_table(TABLE_STATISTIC_TRACK_E_ATTEMPT);
- $table_track_attempt_recording = Database::get_main_table(TABLE_STATISTIC_TRACK_E_ATTEMPT_RECORDING);
- $courseId = intval($courseId);
- $exercise_id = intval($exercise_id);
- $session_id = intval($session_id);
- $user_id = intval($user_id);
- $lp_id = intval($lp_id);
- $lp_item_id = intval($lp_item_id);
- if (!in_array(strtolower($order), array('asc', 'desc'))) {
- $order = 'asc';
- }
- $sql = "SELECT * FROM $table_track_exercises
- WHERE
- status = '' AND
- exe_user_id = $user_id AND
- c_id = $courseId AND
- exe_exo_id = $exercise_id AND
- session_id = $session_id AND
- orig_lp_id = $lp_id AND
- orig_lp_item_id = $lp_item_id
- ORDER by exe_id $order ";
- $res = Database::query($sql);
- $list = array();
- while ($row = Database::fetch_array($res, 'ASSOC')) {
- // Checking if this attempt was revised by a teacher
- $sql = 'SELECT exe_id FROM '.$table_track_attempt_recording.'
- WHERE author != "" AND exe_id = '.$row['exe_id'].'
- LIMIT 1';
- $res_revised = Database::query($sql);
- $row['attempt_revised'] = 0;
- if (Database::num_rows($res_revised) > 0) {
- $row['attempt_revised'] = 1;
- }
- $list[$row['exe_id']] = $row;
- $sql = "SELECT * FROM $table_track_attempt
- WHERE exe_id = {$row['exe_id']}";
- $res_question = Database::query($sql);
- while ($row_q = Database::fetch_array($res_question, 'ASSOC')) {
- $list[$row['exe_id']]['question_list'][$row_q['question_id']][] = $row_q;
- }
- }
- return $list;
- }
- /**
- * Count exercise attempts (NO Exercises in LPs ) from a given exercise id, course, session
- * @param int $user_id
- * @param int exercise id
- * @param int $courseId
- * @param int session id
- * @return array with the results
- *
- */
- public static function count_exercise_attempts_by_user($user_id, $exercise_id, $courseId, $session_id = 0)
- {
- $TABLETRACK_EXERCICES = Database::get_main_table(TABLE_STATISTIC_TRACK_E_EXERCISES);
- $courseId = intval($courseId);
- $exercise_id = intval($exercise_id);
- $session_id = intval($session_id);
- $user_id = intval($user_id);
- $sql = "SELECT count(*) as count FROM $TABLETRACK_EXERCICES
- WHERE status = '' AND
- exe_user_id = '$user_id' AND
- c_id = '$courseId' AND
- exe_exo_id = '$exercise_id' AND
- session_id = $session_id AND
- orig_lp_id =0 AND
- orig_lp_item_id = 0
- ORDER BY exe_id";
- $res = Database::query($sql);
- $result = 0;
- if (Database::num_rows($res) > 0) {
- $row = Database::fetch_array($res, 'ASSOC');
- $result = $row['count'];
- }
- return $result;
- }
- /**
- * Gets all exercise BEST results attempts (NO Exercises in LPs) from a given exercise id, course, session per user
- * @param int $exercise_id
- * @param int $courseId
- * @param int $session_id
- * @return array with the results
- * @todo rename this function
- */
- public static function get_best_exercise_results_by_user($exercise_id, $courseId, $session_id = 0)
- {
- $table_track_exercises = Database::get_main_table(TABLE_STATISTIC_TRACK_E_EXERCISES);
- $table_track_attempt = Database::get_main_table(TABLE_STATISTIC_TRACK_E_ATTEMPT);
- $courseId = intval($courseId);
- $exercise_id = intval($exercise_id);
- $session_id = intval($session_id);
- $sql = "SELECT * FROM $table_track_exercises
- WHERE
- status = '' AND
- c_id = $courseId AND
- exe_exo_id = '$exercise_id' AND
- session_id = $session_id AND
- orig_lp_id = 0 AND
- orig_lp_item_id = 0
- ORDER BY exe_id";
- $res = Database::query($sql);
- $list = array();
- while ($row = Database::fetch_array($res, 'ASSOC')) {
- $list[$row['exe_id']] = $row;
- $sql = "SELECT * FROM $table_track_attempt
- WHERE exe_id = {$row['exe_id']}";
- $res_question = Database::query($sql);
- while ($row_q = Database::fetch_array($res_question, 'ASSOC')) {
- $list[$row['exe_id']]['question_list'][$row_q['question_id']] = $row_q;
- }
- }
- // Getting the best results of every student
- $best_score_return = array();
- foreach ($list as $student_result) {
- $user_id = $student_result['exe_user_id'];
- $current_best_score[$user_id] = $student_result['exe_result'];
- //echo $current_best_score[$user_id].' - '.$best_score_return[$user_id]['exe_result'].'<br />';
- if (!isset($best_score_return[$user_id]['exe_result'])) {
- $best_score_return[$user_id] = $student_result;
- }
- if ($current_best_score[$user_id] > $best_score_return[$user_id]['exe_result']) {
- $best_score_return[$user_id] = $student_result;
- }
- }
- return $best_score_return;
- }
- /**
- * @param int $user_id
- * @param int $exercise_id
- * @param int $courseId
- * @param int $session_id
- * @return array
- */
- public static function get_best_attempt_exercise_results_per_user($user_id, $exercise_id, $courseId, $session_id = 0)
- {
- $table_track_exercises = Database::get_main_table(TABLE_STATISTIC_TRACK_E_EXERCISES);
- $courseId = intval($courseId);
- $exercise_id = intval($exercise_id);
- $session_id = intval($session_id);
- $user_id = intval($user_id);
- $sql = "SELECT * FROM $table_track_exercises
- WHERE
- status = '' AND
- c_id = '$courseId' AND
- exe_exo_id = '$exercise_id' AND
- session_id = $session_id AND
- exe_user_id = $user_id AND
- orig_lp_id =0 AND
- orig_lp_item_id = 0
- ORDER BY exe_id";
- $res = Database::query($sql);
- $list = array();
- while ($row = Database::fetch_array($res, 'ASSOC')) {
- $list[$row['exe_id']] = $row;
- }
- //Getting the best results of every student
- $best_score_return = array();
- $best_score_return['exe_result'] = 0;
- foreach ($list as $result) {
- $current_best_score = $result;
- if ($current_best_score['exe_result'] > $best_score_return['exe_result']) {
- $best_score_return = $result;
- }
- }
- if (!isset($best_score_return['exe_weighting'])) {
- $best_score_return = array();
- }
- return $best_score_return;
- }
- /**
- * @param int $exercise_id
- * @param int $courseId
- * @param int $session_id
- * @return mixed
- */
- public static function count_exercise_result_not_validated($exercise_id, $courseId, $session_id = 0)
- {
- $table_track_exercises = Database::get_main_table(TABLE_STATISTIC_TRACK_E_EXERCISES);
- $table_track_attempt = Database::get_main_table(TABLE_STATISTIC_TRACK_E_ATTEMPT_RECORDING);
- $courseId = intval($courseId);
- $session_id = intval($session_id);
- $exercise_id = intval($exercise_id);
- $sql = "SELECT count(e.exe_id) as count
- FROM $table_track_exercises e
- LEFT JOIN $table_track_attempt a
- ON e.exe_id = a.exe_id
- WHERE
- exe_exo_id = $exercise_id AND
- c_id = '$courseId' AND
- e.session_id = $session_id AND
- orig_lp_id = 0 AND
- marks IS NULL AND
- status = '' AND
- orig_lp_item_id = 0
- ORDER BY e.exe_id";
- $res = Database::query($sql);
- $row = Database::fetch_array($res, 'ASSOC');
- return $row['count'];
- }
- /**
- * Gets all exercise BEST results attempts (NO Exercises in LPs) from a given exercise id, course, session per user
- * @param int exercise id
- * @param int course id
- * @param int session id
- * @return array with the results
- *
- */
- public static function get_count_exercises_attempted_by_course($courseId, $session_id = 0)
- {
- $table_track_exercises = Database::get_main_table(TABLE_STATISTIC_TRACK_E_EXERCISES);
- $courseId = intval($courseId);
- $session_id = intval($session_id);
- $sql = "SELECT DISTINCT exe_exo_id, exe_user_id
- FROM $table_track_exercises
- WHERE
- status = '' AND
- c_id = '$courseId' AND
- session_id = $session_id AND
- orig_lp_id =0 AND
- orig_lp_item_id = 0
- ORDER BY exe_id";
- $res = Database::query($sql);
- $count = 0;
- if (Database::num_rows($res) > 0) {
- $count = Database::num_rows($res);
- }
- return $count;
- }
- /**
- * Gets all exercise events from a Learning Path within a Course nd Session
- * @param int $exercise_id
- * @param int $courseId
- * @param int $session_id
- * @return array
- */
- public static function get_all_exercise_event_from_lp($exercise_id, $courseId, $session_id = 0)
- {
- $table_track_exercises = Database::get_main_table(TABLE_STATISTIC_TRACK_E_EXERCISES);
- $table_track_attempt = Database::get_main_table(TABLE_STATISTIC_TRACK_E_ATTEMPT);
- $courseId = intval($courseId);
- $exercise_id = intval($exercise_id);
- $session_id = intval($session_id);
- $sql = "SELECT * FROM $table_track_exercises
- WHERE
- status = '' AND
- c_id = $courseId AND
- exe_exo_id = '$exercise_id' AND
- session_id = $session_id AND
- orig_lp_id !=0 AND
- orig_lp_item_id != 0";
- $res = Database::query($sql);
- $list = array();
- while ($row = Database::fetch_array($res, 'ASSOC')) {
- $list[$row['exe_id']] = $row;
- $sql = "SELECT * FROM $table_track_attempt
- WHERE exe_id = {$row['exe_id']}";
- $res_question = Database::query($sql);
- while ($row_q = Database::fetch_array($res_question, 'ASSOC')) {
- $list[$row['exe_id']]['question_list'][$row_q['question_id']] = $row_q;
- }
- }
- return $list;
- }
- /**
- * @param int $lp_id
- * @param int $course_id
- *
- * @return array
- */
- public static function get_all_exercises_from_lp($lp_id, $course_id)
- {
- $lp_item_table = Database::get_course_table(TABLE_LP_ITEM);
- $course_id = intval($course_id);
- $lp_id = intval($lp_id);
- $sql = "SELECT * FROM $lp_item_table
- WHERE
- c_id = $course_id AND
- lp_id = '".$lp_id."' AND
- item_type = 'quiz'
- ORDER BY parent_item_id, display_order";
- $res = Database::query($sql);
- $my_exercise_list = array();
- while ($row = Database::fetch_array($res, 'ASSOC')) {
- $my_exercise_list[] = $row;
- }
- return $my_exercise_list;
- }
- /**
- * This function gets the comments of an exercise
- *
- * @param int $exe_id
- * @param int $question_id
- *
- * @return string the comment
- */
- public static function get_comments($exe_id, $question_id)
- {
- $table_track_attempt = Database::get_main_table(TABLE_STATISTIC_TRACK_E_ATTEMPT);
- $sql = "SELECT teacher_comment
- FROM $table_track_attempt
- WHERE
- exe_id='".Database::escape_string($exe_id)."' AND
- question_id = '".Database::escape_string($question_id)."'
- ORDER by question_id";
- $sqlres = Database::query($sql);
- $comm = Database::result($sqlres, 0, 'teacher_comment');
- return $comm;
- }
- /**
- * @param int $exe_id
- *
- * @return array
- */
- public static function getAllExerciseEventByExeId($exe_id)
- {
- $table_track_attempt = Database::get_main_table(TABLE_STATISTIC_TRACK_E_ATTEMPT);
- $exe_id = intval($exe_id);
- $list = array();
- $sql = "SELECT * FROM $table_track_attempt
- WHERE exe_id = $exe_id
- ORDER BY position";
- $res_question = Database::query($sql);
- if (Database::num_rows($res_question)) {
- while ($row = Database::fetch_array($res_question, 'ASSOC')) {
- $list[$row['question_id']][] = $row;
- }
- }
- return $list;
- }
- /**
- *
- * @param int $exe_id
- * @param int $user_id
- * @param int $courseId
- * @param int $session_id
- * @param int $question_id
- */
- public static function delete_attempt($exe_id, $user_id, $courseId, $session_id, $question_id)
- {
- $table_track_attempt = Database::get_main_table(TABLE_STATISTIC_TRACK_E_ATTEMPT);
- $exe_id = intval($exe_id);
- $user_id = intval($user_id);
- $courseId = intval($courseId);
- $session_id = intval($session_id);
- $question_id = intval($question_id);
- $sql = "DELETE FROM $table_track_attempt
- WHERE
- exe_id = $exe_id AND
- user_id = $user_id AND
- c_id = $courseId AND
- session_id = $session_id AND
- question_id = $question_id ";
- Database::query($sql);
- self::addEvent(
- LOG_QUESTION_RESULT_DELETE,
- LOG_EXERCISE_ATTEMPT_QUESTION_ID,
- $exe_id.'-'.$question_id,
- null,
- null,
- $courseId,
- $session_id
- );
- }
- /**
- * @param $exe_id
- * @param $user_id
- * @param int $courseId
- * @param $question_id
- * @param int $sessionId
- */
- public static function delete_attempt_hotspot($exe_id, $user_id, $courseId, $question_id, $sessionId = null)
- {
- $table_track_attempt = Database::get_main_table(TABLE_STATISTIC_TRACK_E_HOTSPOT);
- $exe_id = intval($exe_id);
- $user_id = intval($user_id);
- $courseId = intval($courseId);
- $question_id = intval($question_id);
- if (!isset($sessionId)) {
- $sessionId = api_get_session_id();
- }
- $sql = "DELETE FROM $table_track_attempt
- WHERE
- hotspot_exe_id = $exe_id AND
- hotspot_user_id = $user_id AND
- c_id = $courseId AND
- hotspot_question_id = $question_id ";
- Database::query($sql);
- self::addEvent(
- LOG_QUESTION_RESULT_DELETE,
- LOG_EXERCISE_ATTEMPT_QUESTION_ID,
- $exe_id.'-'.$question_id,
- null,
- null,
- $courseId,
- $sessionId
- );
- }
- /**
- * Registers in track_e_course_access when user logs in for the first time to a course
- * @param int $courseId ID of the course
- * @param int $user_id ID of the user
- * @param int $session_id ID of the session (if any)
- */
- public static function event_course_login($courseId, $user_id, $session_id)
- {
- $course_tracking_table = Database::get_main_table(TABLE_STATISTIC_TRACK_E_COURSE_ACCESS);
- $loginDate = $logoutDate = api_get_utc_datetime();
- //$counter represents the number of time this record has been refreshed
- $counter = 1;
- $courseId = intval($courseId);
- $user_id = intval($user_id);
- $session_id = intval($session_id);
- $ip = api_get_real_ip();
- $sql = "INSERT INTO $course_tracking_table(c_id, user_ip, user_id, login_course_date, logout_course_date, counter, session_id)
- VALUES('".$courseId."', '".$ip."', '".$user_id."', '$loginDate', '$logoutDate', $counter, '".$session_id."')";
- Database::query($sql);
- // Course catalog stats modifications see #4191
- CourseManager::update_course_ranking(
- null,
- null,
- null,
- null,
- true,
- false
- );
- }
- /**
- * Register a "fake" time spent on the platform, for example to match the
- * estimated time he took to author an assignment/work, see configuration
- * setting considered_working_time.
- * This assumes there is already some connection of the student to the
- * course, otherwise he wouldn't be able to upload an assignment.
- * This works by creating a new record, copy of the current one, then
- * updating the current one to be just the considered_working_time and
- * end at the same second as the user connected to the course.
- * @param int $courseId The course in which to add the time
- * @param int $userId The user for whom to add the time
- * @param int $sessionId The session in which to add the time (if any)
- * @param string $virtualTime The amount of time to be added, in a hh:mm:ss format. If int, we consider it is expressed in hours.
- * @param string $ip IP address to go on record for this time record
- *
- * @return True on successful insertion, false otherwise
- */
- public static function eventAddVirtualCourseTime(
- $courseId,
- $userId,
- $sessionId,
- $virtualTime = '',
- $ip = ''
- ) {
- $courseTrackingTable = Database::get_main_table(TABLE_STATISTIC_TRACK_E_COURSE_ACCESS);
- $time = $loginDate = $logoutDate = api_get_utc_datetime();
- $courseId = intval($courseId);
- $userId = intval($userId);
- $sessionId = intval($sessionId);
- $ip = Database::escape_string($ip);
- // Get the current latest course connection register. We need that
- // record to re-use the data and create a new record.
- $sql = "SELECT *
- FROM $courseTrackingTable
- WHERE
- user_id = ".$userId." AND
- c_id = ".$courseId." AND
- session_id = ".$sessionId." AND
- login_course_date > '$time' - INTERVAL 3600 SECOND
- ORDER BY login_course_date DESC LIMIT 0,1";
- $result = Database::query($sql);
- // Ignore if we didn't find any course connection record in the last
- // hour. In this case it wouldn't be right to add a "fake" time record.
- if (Database::num_rows($result) > 0) {
- // Found the latest connection
- $row = Database::fetch_row($result);
- $courseAccessId = $row[0];
- $courseAccessLoginDate = $row[3];
- $counter = $row[5];
- $counter = $counter ? $counter : 0;
- // Insert a new record, copy of the current one (except the logout
- // date that we update to the current time)
- $sql = "INSERT INTO $courseTrackingTable(
- c_id,
- user_ip,
- user_id,
- login_course_date,
- logout_course_date,
- counter,
- session_id
- ) VALUES(
- $courseId,
- '$ip',
- $userId,
- '$courseAccessLoginDate',
- '$logoutDate',
- $counter,
- $sessionId
- )";
- Database::query($sql);
- $loginDate = ChamiloApi::addOrSubTimeToDateTime(
- $virtualTime,
- $courseAccessLoginDate,
- false
- );
- // We update the course tracking table
- $sql = "UPDATE $courseTrackingTable
- SET
- login_course_date = '$loginDate',
- logout_course_date = '$courseAccessLoginDate',
- counter = 0
- WHERE
- course_access_id = ".intval($courseAccessId)." AND
- session_id = ".$sessionId;
- Database::query($sql);
- return true;
- }
- return false;
- }
- /**
- * Removes a "fake" time spent on the platform, for example to match the
- * estimated time he took to author an assignment/work, see configuration
- * setting considered_working_time.
- * This method should be called when something that generated a fake
- * time record is removed. Given the database link is weak (no real
- * relationship kept between the deleted item and this record), this
- * method just looks for the latest record that has the same time as the
- * item's fake time, is in the past and in this course+session. If such a
- * record cannot be found, it doesn't do anything.
- * The IP address is not considered a useful filter here.
- * @param int $courseId The course in which to add the time
- * @param int $userId The user for whom to add the time
- * @param int $sessionId The session in which to add the time (if any)
- * @param string $virtualTime The amount of time to be added, in a hh:mm:ss format. If int, we consider it is expressed in hours.
- * @return True on successful removal, false otherwise
- */
- public static function eventRemoveVirtualCourseTime($courseId, $userId, $sessionId = 0, $virtualTime = '')
- {
- if (empty($virtualTime)) {
- return false;
- }
- $courseTrackingTable = Database::get_main_table(TABLE_STATISTIC_TRACK_E_COURSE_ACCESS);
- $time = $loginDate = $logoutDate = api_get_utc_datetime();
- $courseId = intval($courseId);
- $userId = intval($userId);
- $sessionId = intval($sessionId);
- // Change $virtualTime format from hh:mm:ss to hhmmss which is the
- // format returned by SQL for a subtraction of two datetime values
- // @todo make sure this is portable between DBMSes
- if (preg_match('/:/', $virtualTime)) {
- list($h, $m, $s) = preg_split('/:/', $virtualTime);
- $virtualTime = $h * 3600 + $m * 60 + $s;
- } else {
- $virtualTime *= 3600;
- }
- // Get the current latest course connection register. We need that
- // record to re-use the data and create a new record.
- $sql = "SELECT course_access_id
- FROM $courseTrackingTable
- WHERE
- user_id = $userId AND
- c_id = $courseId AND
- session_id = $sessionId AND
- counter = 0 AND
- (UNIX_TIMESTAMP(logout_course_date) - UNIX_TIMESTAMP(login_course_date)) = '$virtualTime'
- ORDER BY login_course_date DESC LIMIT 0,1";
- $result = Database::query($sql);
- // Ignore if we didn't find any course connection record in the last
- // hour. In this case it wouldn't be right to add a "fake" time record.
- if (Database::num_rows($result) > 0) {
- // Found the latest connection
- $row = Database::fetch_row($result);
- $courseAccessId = $row[0];
- $sql = "DELETE FROM $courseTrackingTable WHERE course_access_id = $courseAccessId";
- $result = Database::query($sql);
- return $result;
- }
- return false;
- }
- /**
- * For the sake of genericity, this function is a switch.
- * It's called by EventsDispatcher and fires the good function
- * with the good require_once.
- *
- * @param string $event_name
- * @param array $params
- */
- public static function event_send_mail($event_name, $params)
- {
- EventsMail::send_mail($event_name, $params);
- }
- /**
- * Internal function checking if the mail was already sent from that user to that user
- * @param string $event_name
- * @param int $user_from
- * @param int $user_to
- * @return boolean
- */
- public static function check_if_mail_already_sent($event_name, $user_from, $user_to = null)
- {
- if ($user_to == null) {
- $sql = 'SELECT COUNT(*) as total FROM '.Database::get_main_table(TABLE_EVENT_SENT).'
- WHERE user_from = '.$user_from.' AND event_type_name = "'.$event_name.'"';
- } else {
- $sql = 'SELECT COUNT(*) as total FROM '.Database::get_main_table(TABLE_EVENT_SENT).'
- WHERE user_from = '.$user_from.' AND user_to = '.$user_to.' AND event_type_name = "'.$event_name.'"';
- }
- $result = Database::store_result(Database::query($sql), 'ASSOC');
- return $result[0]["total"];
- }
- /**
- *
- * Filter EventEmailTemplate Filters see the main/inc/conf/events.conf.dist.php
- *
- */
- /**
- * Basic template event message filter (to be used by other filters as default)
- * @param array $values (passing by reference)
- * @return boolean True if everything is OK, false otherwise
- */
- public function event_send_mail_filter_func(&$values)
- {
- return true;
- }
- /**
- * user_registration - send_mail filter
- * @param array $values (passing by reference)
- * @return boolean True if everything is OK, false otherwise
- */
- public function user_registration_event_send_mail_filter_func(&$values)
- {
- $res = self::event_send_mail_filter_func($values);
- // proper logic for this filter
- return $res;
- }
- /**
- * portal_homepage_edited - send_mail filter
- * @param array $values (passing by reference)
- * @return boolean True if everything is OK, false otherwise
- */
- public function portal_homepage_edited_event_send_mail_filter_func(&$values)
- {
- $res = self::event_send_mail_filter_func($values);
- // proper logic for this filter
- return $res;
- }
- }
|