summaryrefslogblamecommitdiffstats
path: root/mbbsd/admin.c
blob: cab1089fdc5fb4b7e0ad458b8307e1df0a1db3b4 (plain) (tree)
1
2
3
4
5
6
7
8
9
10
11
12
          
                
 
                  
   
                





                                                             
                                                              










                                                                                      

                                                                                       



           
                
   
            
 

                          

                                
                         
                                  
                  
                   

                                             



                                            








                          

                                          


                                       

                                                         
                      
                                      
                                                                
                                    
                                      


                 
                                                           
 
                     















                                                               


















                                                          
          
                                                 


                         
                             
                                                  
                                                               
                                
                                 
 
                

                                                        
            




                                                              



                                                                                               

                                               
                                       
                        
                                                                  

     

                    
                 
     

                

                                                                            




                             
                             
                        
                                             

                      
 


                                                    
                        








                                              
                                                  

                                                  
                                                    

                                                  
                                                    

                                                  
                                                   

                                                  
                                                         

                                                  
                                                            

                                                  





                                                  
                                                   
                                        
         
 
                      
                        
                                             
                      
 
                                   
                                                           
 
                                                           
                                            

                           
 


                                                            
                        

                                                             

                                             







                                                                     
                         
                   

                              

                                             
                             
                 
                                         
                                                      
                                    
                                 




                     
 




                           
   
                       
 




                                  
   
                          

                                



                                                       

                       


                                                         





                                                           




                         
                                   
                     
                                   
              



                                       


             
           
                                      
 



                                                             
            
                                                          

                      
 
               
                                        







                                                        
           
                                                                   




                                 
              

                     
         



                                             



                 



                                          


                               













                                                                   
                                



                                                                 

                                  
                   





                                                                           





                                                           
                                




                                                                  




                                                       
                                                           
                                



                                                   
                    
    
                                                                      


                                

                                     

                   



                                                                          
      

 

                                                     
                                        

                                                                
                     
                   


                                      





                                                     
                                                                




















                                                               





                                                
                                                     
                                                      
                          
                             
         
                                           
                                                                    




                                                                    
                                 







                                                       
     


              
   



                              
                                            
 
                         
                                                                                
                      

                  
                                        


                                                                     
 

                                
                                                                          
                                    
                                                                               
                                                                                            
                                           

                   
             
                                                   
                                          




                                                    
             
                                                   


                                                         
                           

              
             
                                      
                                                
                                                       




                                                              
                                          


                             
             
                                                   

                                         
                                            
                                                                    
                        
                                                                     
                                                                             
                                       
                                                







                                                                    
                           

                                                        

                                        


                                                           
                                                





                                                                    

                                                                    
                  


                                                               
                                             
                                           
         
              

                                                           





                                                                                                   
                                                
                                                                  
                                                                                  
                                                

                                                              
                           


                                            

              



                                                       



                                               

                                                                          

                                       
                                                   
                                                                      


                      
 

                                                                          
                                    


                      
                                    
 

                             


                                                    
                      
                                                                      

                                                                             
                         
                                                        
         
                   
                                      
                                                                    


                                                                              




                                                            

                                  
                                                 





                                                                   











                                                
 
                                                                                
                                                                                     
                              
                                                                                       
                                 
                                


                                                   
 






                                                                
 
                                                                         
 
                                                                  

                         


                                                    


                                             
 



                                             
                                   
                                                

                                                                    
                           
                                                 





                                                                                          
                                                               




             
              
   
             


                              
                       
                                          
                




                       















                                            
                  
   
            
 






                                        
                                       

             
                                                       
                                                                            

                 
 























                                                                 
 



                                   
 
                   
                                                       









                                                                  

                                 

                                              


                                                                       
         
                                   
                                                                          

















                                                                 


                                                        
                                                          





                                                             
                                           
                                                        
                                               
                                   
                                  
                                      

                                                            
                                                           

                                                  

                                                           

                      










                                                    
 


                                  
 


                      
                                                       


                                  
                                            

                                                                       
                       

                                           







                                                                                             





                                                                      
   
                                    


                             
                                
 
                       
                                           
 
                             
                            
                                        

                  
        

                                                       
                      
                                                  


                                                       
                                

                  
 
                                   
 
                            
 
                                                           
                  
                                                                        
                                       
 

                                                                       
                                                 

                  
                                  


                                  
 
                                  






                                                                     
                         
     
                                           

                                       
 











                                                   
 
                       
                                                                           
                   
                                                                           






                                                       
 
                                                                    
                                                                                 
                          
                                                                                         
                            


                                                    
 






                                                                  
 




                                           


                                           


                  
                                         
                  
                               






                                                                 
                                                  




                             
                         
                                        
































                                                                           
                                


                  
                           








                                           
           
                                                                     


                                        
                                                                                          



                                                          
            
                                                                                
                                                                        
     

 
   
                


                                  
                                                                   
                                          
                                           
 
                                                                              
                      
                 
                             


                                                      
                         
                       
                               


                     
                                               

                     
 
            
 

                                                 
                 
 
                                                                              







                                                        
 
                                                      
                                          
               
 
                         
                                             
                    
                 
     
 
                        



                                                        
                         
                                
                                         

                                                   
         
                                                                   
                            
            
                                                       


                        
                                             
                    




                                          
                             
                                         


                                                   

                   
                                                                   

                            
     
 
                
 
                                                        
                                                                     


                      
/* $Id$ */
#include "bbs.h"

