/[LeafOK_CVS]/lbbs/src/section_list_loader.c
ViewVC logotype

Diff of /lbbs/src/section_list_loader.c

Parent Directory Parent Directory | Revision Log Revision Log | View Patch Patch

Revision 1.50 by sysadm, Tue Oct 14 00:57:06 2025 UTC Revision 1.65 by sysadm, Tue Nov 11 00:28:05 2025 UTC
# Line 1  Line 1 
1  /***************************************************************************  /* SPDX-License-Identifier: GPL-3.0-or-later */
2                                          section_list_loader.c  -  description  /*
3                                                           -------------------   * section_list_loader
4          Copyright            : (C) 2004-2025 by Leaflet   *   - load and query operations of section articles
5          Email                : leaflet@leafok.com   *
6   ***************************************************************************/   * Copyright (C) 2004-2025  Leaflet <leaflet@leafok.com>
7     */
8  /***************************************************************************  
9   *                                                                         *  #ifdef HAVE_CONFIG_H
10   *   This program is free software; you can redistribute it and/or modify  *  #include "config.h"
11   *   it under the terms of the GNU General Public License as published by  *  #endif
  *   the Free Software Foundation; either version 3 of the License, or     *  
  *   (at your option) any later version.                                   *  
  *                                                                         *  
  ***************************************************************************/  
