// swad_mark.c: marks /* 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. Copyright (C) 1999-2019 Antonio Caņas Vargas This program is free software: you can redistribute it and/or modify it under the terms of the GNU Affero General Public 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 . */ /*****************************************************************************/ /********************************** Headers **********************************/ /*****************************************************************************/ #define _GNU_SOURCE // For asprintf #include // For PATH_MAX #include // For NULL #include // For malloc #include // For asprintf #include // For string functions #include // For unlink #include "swad_database.h" #include "swad_form.h" #include "swad_global.h" #include "swad_ID.h" #include "swad_mark.h" #include "swad_notification.h" #include "swad_parameter.h" #include "swad_string.h" #include "swad_user.h" /*****************************************************************************/ /************** External global variables from others modules ****************/ /*****************************************************************************/ extern struct Globals Gbl; /*****************************************************************************/ /****************************** Internal types *******************************/ /*****************************************************************************/ typedef enum { Brw_HEADER = 0, Brw_FOOTER = 1, } Brw_HeadOrFoot_t; /*****************************************************************************/ /**************************** Internal constants *****************************/ /*****************************************************************************/ const char *Mrk_HeadOrFootStr[2] = // Names of columns in database, so don't change! { "Header", "Footer", }; #define Mrk_MAX_BYTES_IN_CELL_CONTENT 1024 // Cell of a table containing one or several user's IDs /*****************************************************************************/ /*************************** Internal prototypes *****************************/ /*****************************************************************************/ static void Mrk_GetNumRowsHeaderAndFooter (struct MarksProperties *Marks); static void Mrk_ChangeNumRowsHeaderOrFooter (Brw_HeadOrFoot_t HeaderOrFooter); static bool Mrk_CheckIfCellContainsOnlyIDs (const char *CellContent); static bool Mrk_GetUsrMarks (FILE *FileUsrMarks,struct UsrData *UsrDat, const char *PathFileAllMarks, struct MarksProperties *Marks); /*****************************************************************************/ /****************** Add a new entry of marks into database *******************/ /*****************************************************************************/ void Mrk_AddMarksToDB (long FilCod,struct MarksProperties *Marks) { /***** Add file of marks to the database *****/ DB_QueryINSERT ("can not add properties of marks to database", "INSERT INTO marks_properties" " (FilCod,%s,%s)" " VALUES" " (%ld,%u,%u)", Mrk_HeadOrFootStr[Brw_HEADER], Mrk_HeadOrFootStr[Brw_FOOTER], FilCod, Marks->Header, Marks->Footer); } /*****************************************************************************/ /********* Write number of header and footer rows of a file of marks *********/ /*****************************************************************************/ void Mrk_GetAndWriteNumRowsHeaderAndFooter (Brw_FileType_t FileType, const char *PathInTree, const char *FileName) { extern const char *The_ClassFormInBoxNoWrap[The_NUM_THEMES]; extern const char *Txt_TABLE_Header; extern const char *Txt_TABLE_Footer; struct MarksProperties Marks; if (FileType == Brw_IS_FOLDER) fprintf (Gbl.F.Out,"" "", Gbl.RowEvenOdd, Gbl.RowEvenOdd); else // File or link { /***** Get number of rows in header or footer *****/ Mrk_GetNumRowsHeaderAndFooter (&Marks); /***** Write the number of rows of header *****/ fprintf (Gbl.F.Out,"", The_ClassFormInBoxNoWrap[Gbl.Prefs.Theme], Gbl.RowEvenOdd); if (Gbl.Crs.Grps.GrpCod > 0) // Group zone { Frm_StartForm (ActChgNumRowHeaGrp); Grp_PutParamGrpCod (Gbl.Crs.Grps.GrpCod); } else // Course zone Frm_StartForm (ActChgNumRowHeaCrs); fprintf (Gbl.F.Out,"", Txt_TABLE_Header, Mrk_HeadOrFootStr[Brw_HEADER],Marks.Header, Gbl.RowEvenOdd, Gbl.Form.Id); Brw_PutParamsFileBrowser (ActUnk, PathInTree,FileName, FileType,-1L); Frm_EndForm (); fprintf (Gbl.F.Out,""); /***** Write the number of rows of footer *****/ fprintf (Gbl.F.Out,"", The_ClassFormInBoxNoWrap[Gbl.Prefs.Theme], Gbl.RowEvenOdd); if (Gbl.Crs.Grps.GrpCod > 0) // Group zone { Frm_StartForm (ActChgNumRowFooGrp); Grp_PutParamGrpCod (Gbl.Crs.Grps.GrpCod); } else // Course zone Frm_StartForm (ActChgNumRowFooCrs); fprintf (Gbl.F.Out,"", Txt_TABLE_Footer, Mrk_HeadOrFootStr[Brw_FOOTER],Marks.Footer, Gbl.RowEvenOdd, Gbl.Form.Id); Brw_PutParamsFileBrowser (ActUnk, PathInTree,FileName, FileType,-1L); Frm_EndForm (); fprintf (Gbl.F.Out,""); } } /*****************************************************************************/ /******** Get number of rows of header and of footer of a file of marks ******/ /*****************************************************************************/ static void Mrk_GetNumRowsHeaderAndFooter (struct MarksProperties *Marks) { extern const Brw_FileBrowser_t Brw_FileBrowserForDB_files[Brw_NUM_TYPES_FILE_BROWSER]; long Cod = Brw_GetCodForFiles (); MYSQL_RES *mysql_res; MYSQL_ROW row; unsigned long NumRows; /***** Get number of rows of header and footer from database *****/ /* There should be a single file in database. If, due to an error, there is more than one file, get the number of rows of the more recent file. */ NumRows = DB_QuerySELECT (&mysql_res,"can not get the number of rows" " in header and footer", "SELECT marks_properties.%s,marks_properties.%s" " FROM files,marks_properties" " WHERE files.FileBrowser=%u" " AND files.Cod=%ld" " AND files.Path='%s'" " AND files.FilCod=marks_properties.FilCod" " ORDER BY files.FilCod DESC LIMIT 1", // On duplicate entries, get the more recent Mrk_HeadOrFootStr[Brw_HEADER], Mrk_HeadOrFootStr[Brw_FOOTER], (unsigned) Brw_FileBrowserForDB_files[Gbl.FileBrowser.Type], Cod, Gbl.FileBrowser.Priv.FullPathInTree); /***** The result of the query must have only one row *****/ if (NumRows == 1) { /***** Get number of header and footer rows *****/ row = mysql_fetch_row (mysql_res); /* Header (row[0]) */ if (sscanf (row[0],"%u",&(Marks->Header)) != 1) Lay_ShowErrorAndExit ("Wrong number of header rows."); /* Footer (row[1]) */ if (sscanf (row[1],"%u",&(Marks->Footer)) != 1) Lay_ShowErrorAndExit ("Wrong number of footer rows."); } else // Unknown numbers of header and footer rows Marks->Header = Marks->Footer = 0; /***** Free structure that stores the query result *****/ DB_FreeMySQLResult (&mysql_res); } /*****************************************************************************/ /********* Change the number of rows of header of a file of marks ************/ /*****************************************************************************/ void Mrk_ChangeNumRowsHeader (void) { Mrk_ChangeNumRowsHeaderOrFooter (Brw_HEADER); } /*****************************************************************************/ /********** Change the number of rows of footer of a file of marks ***********/ /*****************************************************************************/ void Mrk_ChangeNumRowsFooter (void) { Mrk_ChangeNumRowsHeaderOrFooter (Brw_FOOTER); } /*****************************************************************************/ /***** Change the number of rows of header or footer of a file of marks ******/ /*****************************************************************************/ static void Mrk_ChangeNumRowsHeaderOrFooter (Brw_HeadOrFoot_t HeaderOrFooter) { extern const Brw_FileBrowser_t Brw_FileBrowserForDB_files[Brw_NUM_TYPES_FILE_BROWSER]; extern const char *Txt_The_number_of_rows_is_now_X; char UnsignedStr[10 + 1]; long Cod; unsigned NumRows; /***** Get parameters related to file browser *****/ Brw_GetParAndInitFileBrowser (); /***** Get the number of rows of the header or footer of the table of marks *****/ Par_GetParToText (Mrk_HeadOrFootStr[HeaderOrFooter],UnsignedStr,10); if (sscanf (UnsignedStr,"%u",&NumRows) == 1) { /***** Update properties of marks in the database *****/ Cod = Brw_GetCodForFiles (); DB_QueryUPDATE ("can not update properties of marks", "UPDATE marks_properties,files" " SET marks_properties.%s=%u" " WHERE files.FileBrowser=%u AND files.Cod=%ld AND files.Path='%s'" " AND files.FilCod=marks_properties.FilCod", Mrk_HeadOrFootStr[HeaderOrFooter],NumRows, (unsigned) Brw_FileBrowserForDB_files[Gbl.FileBrowser.Type], Cod, Gbl.FileBrowser.Priv.FullPathInTree); /***** Write message of success *****/ Ale_ShowAlert (Ale_SUCCESS,Txt_The_number_of_rows_is_now_X, NumRows); } else Lay_ShowErrorAndExit ("Wrong number of rows."); /***** Show again the file browser *****/ Brw_ShowAgainFileBrowserOrWorks (); } /*****************************************************************************/ /************************ Receive a new file of marks ************************/ /*****************************************************************************/ // Returns true if the format of the HTML file of marks is correct // Returns true if the format of the HTML file of marks is wrong // Gbl.Alert.Txt will contain feedback text bool Mrk_CheckFileOfMarks (const char *Path,struct MarksProperties *Marks) { extern const char *Txt_There_are_more_than_one_table_in_the_file_of_marks; extern const char *Txt_Table_not_found_in_the_file_of_marks; char CellContent[Mrk_MAX_BYTES_IN_CELL_CONTENT + 1]; FILE *FileAllMarks; bool EndOfHead = false; bool EndOfTable = false; bool FileIsCorrect = true; unsigned NumRowsStds = 0; Marks->Header = Marks->Footer = 0; /***** Open file with the table of marks *****/ if ((FileAllMarks = fopen (Path,"rb"))) { /***** Check if there is a table in the received file ******/ if (Str_FindStrInFile (FileAllMarks,"",Str_NO_SKIP_HTML_COMMENTS); // Only one table is allowed if (Str_FindStrInFile (FileAllMarks,"",Str_NO_SKIP_HTML_COMMENTS); /* We assume that the structure of the table has several rows of header until the first row of students is found, then it has a number of rows of students, including some dummy rows without students, and finally it has several rows of footer from the last row of students until the end of the table */ /***** Count rows of header *****/ while (!EndOfHead) if (Str_FindStrInFile (FileAllMarks," continue in header Marks->Header++; } else EndOfHead = true; // No more rows /***** Count rows of students and rows of footer *****/ while (!EndOfTable) if (Str_FindStrInFile (FileAllMarks,"Footer = 0; } else // Other stuff found ==> continue in header Marks->Footer++; } else EndOfTable = true; // No more rows } } else { Ale_CreateAlert (Ale_WARNING,NULL, Txt_Table_not_found_in_the_file_of_marks); FileIsCorrect = false; } /***** The file of marks is no more necessary. Close it. *****/ fclose (FileAllMarks); } return FileIsCorrect; } /*****************************************************************************/ /******* Check if only user's IDs or other stuff found in a table cell *******/ /*****************************************************************************/ static bool Mrk_CheckIfCellContainsOnlyIDs (const char *CellContent) { char UsrIDFromTable[ID_MAX_BYTES_USR_ID + 1]; const char *Ptr = CellContent; bool UsrIDFound = false; bool StuffNotUsrIDFound = false; /***** Get strings in this table cell and check if they look like user's IDs or not *****/ while (*Ptr && !StuffNotUsrIDFound) { /* Find next string in text until space, comma or semicolon (leading and trailing spaces are removed) */ Str_GetNextStringUntilSeparator (&Ptr,UsrIDFromTable,ID_MAX_BYTES_USR_ID); // Users' IDs are always stored internally in capitals and without leading zeros Str_RemoveLeadingZeros (UsrIDFromTable); Str_ConvertToUpperText (UsrIDFromTable); if (UsrIDFromTable[0]) // Something found { if (ID_CheckIfUsrIDIsValid (UsrIDFromTable)) UsrIDFound = true; else StuffNotUsrIDFound = true; } } /***** Check if only user's IDs or other stuff found in this table cell *****/ return (UsrIDFound && !StuffNotUsrIDFound); } /*****************************************************************************/ /*************************** Show the marks of a user ************************/ /*****************************************************************************/ static bool Mrk_GetUsrMarks (FILE *FileUsrMarks,struct UsrData *UsrDat, const char *PathFileAllMarks, struct MarksProperties *Marks) { extern const char *Txt_THE_USER_X_is_not_found_in_the_file_of_marks; unsigned Row; char CellContent[Mrk_MAX_BYTES_IN_CELL_CONTENT + 1]; const char *Ptr; char UsrIDFromTable[ID_MAX_BYTES_USR_ID + 1]; FILE *FileAllMarks; unsigned NumID; bool UsrIDFound; bool EndOfTable; /***** Open HTML file with the table of marks *****/ if (!(FileAllMarks = fopen (PathFileAllMarks,"rb"))) { // Can't open the file with the table of marks Ale_CreateAlert (Ale_ERROR,NULL, "Can not open file of marks."); return false; } /***** Check if it exists a user's ID in the first column of the table of marks *****/ /* Jump to table start */ Str_FindStrInFile (FileAllMarks,"",Str_NO_SKIP_HTML_COMMENTS); /* Skip header */ for (Row = 1; Row <= Marks->Header; Row++) Str_FindStrInFile (FileAllMarks,"IDs.Num && !UsrIDFound; NumID++) if (UsrDat->IDs.List[NumID].Confirmed) if (!strcasecmp (UsrDat->IDs.List[NumID].ID,UsrIDFromTable)) UsrIDFound = true; } } else EndOfTable = true; // No more rows if (UsrIDFound) { /***** Write all until the header (included) *****/ /* Go to start of file */ rewind (FileAllMarks); /* Write until table start */ Str_WriteUntilStrFoundInFileIncludingStr (FileUsrMarks,FileAllMarks,"",Str_NO_SKIP_HTML_COMMENTS); /* Write header */ for (Row = 1; Row <= Marks->Header; Row++) Str_WriteUntilStrFoundInFileIncludingStr (FileUsrMarks,FileAllMarks,"",Str_NO_SKIP_HTML_COMMENTS); /****** Write the row corresponding to the student *****/ /* Find user's ID */ UsrIDFound = EndOfTable = false; while (!UsrIDFound && !EndOfTable) if (Str_FindStrInFile (FileAllMarks,"IDs.Num && !UsrIDFound; NumID++) if (UsrDat->IDs.List[NumID].Confirmed) if (!strcasecmp (UsrDat->IDs.List[NumID].ID,UsrIDFromTable)) UsrIDFound = true; } } else EndOfTable = true; // No more rows if (UsrIDFound) // This should happen always, because the check was already made { /* Find backward until "" */ Str_WriteUntilStrFoundInFileIncludingStr (FileUsrMarks,FileAllMarks,"",Str_NO_SKIP_HTML_COMMENTS); /***** Write the footer and all until the end *****/ /* Find the footer of the table */ Str_FindStrInFile (FileAllMarks,"",Str_NO_SKIP_HTML_COMMENTS); Str_FindStrInFileBack (FileAllMarks,"",Str_NO_SKIP_HTML_COMMENTS); for (Row = 1; Row <= Marks->Footer; Row++) Str_FindStrInFileBack (FileAllMarks,"Footer; Row++) Str_WriteUntilStrFoundInFileIncludingStr (FileUsrMarks,FileAllMarks,"",Str_NO_SKIP_HTML_COMMENTS); /* Write the end */ Str_FindStrInFile (FileAllMarks,"",Str_NO_SKIP_HTML_COMMENTS); Str_FindStrInFileBack (FileAllMarks,"",Str_NO_SKIP_HTML_COMMENTS); Fil_FastCopyOfOpenFiles (FileAllMarks,FileUsrMarks); /***** The file of marks is no longer needed. Close it. *****/ fclose (FileAllMarks); return true; } } /***** User's ID not found in table *****/ fclose (FileAllMarks); Ale_CreateAlert (Ale_WARNING,NULL, Txt_THE_USER_X_is_not_found_in_the_file_of_marks, UsrDat->FullName); return false; } /*****************************************************************************/ /*************************** Show the marks of a user ************************/ /*****************************************************************************/ void Mrk_ShowMyMarks (void) { struct MarksProperties Marks; char FileNameUsrMarks[PATH_MAX + 1]; FILE *FileUsrMarks; char PathPrivate[PATH_MAX + 1 + PATH_MAX + 1]; struct UsrData *UsrDat; bool UsrIsOK = true; /***** Get parameters related to file browser *****/ Brw_GetParAndInitFileBrowser (); /***** Get the path of the file of marks *****/ Brw_SetFullPathInTree (Gbl.FileBrowser.Priv.PathInTreeUntilFilFolLnk, Gbl.FileBrowser.FilFolLnkName); snprintf (PathPrivate,sizeof (PathPrivate), "%s/%s", Gbl.FileBrowser.Priv.PathAboveRootFolder, Gbl.FileBrowser.Priv.FullPathInTree); /***** Get number of rows of header or footer *****/ Mrk_GetNumRowsHeaderAndFooter (&Marks); /***** Set the student whose marks will be shown *****/ if (Gbl.Usrs.Me.Role.Logged == Rol_STD) // If I am logged as student... UsrDat = &Gbl.Usrs.Me.UsrDat; // ...use my list of IDs else // If I am logged as non-editing teacher, teacher or admin { /* Select a random student from the course */ if (Gbl.Crs.Grps.GrpCod > 0) // Group zone { if (Grp_CountNumUsrsInGrp (Rol_STD,Gbl.Crs.Grps.GrpCod)) { Gbl.Usrs.Other.UsrDat.UsrCod = Usr_GetRamdomStdFromGrp (Gbl.Crs.Grps.GrpCod); UsrDat = &Gbl.Usrs.Other.UsrDat; } else UsrIsOK = false; } else // Course zone { if (Gbl.Hierarchy.Crs.NumUsrs[Rol_STD]) // If there are students in this course { Gbl.Usrs.Other.UsrDat.UsrCod = Usr_GetRamdomStdFromCrs (Gbl.Hierarchy.Crs.CrsCod); UsrDat = &Gbl.Usrs.Other.UsrDat; } else UsrIsOK = false; } } if (UsrIsOK) { /***** Get list of user's IDs *****/ Usr_GetAllUsrDataFromUsrCod (UsrDat,Usr_DONT_GET_PREFS); /***** Create temporal file to store my marks (in HTML) *****/ /* If the private directory does not exist, create it */ Fil_CreateDirIfNotExists (Cfg_PATH_MARK_PRIVATE); /* Create a new temporary file *****/ snprintf (FileNameUsrMarks,sizeof (FileNameUsrMarks), "%s/%s.html", Cfg_PATH_MARK_PRIVATE,Gbl.UniqueNameEncrypted); if ((FileUsrMarks = fopen (FileNameUsrMarks,"wb")) == NULL) Lay_ShowErrorAndExit ("Can not open file for my marks."); /***** Show my marks *****/ if (Mrk_GetUsrMarks (FileUsrMarks,UsrDat,PathPrivate,&Marks)) { fclose (FileUsrMarks); if ((FileUsrMarks = fopen (FileNameUsrMarks,"rb")) == NULL) Lay_ShowErrorAndExit ("Can not open file with my marks."); /* Start HTML output */ /* Do not write charset here. Instead, delegate to the meta directive (example: ) that is typically included in the HTML document header. */ fprintf (Gbl.F.Out,"Content-type: text/html\r\n\r\n"); // Two \r\n are necessary Gbl.Layout.HTMLStartWritten = true; /* Copy HTML to output file */ Fil_FastCopyOfOpenFiles (FileUsrMarks,Gbl.F.Out); fclose (FileUsrMarks); Gbl.Layout.DivsEndWritten = Gbl.Layout.HTMLEndWritten = true; } else // Problems in table of marks or user's ID not found { fclose (FileUsrMarks); Ale_ShowAlerts (NULL); } unlink (FileNameUsrMarks); // File with marks is no longer necessary } else /***** Show warning indicating no students found *****/ Usr_ShowWarningNoUsersFound (Rol_STD); } /*****************************************************************************/ /******************** Put my marks into a notification ***********************/ /*****************************************************************************/ void Mrk_GetNotifMyMarks (char SummaryStr[Ntf_MAX_BYTES_SUMMARY + 1], char **ContentStr, long MrkCod,long UsrCod,bool GetContent) { MYSQL_RES *mysql_res; MYSQL_ROW row; struct UsrData UsrDat; unsigned UnsignedNum; Brw_FileBrowser_t FileBrowser; long Cod; long InsCod; // Not applicable here long CtrCod; // Not applicable here long DegCod; // Not applicable here long CrsCod; long GrpCod; struct MarksProperties Marks; char FullPathInTreeFromDBMarksTable[PATH_MAX + 1]; char PathUntilFileName[PATH_MAX + 1]; char FileName[NAME_MAX + 1]; char PathMarks[PATH_MAX + 1]; char FileNameUsrMarks[PATH_MAX + 1]; FILE *FileUsrMarks; size_t SizeOfMyMarks; size_t Length; size_t i; char *Ptr; SummaryStr[0] = '\0'; // Return nothing on error // This function may be called inside a web service, so don't report error /***** Initialize structure with user's data *****/ Usr_UsrDataConstructor (&UsrDat); /***** Get user's ID from user's code *****/ UsrDat.UsrCod = UsrCod; ID_GetListIDsFromUsrCod (&UsrDat); /***** Get subject of message from database *****/ if (DB_QuerySELECT (&mysql_res,"can not get the number of rows" " in header and footer", "SELECT files.FileBrowser,files.Cod,files.Path," "marks_properties.Header,marks_properties.Footer" " FROM files,marks_properties" " WHERE files.FilCod=%ld" " AND files.FilCod=marks_properties.FilCod", MrkCod) == 1) // Result should have a unique row { /***** Get data of this file of marks *****/ row = mysql_fetch_row (mysql_res); /* Get file browser type in database (row[0]) */ FileBrowser = Brw_UNKNOWN; if (sscanf (row[0],"%u",&UnsignedNum) == 1) if (UnsignedNum < Brw_NUM_TYPES_FILE_BROWSER) FileBrowser = (Brw_FileBrowser_t) UnsignedNum; /* Course/group code (row[1]) */ Cod = Str_ConvertStrCodToLongCod (row[1]); Brw_GetCrsGrpFromFileMetadata (FileBrowser,Cod, &InsCod,&CtrCod,&DegCod,&CrsCod,&GrpCod); /* Path (row[2]) */ Str_Copy (FullPathInTreeFromDBMarksTable,row[2], PATH_MAX); Str_SplitFullPathIntoPathAndFileName (FullPathInTreeFromDBMarksTable, PathUntilFileName, FileName); Str_Copy (SummaryStr,FileName, Cns_MAX_BYTES_TEXT); if (GetContent) { /* Header (row[3]) */ if (sscanf (row[3],"%u",&(Marks.Header)) != 1) Lay_ShowErrorAndExit ("Wrong number of header rows."); /* Footer (row[4]) */ if (sscanf (row[4],"%u",&(Marks.Footer)) != 1) Lay_ShowErrorAndExit ("Wrong number of footer rows."); if (UsrDat.IDs.Num) { if (GrpCod > 0) snprintf (PathMarks,sizeof (PathMarks), "%s/%ld/grp/%ld/%s", Cfg_PATH_CRS_PRIVATE,CrsCod,GrpCod, FullPathInTreeFromDBMarksTable); else snprintf (PathMarks,sizeof (PathMarks), "%s/%ld/%s", Cfg_PATH_CRS_PRIVATE,CrsCod, FullPathInTreeFromDBMarksTable); /***** Create temporal file to store my marks (in HTML) *****/ /* If the private directory does not exist, create it */ Fil_CreateDirIfNotExists (Cfg_PATH_MARK_PRIVATE); /* Create a new temporary file *****/ snprintf (FileNameUsrMarks,sizeof (FileNameUsrMarks), "%s/%s.html", Cfg_PATH_MARK_PRIVATE,Gbl.UniqueNameEncrypted); if ((FileUsrMarks = fopen (FileNameUsrMarks,"wb"))) { /***** Get user's marks *****/ if (Mrk_GetUsrMarks (FileUsrMarks,&UsrDat,PathMarks,&Marks)) { SizeOfMyMarks = (size_t) ftell (FileUsrMarks); fclose (FileUsrMarks); Length = 9 + SizeOfMyMarks + 3; if ((*ContentStr = (char *) malloc (Length + 1))) { /* 9 starting chars */ Str_Copy (*ContentStr,"", 3); } } else { fclose (FileUsrMarks); if (asprintf (ContentStr,"", Ale_GetTextOfLastAlert ()) < 0) Lay_NotEnoughMemoryExit (); Ale_ResetAllAlerts (); } } else { if (asprintf (ContentStr,"", "Can not open file of marks.") < 0) Lay_NotEnoughMemoryExit (); } unlink (FileNameUsrMarks); // File with marks is no longer necessary } else { if (asprintf (ContentStr,"", "User's IDs not found!") < 0) Lay_NotEnoughMemoryExit (); } } } /***** Free structure that stores the query result *****/ DB_FreeMySQLResult (&mysql_res); /***** Free memory used for user's data *****/ Usr_UsrDataDestructor (&UsrDat); }