/* 進站水球宣傳 */
int
m_loginmsg(void)
{
  char msg[100];
  move(21,0);
  clrtobot();
  if(SHM->loginmsg.pid && SHM->loginmsg.pid != currutmp->pid)
    {
      outs("目前已經有以下的 進站水球設定請先協調好再設定..");
      getmessage(SHM->loginmsg);
    }
  getdata(22, 0, 
     "進站水球:本站活動,不干擾使用者為限,設定者離站自動取消,確定要設?(y/N)",
          msg, 3, LCECHO);

  if(msg[0]=='y' &&

     getdata_str(23, 0, "設定進站水球:", msg, 56, DOECHO, SHM->loginmsg.last_call_in))
    {
          SHM->loginmsg.pid=currutmp->pid; /*站長不多 就不管race condition */
          strlcpy(SHM->loginmsg.last_call_in, msg, sizeof(SHM->loginmsg.last_call_in));
          strlcpy(SHM->loginmsg.userid, cuser.userid, sizeof(SHM->loginmsg.userid));
    }
  return 0;
}

/* 使用者管理 */
int
m_user(void)
{
    userec_t        xuser;
    int             id;
    char            genbuf[200];

    vs_hdr("使用者設定");
    usercomplete(msg_uid, genbuf);
    if (*genbuf) {
    move(2, 0);
    if ((id = getuser(genbuf, &xuser))) {
        user_display(&xuser, 1);
        if( HasUserPerm(PERM_ACCOUNTS) )
        uinfo_query(&xuser, 1, id);
        else
        pressanykey();
    } else {
        outs(err_uid);
        clrtoeol();
        pressanykey();
    }
    }
    return 0;
}

static int retrieve_backup(userec_t *user)
{
    int     uid;
    char    src[PATHLEN], dst[PATHLEN];
    char    ans;

    if ((uid = searchuser(user->userid, user->userid))) {
    userec_t orig;
    passwd_sync_query(uid, &orig);
    strlcpy(user->passwd, orig.passwd, sizeof(orig.passwd));
    setumoney(uid, user->money);
    passwd_sync_update(uid, user);
    return 0;
    }

    ans = vans("目前的 PASSWD 檔沒有此 ID,新增嗎?[y/N]");

    if (ans != 'y') {
    vmsg("目前的 PASSWDS 檔沒有此 ID,請先新增此帳號");
    return -1;
    }

    if (setupnewuser((const userec_t *)user) >= 0) {
    sethomepath(dst, user->userid);
    if (!dashd(dst)) {
        snprintf(src, sizeof(src), "tmp/%s", user->userid);
        if (!dashd(src) || !Rename(src, dst))
        mkuserdir(user->userid);
    }
    return 0;
    }
    return -1;
}

void 
upgrade_passwd(userec_t *puser)
{
    if (puser->version == PASSWD_VERSION)
    return;
    if (!puser->userid[0])
    return;
    if (puser->version == 2275) // chicken change
    {
    memset(puser->career,  0, sizeof(puser->career));
    memset(puser->phone,   0, sizeof(puser->phone));
    memset(puser->chkpad0, 0, sizeof(puser->chkpad0));
    memset(puser->chkpad1, 0, sizeof(puser->chkpad1));
    memset(puser->chkpad2, 0, sizeof(puser->chkpad2));
    puser->version = PASSWD_VERSION;
    return;
    }
}

static int
search_key_user(const char *passwdfile, int mode)
{
    userec_t        user;
    int             ch;
    int             unum = 0;
    FILE            *fp1 = fopen(passwdfile, "r");
    char            friendfile[PATHLEN]="", key[22], *keymatch;
    int         keytype = 0;
    int             isCurrentPwd;

    assert(fp1);

    isCurrentPwd = (strcmp(passwdfile, FN_PASSWD) == 0);
    clear();
    if (!mode)
    {
    getdata(0, 0, "請輸入id :", key, sizeof(key), DOECHO);
    } else {
    // improved search
    vs_hdr("關鍵字搜尋");
    outs("搜尋欄位: [0]全部 1.ID 2.姓名 3.暱稱 4.地址 5.Mail 6.IP 7.職業 8.電話 9.認證\n");
    getdata(2, 0, "要搜尋哪種資料?", key, 2, NUMECHO);

    if (isascii(key[0]) && isdigit(key[0]))
        keytype = key[0] - '0';
    if (keytype < 0 || keytype > 9)
        keytype = 0;
    getdata(3, 0, "請輸入關鍵字: ", key, sizeof(key), DOECHO);
    }

    if(!key[0]) {
    fclose(fp1);
    return 0;
    }
    vs_hdr(key);

    // <= or < ? I'm not sure...
    while ((fread(&user, sizeof(user), 1, fp1)) > 0 && unum++ < MAX_USERS) {

    // skip empty records
    if (!user.userid[0])
        continue;

    if (!(unum & 0xFF)) {
        vs_hdr(key);
        prints("第 [%d] 筆資料\n", unum);
        refresh();
    }

    // XXX 這裡會取舊資料,要小心 PWD 的 upgrade
    upgrade_passwd(&user);

        keymatch = NULL;

    if (!mode)
    {
        // only verify id
        if (!strcasecmp(user.userid, key))
        keymatch = user.userid; 
    } else {
        // search by keytype
        if ((!keytype || keytype == 1) &&
        DBCS_strcasestr(user.userid, key))
        keymatch = user.userid;
        else if ((!keytype || keytype == 2) &&
        DBCS_strcasestr(user.realname, key))
        keymatch = user.realname;
        else if ((!keytype || keytype == 3) &&
        DBCS_strcasestr(user.nickname, key))
        keymatch = user.nickname;
        else if ((!keytype || keytype == 4) &&
        DBCS_strcasestr(user.address, key))
        keymatch = user.address;
        else if ((!keytype || keytype == 5) &&
        strcasestr(user.email, key)) // not DBCS.
        keymatch = user.email;
        else if ((!keytype || keytype == 6) &&
        strcasestr(user.lasthost, key)) // not DBCS.
        keymatch = user.lasthost;
        else if ((!keytype || keytype == 7) &&
        DBCS_strcasestr(user.career, key))
        keymatch = user.career;
        else if ((!keytype || keytype == 8) &&
        DBCS_strcasestr(user.phone, key))
        keymatch = user.phone;
        else if ((!keytype || keytype == 9) &&
        DBCS_strcasestr(user.justify, key))
        keymatch = user.justify;
    }

        if(keymatch) {
        vs_hdr(key);
        prints("第 [%d] 筆資料\n", unum);
        refresh();

        user_display(&user, 1);
        // user_display does not have linefeed in tail.

        if (isCurrentPwd && HasUserPerm(PERM_ACCOUNTS))
        uinfo_query(&user, 1, unum);
        else
        outs("\n");

        outs(ANSI_COLOR(44) "               空白鍵" \
         ANSI_COLOR(37) ":搜尋下一個          " \
         ANSI_COLOR(33)" Q" ANSI_COLOR(37)": 離開");
        outs(mode ? 
                 "      A: add to namelist " ANSI_RESET " " :
         "      S: 取用備份資料    " ANSI_RESET " ");
        while (1) {
        while ((ch = igetch()) == 0);
                if (ch == 'a' || ch=='A' )
                  {
                   if(!friendfile[0])
                    {
                     friend_special();
                     setfriendfile(friendfile, FRIEND_SPECIAL);
                    }
                   friend_add(user.userid, FRIEND_SPECIAL, keymatch);
                   break;
                  }
        if (ch == ' ')
            break;
        if (ch == 'q' || ch == 'Q') {
            fclose(fp1);
            return 0;
        }
        if (ch == 's' && !mode) {
            if (retrieve_backup(&user) >= 0) {
            fclose(fp1);
            return 0;
            }
        }
        }
    }
    }

    fclose(fp1);
    return 0;
}

