summaryrefslogblamecommitdiffstats
path: root/innbbsd/receive_article.c
blob: f79096b66e781e387f95c799d483bac8e595e14c (plain) (tree)
1
2
3
4
5
6
7
8
9
 

                                       
   
                                           
   
                                                                    
   
                                                          
   
                                                                         
   
                                     
   






                                                                                
   



                                                                             
                

                    
                     
                                             
      
                 





                       
                
 
                            
 
                                                                                       









                                                 

                                          






                    
                  


              
                      




                        
   
                                       


                             
 

                                           
 







                                                            

                


                                                   

      






                                                                                                                        
                      
 


                 




                                         

                                     
 






                                                  
     





                                                          
     

                                                
     





                                                                                                                        
     











                                                                                

               






























                                                                                                                            



      
   
                                          


                             
 































                                                                                                                        
                      

 


                 


 
                          


                          
   

                 
                                   
                                    






                                   
                                                       
                                                         

                        
                      
                               
           
                              
           
                              

















                                                                            

                                                                     
                                                                        
                                                                  
                                                                   



                                                                              





                                                                        
                                             
                                                                

      



                                                          
                        

                                          


                                                                      
      
     
 



                                                         
                        

                                          
                                                           

                         
      
     
 



                                                            
                        

                                              


                                                                           
      
     































                                                    
     

                              








                                                                           
     
 







                                                         
                













                                                           
     

      
                                                                                         




                                                                                              
     
                     

                
                  

      



                                         
 

                                                                   
 


















                                                                                                                                                                 
                                    



                                                                
                                                                             
































                                                                                           
     








                                                                   
     
             

 
   
                     
 
                                      
                                                       






                                                                 
                           





















                                                                                                         
     
             

 
   
                           
                          
 


                                                 
                                                                   





                                                      

            
                                                 

      


                                 
     









                                                   
         

                                                                     
                              

                                                      
                                                                 





















                                                                 
                                                                







                                                                          
                                   
































                                                                                                                    

                  












                                                                                                     
      




                                                                          
     



                                                                  

      








                                                          
     















                                                                       
     
             







                                                                                       
                
                                                                     



                                         
 

                                      


                                     












                                                              
     



                     

                                                                                      







                                                              
     

            
                                    

      



                                  

                
                                                           

      



                                                                  
     








                                                            
     
                                           

                



                                           

     
                                  




                              

                                          

                                                                                    




                                                                            
      









                                                                             



                                                  

                


  

                         
 
                            
   
                                        
 
                             
 


                                           
 

                                            
 
      

 
    
                                                
 
                       
                                   



                                                                                        
                                                                   








                                                                






                                  
     
                


                                    
                    



                     

                          
 


                                         

 
   
                                                  
 




                                 
 
                            
 


                                                                        
 
                                                 

                   
                                                 

      





                                                                          

                   
                                                         

      


                   
     






                                                                                 
 
                                                                   
                   
                                                      

      






                               
     


                                                                         

                   
                                                   

      



                           

                       
             
 
      
 
   
                                     

                                 
 

                               
                                     
                                    






                                                                
     

























                                                                        

                                                                           







                                                                                        
     

              


                      

                     

                   
                
                                                                     



                                         
 




                                    
                                 
















                                                                

            
                                        

      
                                      

                
                                                               

      














                                                              
     

                       





                                                                

      
                


                                     

                                 
 
                          

                        

                          
 

                                                                

      



                                                                          

                       
                                                                













                                                                               
                          


      

/*
 * BBS implementation dependendent part
 * 
 * The only two interfaces you must provide
 * 
 * #include "inntobbs.h" int receive_article(); 0 success not 0 fail
 * 
 * if (storeDB(HEADER[MID_H], hispaths) < 0) { .... fail }
 * 
 * int cancel_article_front( char *msgid );          0 success not 0 fail
 * 
 * char *ptr = (char*)DBfetch(msgid);
 * 
 * 收到之文章內容 (body)在 char *BODY, 檔頭 (header)在 char *HEADER[] SUBJECT_H,
 * FROM_H, DATE_H, MID_H, NEWSGROUPS_H, NNTPPOSTINGHOST_H, NNTPHOST_H,
 * CONTROL_H, PATH_H, ORGANIZATION_H
 */

/*
 * Sample Implementation
 * 
 * receive_article()         --> post_article()   --> bbspost_write_post();
 * cacnel_article_front(mid) --> cancel_article() --> bbspost_write_cancel();
 */

#include "bbs.h"
#include "externs.h"
#include <stdlib.h>
#ifndef _XOPEN_SOURCE
#define _XOPEN_SOURCE /* glibc2 needs this */
#endif
#include <time.h>
#ifndef PowerBBS
#include "innbbsconf.h"
#include "daemon.h"
#include "bbslib.h"
#include "inntobbs.h"
#include "antisplam.h"
#include "his.h"

extern int      Junkhistory;

char           *post_article ARG((char *, char *, char *, int (*) (), char *, char *));
int cancel_article ARG((char *, char *, char *));


#ifdef  MapleBBS
#include "config.h"
#include "pttstruct.h"
#define _BBS_UTIL_C_
#else
report()
{
    /* Function called from record.o */
    /* Please leave this function empty */
}
#endif


