swad-core/swad_report.c

1423 lines
49 KiB
C
Raw Normal View History

2016-09-08 18:04:30 +02:00
// swad_report.c: report on my use of the platform
/*
SWAD (Shared Workspace At a Distance),
is a web platform developed at the University of Granada (Spain),
and used to support university teaching.
This file is part of SWAD core.
2017-01-13 01:51:23 +01:00
Copyright (C) 1999-2017 Antonio Ca<EFBFBD>as Vargas
2016-09-08 18:04:30 +02:00
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU Affero General 3 License as
published by the Free Software Foundation, either version 3 of the
License, or (at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Affero General Public License for more details.
You should have received a copy of the GNU Affero General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
/*****************************************************************************/
/*********************************** Headers *********************************/
/*****************************************************************************/
2016-10-04 02:32:25 +02:00
#include <sys/stat.h> // For mkdir
#include <sys/types.h> // For mkdir
2016-09-08 18:04:30 +02:00
2017-06-10 21:38:10 +02:00
#include "swad_box.h"
2016-09-11 21:25:14 +02:00
#include "swad_database.h"
2016-09-08 19:21:25 +02:00
#include "swad_global.h"
2016-09-11 12:46:52 +02:00
#include "swad_ID.h"
2016-09-08 19:21:25 +02:00
#include "swad_profile.h"
2016-10-08 20:25:26 +02:00
#include "swad_tab.h"
2016-09-08 18:04:30 +02:00
/*****************************************************************************/
/****************************** Public constants *****************************/
/*****************************************************************************/
/*****************************************************************************/
/***************************** Private constants *****************************/
/*****************************************************************************/
2016-10-04 17:21:21 +02:00
#define Rep_FILENAME_ROOT "report"
2016-10-03 13:29:05 +02:00
#define Rep_MIN_CLICKS_CRS 100 // Minimum number of clicks to show a course in historic log
2016-10-08 20:25:26 +02:00
#define Rep_MAX_ACTIONS 50 // Maximum number of actions in list of frequent actions
2016-10-04 01:16:52 +02:00
#define Rep_MAX_BAR_WIDTH 50 // Maximum width of graphic bar
2016-10-03 13:29:05 +02:00
2016-09-12 10:02:17 +02:00
// #define Rep_BLOCK "&boxH;" // HTML code for a block in graphic bar
// #define Rep_BLOCK "&blk12;" // HTML code for a block in graphic bar
// #define Rep_BLOCK "&block;" // HTML code for a block in graphic bar
// #define Rep_BLOCK "&equiv;" // HTML code for a block in graphic bar
// #define Rep_BLOCK "&bull;" // HTML code for a block in graphic bar
2016-10-04 01:16:52 +02:00
// #define Rep_BLOCK "&squf;" // HTML code for a block in graphic bar
#define Rep_BLOCK "-"
2016-09-12 10:02:17 +02:00
2016-09-08 18:04:30 +02:00
/*****************************************************************************/
/****************************** Internal types *******************************/
/*****************************************************************************/
2016-10-08 21:38:17 +02:00
struct Rep_CurrentTimeUTC
2016-10-04 00:22:59 +02:00
{
2017-01-28 15:58:46 +01:00
char StrDate[10 + 1]; // Example: 2016-10-02
// 1234567890
char StrTime[8 + 1]; // Example: 19:03:49
// 12345678
unsigned Date; // Example: 20161002
unsigned Time; // Example: 190349
2016-10-04 00:22:59 +02:00
};
2016-10-08 21:19:47 +02:00
struct Rep_Hits
{
unsigned long Num;
unsigned long Max;
};
2016-10-08 21:38:17 +02:00
struct Rep_Report
{
struct UsrFigures UsrFigures;
struct tm tm_FirstClickTime;
struct tm tm_CurrentTime;
struct Rep_CurrentTimeUTC CurrentTimeUTC;
struct Rep_Hits Hits;
unsigned long MaxHitsPerYear;
2017-01-28 15:58:46 +01:00
char FilenameReport[NAME_MAX + 1];
2017-03-13 22:47:57 +01:00
char Permalink[Cns_MAX_BYTES_WWW + 1];
2016-10-08 21:38:17 +02:00
};
2016-09-08 18:04:30 +02:00
/*****************************************************************************/
/************** External global variables from others modules ****************/
/*****************************************************************************/
2016-09-08 19:21:25 +02:00
extern struct Globals Gbl;
2016-09-08 18:04:30 +02:00
/*****************************************************************************/
/************************* Internal global variables *************************/
/*****************************************************************************/
/*****************************************************************************/
/***************************** Private prototypes ****************************/
/*****************************************************************************/
2016-10-08 21:49:04 +02:00
static void Rep_CreateMyUsageReport (struct Rep_Report *Report);
static void Rep_PutLinkToMyUsageReport (struct Rep_Report *Report);
2016-10-08 21:38:17 +02:00
static void Req_TitleReport (struct Rep_CurrentTimeUTC *CurrentTimeUTC);
2016-10-04 00:22:59 +02:00
2016-10-08 23:47:20 +02:00
static void Rep_GetCurrentDateTimeUTC (struct Rep_Report *Report);
2016-10-04 17:21:21 +02:00
2016-10-08 23:50:01 +02:00
static void Rep_CreateNewReportFile (struct Rep_Report *Report);
2016-10-08 23:52:41 +02:00
static void Rep_CreateNewReportEntryIntoDB (const struct Rep_Report *Report);
2016-10-09 00:05:56 +02:00
static void Rep_WriteHeader (const struct Rep_Report *Report);
2016-10-03 09:40:10 +02:00
static void Rep_WriteSectionPlatform (void);
2016-10-03 09:44:32 +02:00
static void Rep_WriteSectionUsrInfo (void);
2016-10-09 00:20:03 +02:00
static void Rep_WriteSectionUsrFigures (const struct Rep_Report *Report);
2016-10-09 01:03:30 +02:00
static void Rep_WriteSectionHitsPerAction (struct Rep_Report *Report);
2016-10-09 00:49:23 +02:00
static void Rep_WriteSectionGlobalHits (struct Rep_Report *Report);
2016-10-09 01:03:30 +02:00
static void Rep_WriteSectionCurrentCourses (struct Rep_Report *Report);
static void Rep_WriteSectionHistoricCourses (struct Rep_Report *Report);
2016-10-03 09:40:10 +02:00
2016-10-09 01:03:30 +02:00
static void Rep_GetMaxHitsPerYear (struct Rep_Report *Report);
2016-10-07 02:23:24 +02:00
static void Rep_GetAndWriteMyCurrentCrss (Rol_Role_t Role,
2016-10-09 01:03:30 +02:00
struct Rep_Report *Report);
static void Rep_GetAndWriteMyHistoricClicsWithoutCrs (struct Rep_Report *Report);
2016-10-07 02:23:24 +02:00
static void Rep_GetAndWriteMyHistoricCrss (Rol_Role_t Role,
2016-10-09 01:03:30 +02:00
struct Rep_Report *Report);
2016-10-02 19:42:47 +02:00
static void Rep_WriteRowCrsData (long CrsCod,Rol_Role_t Role,
2016-10-09 01:03:30 +02:00
struct Rep_Report *Report,
2016-10-07 02:23:24 +02:00
bool WriteNumUsrs);
2016-10-02 17:56:05 +02:00
2016-10-02 19:42:47 +02:00
static void Rep_ShowMyHitsPerYear (bool AnyCourse,long CrsCod,Rol_Role_t Role,
2016-10-09 01:03:30 +02:00
struct Rep_Report *Report);
2016-10-08 21:19:47 +02:00
static void Rep_ComputeMaxAndTotalHits (struct Rep_Hits *Hits,
unsigned long NumRows,
MYSQL_RES *mysql_res,unsigned Field);
static void Rep_DrawBarNumHits (unsigned long HitsNum,unsigned long HitsMax,
2016-09-12 10:02:17 +02:00
unsigned MaxBarWidth);
2016-09-12 00:50:24 +02:00
2016-10-06 22:18:33 +02:00
static void Rep_RemoveUsrReportsFiles (long UsrCod);
static void Rep_RemoveUsrReportsFromDB (long UsrCod);
2016-09-08 18:04:30 +02:00
/*****************************************************************************/
2016-10-04 00:22:59 +02:00
/******* Request my usage report (report on my use of the platform) **********/
2016-09-08 18:04:30 +02:00
/*****************************************************************************/
2016-10-03 13:04:49 +02:00
void Rep_ReqMyUsageReport (void)
{
2016-11-13 23:01:11 +01:00
extern const char *Hlp_STATS_Report;
2016-10-03 18:16:05 +02:00
extern const char *Txt_Report_of_use_of_PLATFORM;
2016-10-03 13:04:49 +02:00
extern const char *Txt_Generate_report;
/***** Form to show my usage report *****/
Act_FormStart (ActSeeMyUsgRep);
2017-06-12 14:16:33 +02:00
/***** Start box *****/
2016-10-03 18:16:05 +02:00
sprintf (Gbl.Title,Txt_Report_of_use_of_PLATFORM,Cfg_PLATFORM_SHORT_NAME);
2017-06-10 21:38:10 +02:00
Box_StartBox (NULL,Gbl.Title,NULL,
2017-06-12 15:03:29 +02:00
Hlp_STATS_Report,Box_NOT_CLOSABLE);
2016-10-03 13:04:49 +02:00
/***** Header *****/
2016-10-06 21:31:51 +02:00
Req_TitleReport (NULL); // NULL means do not write date
2016-10-03 13:04:49 +02:00
2017-06-12 14:16:33 +02:00
/***** Send button and end box *****/
2017-06-11 19:02:40 +02:00
Box_EndBoxWithButton (Btn_CONFIRM_BUTTON,Txt_Generate_report);
2016-10-03 13:04:49 +02:00
2017-06-12 14:16:33 +02:00
/***** End form *****/
2016-10-03 13:04:49 +02:00
Act_FormEnd ();
}
2016-10-04 00:22:59 +02:00
/*****************************************************************************/
/********* Show my usage report (report on my use of the platform) ***********/
/*****************************************************************************/
2016-09-08 18:04:30 +02:00
void Rep_ShowMyUsageReport (void)
2016-10-04 00:22:59 +02:00
{
2016-10-08 21:49:04 +02:00
struct Rep_Report Report;
2016-10-04 00:22:59 +02:00
/***** Create my usage report *****/
2016-10-08 21:49:04 +02:00
Rep_CreateMyUsageReport (&Report);
2016-10-04 00:22:59 +02:00
/***** Put link to my usage report *****/
2016-10-08 21:49:04 +02:00
Rep_PutLinkToMyUsageReport (&Report);
2016-10-04 00:22:59 +02:00
}
/*****************************************************************************/
/******** Create my usage report (report on my use of the platform) **********/
/*****************************************************************************/
2016-10-08 21:49:04 +02:00
static void Rep_CreateMyUsageReport (struct Rep_Report *Report)
2016-09-09 00:36:44 +02:00
{
2016-10-03 18:16:05 +02:00
bool GetUsrFiguresAgain;
2016-10-02 17:56:05 +02:00
2016-10-02 21:14:07 +02:00
/***** Get current date-time *****/
2016-10-08 23:47:20 +02:00
Rep_GetCurrentDateTimeUTC (Report);
2016-10-02 21:14:07 +02:00
2016-10-06 20:35:10 +02:00
/***** Create a new report file *****/
2016-10-08 23:50:01 +02:00
Rep_CreateNewReportFile (Report);
2016-10-03 14:12:01 +02:00
2016-10-06 20:35:10 +02:00
/***** Store report entry into database *****/
2016-10-08 23:52:41 +02:00
Rep_CreateNewReportEntryIntoDB (Report);
2016-10-04 02:32:25 +02:00
2016-10-03 14:12:01 +02:00
/***** Start file *****/
2016-10-08 21:49:04 +02:00
Lay_StartHTMLFile (Gbl.F.Rep,Report->FilenameReport);
2016-10-04 23:57:32 +02:00
fprintf (Gbl.F.Rep,"<body style=\"margin:1em;"
" text-align:left;"
" font-family:Helvetica,Arial,sans-serif;\">\n");
2016-09-11 22:45:53 +02:00
2016-10-03 13:04:49 +02:00
/***** Header *****/
2016-10-09 00:05:56 +02:00
Rep_WriteHeader (Report);
2016-10-02 22:54:30 +02:00
/***** Platform *****/
2016-10-03 09:40:10 +02:00
Rep_WriteSectionPlatform ();
2016-10-02 21:14:07 +02:00
2016-09-11 22:45:53 +02:00
/***** Personal information *****/
2016-10-03 09:44:32 +02:00
Rep_WriteSectionUsrInfo ();
2016-09-11 22:45:53 +02:00
/***** Figures *****/
2016-10-08 21:49:04 +02:00
Prf_GetUsrFigures (Gbl.Usrs.Me.UsrDat.UsrCod,&Report->UsrFigures);
2016-10-09 01:03:30 +02:00
GetUsrFiguresAgain = Prf_GetAndStoreAllUsrFigures (Gbl.Usrs.Me.UsrDat.UsrCod,
&Report->UsrFigures);
2016-10-03 18:16:05 +02:00
if (GetUsrFiguresAgain)
2016-10-08 21:49:04 +02:00
Prf_GetUsrFigures (Gbl.Usrs.Me.UsrDat.UsrCod,&Report->UsrFigures);
if (Report->UsrFigures.FirstClickTimeUTC)
2016-10-09 01:03:30 +02:00
gmtime_r (&Report->UsrFigures.FirstClickTimeUTC,
&Report->tm_FirstClickTime);
2016-10-09 00:20:03 +02:00
Rep_WriteSectionUsrFigures (Report);
2016-09-11 22:45:53 +02:00
2016-10-08 20:25:26 +02:00
/***** Global count of hits *****/
2016-10-09 00:34:47 +02:00
Rep_WriteSectionGlobalHits (Report);
2016-10-02 23:38:40 +02:00
2016-10-08 20:25:26 +02:00
/***** Global hits distributed by action *****/
2016-10-09 01:03:30 +02:00
Rep_WriteSectionHitsPerAction (Report);
2016-10-08 20:25:26 +02:00
2016-10-02 19:42:47 +02:00
/***** Current courses *****/
2016-10-09 01:03:30 +02:00
Rep_GetMaxHitsPerYear (Report);
2016-10-09 00:37:58 +02:00
Rep_WriteSectionCurrentCourses (Report);
2016-09-11 22:45:53 +02:00
2016-10-02 19:42:47 +02:00
/***** Historic courses *****/
2016-10-09 00:49:23 +02:00
Rep_WriteSectionHistoricCourses (Report);
2016-10-02 17:56:05 +02:00
2016-10-03 14:12:01 +02:00
/***** End file *****/
2016-10-04 01:16:52 +02:00
fprintf (Gbl.F.Rep,"</body>\n"
2016-10-03 22:41:45 +02:00
"</html>\n");
2016-10-03 14:12:01 +02:00
2016-10-03 22:41:45 +02:00
/***** Close report file *****/
2016-10-03 14:12:01 +02:00
Fil_CloseReportFile ();
2016-10-04 00:22:59 +02:00
}
/*****************************************************************************/
/******* Put link to my usage report (report on my use of the platform) ******/
/*****************************************************************************/
2016-10-08 21:49:04 +02:00
static void Rep_PutLinkToMyUsageReport (struct Rep_Report *Report)
2016-10-04 00:22:59 +02:00
{
2016-11-13 23:01:11 +01:00
extern const char *Hlp_STATS_Report;
2016-10-04 00:22:59 +02:00
extern const char *Txt_Report_of_use_of_PLATFORM;
extern const char *Txt_Report;
2016-10-06 23:15:14 +02:00
extern const char *Txt_This_link_will_remain_active_as_long_as_your_user_s_account_exists;
2016-10-03 14:12:01 +02:00
2017-06-12 14:16:33 +02:00
/***** Start box *****/
2016-10-03 22:41:45 +02:00
sprintf (Gbl.Title,Txt_Report_of_use_of_PLATFORM,Cfg_PLATFORM_SHORT_NAME);
2017-06-10 21:38:10 +02:00
Box_StartBox (NULL,Gbl.Title,NULL,
2017-06-12 15:03:29 +02:00
Hlp_STATS_Report,Box_NOT_CLOSABLE);
2016-09-09 00:36:44 +02:00
2016-10-03 22:41:45 +02:00
/***** Header *****/
2016-10-08 21:49:04 +02:00
Req_TitleReport (&Report->CurrentTimeUTC);
2016-09-09 00:36:44 +02:00
2016-10-03 22:41:45 +02:00
/***** Put anchor and report filename *****/
2017-03-04 22:23:50 +01:00
fprintf (Gbl.F.Out,"<div class=\"FILENAME_TXT CENTER_MIDDLE\">"
"<a href=\"%s\" class=\"FILENAME_TXT\""
2016-10-04 02:32:25 +02:00
" title=\"%s\" target=\"_blank\">"
"<img src=\"%s/report64x64.png\" alt=\"%s\""
2016-11-14 10:05:41 +01:00
" class=\"ICO64x64\" /><br />"
2016-10-04 02:32:25 +02:00
"%s"
2016-10-03 22:41:45 +02:00
"</a>"
"</div>",
2016-10-08 21:49:04 +02:00
Report->Permalink,
2016-10-04 02:32:25 +02:00
Txt_Report,
Gbl.Prefs.IconsURL,
Txt_Report,
2016-10-08 21:49:04 +02:00
Report->FilenameReport);
2016-10-06 23:15:14 +02:00
fprintf (Gbl.F.Out,"<div class=\"DAT_LIGHT\">%s</div>",
Txt_This_link_will_remain_active_as_long_as_your_user_s_account_exists);
2016-09-09 00:36:44 +02:00
2017-06-12 14:16:33 +02:00
/***** End box *****/
2017-06-10 21:38:10 +02:00
Box_EndBox ();
2016-09-09 00:36:44 +02:00
}
2016-09-11 21:25:14 +02:00
2016-10-04 00:30:46 +02:00
/*****************************************************************************/
/********************* Write title of user's usage report ********************/
/*****************************************************************************/
2016-10-06 21:31:51 +02:00
// CurrentTimeUTC == NULL ==> do not write date
2016-10-04 00:30:46 +02:00
2016-10-08 21:38:17 +02:00
static void Req_TitleReport (struct Rep_CurrentTimeUTC *CurrentTimeUTC)
2016-10-04 00:30:46 +02:00
{
2016-10-04 23:57:32 +02:00
extern const char *Txt_User[Usr_NUM_SEXS];
extern const char *Txt_Date;
fprintf (Gbl.F.Out,"<div class=\"DAT\" style=\"margin-bottom:10px;\">");
/***** User *****/
fprintf (Gbl.F.Out,"%s: <span class=\"DAT_N_BOLD\">%s</span>",
Txt_User[Gbl.Usrs.Me.UsrDat.Sex],
Gbl.Usrs.Me.UsrDat.FullName);
/***** Report date *****/
2016-10-06 21:31:51 +02:00
if (CurrentTimeUTC)
fprintf (Gbl.F.Out,"<br />"
"%s: <span class=\"DAT_N\">%s %s UTC</span>",
Txt_Date,
CurrentTimeUTC->StrDate,
CurrentTimeUTC->StrTime);
2016-10-04 23:57:32 +02:00
2016-10-04 00:30:46 +02:00
fprintf (Gbl.F.Out,"</div>");
}
2016-10-03 09:40:10 +02:00
/*****************************************************************************/
2016-10-04 17:21:21 +02:00
/********************* Get current date and time in UTC **********************/
/*****************************************************************************/
2016-10-08 23:47:20 +02:00
static void Rep_GetCurrentDateTimeUTC (struct Rep_Report *Report)
2016-10-04 17:21:21 +02:00
{
time_t CurrentTime;
/***** Initialize to empty strings *****/
2016-10-08 23:47:20 +02:00
Report->CurrentTimeUTC.StrDate[0] = '\0';
Report->CurrentTimeUTC.StrTime[0] = '\0';
2016-10-04 17:21:21 +02:00
/***** Get current time UTC *****/
time (&CurrentTime);
2016-10-08 23:47:20 +02:00
if ((gmtime_r (&CurrentTime,&Report->tm_CurrentTime)) != NULL)
2016-10-04 17:21:21 +02:00
{
/* Date and time as strings */
2016-10-08 23:47:20 +02:00
sprintf (Report->CurrentTimeUTC.StrDate,"%04d-%02d-%02d",
1900 + Report->tm_CurrentTime.tm_year, // year
1 + Report->tm_CurrentTime.tm_mon, // month
Report->tm_CurrentTime.tm_mday); // day of the month
sprintf (Report->CurrentTimeUTC.StrTime,"%02d:%02d:%02d",
Report->tm_CurrentTime.tm_hour, // hours
Report->tm_CurrentTime.tm_min, // minutes
Report->tm_CurrentTime.tm_sec); // seconds
2016-10-04 17:21:21 +02:00
/* Date and time as unsigned */
2016-10-08 23:47:20 +02:00
Report->CurrentTimeUTC.Date = (1900 + Report->tm_CurrentTime.tm_year) * 10000 +
(1 + Report->tm_CurrentTime.tm_mon) * 100 +
Report->tm_CurrentTime.tm_mday;
Report->CurrentTimeUTC.Time = Report->tm_CurrentTime.tm_hour * 10000 +
Report->tm_CurrentTime.tm_min * 100 +
Report->tm_CurrentTime.tm_sec;
2016-10-04 17:21:21 +02:00
}
}
2016-10-06 20:35:10 +02:00
/*****************************************************************************/
/*************** Create a new file for user's usage report *******************/
/*****************************************************************************/
2016-10-08 23:50:01 +02:00
static void Rep_CreateNewReportFile (struct Rep_Report *Report)
2016-10-06 20:35:10 +02:00
{
2017-01-28 15:58:46 +01:00
char PathReports[PATH_MAX + 1];
char PathUniqueDirL[PATH_MAX + 1];
char PathUniqueDirR[PATH_MAX + 1];
char PathFileReport[PATH_MAX + 1];
2016-10-06 20:35:10 +02:00
/***** Path for reports *****/
sprintf (PathReports,"%s/%s",
Cfg_PATH_SWAD_PUBLIC,Cfg_FOLDER_REP);
Fil_CreateDirIfNotExists (PathReports);
/***** Unique directory for the file with the report *****/
/* 1. Create a directory using the leftmost 2 chars of a unique name */
sprintf (PathUniqueDirL,"%s/%s/%c%c",
Cfg_PATH_SWAD_PUBLIC,Cfg_FOLDER_REP,
Gbl.UniqueNameEncrypted[0],
Gbl.UniqueNameEncrypted[1]);
Fil_CreateDirIfNotExists (PathUniqueDirL);
/* 2. Create a directory using the rightmost 41 chars of a unique name */
sprintf (PathUniqueDirR,"%s/%s",
PathUniqueDirL,
&Gbl.UniqueNameEncrypted[2]);
if (mkdir (PathUniqueDirR,(mode_t) 0xFFF))
Lay_ShowErrorAndExit ("Can not create directory for report.");
/***** Path of the public file with the report */
2016-10-08 23:50:01 +02:00
sprintf (Report->FilenameReport,"%s_%06u_%06u.html",
Rep_FILENAME_ROOT,Report->CurrentTimeUTC.Date,Report->CurrentTimeUTC.Time);
2016-10-06 20:35:10 +02:00
sprintf (PathFileReport,"%s/%s",
2016-10-08 23:50:01 +02:00
PathUniqueDirR,Report->FilenameReport);
2016-10-06 20:35:10 +02:00
if ((Gbl.F.Rep = fopen (PathFileReport,"wb")) == NULL)
Lay_ShowErrorAndExit ("Can not create report file.");
/***** Permalink *****/
2016-10-08 23:50:01 +02:00
sprintf (Report->Permalink,"%s/%s/%c%c/%s/%s",
2016-10-06 20:35:10 +02:00
Cfg_URL_SWAD_PUBLIC,
Cfg_FOLDER_REP,
Gbl.UniqueNameEncrypted[0],
Gbl.UniqueNameEncrypted[1],
&Gbl.UniqueNameEncrypted[2],
2016-10-08 23:50:01 +02:00
Report->FilenameReport);
2016-10-06 20:35:10 +02:00
}
/*****************************************************************************/
/************** Insert a new user's usage report into database ***************/
/*****************************************************************************/
2016-10-08 23:52:41 +02:00
static void Rep_CreateNewReportEntryIntoDB (const struct Rep_Report *Report)
2016-10-06 20:35:10 +02:00
{
2017-03-13 22:47:57 +01:00
char Query[1024 +
NAME_MAX +
Cns_MAX_BYTES_WWW];
2016-10-06 20:35:10 +02:00
/***** Insert a new user's usage report into database *****/
sprintf (Query,"INSERT INTO usr_report"
2016-10-06 21:31:51 +02:00
" (UsrCod,ReportTimeUTC,"
"UniqueDirL,UniqueDirR,Filename,Permalink)"
2016-10-06 20:35:10 +02:00
" VALUES"
2017-03-24 01:09:27 +01:00
" (%ld,'%04d-%02d-%02d %02d:%02d:%02d',"
2016-10-06 21:31:51 +02:00
"'%c%c','%s','%s','%s')",
2016-10-06 20:35:10 +02:00
Gbl.Usrs.Me.UsrDat.UsrCod,
2016-10-08 23:52:41 +02:00
1900 + Report->tm_CurrentTime.tm_year, // year
1 + Report->tm_CurrentTime.tm_mon, // month
Report->tm_CurrentTime.tm_mday, // day of the month
Report->tm_CurrentTime.tm_hour, // hours
Report->tm_CurrentTime.tm_min, // minutes
Report->tm_CurrentTime.tm_sec, // seconds
2016-10-06 20:35:10 +02:00
Gbl.UniqueNameEncrypted[0], // 2 leftmost chars from a unique 43 chars base64url codified from a unique SHA-256 string
Gbl.UniqueNameEncrypted[1],
&Gbl.UniqueNameEncrypted[2], // 41 rightmost chars from a unique 43 chars base64url codified from a unique SHA-256 string
2016-10-08 23:52:41 +02:00
Report->FilenameReport,Report->Permalink);
2016-10-06 22:18:33 +02:00
DB_QueryINSERT (Query,"can not create new user's usage report");
2016-10-06 20:35:10 +02:00
}
2016-10-04 17:21:21 +02:00
/*****************************************************************************/
2016-10-03 09:40:10 +02:00
/******************** Write header of user's usage report ********************/
/*****************************************************************************/
2016-10-09 00:05:56 +02:00
static void Rep_WriteHeader (const struct Rep_Report *Report)
2016-10-03 09:40:10 +02:00
{
extern const char *Txt_Report_of_use_of_PLATFORM;
2016-10-04 02:32:25 +02:00
extern const char *Txt_User[Usr_NUM_SEXS];
extern const char *Txt_Date;
extern const char *Txt_Permalink;
2016-10-03 09:40:10 +02:00
2017-06-12 14:16:33 +02:00
/***** Start header *****/
2016-10-04 01:16:52 +02:00
fprintf (Gbl.F.Rep,"<header>");
2016-10-03 09:40:10 +02:00
/***** Main title *****/
sprintf (Gbl.Title,Txt_Report_of_use_of_PLATFORM,Cfg_PLATFORM_SHORT_NAME);
2016-10-04 02:32:25 +02:00
fprintf (Gbl.F.Rep,"<h1>%s</h1>"
"<ul>",
Gbl.Title);
2016-10-03 09:40:10 +02:00
2016-10-04 02:32:25 +02:00
/***** User *****/
fprintf (Gbl.F.Rep,"<li>%s: <strong>%s</strong></li>",
Txt_User[Gbl.Usrs.Me.UsrDat.Sex],
Gbl.Usrs.Me.UsrDat.FullName);
/***** Date-time *****/
fprintf (Gbl.F.Rep,"<li>%s: %s %s UTC</li>",
Txt_Date,
2016-10-09 00:05:56 +02:00
Report->CurrentTimeUTC.StrDate,
Report->CurrentTimeUTC.StrTime);
2016-10-03 22:41:45 +02:00
/***** Permalink *****/
2016-10-04 23:57:32 +02:00
fprintf (Gbl.F.Rep,"<li>%s: "
"<a href=\"%s\" target=\"_blank\""
" style=\"text-decoration:none;\">"
"%s"
"</a>"
"</li>",
2016-10-04 02:32:25 +02:00
Txt_Permalink,
2016-10-09 00:05:56 +02:00
Report->Permalink,Report->Permalink);
2016-10-03 09:40:10 +02:00
2017-06-12 14:16:33 +02:00
/***** End header *****/
2016-10-04 23:57:32 +02:00
fprintf (Gbl.F.Rep,"</ul>"
"</header>\n");
2016-10-03 09:40:10 +02:00
}
/*****************************************************************************/
/************* Write section for platform in user's usage report *************/
/*****************************************************************************/
static void Rep_WriteSectionPlatform (void)
{
extern const char *Txt_Teaching_platform;
extern const char *Txt_Name;
extern const char *Txt_TAGLINE;
extern const char *Txt_URL;
2017-06-12 14:16:33 +02:00
/***** Start section *****/
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"<section>"
2016-10-04 23:57:32 +02:00
"<h3>%s</h3>"
2016-10-03 09:40:10 +02:00
"<ul>",
Txt_Teaching_platform);
/***** Platform name *****/
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"<li>%s: %s, %s</li>",
2016-10-03 09:40:10 +02:00
Txt_Name,
Cfg_PLATFORM_FULL_NAME,Txt_TAGLINE);
/***** Server URL *****/
2016-10-04 23:57:32 +02:00
fprintf (Gbl.F.Rep,"<li>%s: "
"<a href=\"%s\" target=\"_blank\""
" style=\"text-decoration:none;\">"
"%s"
"</a>"
"</li>",
2016-10-03 09:40:10 +02:00
Txt_URL,Cfg_URL_SWAD_SERVER,Cfg_URL_SWAD_SERVER);
2017-06-12 14:16:33 +02:00
/***** End section *****/
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"</ul>"
2016-10-03 10:18:48 +02:00
"</section>\n");
2016-10-03 09:40:10 +02:00
}
2016-10-02 19:42:47 +02:00
2016-10-03 09:44:32 +02:00
/*****************************************************************************/
/*********** Write section for user's info in user's usage report ************/
/*****************************************************************************/
static void Rep_WriteSectionUsrInfo (void)
{
extern const char *Txt_Personal_information;
extern const char *Txt_Name;
extern const char *Txt_Email;
extern const char *Txt_Country;
extern const char *Txt_Institution;
2017-03-08 14:12:33 +01:00
char CtyName[Hie_MAX_BYTES_FULL_NAME + 1];
2016-10-28 10:03:37 +02:00
struct Instit Ins;
2016-10-03 09:44:32 +02:00
2017-06-12 14:16:33 +02:00
/***** Start section *****/
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"<section>"
2016-10-04 23:57:32 +02:00
"<h3>%s</h3>"
2016-10-03 09:44:32 +02:00
"<ul>",
Txt_Personal_information);
/***** User's name *****/
2016-10-04 02:32:25 +02:00
fprintf (Gbl.F.Rep,"<li>%s: %s</li>",
2016-10-03 09:44:32 +02:00
Txt_Name,
Gbl.Usrs.Me.UsrDat.FullName);
2016-11-16 23:19:52 +01:00
/***** User's email *****/
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"<li>%s: %s</li>",
2016-10-03 09:44:32 +02:00
Txt_Email,
Gbl.Usrs.Me.UsrDat.Email);
/***** User's country *****/
Cty_GetCountryName (Gbl.Usrs.Me.UsrDat.CtyCod,CtyName);
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"<li>%s: %s</li>",
2016-10-03 09:44:32 +02:00
Txt_Country,
CtyName);
/***** User's institution *****/
Ins.InsCod = Gbl.Usrs.Me.UsrDat.InsCod;
Ins_GetDataOfInstitutionByCod (&Ins,Ins_GET_BASIC_DATA);
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"<li>%s: %s</li>",
2016-10-03 09:44:32 +02:00
Txt_Institution,
Ins.FullName);
2017-06-12 14:16:33 +02:00
/***** End section *****/
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"</ul>"
2016-10-03 10:18:48 +02:00
"</section>\n");
2016-10-03 09:44:32 +02:00
}
2016-10-03 09:59:57 +02:00
/*****************************************************************************/
/********* Write section for user's figures in user's usage report ***********/
/*****************************************************************************/
2016-10-09 00:20:03 +02:00
static void Rep_WriteSectionUsrFigures (const struct Rep_Report *Report)
2016-10-03 09:59:57 +02:00
{
extern const char *Txt_Figures;
extern const char *Txt_TIME_Since;
extern const char *Txt_TIME_until;
extern const char *Txt_day;
extern const char *Txt_days;
extern const char *Txt_Clicks;
extern const char *Txt_Files_uploaded;
extern const char *Txt_file;
extern const char *Txt_files;
extern const char *Txt_public_FILES;
extern const char *Txt_Downloads;
extern const char *Txt_download;
extern const char *Txt_downloads;
extern const char *Txt_Forum_posts;
extern const char *Txt_post;
extern const char *Txt_posts;
extern const char *Txt_Messages_sent;
extern const char *Txt_message;
extern const char *Txt_messages;
unsigned NumFiles;
unsigned NumPublicFiles;
2017-06-12 14:16:33 +02:00
/***** Start section *****/
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"<section>"
2016-10-04 23:57:32 +02:00
"<h3>%s</h3>"
2016-10-03 09:59:57 +02:00
"<ul>",
Txt_Figures);
/***** Time since first click until now *****/
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"<li>%s ",Txt_TIME_Since);
2016-10-09 00:20:03 +02:00
if (Report->UsrFigures.FirstClickTimeUTC)
2016-10-03 09:59:57 +02:00
{
2016-10-09 00:20:03 +02:00
fprintf (Gbl.F.Rep,"%04d-%02d-%02d %02d:%02d:%02d UTC",
1900 + Report->tm_FirstClickTime.tm_year, // year
1 + Report->tm_FirstClickTime.tm_mon, // month
Report->tm_FirstClickTime.tm_mday, // day of the month
Report->tm_FirstClickTime.tm_hour, // hours
Report->tm_FirstClickTime.tm_min, // minutes
Report->tm_FirstClickTime.tm_sec); // seconds
if (Report->CurrentTimeUTC.StrDate[0])
fprintf (Gbl.F.Rep," %s %s %s UTC",
Txt_TIME_until,
Report->CurrentTimeUTC.StrDate,
Report->CurrentTimeUTC.StrTime);
if (Report->UsrFigures.NumDays > 0)
fprintf (Gbl.F.Rep," (%d %s)",
Report->UsrFigures.NumDays,
(Report->UsrFigures.NumDays == 1) ? Txt_day :
Txt_days);
2016-10-03 09:59:57 +02:00
}
else // Time of first click is unknown
{
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"?");
2016-10-09 00:20:03 +02:00
if (Report->CurrentTimeUTC.StrDate[0])
2016-10-04 00:22:59 +02:00
fprintf (Gbl.F.Rep," - %s %s UTC",
2016-10-09 00:20:03 +02:00
Report->CurrentTimeUTC.StrDate,
Report->CurrentTimeUTC.StrTime);
2016-10-03 09:59:57 +02:00
}
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"</li>");
2016-10-03 09:59:57 +02:00
/***** Number of clicks *****/
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"<li>%s: ",Txt_Clicks);
2016-10-09 00:20:03 +02:00
if (Report->UsrFigures.NumClicks >= 0)
2016-10-03 09:59:57 +02:00
{
2016-10-09 00:20:03 +02:00
fprintf (Gbl.F.Rep,"%ld",Report->UsrFigures.NumClicks);
if (Report->UsrFigures.NumDays > 0)
2016-10-03 09:59:57 +02:00
{
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep," (");
Str_WriteFloatNum (Gbl.F.Rep,
2016-10-09 00:20:03 +02:00
(float) Report->UsrFigures.NumClicks /
(float) Report->UsrFigures.NumDays);
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep," / %s)",Txt_day);
2016-10-03 09:59:57 +02:00
}
}
else // Number of clicks is unknown
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"?");
fprintf (Gbl.F.Rep,"</li>");
2016-10-03 09:59:57 +02:00
/***** Number of files currently published *****/
if ((NumFiles = Brw_GetNumFilesUsr (Gbl.Usrs.Me.UsrDat.UsrCod)))
NumPublicFiles = Brw_GetNumPublicFilesUsr (Gbl.Usrs.Me.UsrDat.UsrCod);
else
NumPublicFiles = 0;
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"<li>"
2016-10-03 09:59:57 +02:00
"%s: %u %s (%u %s)"
"</li>",
Txt_Files_uploaded,
NumFiles,
(NumFiles == 1) ? Txt_file :
Txt_files,
NumPublicFiles,Txt_public_FILES);
/***** Number of file views *****/
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"<li>%s: ",Txt_Downloads);
2016-10-09 00:20:03 +02:00
if (Report->UsrFigures.NumFileViews >= 0)
2016-10-03 09:59:57 +02:00
{
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"%ld %s",
2016-10-09 00:20:03 +02:00
Report->UsrFigures.NumFileViews,
(Report->UsrFigures.NumFileViews == 1) ? Txt_download :
2016-10-09 01:05:45 +02:00
Txt_downloads);
2016-10-09 00:20:03 +02:00
if (Report->UsrFigures.NumDays > 0)
2016-10-03 09:59:57 +02:00
{
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep," (");
Str_WriteFloatNum (Gbl.F.Rep,
2016-10-09 00:20:03 +02:00
(float) Report->UsrFigures.NumFileViews /
(float) Report->UsrFigures.NumDays);
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep," / %s)",Txt_day);
2016-10-03 09:59:57 +02:00
}
}
else // Number of file views is unknown
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"?");
fprintf (Gbl.F.Rep,"</li>");
2016-10-03 09:59:57 +02:00
/***** Number of posts in forums *****/
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"<li>%s: ",Txt_Forum_posts);
2016-10-09 00:20:03 +02:00
if (Report->UsrFigures.NumForPst >= 0)
2016-10-03 09:59:57 +02:00
{
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"%ld %s",
2016-10-09 00:20:03 +02:00
Report->UsrFigures.NumForPst,
(Report->UsrFigures.NumForPst == 1) ? Txt_post :
2016-10-09 01:05:45 +02:00
Txt_posts);
2016-10-09 00:20:03 +02:00
if (Report->UsrFigures.NumDays > 0)
2016-10-03 09:59:57 +02:00
{
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep," (");
Str_WriteFloatNum (Gbl.F.Rep,
2016-10-09 00:20:03 +02:00
(float) Report->UsrFigures.NumForPst /
(float) Report->UsrFigures.NumDays);
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep," / %s)",Txt_day);
2016-10-03 09:59:57 +02:00
}
}
else // Number of forum posts is unknown
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"?");
fprintf (Gbl.F.Rep,"</li>");
2016-10-03 09:59:57 +02:00
/***** Number of messages sent *****/
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"<li>%s: ",Txt_Messages_sent);
2016-10-09 00:20:03 +02:00
if (Report->UsrFigures.NumMsgSnt >= 0)
2016-10-03 09:59:57 +02:00
{
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"%ld %s",
2016-10-09 00:20:03 +02:00
Report->UsrFigures.NumMsgSnt,
(Report->UsrFigures.NumMsgSnt == 1) ? Txt_message :
Txt_messages);
if (Report->UsrFigures.NumDays > 0)
2016-10-03 09:59:57 +02:00
{
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep," (");
Str_WriteFloatNum (Gbl.F.Rep,
2016-10-09 00:20:03 +02:00
(float) Report->UsrFigures.NumMsgSnt /
(float) Report->UsrFigures.NumDays);
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep," / %s)",Txt_day);
2016-10-03 09:59:57 +02:00
}
}
else // Number of messages sent is unknown
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"?");
fprintf (Gbl.F.Rep,"</li>");
2016-10-03 09:59:57 +02:00
2017-06-12 14:16:33 +02:00
/***** End section *****/
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"</ul>"
2016-10-03 10:18:48 +02:00
"</section>\n");
2016-10-03 09:59:57 +02:00
}
2016-10-03 10:07:49 +02:00
/*****************************************************************************/
/******** Write section for user's global hits in user's usage report ********/
/*****************************************************************************/
2016-10-09 00:49:23 +02:00
static void Rep_WriteSectionGlobalHits (struct Rep_Report *Report)
2016-10-03 10:07:49 +02:00
{
2016-10-08 20:37:01 +02:00
extern const char *Txt_Hits_per_year;
2017-06-12 14:16:33 +02:00
/***** Start section *****/
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"<section>"
2016-10-04 23:57:32 +02:00
"<h3>%s</h3>",
2016-10-08 20:37:01 +02:00
Txt_Hits_per_year);
2016-10-03 10:07:49 +02:00
/***** Global (in any course) hits per year *****/
2016-10-09 00:49:23 +02:00
Report->MaxHitsPerYear = 0; // MaxHitsPerYear not passed as an argument but computed inside the function
2016-10-03 10:07:49 +02:00
Rep_ShowMyHitsPerYear (true,-1L, // Any course
2017-05-18 19:13:41 +02:00
Rol_UNK, // Any role
2016-10-09 00:49:23 +02:00
Report);
2016-10-03 10:07:49 +02:00
2017-06-12 14:16:33 +02:00
/***** End section *****/
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"</section>\n");
2016-10-03 10:18:48 +02:00
}
2016-10-08 20:25:26 +02:00
/*****************************************************************************/
/******** Write section for user's global hits in user's usage report ********/
/*****************************************************************************/
2016-10-09 01:03:30 +02:00
static void Rep_WriteSectionHitsPerAction (struct Rep_Report *Report)
2016-10-08 20:25:26 +02:00
{
extern struct Act_Actions Act_Actions[Act_NUM_ACTIONS];
2017-01-28 15:58:46 +01:00
extern Act_Action_t Act_FromActCodToAction[1 + Act_MAX_ACTION_COD];
2016-10-08 20:41:25 +02:00
extern const char *Txt_Hits_per_action;
2016-12-28 17:22:25 +01:00
extern const char *Txt_TABS_TXT[Tab_NUM_TABS];
2016-10-08 20:51:39 +02:00
extern const char *Txt_Other_actions;
2016-10-08 20:25:26 +02:00
char Query[512];
MYSQL_RES *mysql_res;
MYSQL_ROW row;
unsigned long NumRows;
unsigned long NumRow;
long ActCod;
2016-10-09 01:41:36 +02:00
Act_Action_t Action;
2016-10-20 22:58:11 +02:00
Act_Action_t SuperAction;
2016-10-12 14:02:56 +02:00
Tab_Tab_t Tab;
2017-03-07 01:56:41 +01:00
char ActTxt[Act_MAX_BYTES_ACTION_TXT + 1];
2016-10-08 21:19:47 +02:00
unsigned long NumClicks;
2016-10-08 20:25:26 +02:00
2017-06-12 14:16:33 +02:00
/***** Start section *****/
2016-10-08 20:25:26 +02:00
fprintf (Gbl.F.Rep,"<section>"
"<h3>%s</h3>",
2016-10-08 20:41:25 +02:00
Txt_Hits_per_action);
2016-10-08 20:25:26 +02:00
/***** Make the query *****/
sprintf (Query,"SELECT SQL_NO_CACHE ActCod,COUNT(*) AS N FROM log_full"
2017-03-24 01:09:27 +01:00
" WHERE ClickTime>=FROM_UNIXTIME(%ld) AND UsrCod=%ld"
2016-10-08 20:25:26 +02:00
" GROUP BY ActCod ORDER BY N DESC LIMIT %u",
2016-10-09 01:03:30 +02:00
(long) Report->UsrFigures.FirstClickTimeUTC,Gbl.Usrs.Me.UsrDat.UsrCod,
2016-10-08 20:25:26 +02:00
Rep_MAX_ACTIONS);
NumRows = DB_QuerySELECT (Query,&mysql_res,"can not get clicks");
/***** Compute maximum number of hits per action *****/
2016-10-09 01:03:30 +02:00
Rep_ComputeMaxAndTotalHits (&Report->Hits,NumRows,mysql_res,1);
2016-10-08 20:25:26 +02:00
mysql_data_seek (mysql_res,0);
/***** Write rows *****/
for (NumRow = 1, NumClicks = 0;
NumRow <= NumRows;
NumRow++)
{
row = mysql_fetch_row (mysql_res);
/* Get the action (row[0]) */
ActCod = Str_ConvertStrCodToLongCod (row[0]);
2016-10-08 21:19:47 +02:00
/* Get number of hits (row[1]) */
2016-10-09 01:03:30 +02:00
if (sscanf (row[1],"%lu",&Report->Hits.Num) != 1)
Report->Hits.Num = 0;
NumClicks += Report->Hits.Num;
2016-10-08 20:25:26 +02:00
/* Draw bar proportional to number of hits */
2016-10-09 01:03:30 +02:00
Rep_DrawBarNumHits (Report->Hits.Num,Report->Hits.Max,Rep_MAX_BAR_WIDTH);
2016-10-08 20:25:26 +02:00
/* Write action text */
fprintf (Gbl.F.Rep,"&nbsp;");
if (ActCod >= 0)
{
2016-10-09 01:41:36 +02:00
Action = Act_FromActCodToAction[ActCod];
if (Action >= 0)
{
2016-10-20 22:58:11 +02:00
SuperAction = Act_Actions[Action].SuperAction;
Tab = Act_Actions[SuperAction].Tab;
2016-12-28 17:22:25 +01:00
if (Txt_TABS_TXT[Tab])
fprintf (Gbl.F.Rep,"%s &gt; ",Txt_TABS_TXT[Tab]);
2016-10-09 01:41:36 +02:00
}
2016-10-08 20:25:26 +02:00
fprintf (Gbl.F.Rep,"%s",Act_GetActionTextFromDB (ActCod,ActTxt));
}
else
fprintf (Gbl.F.Rep,"?");
fprintf (Gbl.F.Rep,"<br />");
}
/***** Draw bar for the rest of the clicks *****/
2016-10-09 01:03:30 +02:00
if ((unsigned long) Report->UsrFigures.NumClicks > NumClicks)
2016-10-08 20:51:39 +02:00
fprintf (Gbl.F.Rep,"%ld&nbsp;%s<br />",
2016-10-09 01:03:30 +02:00
Report->UsrFigures.NumClicks - NumClicks,
2016-10-08 20:51:39 +02:00
Txt_Other_actions);
2016-10-08 20:25:26 +02:00
/***** Free structure that stores the query result *****/
DB_FreeMySQLResult (&mysql_res);
2017-06-12 14:16:33 +02:00
/***** End section *****/
2016-10-08 20:25:26 +02:00
fprintf (Gbl.F.Rep,"</section>\n");
}
2016-10-03 10:18:48 +02:00
/*****************************************************************************/
/****** Write section for user's current courses in user's usage report ******/
/*****************************************************************************/
2016-10-09 01:03:30 +02:00
static void Rep_WriteSectionCurrentCourses (struct Rep_Report *Report)
2016-10-03 10:18:48 +02:00
{
extern const char *Txt_Courses;
Rol_Role_t Role;
2017-06-12 14:16:33 +02:00
/***** Start section *****/
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"<section>"
2016-10-04 23:57:32 +02:00
"<h3>%s",
2016-10-03 10:18:48 +02:00
Txt_Courses);
2016-10-09 00:37:58 +02:00
if (Report->CurrentTimeUTC.StrDate[0])
fprintf (Gbl.F.Rep," (%s)",Report->CurrentTimeUTC.StrDate);
2016-10-04 23:57:32 +02:00
fprintf (Gbl.F.Rep,"</h3>"
2016-10-03 10:18:48 +02:00
"<ul>");
/***** Number of courses in which the user is student/teacher *****/
2017-05-18 19:13:41 +02:00
for (Role = Rol_STD;
Role <= Rol_TCH;
2016-10-03 10:18:48 +02:00
Role++)
/* List my courses with this role */
2016-10-09 00:41:09 +02:00
Rep_GetAndWriteMyCurrentCrss (Role,Report);
2016-10-03 10:18:48 +02:00
2017-06-12 14:16:33 +02:00
/***** End section *****/
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"</ul>"
2016-10-03 10:18:48 +02:00
"</section>\n");
2016-10-03 10:07:49 +02:00
}
2016-10-03 11:38:13 +02:00
/*****************************************************************************/
/***** Write section for user's historic courses in user's usage report ******/
/*****************************************************************************/
2016-10-09 01:03:30 +02:00
static void Rep_WriteSectionHistoricCourses (struct Rep_Report *Report)
2016-10-03 11:38:13 +02:00
{
extern const char *Txt_Courses;
extern const char *Txt_historical_log;
2016-10-03 13:29:05 +02:00
extern const char *Txt_Only_courses_with_more_than_X_clicks_are_shown;
2016-10-03 11:38:13 +02:00
Rol_Role_t Role;
2017-06-12 14:16:33 +02:00
/***** Start section *****/
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"<section>"
2016-10-04 23:57:32 +02:00
"<h3>%s (%s)</h3>",
2016-10-03 11:38:13 +02:00
Txt_Courses,Txt_historical_log);
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,Txt_Only_courses_with_more_than_X_clicks_are_shown,
2016-10-03 13:29:05 +02:00
Rep_MIN_CLICKS_CRS);
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"<ul>");
2016-10-03 11:38:13 +02:00
2016-10-07 02:23:24 +02:00
/********* Historic clicks of a user without course selected ***********/
2016-10-09 00:49:23 +02:00
Rep_GetAndWriteMyHistoricClicsWithoutCrs (Report);
2016-10-07 02:23:24 +02:00
/***** Historic courses in which the user clicked as student/teacher *****/
2017-05-18 19:13:41 +02:00
for (Role = Rol_STD;
Role <= Rol_TCH;
2016-10-03 11:38:13 +02:00
Role++)
/* List my courses with this role */
2016-10-09 00:49:23 +02:00
Rep_GetAndWriteMyHistoricCrss (Role,Report);
2016-10-03 11:38:13 +02:00
2017-06-12 14:16:33 +02:00
/***** End section *****/
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"</ul>"
2016-10-03 11:38:13 +02:00
"</section>\n");
}
2016-10-02 19:42:47 +02:00
/*****************************************************************************/
/************ Get the maximum number of hits per course-year-role ************/
/*****************************************************************************/
// Return the maximum number of hits per year
2016-10-09 01:03:30 +02:00
static void Rep_GetMaxHitsPerYear (struct Rep_Report *Report)
2016-10-02 19:42:47 +02:00
{
2016-10-07 09:57:28 +02:00
char Query[1024];
2016-10-02 19:42:47 +02:00
MYSQL_RES *mysql_res;
MYSQL_ROW row;
2016-10-07 09:57:28 +02:00
sprintf (Query,"SELECT MAX(N) FROM ("
// Clicks without course selected ---------------------------
"SELECT "
2017-03-24 01:09:27 +01:00
"-1 AS CrsCod,"
2016-10-07 09:57:28 +02:00
"YEAR(CONVERT_TZ(ClickTime,@@session.time_zone,'UTC')) AS Year,"
2017-03-24 01:09:27 +01:00
"%u AS Role,"
2016-10-07 09:57:28 +02:00
"COUNT(*) AS N"
" FROM log_full"
2017-03-24 01:09:27 +01:00
" WHERE ClickTime>=FROM_UNIXTIME(%ld)"
" AND UsrCod=%ld"
" AND CrsCod<=0"
2016-10-07 09:57:28 +02:00
" GROUP BY Year"
// ----------------------------------------------------------
" UNION "
2017-05-22 20:37:46 +02:00
// Clicks as student, non-editing teacher or teacher in courses
2016-10-07 09:57:28 +02:00
"SELECT "
2016-10-02 19:42:47 +02:00
"CrsCod,"
2016-10-02 23:26:59 +02:00
"YEAR(CONVERT_TZ(ClickTime,@@session.time_zone,'UTC')) AS Year,"
2016-10-02 19:42:47 +02:00
"Role,"
"COUNT(*) AS N"
" FROM log_full"
2017-03-24 01:09:27 +01:00
" WHERE ClickTime>=FROM_UNIXTIME(%ld)"
" AND UsrCod=%ld"
" AND Role>=%u" // Student
" AND Role<=%u" // Teacher
" AND CrsCod>0"
2016-10-07 09:57:28 +02:00
" GROUP BY CrsCod,Year,Role"
// ----------------------------------------------------------
") AS hits_per_crs_year",
2017-05-18 19:13:41 +02:00
(unsigned) Rol_UNK,
2016-10-09 01:03:30 +02:00
(long) Report->UsrFigures.FirstClickTimeUTC,
2016-10-02 23:26:59 +02:00
Gbl.Usrs.Me.UsrDat.UsrCod,
2016-10-09 01:03:30 +02:00
(long) Report->UsrFigures.FirstClickTimeUTC,
2016-10-07 09:57:28 +02:00
Gbl.Usrs.Me.UsrDat.UsrCod,
2017-05-18 19:13:41 +02:00
(unsigned) Rol_STD,
(unsigned) Rol_TCH);
2016-10-02 19:42:47 +02:00
DB_QuerySELECT (Query,&mysql_res,"can not get last question index");
/***** Get number of users *****/
2016-10-09 01:03:30 +02:00
Report->MaxHitsPerYear = 0;
2016-10-02 19:42:47 +02:00
row = mysql_fetch_row (mysql_res);
if (row[0]) // There are questions
2016-10-09 01:03:30 +02:00
if (sscanf (row[0],"%lu",&Report->MaxHitsPerYear) != 1)
2016-10-02 19:42:47 +02:00
Lay_ShowErrorAndExit ("Error when getting maximum hits.");
2016-10-08 20:25:26 +02:00
/***** Free structure that stores the query result *****/
DB_FreeMySQLResult (&mysql_res);
2016-10-02 19:42:47 +02:00
}
2016-09-11 21:25:14 +02:00
/*****************************************************************************/
2016-10-07 02:23:24 +02:00
/************************* Write my current courses **************************/
2016-09-11 21:25:14 +02:00
/*****************************************************************************/
2016-10-07 02:23:24 +02:00
static void Rep_GetAndWriteMyCurrentCrss (Rol_Role_t Role,
2016-10-09 01:03:30 +02:00
struct Rep_Report *Report)
2016-09-11 21:25:14 +02:00
{
2016-10-02 19:42:47 +02:00
extern const char *Txt_USER_in_COURSE;
extern const char *Txt_ROLES_SINGUL_Abc[Rol_NUM_ROLES][Usr_NUM_SEXS];
extern const char *Txt_course;
extern const char *Txt_courses;
extern const char *Txt_teachers_ABBREVIATION;
extern const char *Txt_students_ABBREVIATION;
2016-10-03 13:29:05 +02:00
char Query[512];
2016-10-02 19:42:47 +02:00
MYSQL_RES *mysql_res;
MYSQL_ROW row;
unsigned NumCrss;
unsigned NumCrs;
long CrsCod;
2016-10-07 02:23:24 +02:00
NumCrss = Usr_GetNumCrssOfUsrWithARole (Gbl.Usrs.Me.UsrDat.UsrCod,Role);
2016-10-02 19:42:47 +02:00
sprintf (Gbl.Title,Txt_USER_in_COURSE,Txt_ROLES_SINGUL_Abc[Role][Gbl.Usrs.Me.UsrDat.Sex]);
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"<li>%s %u %s",
2016-10-02 19:42:47 +02:00
Gbl.Title,
NumCrss,
NumCrss == 1 ? Txt_course :
Txt_courses);
if (NumCrss)
{
2016-10-04 23:57:32 +02:00
fprintf (Gbl.F.Rep," (%u %s / %u %s):",
2017-05-18 19:13:41 +02:00
Usr_GetNumUsrsInCrssOfAUsr (Gbl.Usrs.Me.UsrDat.UsrCod,Role,Rol_TCH),
2016-10-02 19:42:47 +02:00
Txt_teachers_ABBREVIATION,
2017-05-18 19:13:41 +02:00
Usr_GetNumUsrsInCrssOfAUsr (Gbl.Usrs.Me.UsrDat.UsrCod,Role,Rol_STD),
2016-10-02 19:42:47 +02:00
Txt_students_ABBREVIATION);
/***** Get courses of a user from database *****/
2016-10-07 02:23:24 +02:00
sprintf (Query,"SELECT crs_usr.CrsCod,log_full.CrsCod,COUNT(*) AS N"
" FROM crs_usr LEFT JOIN log_full ON"
" (crs_usr.CrsCod=log_full.CrsCod"
" AND crs_usr.UsrCod=log_full.UsrCod"
" AND crs_usr.Role=log_full.Role)"
2017-03-24 01:09:27 +01:00
" WHERE crs_usr.UsrCod=%ld"
" AND crs_usr.Role=%u"
2016-10-07 02:23:24 +02:00
" GROUP BY crs_usr.CrsCod"
" ORDER BY N DESC,log_full.CrsCod DESC",
Gbl.Usrs.Me.UsrDat.UsrCod,(unsigned) Role);
2016-10-02 19:42:47 +02:00
/***** List the courses (one row per course) *****/
if ((NumCrss = (unsigned) DB_QuerySELECT (Query,&mysql_res,"can not get courses of a user")))
{
/* Heading row */
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"<ol>");
2016-10-02 19:42:47 +02:00
/* Write courses */
for (NumCrs = 1;
NumCrs <= NumCrss;
NumCrs++)
{
/* Get next course */
row = mysql_fetch_row (mysql_res);
/* Get course code (row[0]) */
CrsCod = Str_ConvertStrCodToLongCod (row[0]);
/* Write data of this course */
2016-10-09 00:49:23 +02:00
Rep_WriteRowCrsData (CrsCod,Role,Report,
2016-10-07 02:23:24 +02:00
true); // Write number of users in course
2016-10-02 19:42:47 +02:00
}
/* End table */
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"</ol>");
2016-10-02 19:42:47 +02:00
}
/***** Free structure that stores the query result *****/
DB_FreeMySQLResult (&mysql_res);
}
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"</li>");
2016-10-02 19:42:47 +02:00
}
/*****************************************************************************/
2016-10-07 02:23:24 +02:00
/************* Write my historic clicks without course selected **************/
2016-10-02 19:42:47 +02:00
/*****************************************************************************/
2016-10-09 01:03:30 +02:00
static void Rep_GetAndWriteMyHistoricClicsWithoutCrs (struct Rep_Report *Report)
2016-10-07 02:23:24 +02:00
{
extern const char *Txt_Hits_without_course_selected;
/***** Heading row *****/
fprintf (Gbl.F.Rep,"<li>%s:"
"<ol>",
Txt_Hits_without_course_selected);
/***** Historic clicks *****/
Rep_WriteRowCrsData (-1L,
2017-05-18 19:13:41 +02:00
Rol_UNK, // Role does not matter
2016-10-09 00:49:23 +02:00
Report,
2016-10-07 02:23:24 +02:00
false); // Do not write number of users in course
2017-06-12 14:16:33 +02:00
/***** End list *****/
2016-10-07 02:23:24 +02:00
fprintf (Gbl.F.Rep,"</ol>"
"</li>");
}
/*****************************************************************************/
/********************** Write historic courses of a user *********************/
/*****************************************************************************/
static void Rep_GetAndWriteMyHistoricCrss (Rol_Role_t Role,
2016-10-09 01:03:30 +02:00
struct Rep_Report *Report)
2016-10-02 19:42:47 +02:00
{
2016-10-02 22:54:30 +02:00
extern const char *Txt_Hits_as_a_USER;
2016-10-02 19:42:47 +02:00
extern const char *Txt_ROLES_SINGUL_abc[Rol_NUM_ROLES][Usr_NUM_SEXS];
2016-10-03 13:29:05 +02:00
char Query[256];
2016-09-11 21:25:14 +02:00
MYSQL_RES *mysql_res;
MYSQL_ROW row;
unsigned NumCrss;
unsigned NumCrs;
2016-10-02 19:42:47 +02:00
long CrsCod;
2016-09-11 21:25:14 +02:00
2016-10-07 02:23:24 +02:00
/***** Get historic courses of a user from log *****/
2016-10-03 13:29:05 +02:00
sprintf (Query,"SELECT CrsCod,COUNT(*) AS N"
2016-10-02 19:42:47 +02:00
" FROM log_full"
2017-03-24 01:09:27 +01:00
" WHERE UsrCod=%ld AND Role=%u AND CrsCod>0"
2016-10-03 13:29:05 +02:00
" GROUP BY CrsCod"
2017-03-24 01:09:27 +01:00
" HAVING N>%u"
2016-10-03 13:29:05 +02:00
" ORDER BY N DESC",
2016-10-07 02:23:24 +02:00
Gbl.Usrs.Me.UsrDat.UsrCod,(unsigned) Role,
2016-10-03 13:29:05 +02:00
Rep_MIN_CLICKS_CRS);
2016-09-11 21:25:14 +02:00
/***** List the courses (one row per course) *****/
if ((NumCrss = (unsigned) DB_QuerySELECT (Query,&mysql_res,"can not get courses of a user")))
{
/* Heading row */
2016-10-02 22:54:30 +02:00
sprintf (Gbl.Title,Txt_Hits_as_a_USER,
2016-10-02 19:42:47 +02:00
Txt_ROLES_SINGUL_abc[Role][Gbl.Usrs.Me.UsrDat.Sex]);
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"<li>%s:"
2016-10-02 19:42:47 +02:00
"<ol>",
Gbl.Title);
2016-09-11 21:25:14 +02:00
/* Write courses */
for (NumCrs = 1;
NumCrs <= NumCrss;
NumCrs++)
{
/* Get next course */
row = mysql_fetch_row (mysql_res);
2016-10-02 19:42:47 +02:00
/* Get course code (row[0]) */
CrsCod = Str_ConvertStrCodToLongCod (row[0]);
2016-09-11 21:25:14 +02:00
/* Write data of this course */
2016-10-09 00:49:23 +02:00
Rep_WriteRowCrsData (CrsCod,Role,Report,
2016-10-07 02:23:24 +02:00
false); // Do not write number of users in course
}
2016-09-11 21:25:14 +02:00
2017-06-12 14:16:33 +02:00
/* End list */
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"</ol>"
2016-10-02 19:42:47 +02:00
"</li>");
2016-09-11 21:25:14 +02:00
}
/***** Free structure that stores the query result *****/
DB_FreeMySQLResult (&mysql_res);
}
/*****************************************************************************/
/************** Write the data of a course (result of a query) ***************/
/*****************************************************************************/
2016-10-02 19:42:47 +02:00
static void Rep_WriteRowCrsData (long CrsCod,Rol_Role_t Role,
2016-10-09 01:03:30 +02:00
struct Rep_Report *Report,
2016-10-07 02:23:24 +02:00
bool WriteNumUsrs)
2016-09-11 21:25:14 +02:00
{
2017-01-28 15:58:46 +01:00
extern const char *Txt_YEAR_OF_DEGREE[1 + Deg_MAX_YEARS_PER_DEGREE];
2016-09-11 21:25:14 +02:00
extern const char *Txt_teachers_ABBREVIATION;
extern const char *Txt_students_ABBREVIATION;
2016-10-02 20:03:21 +02:00
extern const char *Txt_unknown_removed_course;
2016-10-02 19:55:12 +02:00
extern const char *Txt_no_course_selected;
2016-10-02 19:42:47 +02:00
struct Course Crs;
2016-09-11 21:25:14 +02:00
struct Degree Deg;
2016-10-02 19:42:47 +02:00
/***** Get course data *****/
Crs.CrsCod = CrsCod;
Crs_GetDataOfCourseByCod (&Crs);
/***** Get degree data *****/
Deg.DegCod = Crs.DegCod;
Deg_GetDataOfDegreeByCod (&Deg);
2016-09-11 21:25:14 +02:00
/***** Start row *****/
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"<li>");
2016-09-11 21:25:14 +02:00
2016-10-02 19:42:47 +02:00
if (CrsCod > 0) // CrsCod > 0 in log ==> course selected
{
if (Crs.CrsCod > 0) // Course exists
{
/***** Write course full name *****/
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"<strong>%s</strong> -",Crs.FullName);
2016-09-11 21:45:29 +02:00
2016-10-02 19:42:47 +02:00
/***** Write year *****/
if (Crs.Year)
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep," %s",Txt_YEAR_OF_DEGREE[Crs.Year]);
2016-09-11 21:45:29 +02:00
2016-10-02 19:42:47 +02:00
/***** Write degree full name *****/
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep," %s",Deg.FullName);
2016-09-11 21:25:14 +02:00
2016-10-02 19:42:47 +02:00
/***** Write number of teachers / students in course *****/
2016-10-07 02:23:24 +02:00
if (WriteNumUsrs)
fprintf (Gbl.F.Rep," (%u %s / %u %s)",
2017-05-22 20:37:46 +02:00
Usr_GetNumUsrsInCrs (Rol_NET,Crs.CrsCod) +
Usr_GetNumUsrsInCrs (Rol_TCH,Crs.CrsCod),
Txt_teachers_ABBREVIATION,
Usr_GetNumUsrsInCrs (Rol_STD,Crs.CrsCod),
Txt_students_ABBREVIATION);
2016-10-02 19:42:47 +02:00
}
else
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"(%s)",Txt_unknown_removed_course);
2016-10-02 19:42:47 +02:00
}
else // CrsCod <= 0 in log ==> no course selected
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"(%s)",Txt_no_course_selected);
2016-09-29 01:34:17 +02:00
/***** Write hits per year for this course *****/
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"<br />");
2016-10-09 00:49:23 +02:00
Rep_ShowMyHitsPerYear (false,CrsCod,Role,Report);
2016-09-29 01:34:17 +02:00
2016-10-03 14:12:01 +02:00
fprintf (Gbl.F.Rep,"</li>");
2016-09-11 21:25:14 +02:00
}
2016-09-12 00:50:24 +02:00
2016-10-02 17:56:05 +02:00
/*****************************************************************************/
/********************** Write my hits grouped by years ***********************/
/*****************************************************************************/
2016-10-02 19:42:47 +02:00
static void Rep_ShowMyHitsPerYear (bool AnyCourse,long CrsCod,Rol_Role_t Role,
2016-10-09 01:03:30 +02:00
struct Rep_Report *Report)
2016-10-02 17:56:05 +02:00
{
2016-11-14 10:30:33 +01:00
char Query[1024];
2016-10-02 17:56:05 +02:00
char SubQueryCrs[128];
2016-10-02 19:42:47 +02:00
char SubQueryRol[128];
2016-09-12 17:43:23 +02:00
MYSQL_RES *mysql_res;
MYSQL_ROW row;
unsigned long NumRows;
unsigned long NumRow;
unsigned ReadYear;
unsigned LastYear;
unsigned Year;
/***** Make the query *****/
2016-10-02 19:42:47 +02:00
if (AnyCourse)
SubQueryCrs[0] = '\0';
else
2017-03-24 01:09:27 +01:00
sprintf (SubQueryCrs," AND CrsCod=%ld",CrsCod);
2016-10-02 19:42:47 +02:00
2017-05-18 19:13:41 +02:00
if (Role == Rol_UNK) // Here Rol_UNK means any role
2016-10-02 19:42:47 +02:00
SubQueryRol[0] = '\0';
2016-09-29 01:34:17 +02:00
else
2017-03-24 01:09:27 +01:00
sprintf (SubQueryRol," AND Role=%u",(unsigned) Role);
2016-10-02 17:56:05 +02:00
2016-09-12 17:43:23 +02:00
sprintf (Query,"SELECT SQL_NO_CACHE "
2016-10-02 23:26:59 +02:00
"YEAR(CONVERT_TZ(ClickTime,@@session.time_zone,'UTC')) AS Year,"
2016-09-12 17:43:23 +02:00
"COUNT(*) FROM log_full"
2017-03-24 01:09:27 +01:00
" WHERE ClickTime>=FROM_UNIXTIME(%ld)"
" AND UsrCod=%ld%s%s"
2016-09-12 17:43:23 +02:00
" GROUP BY Year DESC",
2016-10-09 00:49:23 +02:00
(long) Report->UsrFigures.FirstClickTimeUTC,
2016-09-29 01:34:17 +02:00
Gbl.Usrs.Me.UsrDat.UsrCod,
2016-10-02 19:42:47 +02:00
SubQueryCrs,
SubQueryRol);
2016-09-12 17:43:23 +02:00
NumRows = DB_QuerySELECT (Query,&mysql_res,"can not get clicks");
/***** Initialize first year *****/
2016-10-09 00:49:23 +02:00
Gbl.DateRange.DateIni.Date.Year = 1900 + Report->tm_FirstClickTime.tm_year;
2016-09-12 17:43:23 +02:00
/***** Initialize LastYear *****/
LastYear = Gbl.Now.Date.Year;
2016-10-02 17:56:05 +02:00
/***** Set maximum number of hits per year *****/
2016-10-09 00:49:23 +02:00
if (Report->MaxHitsPerYear)
2016-10-02 17:56:05 +02:00
/* Set maximum number of hits per year from parameter */
2016-10-09 01:03:30 +02:00
Report->Hits.Max = Report->MaxHitsPerYear;
2016-09-29 01:48:52 +02:00
else
2016-10-02 17:56:05 +02:00
{
/* Compute maximum number of hits per year */
2016-10-09 01:03:30 +02:00
Rep_ComputeMaxAndTotalHits (&Report->Hits,NumRows,mysql_res,1);
2016-10-02 17:56:05 +02:00
mysql_data_seek (mysql_res,0);
}
2016-09-12 17:43:23 +02:00
/***** Write rows *****/
for (NumRow = 1;
NumRow <= NumRows;
NumRow++)
{
row = mysql_fetch_row (mysql_res);
/* Get the year (in row[0] is the date in YYYY format) */
if (sscanf (row[0],"%04u",&ReadYear) != 1)
Lay_ShowErrorAndExit ("Wrong date.");
/* Get number hits (in row[1]) */
2016-10-09 01:03:30 +02:00
if (sscanf (row[1],"%lu",&Report->Hits.Num) != 1)
Report->Hits.Num = 0;
2016-09-12 17:43:23 +02:00
for (Year = LastYear;
Year >= ReadYear;
Year--)
{
/* Write the year */
2016-10-03 22:41:45 +02:00
fprintf (Gbl.F.Rep,"%04u&nbsp;",Year);
2016-09-12 17:43:23 +02:00
/* Draw bar proportional to number of hits */
2016-11-14 10:30:33 +01:00
Rep_DrawBarNumHits (Year == ReadYear ? Report->Hits.Num :
2016-10-08 21:19:47 +02:00
0,
2016-10-09 01:03:30 +02:00
Report->Hits.Max,Rep_MAX_BAR_WIDTH);
2016-10-08 20:25:26 +02:00
fprintf (Gbl.F.Rep,"<br />");
2016-09-12 17:43:23 +02:00
}
LastYear = Year;
}
2016-10-08 20:25:26 +02:00
/***** Free structure that stores the query result *****/
DB_FreeMySQLResult (&mysql_res);
2016-09-12 17:43:23 +02:00
/***** Finally, show the oldest years without clicks *****/
for (Year = LastYear;
Year >= Gbl.DateRange.DateIni.Date.Year;
Year--)
{
/* Write the year */
2016-10-03 22:41:45 +02:00
fprintf (Gbl.F.Rep,"%04u&nbsp;",Year);
2016-09-12 17:43:23 +02:00
/* Draw bar proportional to number of hits */
2016-10-09 01:03:30 +02:00
Rep_DrawBarNumHits (0,Report->Hits.Max,Rep_MAX_BAR_WIDTH);
2016-10-08 20:25:26 +02:00
fprintf (Gbl.F.Rep,"<br />");
2016-09-12 17:43:23 +02:00
}
}
2016-10-08 21:19:47 +02:00
/*****************************************************************************/
/*************** Compute maximum and total number of hits ********************/
/*****************************************************************************/
static void Rep_ComputeMaxAndTotalHits (struct Rep_Hits *Hits,
unsigned long NumRows,
MYSQL_RES *mysql_res,unsigned Field)
{
unsigned long NumRow;
MYSQL_ROW row;
/***** For each row... *****/
for (NumRow = 1, Hits->Max = 0;
NumRow <= NumRows;
NumRow++)
{
/* Get row */
row = mysql_fetch_row (mysql_res);
/* Get number of hits */
if (sscanf (row[Field],"%lu",&Hits->Num) != 1)
Hits->Num = 0;
/* Update maximum hits */
if (Hits->Num > Hits->Max)
Hits->Max = Hits->Num;
}
}
2016-09-12 01:06:36 +02:00
/*****************************************************************************/
/********************* Draw a bar with the number of hits ********************/
/*****************************************************************************/
2016-10-08 21:19:47 +02:00
static void Rep_DrawBarNumHits (unsigned long HitsNum,unsigned long HitsMax,
2016-09-12 10:02:17 +02:00
unsigned MaxBarWidth)
2016-09-12 01:06:36 +02:00
{
unsigned BarWidth;
2016-09-12 10:02:17 +02:00
unsigned i;
2016-09-12 01:06:36 +02:00
2016-10-08 21:19:47 +02:00
if (HitsNum)
2016-09-12 01:06:36 +02:00
{
/***** Draw bar with a with proportional to the number of hits *****/
2016-10-08 21:19:47 +02:00
BarWidth = (unsigned) ((((float) HitsNum * (float) MaxBarWidth) / (float) HitsMax) + 0.5);
2016-10-04 01:16:52 +02:00
if (BarWidth)
{
fprintf (Gbl.F.Rep,"<strong>");
for (i = 0;
i < BarWidth;
i++)
fprintf (Gbl.F.Rep,Rep_BLOCK);
fprintf (Gbl.F.Rep,"</strong>");
}
2016-09-12 01:06:36 +02:00
/***** Write the number of hits *****/
2016-10-03 22:41:45 +02:00
fprintf (Gbl.F.Rep,"&nbsp;");
2016-10-03 14:12:01 +02:00
Str_WriteFloatNum (Gbl.F.Rep,HitsNum);
2016-09-12 01:06:36 +02:00
}
}
2016-10-06 22:18:33 +02:00
/*****************************************************************************/
/********** Remove all user's usage report of a user from database ***********/
/*****************************************************************************/
void Rep_RemoveUsrUsageReports (long UsrCod)
{
/***** Remove all user's usage report files of a user *****/
Rep_RemoveUsrReportsFiles (UsrCod);
/***** Remove all user's usage reports of a user from database *****/
Rep_RemoveUsrReportsFromDB (UsrCod);
}
/*****************************************************************************/
/********** Remove all user's usage reports of a user from database **********/
/*****************************************************************************/
static void Rep_RemoveUsrReportsFiles (long UsrCod)
{
char Query[128];
MYSQL_RES *mysql_res;
MYSQL_ROW row;
unsigned NumReports;
unsigned NumReport;
2017-01-28 15:58:46 +01:00
char PathUniqueDirReport[PATH_MAX + 1];
2016-10-06 22:18:33 +02:00
/***** Get directories for the reports *****/
sprintf (Query,"SELECT UniqueDirL,UniqueDirR FROM usr_report"
2017-03-24 01:09:27 +01:00
" WHERE UsrCod=%ld",
2016-10-06 22:18:33 +02:00
UsrCod);
NumReports = (unsigned) DB_QuerySELECT (Query,&mysql_res,"can not get user's usage reports");
/***** Remove the reports *****/
for (NumReport = 0;
NumReport < NumReports;
NumReport++)
{
/* Get next report */
row = mysql_fetch_row (mysql_res);
/* Remove report directory and file */
sprintf (PathUniqueDirReport,"%s/%s/%s/%s",
Cfg_PATH_SWAD_PUBLIC,Cfg_FOLDER_REP,row[0],row[1]);
Fil_RemoveTree (PathUniqueDirReport);
}
/***** Free structure that stores the query result *****/
DB_FreeMySQLResult (&mysql_res);
}
/*****************************************************************************/
/********** Remove all user's usage reports of a user from database **********/
/*****************************************************************************/
static void Rep_RemoveUsrReportsFromDB (long UsrCod)
{
char Query[128];
/***** Insert a new user's usage report into database *****/
2017-03-24 01:09:27 +01:00
sprintf (Query,"DELETE FROM usr_report WHERE UsrCod=%ld",UsrCod);
2016-10-06 22:18:33 +02:00
DB_QueryDELETE (Query,"can not remove user's usage reports");
}