/* 以任意 key 尋找使用者 */
int
search_user_bypwd(void)
{
    search_key_user(FN_PASSWD, 1);
    return 0;
}

/* 尋找備份的使用者資料 */
int
search_user_bybakpwd(void)
{
    char           *choice[] = {
    "PASSWDS.NEW1", "PASSWDS.NEW2", "PASSWDS.NEW3",
    "PASSWDS.NEW4", "PASSWDS.NEW5", "PASSWDS.NEW6",
    "PASSWDS.BAK"
    };
    int             ch;

    clear();
    move(1, 1);
    outs("請輸入你要用來尋找備份的檔案 或按 'q' 離開\n");
    outs(" [" ANSI_COLOR(1;31) "1" ANSI_RESET "]一天前,"
     " [" ANSI_COLOR(1;31) "2" ANSI_RESET "]兩天前," 
     " [" ANSI_COLOR(1;31) "3" ANSI_RESET "]三天前\n");
    outs(" [" ANSI_COLOR(1;31) "4" ANSI_RESET "]四天前,"
     " [" ANSI_COLOR(1;31) "5" ANSI_RESET "]五天前,"
     " [" ANSI_COLOR(1;31) "6" ANSI_RESET "]六天前\n");
    outs(" [7]備份的\n");
    do {
    move(5, 1);
    outs("選擇 => ");
    ch = igetch();
    if (ch == 'q' || ch == 'Q')
        return 0;
    } while (ch < '1' || ch > '7');
    ch -= '1';
    if( access(choice[ch], R_OK) != 0 )
    vmsg("檔案不存在");
    else
    search_key_user(choice[ch], 0);
    return 0;
}

static void
bperm_msg(const boardheader_t * board)
{
    prints("\n設定 [%s] 看板之(%s)權限:", board->brdname,
       board->brdattr & BRD_POSTMASK ? "發表" : "閱\讀");
}

unsigned int
setperms(unsigned int pbits, const char * const pstring[])
{
    register int    i;

    move(4, 0);
    for (i = 0; i < NUMPERMS / 2; i++) {
    prints("%c. %-20s %-15s %c. %-20s %s\n",
           'A' + i, pstring[i],
           ((pbits >> i) & 1 ? "ˇ" : "X"),
           i < 10 ? 'Q' + i : '0' + i - 10,
           pstring[i + 16],
           ((pbits >> (i + 16)) & 1 ? "ˇ" : "X"));
    }
    clrtobot();
    while (
       (i = vmsg("請按 [A-5] 切換設定,按 [Return] 結束:"))!='\r')
    {
    if (isdigit(i))
        i = i - '0' + 26;
    else if (isalpha(i))
        i = tolower(i) - 'a';
    else {
        bell();
        continue;
    }

    pbits ^= (1 << i);
    move(i % 16 + 4, i <= 15 ? 24 : 64);
    outs((pbits >> i) & 1 ? "ˇ" : "X");
    }
    return pbits;
}

#ifdef CHESSCOUNTRY
static void
AddingChessCountryFiles(const char* apath)
{
    char filename[PATHLEN];
    char symbolicname[PATHLEN];
    char adir[PATHLEN];
    FILE* fp;
    fileheader_t fh;

    setadir(adir, apath);

    /* creating chess country regalia */
    snprintf(filename, sizeof(filename), "%s/chess_ensign", apath);
    close(open(filename, O_CREAT | O_WRONLY, 0644));

    strlcpy(symbolicname, apath, sizeof(symbolicname));
    stampfile(symbolicname, &fh);
    symlink("chess_ensign", symbolicname);

    strcpy(fh.title, "◇ 棋國國徽 (不能刪除,系統需要)");
    strcpy(fh.owner, str_sysop);
    append_record(adir, &fh, sizeof(fileheader_t));

    /* creating member list */
    snprintf(filename, sizeof(filename), "%s/chess_list", apath);
    if (!dashf(filename)) {
    fp = fopen(filename, "w");
    assert(fp);
    fputs("棋國國名\n"
        "帳號            階級    加入日期        等級或被誰俘虜\n"
        "──────    ───  ─────      ───────\n",
        fp);
    fclose(fp);
    }

    strlcpy(symbolicname, apath, sizeof(symbolicname));
    stampfile(symbolicname, &fh);
    symlink("chess_list", symbolicname);

    strcpy(fh.title, "◇ 棋國成員表 (不能刪除,系統需要)");
    strcpy(fh.owner, str_sysop);
    append_record(adir, &fh, sizeof(fileheader_t));

    /* creating profession photos' dir */
    snprintf(filename, sizeof(filename), "%s/chess_photo", apath);
    mkdir(filename, 0755);

    strlcpy(symbolicname, apath, sizeof(symbolicname));
    stampfile(symbolicname, &fh);
    symlink("chess_photo", symbolicname);

    strcpy(fh.title, "◆ 棋國照片檔 (不能刪除,系統需要)");
    strcpy(fh.owner, str_sysop);
    append_record(adir, &fh, sizeof(fileheader_t));
}
#endif /* defined(CHESSCOUNTRY) */