#if defined(PalmBBS)

#ifndef PATH
#define PATH XPATH
#endif

#ifndef HEADER
#define HEADER XHEADER
#endif

#endif

/* process post write */
int
bbspost_write_post(fh, board, filename)
    int             fh;
    char           *board;
    char           *filename;
{
    char           *fptr, *ptr;
    FILE           *fhfd = fdopen(fh, "w");

    if (fhfd == NULL) {
    bbslog("can't fdopen, maybe disk full\n");
    return -1;
    }
    fprintf(fhfd, "發信人: %.60s, 看板: %s\n", FROM, board);
    fprintf(fhfd, "標  題: %.70s\n", SUBJECT);
    fprintf(fhfd, "發信站: %.43s (%s)\n", SITE, DATE);
    fprintf(fhfd, "轉信站: %.70s\n", PATH);

#ifndef MapleBBS
    if (POSTHOST != NULL) {
    fprintf(fhfd, "Origin: %.70s\n", POSTHOST);
    }
#endif

    fprintf(fhfd, "\n");
    for (fptr = BODY, ptr = strchr(fptr, '\r'); ptr != NULL && *ptr != '\0'; fptr = ptr + 1, ptr = strchr(fptr, '\r')) {
    int             ch = *ptr;
    *ptr = '\0';
    fputs(fptr, fhfd);
    *ptr = ch;
    }
    fputs(fptr, fhfd);

    fflush(fhfd);
    fclose(fhfd);
    return 0;
}

#ifdef  KEEP_NETWORK_CANCEL
/* process cancel write */
bbspost_write_cancel(fh, board, filename)
    int             fh;
    char           *board, *filename;
{
    char           *fptr, *ptr;
    FILE           *fhfd = fdopen(fh, "w"), *fp;
    char            buffer[256];

    if (fhfd == NULL) {
    bbslog("can't fdopen, maybe disk full\n");
    return -1;
    }
    fprintf(fhfd, "發信人: %s, 信區: %s\n", FROM, board);
    fprintf(fhfd, "標  題: %s\n", SUBJECT);
    fprintf(fhfd, "發信站: %.43s (%s)\n", SITE, DATE);
    fprintf(fhfd, "轉信站: %.70s\n", PATH);
    if (HEADER[CONTROL_H] != NULL) {
    fprintf(fhfd, "Control: %s\n", HEADER[CONTROL_H]);
    }
    if (POSTHOST != NULL) {
    fprintf(fhfd, "Origin: %s\n", POSTHOST);
    }
    fprintf(fhfd, "\n");
    for (fptr = BODY, ptr = strchr(fptr, '\r'); ptr != NULL && *ptr != '\0'; fptr = ptr + 1, ptr = strchr(fptr, '\r')) {
    int             ch = *ptr;
    *ptr = '\0';
    fputs(fptr, fhfd);
    *ptr = ch;
    }
    fputs(fptr, fhfd);
    if (POSTHOST != NULL) {
    fprintf(fhfd, "\n * Origin: ● %.26s ● From: %.40s\n", SITE, POSTHOST);
    }
    fprintf(fhfd, "\n---------------------\n");
    fp = fopen(filename, "r");
    if (fp == NULL) {
    bbslog("can't open %s\n", filename);
    return -1;
    }
    while (fgets(buffer, sizeof buffer, fp) != NULL) {
    fputs(buffer, fhfd);
    }
    fclose(fp);
    fflush(fhfd);
    fclose(fhfd);

    {
    fp = fopen(filename, "w");
    if (fp == NULL) {
        bbslog("can't write %s\n", filename);
        return -1;
    }
    fprintf(fp, "發信人: %s, 信區: %s\n", FROM, board);
    fprintf(fp, "標  題: %.70s\n", SUBJECT);
    fprintf(fp, "發信站: %.43s (%s)\n", SITE, DATE);
    fprintf(fp, "轉信站: %.70s\n", PATH);
    if (POSTHOST != NULL) {
        fprintf(fhfd, "Origin: %s\n", POSTHOST);
    }
    if (HEADER[CONTROL_H] != NULL) {
        fprintf(fhfd, "Control: %s\n", HEADER[CONTROL_H]);
    }
    fprintf(fp, "\n");
    for (fptr = BODY, ptr = strchr(fptr, '\r'); ptr != NULL && *ptr != '\0'; fptr = ptr + 1, ptr = strchr(fptr, '\r')) {
        *ptr = '\0';
        fputs(fptr, fp);
    }
    fputs(fptr, fp);
    if (POSTHOST != NULL) {
        fprintf(fp, "\n * Origin: ● %.26s ● From: %.40s\n", SITE, POSTHOST);
    }
    fclose(fp);
    }
    return 0;
}
#endif