12    
13  #include "article_cache.h"  #include "article_cache.h"
14  #include "article_view_log.h"  #include "article_view_log.h"
# Line 22  Line 18 
18  #include "log.h"  #include "log.h"
19  #include "menu.h"  #include "menu.h"
20  #include "section_list_loader.h"  #include "section_list_loader.h"
21    #include "user_list.h"
22  #include "user_priv.h"  #include "user_priv.h"
23  #include <errno.h>  #include <errno.h>
24  #include <signal.h>  #include <signal.h>
# Line 137  int load_section_config_from_db(int upda Line 134  int load_section_config_from_db(int upda
134                                  break;                                  break;
135                          }                          }
136    
137                          strncpy(p_section->sname, row[1], sizeof(p_section->sname) - 1);                          if (section_list_update(p_section, row[1], row[2], master_list) < 0)
138                          p_section->sname[sizeof(p_section->sname) - 1] = '\0';                          {
139                          strncpy(p_section->stitle, row[2], sizeof(p_section->stitle) - 1);                                  log_error("section_list_update(sid=%d) error\n", p_section->sid);
140                          p_section->stitle[sizeof(p_section->stitle) - 1] = '\0';                                  ret = -4;
141                          strncpy(p_section->master_list, master_list, sizeof(p_section->master_list) - 1);                                  break;
142                          p_section->master_list[sizeof(p_section->master_list) - 1] = '\0';                          }
143                  }                  }
144    
145                  p_section->class_id = atoi(row[3]);                  p_section->class_id = atoi(row[3]);
# Line 689  int section_list_loader_launch(void) Line 686  int section_list_loader_launch(void)
686          int article_count;          int article_count;
687          int load_count;          int load_count;
688          int last_mid;          int last_mid;
689          int i;          time_t tm_section_list_reload = 0;
690            time_t tm_user_list_reload = time(NULL);
691            time_t tm_user_online_list_reload = time(NULL);
692    
693          if (section_list_loader_pid != 0)          if (section_list_loader_pid != 0)
694          {          {
# Line 720  int section_list_loader_launch(void) Line 719  int section_list_loader_launch(void)
719          detach_menu_shm(&top10_menu);          detach_menu_shm(&top10_menu);
720    
721          // Set signal handler          // Set signal handler
722          act.sa_handler = SIG_DFL;          act.sa_handler = SIG_IGN;
723          if (sigaction(SIGHUP, &act, NULL) == -1)          if (sigaction(SIGHUP, &act, NULL) == -1)
724          {          {
725                  log_error("set signal action of SIGHUP error: %d\n", errno);                  log_error("set signal action of SIGHUP error: %d\n", errno);
# Line 734  int section_list_loader_launch(void) Line 733  int section_list_loader_launch(void)
733          // Do section data loader periodically          // Do section data loader periodically
734          while (!SYS_server_exit)          while (!SYS_server_exit)
735          {          {
736                    tm_section_list_reload = time(NULL);
737    
738                  if (SYS_conf_reload)                  if (SYS_conf_reload)
739                  {                  {
740                          SYS_conf_reload = 0;                          SYS_conf_reload = 0;
# Line 799  int section_list_loader_launch(void) Line 800  int section_list_loader_launch(void)
800                          log_common("Proceeded %d article logs, last_mid = %d\n", last_article_op_log_mid - last_mid, last_article_op_log_mid);                          log_common("Proceeded %d article logs, last_mid = %d\n", last_article_op_log_mid - last_mid, last_article_op_log_mid);
801                  }                  }
802    
803                  if (SYS_conf_reload)                  // Reload user list
804                    if (time(NULL) - tm_user_list_reload >= BBS_user_list_load_interval)
805                  {                  {
806                          continue;                          if (user_list_pool_reload(0) < 0)
807                            {
808                                    log_error("user_list_pool_reload(all_user) error\n");
809                            }
810    
811                            if (user_stat_update() < 0)
812                            {
813                                    log_error("user_stat_update() error\n");
814                            }
815    
816                            tm_user_list_reload = time(NULL);
817                  }                  }
818    
819                  for (i = 0; i < BBS_section_list_load_interval && !SYS_server_exit && !SYS_conf_reload; i++)                  // Reload user online list
820                    if (time(NULL) - tm_user_online_list_reload >= BBS_user_online_list_load_interval)
821                    {
822                            if (user_list_pool_reload(1) < 0)
823                            {
824                                    log_error("user_list_pool_reload(online_user) error\n");
825                            }
826    
827                            tm_user_online_list_reload = time(NULL);
828                    }
829    
830                    // Wait for BBS_section_list_load_interval seconds
831                    while (!SYS_server_exit && time(NULL) - tm_section_list_reload < BBS_section_list_load_interval)
832                  {                  {
833                          sleep(1);                          sleep(1);
834                  }                  }
# Line 821  int section_list_loader_launch(void) Line 845  int section_list_loader_launch(void)
845          detach_section_list_shm();          detach_section_list_shm();
846          detach_article_block_shm();          detach_article_block_shm();
847          detach_trie_dict_shm();          detach_trie_dict_shm();
848            detach_user_list_pool_shm();
849    
850          log_common("Section list loader process exit normally\n");          log_common("Section list loader process exit normally\n");
851          log_end();          log_end();
# Line 863  int query_section_articles(SECTION_LIST Line 888  int query_section_articles(SECTION_LIST
888          }          }
889          else if (page_id < 0 || page_id >= *p_page_count)          else if (page_id < 0 || page_id >= *p_page_count)
890          {          {
891    #ifdef _DEBUG
892                  log_error("Invalid page_id=%d, not in range [0, %d)\n", page_id, *p_page_count);                  log_error("Invalid page_id=%d, not in range [0, %d)\n", page_id, *p_page_count);
893    #endif
894                  ret = -3;                  ret = -3;
895          }          }
896          else          else
# Line 902  int query_section_articles(SECTION_LIST Line 929  int query_section_articles(SECTION_LIST
929                                           : (page_id - p_section->page_count + 1) * BBS_article_limit_per_page - p_section->last_page_visible_article_count);                                           : (page_id - p_section->page_count + 1) * BBS_article_limit_per_page - p_section->last_page_visible_article_count);
930                          while (*p_article_count < BBS_article_limit_per_page && i < p_section->ontop_article_count)                          while (*p_article_count < BBS_article_limit_per_page && i < p_section->ontop_article_count)
931                          {                          {
932                                  p_articles[(*p_article_count)++] = p_section->p_ontop_articles[i++];                                  p_articles[*p_article_count] = p_section->p_ontop_articles[i++];
933                                    (*p_article_count)++;
934                          }                          }
935                  }                  }
936          }          }
# Line 986  int locate_article_in_section(SECTION_LI Line 1014  int locate_article_in_section(SECTION_LI
1014                  p_article = section_list_find_article_with_offset(p_section, aid, &page_id, &offset, &p_tmp);                  p_article = section_list_find_article_with_offset(p_section, aid, &page_id, &offset, &p_tmp);
1015                  if (p_article != NULL)                  if (p_article != NULL)
1016                  {                  {
1017                          *p_article_count = (page_id == p_section->page_count - 1 ? p_section->last_page_visible_article_count : BBS_article_limit_per_page);                          *p_article_count = (page_id >= p_section->page_count - 1 ? p_section->last_page_visible_article_count : BBS_article_limit_per_page);
1018    
1019                          p_article = p_section->p_page_first_article[page_id];                          p_article = p_section->p_page_first_article[page_id];
1020                          for (i = 0; i < *p_article_count && p_article != NULL;)                          for (i = 0; i < *p_article_count && p_article != NULL;)
# Line 1007  int locate_article_in_section(SECTION_LI Line 1035  int locate_article_in_section(SECTION_LI
1035                                                  break;                                                  break;
1036                                          }                                          }
1037                                  }                                  }
1038    
1039                                  p_article = p_article->p_next;                                  p_article = p_article->p_next;
1040                                    if (p_article == p_section->p_page_first_article[page_id])
1041                                    {
1042                                            log_error("Dead loop detected at page=%d, article_count=%d\n", page_id, *p_article_count);
1043                                            break;
1044                                    }
1045                          }                          }
1046    
1047                          // Include ontop articles                          // Include ontop articles
# Line 1025  int locate_article_in_section(SECTION_LI Line 1059  int locate_article_in_section(SECTION_LI
1059          return (ret < 0 ? ret : (p_article == NULL ? 0 : 1));          return (ret < 0 ? ret : (p_article == NULL ? 0 : 1));
1060  }  }
1061    
1062    int last_article_in_section(SECTION_LIST *p_section, const ARTICLE **pp_article)
1063    {
1064            int ret = 0;
1065    
1066            const ARTICLE *p_article;
1067    
1068            if (p_section == NULL || pp_article == NULL)
1069            {
1070                    log_error("NULL pointer error\n");
1071                    return -1;
1072            }
1073    
1074            *pp_article = NULL;
1075    
1076            // acquire lock of section
1077            if ((ret = section_list_rd_lock(p_section)) < 0)
1078            {
1079                    log_error("section_list_rd_lock(sid = %d) error\n", p_section->sid);
1080                    return -2;
1081            }
1082    
1083            for (p_article = p_section->p_article_tail;
1084                     p_article && p_article != p_section->p_article_head && !p_article->visible;
1085                     p_article = p_article->p_prior)
1086                    ;
1087    
1088            if (p_article && p_article->visible)
1089            {
1090                    *pp_article = p_article;
1091                    ret = 1;
1092            }
1093    
1094            // release lock of section
1095            if (section_list_rd_unlock(p_section) < 0)
1096            {
1097                    log_error("section_list_rd_unlock(sid = %d) error\n", p_section->sid);
1098                    ret = -2;
1099            }
1100    
1101            return ret;
1102    }
1103    
1104  int scan_unread_article_in_section(SECTION_LIST *p_section, const ARTICLE *p_article_cur, const ARTICLE **pp_article_unread)  int scan_unread_article_in_section(SECTION_LIST *p_section, const ARTICLE *p_article_cur, const ARTICLE **pp_article_unread)
1105  {  {
1106          ARTICLE *p_article;          ARTICLE *p_article;
# Line 1060  int scan_unread_article_in_section(SECTI Line 1136  int scan_unread_article_in_section(SECTI
1136                  }                  }
1137          }          }
1138    
1139            // release lock of section
1140            if (section_list_rd_unlock(p_section) < 0)
1141            {
1142                    log_error("section_list_rd_unlock(sid = %d) error\n", p_section->sid);
1143                    return -2;
1144            }
1145    
1146            return (p_article != NULL && p_article != p_article_cur ? 1 : 0);
1147    }
1148    
1149    int scan_article_in_section_by_uid(SECTION_LIST *p_section, const ARTICLE *p_article_cur,
1150                                                                       int direction, int32_t uid, const ARTICLE **pp_article)
1151    {
1152            ARTICLE *p_article;
1153            int ret = 0;
1154    
1155            if (p_section == NULL || p_article_cur == NULL || pp_article == NULL)
1156            {
1157                    log_error("NULL pointer error\n");
1158                    return -1;
1159            }
1160    
1161            if (p_article_cur->sid != p_section->sid)
1162            {
1163                    log_error("Inconsistent SID\n");
1164                    return -1;
1165            }
1166    
1167            // acquire lock of section
1168            if ((ret = section_list_rd_lock(p_section)) < 0)
1169            {
1170                    log_error("section_list_rd_lock(sid = %d) error\n", p_section->sid);
1171                    return -2;
1172            }
1173    
1174            *pp_article = NULL;
1175    
1176            if (direction >= 0)
1177            {
1178                    for (p_article = p_article_cur->p_next; p_article != NULL && p_article != p_article_cur; p_article = p_article->p_next)
1179                    {
1180                            if (p_article->visible && p_article->uid == uid)
1181                            {
1182                                    *pp_article = p_article;
1183                                    break;
1184                            }
1185                    }
1186            }
1187            else // direction < 0
1188            {
1189                    for (p_article = p_article_cur->p_prior; p_article != NULL && p_article != p_article_cur; p_article = p_article->p_prior)
1190                    {
1191                            if (p_article->visible && p_article->uid == uid)
1192                            {
1193                                    *pp_article = p_article;
1194                                    break;
1195                            }
1196                    }
1197            }
1198    
1199            // release lock of section
1200            if (section_list_rd_unlock(p_section) < 0)
1201            {
1202                    log_error("section_list_rd_unlock(sid = %d) error\n", p_section->sid);
1203                    return -2;
1204            }
1205    
1206            return (p_article != NULL && p_article != p_article_cur ? 1 : 0);
1207    }
1208    
1209    int scan_article_in_section_by_username(SECTION_LIST *p_section, const ARTICLE *p_article_cur,
1210                                                                                    int direction, const char *username, const ARTICLE **pp_article)
1211    {
1212            ARTICLE *p_article;
1213            int ret = 0;
1214    
1215            if (p_section == NULL || p_article_cur == NULL || username == NULL || pp_article == NULL)
1216            {
1217                    log_error("NULL pointer error\n");
1218                    return -1;
1219            }
1220    
1221            if (p_article_cur->sid != p_section->sid)
1222            {
1223                    log_error("Inconsistent SID\n");
1224                    return -1;
1225            }
1226    
1227            // acquire lock of section
1228            if ((ret = section_list_rd_lock(p_section)) < 0)
1229            {
1230                    log_error("section_list_rd_lock(sid = %d) error\n", p_section->sid);
1231                    return -2;
1232            }
1233    
1234            *pp_article = NULL;
1235    
1236            if (direction >= 0)
1237            {
1238                    for (p_article = p_article_cur->p_next; p_article != NULL && p_article != p_article_cur; p_article = p_article->p_next)
1239                    {
1240                            if (p_article->visible && strcasecmp(p_article->username, username) == 0)
1241                            {
1242                                    *pp_article = p_article;
1243                                    break;
1244                            }
1245                    }
1246            }
1247            else // direction < 0
1248            {
1249                    for (p_article = p_article_cur->p_prior; p_article != NULL && p_article != p_article_cur; p_article = p_article->p_prior)
1250                    {
1251                            if (p_article->visible && strcasecmp(p_article->username, username) == 0)
1252                            {
1253                                    *pp_article = p_article;
1254                                    break;
1255                            }
1256                    }
1257            }
1258    
1259            // release lock of section
1260            if (section_list_rd_unlock(p_section) < 0)
1261            {
1262                    log_error("section_list_rd_unlock(sid = %d) error\n", p_section->sid);
1263                    return -2;
1264            }
1265    
1266            return (p_article != NULL && p_article != p_article_cur ? 1 : 0);
1267    }
1268    
1269    int scan_article_in_section_by_title(SECTION_LIST *p_section, const ARTICLE *p_article_cur,
1270                                                                             int direction, const char *title, const ARTICLE **pp_article)
1271    {
1272            ARTICLE *p_article;
1273            int ret = 0;
1274    
1275            if (p_section == NULL || p_article_cur == NULL || title == NULL || pp_article == NULL)
1276            {
1277                    log_error("NULL pointer error\n");
1278                    return -1;
1279            }
1280    
1281            if (p_article_cur->sid != p_section->sid)
1282            {
1283                    log_error("Inconsistent SID\n");
1284                    return -1;
1285            }
1286    
1287            // acquire lock of section
1288            if ((ret = section_list_rd_lock(p_section)) < 0)
1289            {
1290                    log_error("section_list_rd_lock(sid = %d) error\n", p_section->sid);
1291                    return -2;
1292            }
1293    
1294            *pp_article = NULL;
1295    
1296            if (direction >= 0)
1297            {
1298                    for (p_article = p_article_cur->p_next; p_article != NULL && p_article != p_article_cur; p_article = p_article->p_next)
1299                    {
1300                            if (p_article->visible && strcasestr(p_article->title, title) != NULL)
1301                            {
1302                                    *pp_article = p_article;
1303                                    break;
1304                            }
1305                    }
1306            }
1307            else // direction < 0
1308            {
1309                    for (p_article = p_article_cur->p_prior; p_article != NULL && p_article != p_article_cur; p_article = p_article->p_prior)
1310                    {
1311                            if (p_article->visible && strcasestr(p_article->title, title) != NULL)
1312                            {
1313                                    *pp_article = p_article;
1314                                    break;
1315                            }
1316                    }
1317            }
1318    
1319          // release lock of section          // release lock of section
1320          if (section_list_rd_unlock(p_section) < 0)          if (section_list_rd_unlock(p_section) < 0)
1321          {          {


Legend:
Removed lines/characters  
Changed lines/characters
  Added lines/characters

webmaster@leafok.com
ViewVC Help
Powered by ViewVC 1.3.0-beta1