/* 自動設立精華區 */
void
setup_man(const boardheader_t * board, const boardheader_t * oldboard)
{
    char            genbuf[200];

    setapath(genbuf, board->brdname);
    mkdir(genbuf, 0755);

#ifdef CHESSCOUNTRY
    if (oldboard == NULL || oldboard->chesscountry != board->chesscountry)
    if (board->chesscountry != CHESSCODE_NONE)
        AddingChessCountryFiles(genbuf);
    // else // doesn't remove files..
#endif
}

void delete_symbolic_link(boardheader_t *bh, int bid)
{
    assert(0<=bid-1 && bid-1<MAX_BOARD);
    memset(bh, 0, sizeof(boardheader_t));
    substitute_record(fn_board, bh, sizeof(boardheader_t), bid);
    reset_board(bid);
    sort_bcache(); 
    log_usies("DelLink", bh->brdname);
}

int dir_cmp(const void *a, const void *b)
{
  return (atoi( &((fileheader_t *)a)->filename[2] ) -
          atoi( &((fileheader_t *)b)->filename[2] ));
}

void merge_dir(const char *dir1, const char *dir2, int isoutter)
{
     int i, pn, sn;
     fileheader_t *fh;
     char *p1, *p2, bakdir[128], file1[128], file2[128];
     strcpy(file1,dir1);
     strcpy(file2,dir2);
     if((p1=strrchr(file1,'/')))
     p1 ++;
     else
     p1 = file1;
     if((p2=strrchr(file2,'/')))
     p2 ++;
     else
     p2 = file2;

     pn=get_num_records(dir1, sizeof(fileheader_t));
     sn=get_num_records(dir2, sizeof(fileheader_t));
     if(!sn) return;
     fh= (fileheader_t *)malloc( (pn+sn)*sizeof(fileheader_t));
     get_records(dir1, fh, sizeof(fileheader_t), 1, pn);
     get_records(dir2, fh+pn, sizeof(fileheader_t), 1, sn);
     if(isoutter)
         {
             for(i=0; i<sn; i++)
               if(fh[pn+i].owner[0])
                   strcat(fh[pn+i].owner, "."); 
         }
     qsort(fh, pn+sn, sizeof(fileheader_t), dir_cmp);
     snprintf(bakdir, sizeof(bakdir), "%s.bak", dir1);
     Rename(dir1, bakdir);
     for(i=1; i<=pn+sn; i++ )
        {
         if(!fh[i-1].filename[0]) continue;
         if(i == pn+sn ||  strcmp(fh[i-1].filename, fh[i].filename))
     {
                fh[i-1].recommend =0;
        fh[i-1].filemode |= 1;
                append_record(dir1, &fh[i-1], sizeof(fileheader_t));
        strcpy(p1, fh[i-1].filename);
                if(!dashf(file1))
              {
              strcpy(p2, fh[i-1].filename);
              Copy(file2, file1);
              } 
     }
         else
                fh[i].filemode |= fh[i-1].filemode;
        }
     
     free(fh);
}