int
bbspost_write_control(fh, board, filename)
    int             fh;
    char           *board;
    char           *filename;
{
    char           *fptr, *ptr;
    FILE           *fhfd = fdopen(fh, "w");

    if (fhfd == NULL) {
    bbslog("can't fdopen, maybe disk full\n");
    return -1;
    }
    fprintf(fhfd, "Path: %s!%s\n", MYBBSID, HEADER[PATH_H]);
    fprintf(fhfd, "From: %s\n", FROM);
    fprintf(fhfd, "Newsgroups: %s\n", GROUPS);
    fprintf(fhfd, "Subject: %s\n", SUBJECT);
    fprintf(fhfd, "Date: %s\n", DATE);
    fprintf(fhfd, "Organization: %s\n", SITE);
    if (POSTHOST != NULL) {
    fprintf(fhfd, "NNTP-Posting-Host: %.70s\n", POSTHOST);
    }
    if (HEADER[CONTROL_H] != NULL) {
    fprintf(fhfd, "Control: %s\n", HEADER[CONTROL_H]);
    }
    if (HEADER[APPROVED_H] != NULL) {
    fprintf(fhfd, "Approved: %s\n", HEADER[APPROVED_H]);
    }
    if (HEADER[DISTRIBUTION_H] != NULL) {
    fprintf(fhfd, "Distribution: %s\n", HEADER[DISTRIBUTION_H]);
    }
    fprintf(fhfd, "\n");
    for (fptr = BODY, ptr = strchr(fptr, '\r'); ptr != NULL && *ptr != '\0'; fptr = ptr + 1, ptr = strchr(fptr, '\r')) {
    int             ch = *ptr;
    *ptr = '\0';
    fputs(fptr, fhfd);
    *ptr = ch;
    }
    fputs(fptr, fhfd);


    fflush(fhfd);
    fclose(fhfd);
    return 0;
}


time_t          datevalue;


