news.c
上传用户:tany51
上传日期:2013-06-12
资源大小:1397k
文件大小:9k
源码类别:

MySQL数据库

开发平台:

Visual C++

  1. /*
  2.  * Copyright (C) 2000 Alexey Belyaev (spider@omskart.ru)
  3.  *
  4.  * This program is free software; you can redistribute it and/or
  5.  * modify it under the terms of the GNU General Public License
  6.  * as published by the Free Software Foundation; either version 2
  7.  * of the License, or (at your option) any later version.
  8.  *
  9.  * This program is distributed in the hope that it will be useful,
  10.  * but WITHOUT ANY WARRANTY; without even the implied warranty of
  11.  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  12.  * GNU General Public License for more details.
  13.  *
  14.  * You should have received a copy of the GNU General Public License
  15.  * along with this program; if not, write to the Free Software
  16.  * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
  17.  */
  18. #define NEWS_INTERNAL_ACCESS
  19. #include "common/setup_before.h"
  20. #include <stdio.h>
  21. #ifdef HAVE_STDDEF_H
  22. # include <stddef.h>
  23. #else
  24. # ifndef NULL
  25. #  define NULL ((void *)0)
  26. # endif
  27. #endif
  28. #ifdef STDC_HEADERS
  29. # include <stdlib.h>
  30. #else
  31. # ifdef HAVE_MALLOC_H
  32. #  include <malloc.h>
  33. # endif
  34. #endif
  35. #include "compat/strtoul.h"
  36. #ifdef HAVE_STRING_H
  37. # include <string.h>
  38. #else
  39. # ifdef HAVE_STRINGS_H
  40. #  include <strings.h>
  41. # endif
  42. #endif
  43. #include "compat/strchr.h"
  44. #include "compat/strdup.h"
  45. #include <errno.h>
  46. #include "compat/strerror.h"
  47. #include "common/eventlog.h"
  48. #include "common/list.h"
  49. #include "common/util.h"
  50. #include "common/proginfo.h"
  51. #include "news.h"
  52. #include <time.h>
  53. #include "common/setup_after.h"
  54. static char * news_read_file(FILE * fp);
  55. static t_list * news_head=NULL;
  56. static FILE * fp = NULL;
  57. extern int news_load(const char *filename)
  58. {
  59.     unsigned int line;
  60.     unsigned int len;
  61.     char *buff;
  62.     struct tm *date;
  63.     char date_set;
  64.     t_news_index *ni, *previous_ni;
  65.     
  66.     date_set = 0;
  67.     previous_ni = NULL;
  68.     if (!filename) {
  69. eventlog(eventlog_level_error, __FUNCTION__,"got NULL fullname");
  70. return -1;
  71.     }
  72.     if (!(news_head = list_create())) {
  73. eventlog(eventlog_level_error, __FUNCTION__,"could create list");
  74. return -1;
  75.     }
  76.     if ((fp = fopen(filename,"r"))==NULL) {
  77. eventlog(eventlog_level_error, __FUNCTION__,"can't open news file");
  78. return -1;
  79.     }
  80. setbuf(fp,NULL);
  81.     date=malloc(sizeof(struct tm));
  82.     
  83.     for (line=1; (buff = news_read_file(fp)); line++) {
  84. len = strlen(buff);
  85. if (buff[0]=='{') {
  86.     int flag;
  87.     char *dpart = malloc(5);
  88.     int dpos;
  89.     unsigned  pos;
  90.     date->tm_hour= 6; 
  91.     date->tm_min = 6;  // need to set non-zero values or else date is displayed wrong
  92.     date->tm_sec = 6;  
  93.     date->tm_isdst=-1;
  94.     dpos=0;
  95.     for (pos=1, flag=0; pos<len; pos++) {
  96. if ((buff[pos]=='/') || (buff[pos]=='}')) {
  97.          pos++;
  98.     dpart[dpos]='';
  99.          
  100.     switch (flag++) {
  101. case 0:
  102.          date->tm_mon=atoi(dpart)-1;
  103. if ((date->tm_mon<0) || (date->tm_mon>11))
  104. eventlog(eventlog_level_error,__FUNCTION__,"found invalid month (%i) in news date. (format: {MM/DD/YYYY}) on line %u",date->tm_mon,line);
  105.          break;
  106. case 1:
  107.          date->tm_mday=atoi(dpart);
  108. if ((date->tm_mday<1) || (date->tm_mday>31))
  109. eventlog(eventlog_level_error,__FUNCTION__,"found invalid month day (%i) in news date. (format: {MM/DD/YYYY}) on line %u",date->tm_mday,line);
  110.          break;
  111. case 2:
  112.          date->tm_year=atoi(dpart)-1900;
  113.          break;
  114. default:
  115.          eventlog(eventlog_level_error,__FUNCTION__,"error parsing news date on line %u",line);
  116.          free((void *)dpart);
  117. free((void *)date);
  118. free((void *)buff);
  119. fclose(fp);
  120.      return -1;
  121.     }
  122.          
  123.     dpos=0;
  124. if (buff[pos]=='}')
  125.          break;
  126. dpart[dpos++]=buff[pos];
  127.     }
  128.     if (((dpos>1) && (flag<2)) || ((dpos>3) && (flag>1))) {
  129. eventlog(eventlog_level_error,__FUNCTION__,"dpos: %d, flag: %d", dpos, flag);
  130.      eventlog(eventlog_level_error,"news_load","error parsing news date");
  131.      free((void *)dpart);
  132. free((void *)date);
  133. free((void *)buff);
  134. fclose(fp);
  135. return -1;
  136.     }
  137.     date_set = 1;
  138.     free((void *)dpart);
  139. } else {
  140.     if (!(ni = (t_news_index*)malloc(sizeof(t_news_index)))) {
  141. eventlog(eventlog_level_error,"news_load","could not allocate memory for news index");
  142. return -1;
  143.     }
  144.     
  145.     if (date_set==1) 
  146. ni->date=mktime(date);
  147. else
  148. {
  149. ni->date=time(0);
  150. eventlog(eventlog_level_error,__FUNCTION__,"(first) news entry seems to be missing a timestamp, please check your news file on line %u",line);
  151. }
  152.     if ((previous_ni) && (ni->date == previous_ni->date))
  153.     {
  154. eventlog(eventlog_level_error,__FUNCTION__,"found another news item for same date, trying to join both");
  155. if ((strlen(previous_ni->body) + strlen(buff) +2) > 1023)
  156. {
  157.   eventlog(eventlog_level_error,__FUNCTION__,"failed in joining news, cause news too long - skipping");
  158.   free((void *)ni);
  159. }
  160. else
  161. {
  162.   previous_ni->body = realloc(previous_ni->body,strlen(previous_ni->body)+1+strlen(buff)+1);
  163.   sprintf(previous_ni->body,"%sn%s",previous_ni->body,buff);
  164.   free((void *)ni);
  165. }
  166.     }
  167.     else
  168.     {
  169.       if ( strlen(buff)<1023 )
  170.       {
  171.         ni->body=strdup(buff);
  172.     
  173.         if (list_append_data(news_head,ni)<0) {
  174.   eventlog(eventlog_level_error,"news_load","could not append item");
  175.   if (ni)
  176.   {
  177.     if (ni->body) free(ni->body);
  178.     free(ni);
  179.   }
  180.   continue;
  181.         }
  182.         previous_ni = ni;
  183.       }
  184.       else
  185.       {
  186. eventlog(eventlog_level_error,__FUNCTION__,"news too long - skipping");
  187. free((void*)ni);
  188.       }
  189.     }
  190. }
  191. free((void *)buff);
  192.     }
  193.     free((void *)date);
  194.     
  195.     fclose(fp);
  196.     fp = NULL;
  197.     return 0;
  198. }
  199. static char * news_read_file(FILE * fp)
  200. {
  201.     char *   line;
  202.     char *       newline;
  203.     unsigned int len=256;
  204.     unsigned int pos=0;
  205.     int          curr_char;
  206.     
  207.     if (!(line = malloc(256)))
  208. return NULL;
  209.     while ((curr_char = fgetc(fp))!=EOF) {
  210. if (((char)curr_char)=='r')
  211.     continue; /* make DOS line endings look Unix-like */
  212. if (((char)curr_char)=='{'&& pos>0 ) {
  213.     fseek(fp,ftell(fp)-1,SEEK_SET);
  214.     break; /* end of news body */
  215. }
  216. line[pos++] = (char)curr_char;
  217. if ((pos+1)>=len) {
  218.     len += 64;
  219.     if (!(newline = realloc(line,len))) {
  220. free(line);
  221. return NULL;
  222.     }
  223.     line = newline;
  224. }
  225. if (((char)curr_char)=='}'&& pos>0 ) {
  226.     if ((curr_char = fgetc(fp))!=EOF)
  227. if (((char)curr_char)!='n') /* check for return */
  228.     fseek(fp,ftell(fp)-1,SEEK_SET); /* if not assume body starts and reset fp pos */
  229.     break; /* end of news date */
  230. }
  231.     }
  232.     
  233.     if (curr_char==EOF && pos<1)  { /* not even an empty line */
  234. free(line);
  235. return NULL;
  236.     }
  237.     
  238.     if (pos+1<len)
  239. if ((newline = realloc(line,pos+1))) /* bump the size back down to what we need */
  240.     line = newline; /* if it fails just ignore it */
  241.     line[pos] = '';
  242.     return line;
  243. }
  244.     
  245. /* Free up all of the elements in the linked list */
  246. extern int news_unload(void)
  247. {
  248.     t_elem *       curr;
  249.     t_news_index * ni;
  250.     
  251.     if (news_head) {
  252. LIST_TRAVERSE(news_head,curr)
  253. {
  254.     if (!(ni = elem_get_data(curr))) {
  255.      eventlog(eventlog_level_error,"news_unload","found NULL entry in list");
  256. continue;
  257.     }
  258.     
  259.     free((void *)ni->body);
  260.     free((void *)ni);
  261.     list_remove_elem(news_head,curr);
  262. }
  263. if (list_destroy(news_head)<0)
  264.     return -1;
  265. news_head = NULL;
  266.     }
  267.     return 0;
  268. }
  269. extern unsigned int news_get_lastnews(void)
  270. {
  271.     t_elem  *curr;
  272.     t_news_index *ni;
  273.     unsigned int last_news = 0;
  274.     
  275.     if (news_head) {
  276. LIST_TRAVERSE(news_head,curr)
  277. {
  278.     if (!(ni = elem_get_data(curr))) {
  279. eventlog(eventlog_level_error,"news_get_lastnews","found NULL entry in list");
  280. continue;
  281.     } else
  282. if (last_news<ni->date)
  283.     last_news=ni->date;
  284. }
  285.     }
  286.     return last_news;
  287. }
  288. extern unsigned int news_get_firstnews(void)
  289. {
  290.     t_elem  *curr;
  291.     t_news_index *ni;
  292.     unsigned int first_news = 0;
  293.     
  294.     if (news_head) {
  295. LIST_TRAVERSE(news_head,curr)
  296. {
  297.     if (!(ni = elem_get_data(curr))) {
  298. eventlog(eventlog_level_error,"news_get_lastnews","found NULL entry in list");
  299. continue;
  300.     } else
  301.         if (first_news)
  302. {
  303.   if (first_news>ni->date)
  304.       first_news=ni->date;
  305. }
  306. else
  307.   first_news=ni->date;
  308. }
  309.     }
  310.     return first_news;
  311. }
  312. extern t_list * newslist(void)
  313. {
  314.     return news_head;
  315. }
  316. extern char * news_get_body(t_news_index const * news)
  317. {
  318.     if (!news) {
  319. eventlog(eventlog_level_warn,"news_get_date","got NULL news");
  320. return 0;
  321.     }
  322.     
  323.     return news->body;
  324. }
  325. extern unsigned int news_get_date(t_news_index const * news)
  326. {
  327.     if (!news) {
  328. eventlog(eventlog_level_warn,"news_get_date","got NULL news");
  329. return 0;
  330.     }
  331.     
  332.     return news->date;
  333. }