int
m_mod_board(char *bname)
{
    boardheader_t   bh, newbh;
    int             bid;
    char            genbuf[PATHLEN], ans[4];

    bid = getbnum(bname);
    if (!bid || !bname[0] || get_record(fn_board, &bh, sizeof(bh), bid) == -1) {
    vmsg(err_bid);
    return -1;
    }
    assert(0<=bid-1 && bid-1<MAX_BOARD);
    prints("看板名稱:%s\n看板說明:%s\n看板bid:%d\n看板GID:%d\n"
       "板主名單:%s", bh.brdname, bh.title, bid, bh.gid, bh.BM);
    bperm_msg(&bh);

    /* Ptt 這邊斷行會檔到下面 */
    move(9, 0);
    snprintf(genbuf, sizeof(genbuf), "(E)設定 (V)違法/解除%s%s [Q]取消?",
        HasUserPerm(PERM_SYSOP |
             PERM_BOARD) ? " (B)Vote (S)救回 (C)合併 (G)賭盤解卡" : "",
        HasUserPerm(PERM_SYSSUBOP | PERM_SYSSUPERSUBOP | PERM_BOARD) ? " (D)刪除" : "");
    getdata(10, 0, genbuf, ans, 3, LCECHO);

    switch (*ans) {
    case 'g':
    if (HasUserPerm(PERM_SYSOP | PERM_BOARD)) {
        char            path[PATHLEN];
        setbfile(genbuf, bname, FN_TICKET_LOCK);
        setbfile(path, bname, FN_TICKET_END);
        rename(genbuf, path);
    }
    break;
    case 's':
    if (HasUserPerm(PERM_SYSOP | PERM_BOARD)) {
      snprintf(genbuf, sizeof(genbuf),
           BBSHOME "/bin/buildir boards/%c/%s &",
           bh.brdname[0], bh.brdname);
        system(genbuf);
    }
    break;
    case 'c':
    if (HasUserPerm(PERM_SYSOP)) {
       char frombname[20], fromdir[PATHLEN];
        CompleteBoard(MSG_SELECT_BOARD, frombname);
            if (frombname[0] == '\0' || !getbnum(frombname) ||
        !strcmp(frombname,bname))
                         break;
            setbdir(genbuf, bname);
            setbdir(fromdir, frombname);
            merge_dir(genbuf, fromdir, 0);
        touchbtotal(bid);
    }
    break;
    case 'b':
    if (HasUserPerm(PERM_SYSOP | PERM_BOARD)) {
        char            bvotebuf[10];

        memcpy(&newbh, &bh, sizeof(bh));
        snprintf(bvotebuf, sizeof(bvotebuf), "%d", newbh.bvote);
        move(20, 0);
        prints("看板 %s 原來的 BVote:%d", bh.brdname, bh.bvote);
        getdata_str(21, 0, "新的 Bvote:", genbuf, 5, NUMECHO, bvotebuf);
        newbh.bvote = atoi(genbuf);
        assert(0<=bid-1 && bid-1<MAX_BOARD);
        substitute_record(fn_board, &newbh, sizeof(newbh), bid);
        reset_board(bid);
        log_usies("SetBoardBvote", newbh.brdname);
        break;
    } else
        break;
    case 'v':
    memcpy(&newbh, &bh, sizeof(bh));
    outs("看板目前為");
    outs((bh.brdattr & BRD_BAD) ? "違法" : "正常");
    getdata(21, 0, "確定更改?", genbuf, 5, LCECHO);
    if (genbuf[0] == 'y') {
        if (newbh.brdattr & BRD_BAD)
        newbh.brdattr = newbh.brdattr & (!BRD_BAD);
        else
        newbh.brdattr = newbh.brdattr | BRD_BAD;
        assert(0<=bid-1 && bid-1<MAX_BOARD);
        substitute_record(fn_board, &newbh, sizeof(newbh), bid);
        reset_board(bid);
        log_usies("ViolateLawSet", newbh.brdname);
    }
    break;
    case 'd':
    if (!(HasUserPerm(PERM_SYSOP | PERM_BOARD) ||
            (HasUserPerm(PERM_SYSSUPERSUBOP) && GROUPOP())))
        break;
    getdata_str(9, 0, msg_sure_ny, genbuf, 3, LCECHO, "N");
    if (genbuf[0] != 'y' || !bname[0])
        outs(MSG_DEL_CANCEL);
    else if (bh.brdattr & BRD_SYMBOLIC) {
        delete_symbolic_link(&bh, bid);
    }
    else {
        strlcpy(bname, bh.brdname, sizeof(bh.brdname));
        snprintf(genbuf, sizeof(genbuf),
            "/bin/tar zcvf tmp/board_%s.tgz boards/%c/%s man/boards/%c/%s >/dev/null 2>&1;"
            "/bin/rm -fr boards/%c/%s man/boards/%c/%s",
            bname, bname[0], bname, bname[0],
            bname, bname[0], bname, bname[0], bname);
        system(genbuf);
        memset(&bh, 0, sizeof(bh));
        snprintf(bh.title, sizeof(bh.title),
             "     %s 看板 %s 刪除", bname, cuser.userid);
        post_msg(BN_SECURITY, bh.title, "請注意刪除的合法性", "[系統安全局]");
        assert(0<=bid-1 && bid-1<MAX_BOARD);
        substitute_record(fn_board, &bh, sizeof(bh), bid);
        reset_board(bid);
            sort_bcache(); 
        log_usies("DelBoard", bh.title);
        outs("刪板完畢");
    }
    break;
    case 'e':
    if( bh.brdattr & BRD_SYMBOLIC ){
        vmsg("禁止更動連結看板,請直接修正原看板");
        break;
    }
    move(8, 0);
    outs("直接按 [Return] 不修改該項設定");
    memcpy(&newbh, &bh, sizeof(bh));

    while (getdata(9, 0, "新看板名稱:", genbuf, IDLEN + 1, DOECHO)) {
        if (getbnum(genbuf)) {
        move(3, 0);
        outs("錯誤! 板名雷同");
        } else if ( is_valid_brdname(genbuf) ){
        strlcpy(newbh.brdname, genbuf, sizeof(newbh.brdname));
        break;
        }
    }

    do {
        getdata_str(12, 0, "看板類別:", genbuf, 5, DOECHO, bh.title);
        if (strlen(genbuf) == 4)
        break;
    } while (1);

    strcpy(newbh.title, genbuf);

    newbh.title[4] = ' ';

    // 7 for category
    getdata_str(14, 0, "看板主題:", genbuf, 
        BTLEN + 1 -7, DOECHO, bh.title + 7);
    if (genbuf[0])
        strlcpy(newbh.title + 7, genbuf, sizeof(newbh.title) - 7);
    if (getdata_str(15, 0, "新板主名單:", genbuf, IDLEN * 3 + 3, DOECHO,
            bh.BM)) {
        trim(genbuf);
        strlcpy(newbh.BM, genbuf, sizeof(newbh.BM));
    }
#ifdef CHESSCOUNTRY
    if (HasUserPerm(PERM_SYSOP)) {
        snprintf(genbuf, sizeof(genbuf), "%d", bh.chesscountry);
        if (getdata_str(16, 0,
            "設定棋國 (0)無 (1)五子棋 (2)象棋 (3)圍棋 (4) 黑白棋",
            ans, sizeof(ans), NUMECHO, genbuf)){
        newbh.chesscountry = atoi(ans);
        if (newbh.chesscountry > CHESSCODE_MAX ||
            newbh.chesscountry < CHESSCODE_NONE)
            newbh.chesscountry = bh.chesscountry;
        }
    }
#endif /* defined(CHESSCOUNTRY) */
    if (HasUserPerm(PERM_SYSOP|PERM_BOARD)) {
        move(1, 0);
        clrtobot();
        newbh.brdattr = setperms(newbh.brdattr, str_permboard);
        move(1, 0);
        clrtobot();
    }
    {
        const char* brd_symbol;
        if (newbh.brdattr & BRD_GROUPBOARD)
            brd_symbol = "Σ";
        else if (newbh.brdattr & BRD_NOTRAN)
        brd_symbol = "◎";
        else
        brd_symbol = "●";

        newbh.title[5] = brd_symbol[0];
        newbh.title[6] = brd_symbol[1];
    }

    if (HasUserPerm(PERM_SYSOP|PERM_BOARD) && !(newbh.brdattr & BRD_HIDE)) {
        getdata_str(14, 0, "設定讀寫權限(Y/N)?", ans, sizeof(ans), LCECHO, "N");
        if (*ans == 'y') {
        getdata_str(15, 0, "限制 [R]閱\讀 (P)發表?", ans, sizeof(ans), LCECHO,
                "R");
        if (*ans == 'p')
            newbh.brdattr |= BRD_POSTMASK;
        else
            newbh.brdattr &= ~BRD_POSTMASK;

        move(1, 0);
        clrtobot();
        bperm_msg(&newbh);
        newbh.level = setperms(newbh.level, str_permid);
        clear();
        }
    }

    getdata(b_lines - 1, 0, "請您確定(Y/N)?[Y]", genbuf, 4, LCECHO);

    if ((*genbuf != 'n') && memcmp(&newbh, &bh, sizeof(bh))) {
        char buf[64];

        if (strcmp(bh.brdname, newbh.brdname)) {
        char            src[60], tar[60];

        setbpath(src, bh.brdname);
        setbpath(tar, newbh.brdname);
        Rename(src, tar);

        setapath(src, bh.brdname);
        setapath(tar, newbh.brdname);
        Rename(src, tar);
        }
        setup_man(&newbh, &bh);
        assert(0<=bid-1 && bid-1<MAX_BOARD);
        substitute_record(fn_board, &newbh, sizeof(newbh), bid);
        reset_board(bid);
            sort_bcache(); 
        log_usies("SetBoard", newbh.brdname);

        snprintf(buf, sizeof(buf), "[看板變更] %s (by %s)", bh.brdname, cuser.userid);
        snprintf(genbuf, sizeof(genbuf),
            "板名: %s => %s\n"
            "板主: %s => %s\n",
            bh.brdname, newbh.brdname, bh.BM, newbh.BM);
        post_msg(BN_SECURITY, buf, genbuf, "[系統安全局]");
    }
    }
    return 0;
}