/* process cancel write */
int
receive_article()
{
    char           *user, *userptr;
    char           *ngptr, *pathptr;
    char          **splitptr;
    static char     userid[32];
    static char     xdate[32];
    static char     xpath[180];
    newsfeeds_t    *nf;
    char           *boardhome;
    char            hispaths[4096];
    char            firstpath[PATHLEN], *firstpathbase;
    char           *lesssym, *nameptrleft, *nameptrright;

#ifdef HMM_USE_ANTI_SPAM
    int             i;
    char           *notitle[] =
    {NULL},
                   *nofrom[] =
    {NULL},
                   *nocont[] =
    {"http://msi-team.com", "http://affluence.rithosts.net",
     "http://www.fly-team.com", "http://fly-team.com",
     "http://whymis.com", "http://www.msihk.com",
     "http://Autopost.e8d8d.com", "http://www.e8d8d.com",
     "http://whymsi.com", "httpwww.e8d8d.com", "http://freeway.163.to/",
     "MSI團隊", "MSI系統", "MSI經營", "本訊息由AUTO POST發送",
     "MSI團隊", "USANA", "http://uuu.to/ommplan",
     "靠一份網路事業創業與加盟圓夢", "http://www.usanaomm.net",
     "優莎娜MSI", "Robert G.Allen", "http://www.whyomm.cn",
     "http://home.pchome.com.tw/store/deryes/",
     "http://xyzoo.hkoo.net/", "mypiece.com/",
     "【新舊歐盟醫學院招生】", "《台鹽多寶在家工作系統》",
     "jin-hua@hotmail.com", "010-82330590", "http://www.s-bus.com",
     "http://fone.4hk.cc/mkslk", "http://kiki.hkoo.net/fongy",
     "http://myokay.nowgo.net/jojo/", "0911685648",
     "http://digi.hkgo.cc/mile", "http://love520.hk852.cc/mytw",
     "美容保養品公司USANA", "Robert G. Allen", "Multiple Streams of Income",
     "http://www.twstars.com", "36005081", "3123835",
     ".hkgo.cc/", ".hk852.cc/", ".4hk.cc/", ".2hk.cc/", ".xdd.cc/",
     ".hkoo.net/", ".nowgo.net/", "http://www.taconet.com.tw/jscha/",
     "www.ejiajia.net", "ufjt0356@ms9.hinet.net", "jt0356@yahoo.com.tw",
     "http://www.IT-Test.Net", "http://uuu.to/", "greenhouse6688",
     "http://www.s-bus.com", "http://goods.sytes.net/", ".uni.cc/",
     "http://www.1-care.com", "美商優莎納生技公司", "Http://www.It-Test.Net",
     "http://home.pchome.com.tw/happy/eykk6767/", "http://www.agelopp.com/",
     "http://www.togetrich.net", "http://www.newchance.ligsystem.com/",
     "jimtist@yahoo.com", "http://fleamarket.mine.nu", "http://e-car.mine.nu",
     "漂亮美媚", "超 多 美 眉", "漂亮妹妹", "http://%6F",
     ".mini.to/", "-------- ### --------", "babylove.24cc.cc",
     "This is a multi-part message in MIME format.",
     "OCAgIFNreXBlOmp1bnNreWVwCg==", "ooqq.bbs@bbs.wretch.cc",
     "http://jsvcd.3cc.cc", "http://bestgirl.mytw.net", "http://98.to/",
     "http://www.boss888.net", "amuro.bbs@bbs.csie.nctu.edu.tw",
     "請問有需要幫忙報告.論文spss統計分析嗎",
     "http://www.whymsi.com", "http://www.msi-team.com/", NULL};
#endif

    if (FROM == NULL) {
    bbslog(":Err: article without usrid %s\n", MSGID);
    return 0;
    } else {
#ifdef HMM_USE_ANTI_SPAM
    for (i = 0; nofrom[i]; i++)
        if (strstr(FROM, nofrom[i])) {
        bbslog(":Ptt: spam from [%s]: %s\n", nofrom[i], FROM);
        return 0;
        }
#endif
    }

    if (!BODY) {
    bbslog(":Err: article without body %s\n", MSGID);
    return 0;
    } else {
#ifdef HMM_USE_ANTI_SPAM
    for (i = 0; nocont[i]; i++)
        if (strstr(BODY, nocont[i])) {
        bbslog(":Ptt: spam body: %s\n", nocont[i]);
        return 0;
        }
#endif
    }

    if (!SUBJECT) {
    bbslog(":Err: article without subject %s\n", MSGID);
    return 0;
    } else {
#ifdef HMM_USE_ANTI_SPAM
    for (i = 0; notitle[i]; i++)
        if (strstr(SUBJECT, notitle[i])) {
        bbslog(":Ptt: spam title [%s]: %s\n", notitle[i], SUBJECT);
        return 0;
        }
#endif
    }


    user = (char *)strchr(FROM, '@');
    lesssym = (char *)strchr(FROM, '<');
    nameptrleft = NULL, nameptrright = NULL;
    if (lesssym == NULL || lesssym >= user) {
    lesssym = FROM;
    nameptrleft = strchr(FROM, '(');
    if (nameptrleft != NULL)
        nameptrleft++;
    nameptrright = strrchr(FROM, ')');
    } else {
    nameptrleft = FROM;
    nameptrright = strrchr(FROM, '<');
    lesssym++;
    }
    if (user != NULL) {
    *user = '\0';
    userptr = (char *)strchr(FROM, '.');
    if (userptr != NULL) {
        *userptr = '\0';
        strncpy(userid, lesssym, sizeof userid);
        *userptr = '.';
    } else {
        strncpy(userid, lesssym, sizeof userid);
    }
    *user = '@';
    } else {
    strncpy(userid, lesssym, sizeof userid);
    }
    strcat(userid, ".");

    {
    struct tm       tmbuf;

    /* RFC-1036 says the second format is the correct one.  But we keep
     * the first format for backward compatible.
     */
    if (strptime(DATE, "%d %b %Y %X ", &tmbuf) != NULL)
        datevalue = timegm(&tmbuf);
    else if (strptime(DATE, "%a, %d %b %Y %X ", &tmbuf) != NULL)
        datevalue = timegm(&tmbuf);
    else
        datevalue = -1;
    }

    if (datevalue > 0) {
    char           *p;
    strncpy(xdate, ctime(&datevalue), sizeof(xdate));
    p = (char *)strchr(xdate, '\n');
    if (p != NULL)
        *p = '\0';
    DATE = xdate;
    }
#ifndef MapleBBS
    if (SITE == NULL || *SITE == '\0') {
    if (nameptrleft != NULL && nameptrright != NULL) {
        char            savech = *nameptrright;
        *nameptrright = '\0';
        strncpy(sitebuf, nameptrleft, sizeof sitebuf);
        *nameptrright = savech;
        SITE = sitebuf;
    } else
        /* SITE = "(Unknown)"; */
        SITE = "";
    }
    if (strlen(MYBBSID) > 70) {
    bbslog(" :Err: your bbsid %s too long\n", MYBBSID);
    return 0;
    }
#endif

    sprintf(xpath, "%s!%.*s", MYBBSID, (int)(sizeof(xpath) - strlen(MYBBSID) - 2), PATH);
    PATH = xpath;
    for (pathptr = PATH; pathptr != NULL && (pathptr = strstr(pathptr, ".edu.tw")) != NULL;) {
    if (pathptr != NULL) {
        strcpy(pathptr, pathptr + 7);
    }
    }
    xpath[71] = '\0';

#ifndef MapleBBS
    echomaillog();
#endif

    *hispaths = '\0';
    splitptr = (char **)BNGsplit(GROUPS);
    firstpath[0] = '\0';
    firstpathbase = firstpath;

    for (ngptr = *splitptr; ngptr != NULL; ngptr = *(++splitptr)) {
    char           *boardptr, *nboardptr;

    if (*ngptr == '\0')
        continue;
    nf = (newsfeeds_t *) search_group(ngptr);
    if (nf == NULL) {
        bbslog("unwanted \'%s\'\n", ngptr);
        continue;
    }
    if (nf->board == NULL || !*nf->board)
        continue;
    if (nf->path == NULL || !*nf->path)
        continue;
    for (boardptr = nf->board, nboardptr = (char *)strchr(boardptr, ','); boardptr != NULL && *boardptr != '\0'; nboardptr = (char *)strchr(boardptr, ',')) {
        if (nboardptr != NULL) {
        *nboardptr = '\0';
        }
        if (*boardptr == '\t') {
        goto boardcont;
        }
        boardhome = (char *)fileglue("%s/boards/%c/%s", BBSHOME, boardptr[0], boardptr);
        if (!dashd(boardhome)) {
        bbslog(":Err: unable to write %s\n", boardhome);
        } else {
        char           *fname;
        /*
         * if ( !dashd( boardhome )) { bbslog( ":Err: unable to write
         * %s\n",boardhome); testandmkdir(boardhome); }
         */
        fname = (char *)post_article(boardhome, userid, boardptr,
                       bbspost_write_post, NULL, firstpath);
        if (fname != NULL) {
            fname = (char *)fileglue("%s/%s", boardptr, fname);
            if (firstpath[0] == '\0') {
            sprintf(firstpath, "%s/boards/%c, %s", BBSHOME, fname[0], fname);
            firstpathbase = firstpath + strlen(BBSHOME) + strlen("/boards/x/");
            }
            if (strlen(fname) + strlen(hispaths) + 1 < sizeof(hispaths)) {
            strcat(hispaths, fname);
            strcat(hispaths, " ");
            }
        } else {
            bbslog("fname is null %s\n", boardhome);
            return -1;
        }
        }

    boardcont:
        if (nboardptr != NULL) {
        *nboardptr = ',';
        boardptr = nboardptr + 1;
        } else
        break;

    }           /* for board1,board2,... */
    /*
     * if (nngptr != NULL) ngptr = nngptr + 1; else break;
     */
    if (*firstpathbase)
        feedfplog(nf, firstpathbase, 'P');
    }
    if (*hispaths)
    bbsfeedslog(hispaths, 'P');

    if (Junkhistory || *hispaths) {
    if (storeDB(HEADER[MID_H], hispaths) < 0) {
        bbslog("store DB fail\n");
        /* I suspect here will introduce duplicated articles */
        /* return -1; */
    }
    }
    return 0;
}

int
receive_control(void)
{
    char           *boardhome, *fname;
    char            firstpath[PATHLEN], *firstpathbase;
    char          **splitptr, *ngptr;
    newsfeeds_t    *nf;

    bbslog("control post %s\n", HEADER[CONTROL_H]);
    boardhome = (char *)fileglue("%s/boards/c/control", BBSHOME);
    testandmkdir(boardhome);
    *firstpath = '\0';
    if (dashd(boardhome)) {
    fname = (char *)post_article(boardhome, FROM, "control", bbspost_write_control, NULL, firstpath);
    if (fname != NULL) {
        if (firstpath[0] == '\0')
        sprintf(firstpath, "%s/boards/c/control/%s", BBSHOME, fname);
        if (storeDB(HEADER[MID_H], (char *)fileglue("control/%s", fname)) < 0) {
        }
        bbsfeedslog(fileglue("control/%s", fname), 'C');
        firstpathbase = firstpath + strlen(BBSHOME) + strlen("/boards/x/");
        splitptr = (char **)BNGsplit(GROUPS);
        for (ngptr = *splitptr; ngptr != NULL; ngptr = *(++splitptr)) {
        if (*ngptr == '\0')
            continue;
        nf = (newsfeeds_t *) search_group(ngptr);
        if (nf == NULL)
            continue;
        if (nf->board == NULL)
            continue;
        if (nf->path == NULL)
            continue;
        feedfplog(nf, firstpathbase, 'C');
        }
    }
    }
    return 0;
}