/* 設定看板 */
int
m_board(void)
{
    char            bname[32];

    vs_hdr("看板設定");
    CompleteBoardAndGroup(msg_bid, bname);
    if (!*bname)
    return 0;
    m_mod_board(bname);
    return 0;
}

static void
str_unify_blank(char *s)
{
    while(*s)
    {
    if (*s == '\t') *s = ' ';
    s++;
    }
}

// 偷懶一下,寫死最大上限。
#define MAX_ENTRIES (100)
#define min(a,b) ((a)<(b) ? (a) : (b))

// TODO 哪天把這種 UI 寫的更 general 一點...

/* 設定系統檔案 */
int
x_file(void)
{
    char *entries[MAX_ENTRIES] = {NULL};
    int  centries = 0, i = 0;
    char *fn, *v;
    int sel = 0, page = 0;
    char buf[PATHLEN];
    FILE *fp = NULL;

    fp = fopen(FN_CONF_EDITABLE, "rt");
    if (!fp) 
    {
    // you can find a sample in sample/etc/editable
    vmsgf("未設定可編輯檔案列表[%s],請洽系統站長。", FN_CONF_EDITABLE);
    return 0;
    }

    // load the editable file.
    // format: filename [ \t]* description
    while (centries < MAX_ENTRIES && fgets(buf, sizeof(buf), fp))
    {
    if (!buf[0] || buf[0] == '#' || buf[0] == '.' 
        || buf[0] == '/' || buf[0] == ' ')
        continue;
    str_unify_blank(buf); // replace all \t to ' ' in buf.
    v = strchr(buf, ' '); // find if description exists.
    if (v == NULL) continue;
    fn = strstr(buf, ".."); // see if someone trying to crack
    if (fn && fn < v) continue;
    // reject anything outside etc/ folder.
    if (strncmp(buf, "etc/", strlen("etc/")) != 0)
        continue;
    chomp(buf);
    entries[centries++] = strdup(buf);
    }
    fclose(fp);
    if (centries == 0)
    {
    vmsg("無可編輯檔案,請洽系統站長。");
    return 0;
    }

    // edit the files!
    while (sel >= 0)
    {
    const int rows = t_lines-2;

    // display.
    clear(); showtitle("系統檔案", "編輯系統檔案");
    for (i = page*rows; i < min(centries, (page+1)*rows); i++)
    {
        // parse entry
        strlcpy(buf, entries[i], sizeof(buf));
        fn = buf;
        v = strchr(fn, ' ');
        *v++ = 0;
        while (*v == ' ') v++;
        if (strlen(fn) > 30)
        strcpy(fn+30-2, "..");
        prints("  %3d. %s" 
             "%-36.36s " 
            ANSI_COLOR(0;1) "%-30.30s"
            ANSI_RESET "\n", 
            i+1, 
            dashf(fn) ? ANSI_COLOR(1;32) : ANSI_COLOR(1;30;40),
            v, fn);
    }
    vs_footer(" 編輯系統檔案 ",
        " (jk/↑↓/0-9)移動 (Enter/→)編輯 (d)刪除 \t(q/←)跳出");
    cursor_show(1+sel-page*rows, 0);
    switch((i = vkey()))
    {
        case 'q': case KEY_LEFT:
        sel = -1; continue;
        case KEY_HOME: sel = 0; break;
        case KEY_END:  sel = centries-1; break;
        case KEY_PGDN: sel += rows; 
               if (sel >= centries) sel = centries-1;
               break;
        case KEY_PGUP: sel -= rows;
               if (sel < 0) sel = 0;
               break;
        case 'k': case KEY_UP: 
        if (sel > 0) sel--; break;
        case 'j': case KEY_DOWN:
        if (sel < centries-1) sel++; break;

        case 'd':
        strlcpy(buf, entries[sel], sizeof(buf));
        v = strchr(buf, ' '); *v++ = 0;
        i = vansf("確定要刪除 %s 嗎? (y/N) ", v);
        if (i == 'y')
            unlink(buf);
        vmsgf("系統檔案[%s]: %s", buf, !dashf(buf) ? 
            "刪除成功\ " : "未刪除");
        break;

        case KEY_ENTER: case KEY_RIGHT:
        strlcpy(buf, entries[sel], sizeof(buf));
        v = strchr(buf, ' '); *v++ = 0;
        i = veditfile(buf);
        // log file change
        if (i != EDIT_ABORTED)
        {
            log_filef("log/etc_edit.log", LOG_CREAT,
                "%s %s %s # %s\n", Cdate(&now),
                cuser.userid, buf, v);
        }
        vmsgf("系統檔案[%s]: %s", buf, (i == -1) ? 
            "未改變" : "更新完畢");
        break;

        default:
        if (i >= '0' && i <= '9')
        {
            sel = search_num(i, centries-1);
            if (sel < 0) sel = 0;
        }
        break;
    }
    // change page if required.
    page = sel / rows;
    }

    // free the entries
    for (i = 0; i < centries; i++)
    free(entries[i]);

    return FULLUPDATE;
}