int
cancel_article_front(msgid)
    char           *msgid;
{
    char           *ptr = (char *)DBfetch(msgid);
    char           *filelist, filename[2048];
    char            histent[4096];
    char            firstpath[PATHLEN], *firstpathbase = firstpath;
    if (ptr == NULL) {
    bbslog("cancel failed(DBfetch): %s\n", msgid);
    return 0;
    }
    strncpy(histent, ptr, sizeof histent);
    ptr = histent;

#ifdef DEBUG
    printf("**** try to cancel %s *****\n", ptr);
#endif

    filelist = strchr(ptr, '\t');
    if (filelist != NULL) {
    filelist++;
    }
    *firstpath = '\0';
    for (ptr = filelist; ptr && *ptr;) {
    char           *file;
    for (; *ptr && isspace(*ptr); ptr++);
    if (*ptr == '\0')
        break;
    file = ptr;
    for (ptr++; *ptr && !isspace(*ptr); ptr++);
    if (*ptr != '\0') {
        *ptr++ = '\0';
    }
    sprintf(filename, "%s/boards/%c/%s", BBSHOME, file[0], file);
    bbslog("cancel post %s\n", filename);
    if (dashf(filename)) {
        FILE           *fp = fopen(filename, "r");
        char            buffer[1024];
        char            xfrom0[100], xfrom[100], xpath[1024];

        if (fp == NULL)
        continue;
        strncpy(xfrom0, HEADER[FROM_H], 99);
        xfrom0[99] = 0;
        strtok(xfrom0, ", ");
        while (fgets(buffer, sizeof buffer, fp) != NULL) {
        char           *hptr;
        if (buffer[0] == '\n')
            break;
        hptr = strchr(buffer, '\n');
        if (hptr != NULL)
            *hptr = '\0';
        if (strncmp(buffer, "發信人: ", 8) == 0) {
            strncpy(xfrom, buffer + 8, 99);
            xfrom[99] = 0;
            strtok(xfrom, ", ");
        } else if (strncmp(buffer, "轉信站: ", 8) == 0) {
            strcpy(xpath, buffer + 8);
        }
        }
        fclose(fp);
        if (strcmp(xfrom0, xfrom) && !search_issuer(FROM)) {
        bbslog("Invalid cancel %s, path: %s!%s, [`%s` != `%s`]\n",
               FROM, MYBBSID, PATH, xfrom0, xfrom);
        return 0;
        }
#ifdef  KEEP_NETWORK_CANCEL
        bbslog("cancel post %s\n", filename);
        boardhome = (char *)fileglue("%s/boards/d/deleted", BBSHOME);
        testandmkdir(boardhome);
        if (dashd(boardhome)) {
        char            subject[1024];
        char           *fname;
        if (POSTHOST) {
            sprintf(subject, "cancel by: %.1000s", POSTHOST);
        } else {
            char           *body, *body2;
            body = strchr(BODY, '\r');
            if (body != NULL)
            *body = '\0';
            body2 = strchr(BODY, '\n');
            if (body2 != NULL)
            *body = '\0';
            sprintf(subject, "%.1000s", BODY);
            if (body != NULL)
            *body = '\r';
            if (body2 != NULL)
            *body = '\n';
        }
        if (*subject) {
            SUBJECT = subject;
        }
        fname = (char *)post_article(boardhome, FROM, "deleted", bbspost_write_cancel, filename, firstpath);
        if (fname != NULL) {
            if (firstpath[0] == '\0') {
            sprintf(firstpath, "%s/boards/d/deleted/%s", BBSHOME, fname);
            firstpathbase = firstpath + strlen(BBSHOME) + strlen("/boards/x/");
            }
            if (storeDB(HEADER[MID_H], (char *)fileglue("deleted/%s", fname)) < 0) {
            /* should do something */
            bbslog("store DB fail\n");
            /* return -1; */
            }
            bbsfeedslog(fileglue("deleted/%s", fname), 'D');

#ifdef OLDDISPATCH
            {
            char            board[256];
            newsfeeds_t    *nf;
            char           *filebase = filename + strlen(BBSHOME) + strlen("/boards/x/");
            char           *filetail = strrchr(filename, '/');
            if (filetail != NULL) {
                strncpy(board, filebase, filetail - filebase);
                nf = (newsfeeds_t *) search_board(board);
                if (nf != NULL && nf->board && nf->path) {
                feedfplog(nf, firstpathbase, 'D');
                }
            }
            }
#endif
        } else {
            bbslog(" fname is null %s %s\n", boardhome, filename);
            return -1;
        }
        }
#else
        /* bbslog("**** %s should be removed\n", filename); */
        /*
         * unlink(filename);
         */
#endif

        {
        char           *fp = strrchr(file, '/');
        if (fp != NULL) {
            *fp = '\0';
            cancel_article(BBSHOME, file, fp + 1);
            *fp = '/';
        }
        }
    }
    }
    if (*firstpath) {
    char          **splitptr, *ngptr;
    newsfeeds_t    *nf;
    splitptr = (char **)BNGsplit(GROUPS);
    for (ngptr = *splitptr; ngptr != NULL; ngptr = *(++splitptr)) {
        if (*ngptr == '\0')
        continue;
        nf = (newsfeeds_t *) search_group(ngptr);
        if (nf == NULL)
        continue;
        if (nf->board == NULL)
        continue;
        if (nf->path == NULL)
        continue;
        feedfplog(nf, firstpathbase, 'D');
    }
    }
    return 0;
}


#if defined(PhoenixBBS) || defined(SecretBBS) || defined(PivotBBS) || defined(MapleBBS)
/* for PhoenixBBS's post article and cancel article */
#include "config.h"


char           *
post_article(homepath, userid, board, writebody, pathname, firstpath)
    char           *homepath;
    char           *userid, *board;
    int             (*writebody) ();
    char           *pathname, *firstpath;
{
    struct fileheader_t header;
    char           *subject = SUBJECT;
    char            index[PATHLEN];
    static char     name[PATHLEN];
    char            article[PATHLEN];
    FILE           *fidx;
    int             fh, bid;
    time_t          now;
    int             linkflag;
    /*
     * Ptt if(bad_subject(subject))  return NULL;
     */
    sprintf(index, "%s/.DIR", homepath);
    if ((fidx = fopen(index, "r")) == NULL) {
    if ((fidx = fopen(index, "w")) == NULL) {
        bbslog(":Err: Unable to post in %s.\n", homepath);
        return NULL;
    }
    }
    fclose(fidx);

    now = time(NULL);
    while (1) {
        /* TODO: extract record.c:stampfile_u() to common lib */
    sprintf(name, "M.%ld.A.%3.3X", (long)++now, (unsigned int)(random() & 0xFFF));
    sprintf(article, "%s/%s", homepath, name);
    fh = open(article, O_CREAT | O_EXCL | O_WRONLY, 0644);
    if (fh >= 0)
        break;
    if (errno != EEXIST) {
        bbslog(" Err: can't writable or other errors\n");
        return NULL;
    }
    }

#ifdef DEBUG
    printf("post to %s\n", article);
#endif

    linkflag = 1;
    if (firstpath && *firstpath) {
    close(fh);
    unlink(article);

#ifdef DEBUGLINK
    bbslog("try to link %s to %s", firstpath, article);
#endif

    linkflag = link(firstpath, article);
    if (linkflag) {
        fh = open(article, O_CREAT | O_EXCL | O_WRONLY, 0644);
    }
    }
    if (linkflag) {
    if (writebody) {
        if ((*writebody) (fh, board, pathname) < 0)
        return NULL;
    } else {
        if (bbspost_write_post(fh, board, pathname) < 0)
        return NULL;
    }
    close(fh);
    }
    bzero((void *)&header, sizeof(header));

#ifndef MapleBBS
    strcpy(header.filename, name);
    strncpy(header.owner, userid, IDLEN);
    strncpy(header.title, subject, STRLEN);
    header.filename[STRLEN - 1] = 'M';
#else

    strcpy(header.filename, name);
    if (userid[IDLEN-1])
    {
    userid[IDLEN-1] = '.';
    userid[IDLEN] = '\0';
    }   
    strcpy(header.owner, userid);
    strncpy(header.title, subject, TTLEN);
    /* no need to apply this... FILE_MULTI is used for mail group reply only now. */
    // header.filemode |= FILE_MULTI;
    {
    struct tm      *ptime;
    ptime = localtime(&datevalue);
    sprintf(header.date, "%2d/%02d", ptime->tm_mon + 1, ptime->tm_mday);
    }
#endif
    {
    int     i;
    for( i = 0 ; header.title[i] != 0 && i < sizeof(header.title) ; ++i )
        if( header.title[i] == '\n' ||
        header.title[i] == '\r' ||
        header.title[i] == '\033' ){
        header.title[i] = 0;
        break;
        }
    }
    append_record(index, &header, sizeof(header));

    if ((bid = getbnum(board)) > 0) {
    touchbtotal(bid);
    }
    return name;
}

/*
 * woju Cross-fs rename()
 */

#if 0 // moved to libbbsutil
int
Rename(const char *src, const char *dst)
{
    char            cmd[200];

    bbslog("Rename: %s -> %s\n", src, dst);
    if (rename(src, dst) == 0)
    return 0;

    sprintf(cmd, "/bin/mv %s %s", src, dst);
    return system(cmd);
}
#endif


void
cancelpost(fileheader_t * fhdr, char *boardname)
{
    int             fd;
    char            fpath[PATHLEN];

    sprintf(fpath, BBSHOME "/boards/%c/%s/%s", boardname[0], boardname, fhdr->filename);
    if ((fd = open(fpath, O_RDONLY)) >= 0) {
    fileheader_t    postfile;
    char            fn2[PATHLEN] = BBSHOME "/boards/d/deleted";

    stampfile(fn2, &postfile);
    memcpy(postfile.owner, fhdr->owner, IDLEN + TTLEN + 10);
    close(fd);
    Rename(fpath, fn2);
    strcpy(strrchr(fn2, '/') + 1, ".DIR");
    append_record(fn2, &postfile, sizeof(postfile));
    } else
    bbslog("cancelpost: %s opened error\n", fpath);
}


/* ---------------------------- */
/* new/old/lock file processing */
/* ---------------------------- */

#if 0
typedef struct {
    char            newfn[PATHLEN];
    char            oldfn[PATHLEN];
    char            lockfn[PATHLEN];
}               nol;


static void
nolfilename(n, fpath)
    nol            *n;
    char           *fpath;
{
    sprintf(n->newfn, "%s.new", fpath);
    sprintf(n->oldfn, "%s.old", fpath);
    sprintf(n->lockfn, "%s.lock", fpath);
}

int
delete_record(const char *fpath, int size, int id)
{
    nol             my;
    char            abuf[512];
    int             fdr, fdw, fd;
    int             count;
    fileheader_t    fhdr;

    nolfilename(&my, fpath);

    if ((fd = open(my.lockfn, O_RDWR | O_CREAT | O_APPEND, 0644)) == -1)
    return -1;
    flock(fd, LOCK_EX);

    if ((fdr = open(fpath, O_RDONLY, 0)) == -1) {

#ifdef  HAVE_REPORT
    report("delete_record failed!!! (open)");
#endif

    flock(fd, LOCK_UN);
    close(fd);
    return -1;
    }
    if ((fdw = open(my.newfn, O_WRONLY | O_CREAT | O_EXCL, 0644)) == -1) {
    flock(fd, LOCK_UN);

#ifdef  HAVE_REPORT
    report("delete_record failed!!! (open tmpfile)");
#endif

    close(fd);
    close(fdr);
    return -1;
    }
    count = 1;
    while (read(fdr, abuf, size) == size) {
    if (id == count) {
        memcpy(&fhdr, abuf, sizeof(fhdr));
        bbslog("delete_record: %d, %s, %s\n", count, fhdr.owner, fhdr.title);
    }
    if (id != count++ && (write(fdw, abuf, size) == -1)) {

        bbslog("delete_record: %s failed!!! (write)\n", fpath);
#ifdef  HAVE_REPORT
        report("delete_record failed!!! (write)");
#endif

        unlink(my.newfn);
        close(fdr);
        close(fdw);
        flock(fd, LOCK_UN);
        close(fd);
        return -1;
    }
    }
    close(fdr);
    close(fdw);
    if (Rename(fpath, my.oldfn) == -1 || Rename(my.newfn, fpath) == -1) {

#ifdef  HAVE_REPORT
    report("delete_record failed!!! (Rename)");
#endif

    flock(fd, LOCK_UN);
    close(fd);
    return -1;
    }
    flock(fd, LOCK_UN);
    close(fd);
    return 0;
}
#endif