static int add_board_record(const boardheader_t *board)
{
    int bid;
    if ((bid = getbnum("")) > 0) {
    assert(0<=bid-1 && bid-1<MAX_BOARD);
    substitute_record(fn_board, board, sizeof(boardheader_t), bid);
    reset_board(bid);
        sort_bcache(); 
    } else if (SHM->Bnumber >= MAX_BOARD) {
    return -1;
    } else if (append_record(fn_board, (fileheader_t *)board, sizeof(boardheader_t)) == -1) {
    return -1;
    } else {
    addbrd_touchcache();
    }
    return 0;
}

/**
 * open a new board
 * @param whatclass In which sub class
 * @param recover   Forcely open a new board, often used for recovery.
 * @return -1 if failed
 */
int
m_newbrd(int whatclass, int recover)
{
    boardheader_t   newboard;
    char            ans[4];
    char            genbuf[200];

    vs_hdr("建立新板");
    memset(&newboard, 0, sizeof(newboard));

    newboard.gid = whatclass;
    if (newboard.gid == 0) {
    vmsg("請先選擇一個類別再開板!");
    return -1;
    }
    do {
    if (!getdata(3, 0, msg_bid, newboard.brdname,
             sizeof(newboard.brdname), DOECHO))
        return -1;
    } while (!is_valid_brdname(newboard.brdname));

    do {
    getdata(6, 0, "看板類別:", genbuf, 5, DOECHO);
    if (strlen(genbuf) == 4)
        break;
    } while (1);

    strcpy(newboard.title, genbuf);

    newboard.title[4] = ' ';

    getdata(8, 0, "看板主題:", genbuf, BTLEN + 1, DOECHO);
    if (genbuf[0])
    strlcpy(newboard.title + 7, genbuf, sizeof(newboard.title) - 7);
    setbpath(genbuf, newboard.brdname);

    if (!recover && 
        (getbnum(newboard.brdname) > 0 || mkdir(genbuf, 0755) == -1)) {
    vmsg("此看板已經存在! 請取不同英文板名");
    return -1;
    }
    newboard.brdattr = BRD_NOTRAN;
#ifdef DEFAULT_AUTOCPLOG
    newboard.brdattr |= BRD_CPLOG;
#endif

    if (HasUserPerm(PERM_SYSOP)) {
    move(1, 0);
    clrtobot();
    newboard.brdattr = setperms(newboard.brdattr, str_permboard);
    move(1, 0);
    clrtobot();
    }
    getdata(9, 0, "是看板? (N:目錄) (Y/n):", genbuf, 3, LCECHO);
    if (genbuf[0] == 'n')
    {
    newboard.brdattr |= BRD_GROUPBOARD;
    newboard.brdattr &= ~BRD_CPLOG;
    }

    {
        const char* brd_symbol;
        if (newboard.brdattr & BRD_GROUPBOARD)
            brd_symbol = "Σ";
        else if (newboard.brdattr & BRD_NOTRAN)
        brd_symbol = "◎";
        else
        brd_symbol = "●";

        newboard.title[5] = brd_symbol[0];
        newboard.title[6] = brd_symbol[1];
    }

    newboard.level = 0;
    getdata(11, 0, "板主名單:", newboard.BM, sizeof(newboard.BM), DOECHO);
#ifdef CHESSCOUNTRY
    if (getdata_str(12, 0, "設定棋國 (0)無 (1)五子棋 (2)象棋 (3)圍棋", ans,
        sizeof(ans), LCECHO, "0")){
    newboard.chesscountry = atoi(ans);
    if (newboard.chesscountry > CHESSCODE_MAX ||
        newboard.chesscountry < CHESSCODE_NONE)
        newboard.chesscountry = CHESSCODE_NONE;
    }
#endif /* defined(CHESSCOUNTRY) */

    if (HasUserPerm(PERM_SYSOP) && !(newboard.brdattr & BRD_HIDE)) {
    getdata_str(14, 0, "設定讀寫權限(Y/N)?", ans, sizeof(ans), LCECHO, "N");
    if (*ans == 'y') {
        getdata_str(15, 0, "限制 [R]閱\讀 (P)發表?", ans, sizeof(ans), LCECHO, "R");
        if (*ans == 'p')
        newboard.brdattr |= BRD_POSTMASK;
        else
        newboard.brdattr &= (~BRD_POSTMASK);

        move(1, 0);
        clrtobot();
        bperm_msg(&newboard);
        newboard.level = setperms(newboard.level, str_permid);
        clear();
    }
    }

    if (add_board_record(&newboard) < 0) {
    if (SHM->Bnumber >= MAX_BOARD)
        vmsg("看板已滿,請洽系統站長");
    else
            vmsg("看板寫入失敗");

    setbpath(genbuf, newboard.brdname);
    rmdir(genbuf);
    return -1;
    }

    getbcache(whatclass)->childcount = 0;
    pressanykey();
    setup_man(&newboard, NULL);
    outs("\n新板成立");
    post_newboard(newboard.title, newboard.brdname, newboard.BM);
    log_usies("NewBoard", newboard.title);
    pressanykey();
    return 0;
}