int
cancel_article(homepath, board, file)
    char           *homepath;
    char           *board, *file;
{
    struct fileheader_t header;
    struct stat     state;
    char            dirname[PATHLEN];
    long            size, time, now;
    int             fd, lower, ent;


    if (file == NULL || file[0] != 'M' || file[1] != '.' ||
    (time = atoi(file + 2)) <= 0) {
    bbslog("cancel_article: invalid filename `%s`\n", file);
    return 0;
    }
    size = sizeof(header);
    sprintf(dirname, "%s/boards/%c/%s/.DIR", homepath, board[0], board);
    if ((fd = open(dirname, O_RDONLY)) == -1) {
    bbslog("cancel_article: open `%s` error\n", dirname);
    return 0;
    }
    fstat(fd, &state);
    ent = ((long)state.st_size) / size;
    lower = 0;
    while (1) {
    ent -= 8;
    if (ent <= 0 || lower >= 2)
        break;
    lseek(fd, size * ent, SEEK_SET);
    if (read(fd, &header, size) != size) {
        ent = 0;
        break;
    }
    now = atoi(header.filename + 2);
    lower = (now < time) ? lower + 1 : 0;
    }
    if (ent < 0)
    ent = 0;
    while (read(fd, &header, size) == size) {
    if (strcmp(file, header.filename) == 0) {
        if ((header.filemode & FILE_MARKED)
         || (header.filemode & FILE_DIGEST) || (header.owner[0] == '-')
         || !strchr(header.owner,'.'))
        break;
        delete_record(dirname, sizeof(fileheader_t), lseek(fd, 0, SEEK_CUR) / size);
        cancelpost(&header, board);
        break;
    }
    now = atoi(header.filename + 2);
    if (now > time)
        break;
    }
    close(fd);
    return 0;
}

#elif defined(PalmBBS)
#undef PATH XPATH
#undef HEADER XHEADER
#include "server.h"

char           *
post_article(homepath, userid, board, writebody, pathname, firstpath)
    char           *homepath;
    char           *userid, *board;
    int             (*writebody) ();
    char           *pathname, *firstpath;
{
    PATH            msgdir, msgfile;
    static PATH     name;

    READINFO        readinfo;
    SHORT           fileid;
    char            buf[PATHLEN];
    struct stat     stbuf;
    int             fh;

    strcpy(msgdir, homepath);
    if (stat(msgdir, &stbuf) == -1 || !S_ISDIR(stbuf.st_mode)) {
    /* A directory is missing! */
    bbslog(":Err: Unable to post in %s.\n", msgdir);
    return NULL;
    }
    get_filelist_ids(msgdir, &readinfo);

    for (fileid = 1; fileid <= BBS_MAX_FILES; fileid++) {
    int             oumask;
    if (test_readbit(&readinfo, fileid))
        continue;
    fileid_to_fname(msgdir, fileid, msgfile);
    sprintf(name, "%04x", fileid);

#ifdef DEBUG
    printf("post to %s\n", msgfile);
#endif

    if (firstpath && *firstpath) {

#ifdef DEBUGLINK
        bbslog("try to link %s to %s", firstpath, msgfile);
#endif

        if (link(firstpath, msgfile) == 0)
        break;
    }
    oumask = umask(0);
    fh = open(msgfile, O_CREAT | O_EXCL | O_WRONLY, 0664);
    umask(oumask);
    if (writebody) {
        if ((*writebody) (fh, board, pathname) < 0)
        return NULL;
    } else {
        if (bbspost_write_post(fh, board, pathname) < 0)
        return NULL;
    }
    close(fh);
    break;
    }

#ifdef CACHED_OPENBOARD
    {
    char           *bname;
    bname = strrchr(msgdir, '/');
    if (bname)
        notify_new_post(++bname, 1, fileid, stbuf.st_mtime);
    }
#endif

    return name;
}

cancel_article(homepath, board, file)
    char           *homepath;
    char           *board, *file;
{
    PATH            fname;

#ifdef  CACHED_OPENBOARD
    PATH            bdir;
    struct stat     stbuf;

    sprintf(bdir, "%s/boards/%c/%s", homepath, board[0], board);
    stat(bdir, &stbuf);
#endif

    sprintf(fname, "%s/boards/%c/%s/%s", homepath, board[0], board, file);
    unlink(fname);
    /* kill it now! the function is far small then original..  :) */
    /* because it won't make system load heavy like before */

#ifdef CACHED_OPENBOARD
    notify_new_post(board, -1, hex2SHORT(file), stbuf.st_mtime);
#endif
}

#else
error("You should choose one of the systems: PhoenixBBS, PowerBBS, or PalmBBS")
#endif

#else

receive_article()
{
}

cancel_article_front(msgid)
    char           *msgid;
{
}
#endif