int make_symbolic_link(const char *bname, int gid)
{
    boardheader_t   newboard;
    int bid;
    
    bid = getbnum(bname);
    if(bid==0) return -1;
    assert(0<=bid-1 && bid-1<MAX_BOARD);
    memset(&newboard, 0, sizeof(newboard));

    /*
     * known issue:
     *   These two stuff will be used for sorting.  But duplicated brdnames
     *   may cause wrong binary-search result.  So I replace the last 
     *   letters of brdname to '~'(ascii code 126) in order to correct the
     *   resuilt, thought I think it's a dirty solution.
     *
     *   Duplicate entry with same brdname may cause wrong result, if
     *   searching by key brdname.  But people don't need to know a board
     *   is symbolic, so just let SYSOP know it. You may want to read
     *   board.c:load_boards().
     */

    strlcpy(newboard.brdname, bname, sizeof(newboard.brdname));
    newboard.brdname[strlen(bname) - 1] = '~';
    strlcpy(newboard.title, bcache[bid - 1].title, sizeof(newboard.title));
    strcpy(newboard.title + 5, "@看板連結");

    newboard.gid = gid;
    BRD_LINK_TARGET(&newboard) = bid;
    newboard.brdattr = BRD_NOTRAN | BRD_SYMBOLIC;

    if (add_board_record(&newboard) < 0)
    return -1;
    return bid;
}

int make_symbolic_link_interactively(int gid)
{
    char buf[32];

    CompleteBoard(msg_bid, buf);
    if (!buf[0])
    return -1;

    vs_hdr("建立看板連結");

    if (make_symbolic_link(buf, gid) < 0) {
    vmsg("看板連結建立失敗");
    return -1;
    }
    log_usies("NewSymbolic", buf);
    return 0;
}

static void
give_id_money(const char *user_id, int money, const char *mail_title)
{
    char            tt[TTLEN + 1] = {0};

    if (deumoney(searchuser(user_id, NULL), money) < 0) { // TODO if searchuser() return 0
    move(12, 0);
    clrtoeol();
    prints("id:%s money:%d 不對吧!!", user_id, money);
    pressanykey();
    } else {
    snprintf(tt, sizeof(tt), "%s : %d " MONEYNAME " 幣", mail_title, money);
    mail_id(user_id, tt, "etc/givemoney.why", "[" BBSMNAME "銀行]");
    }
}

int
give_money(void)
{
    FILE           *fp, *fp2;
    char           *ptr, *id, *mn;
    char            buf[200] = "", reason[100], tt[TTLEN + 1] = "";
    int             to_all = 0, money = 0;
    int             total_money=0, count=0;

    getdata(0, 0, "指定使用者(S) 全站使用者(A) 取消(Q)?[S]", buf, 3, LCECHO);
    if (buf[0] == 'q')
    return 1;
    else if (buf[0] == 'a') {
    to_all = 1;
    getdata(1, 0, "發多少錢呢?", buf, 20, DOECHO);
    money = atoi(buf);
    if (money <= 0) {
        move(2, 0);
        vmsg("輸入錯誤!!");
        return 1;
    }
    } else {
    if (veditfile("etc/givemoney.txt") < 0)
        return 1;
    }

    clear();

    unlink("etc/givemoney.log");
    if (!(fp2 = fopen("etc/givemoney.log", "w")))
    return 1;

    getdata(0, 0, "動用國庫!請輸入正當理由(如活動名稱):", reason, 40, DOECHO);
    fprintf(fp2,"\n使用理由: %s\n", reason);

    getdata(1, 0, "要發錢了嗎(Y/N)[N]", buf, 3, LCECHO);
    if (buf[0] != 'y')
       {
        fclose(fp2);
    return 1;
       }

    getdata(1, 0, "紅包袋標題 :", tt, TTLEN, DOECHO);
    fprintf(fp2,"\n紅包袋標題: %s\n", tt);
    move(2, 0);

    vmsg("編紅包袋內容");
    if (veditfile("etc/givemoney.why") < 0) {
        fclose(fp2);
    return 1;
    }

    vs_hdr("發錢中...");
    if (to_all) {
    int             i, unum;
    for (unum = SHM->number, i = 0; i < unum; i++) {
        if (bad_user_id(SHM->userid[i]))
        continue;
        id = SHM->userid[i];
        give_id_money(id, money, tt);
            fprintf(fp2,"給 %s : %d\n", id, money);
            count++;
    }
        sprintf(buf, "(%d人:%d"MONEYNAME"幣)", count, count*money);
        strcat(reason, buf);
    } else {
    if (!(fp = fopen("etc/givemoney.txt", "r+"))) {
        fclose(fp2);
        return 1;
    }
    while (fgets(buf, sizeof(buf), fp)) {
        clear();
        if (!(ptr = strchr(buf, ':')))
        continue;
        *ptr = '\0';
        id = buf;
        mn = ptr + 1;
            money = atoi(mn);
        give_id_money(id, money, tt);
            fprintf(fp2,"給 %s : %d\n", id, money);
            total_money += money;
            count++;
    }
    fclose(fp);
        sprintf(buf, "(%d人:%d"MONEYNAME"幣)", count, total_money);
        strcat(reason, buf);
    
    }

    fclose(fp2);

    sprintf(buf, "%s 紅包機: %s", cuser.userid, reason);
    post_file(BN_SECURITY, buf, "etc/givemoney.log", "[紅包機報告]");
    pressanykey();
    return FULLUPDATE;
}