summaryrefslogtreecommitdiffstats
path: root/console/register.c
diff options
context:
space:
mode:
Diffstat (limited to 'console/register.c')
-rw-r--r--console/register.c3040
1 files changed, 3040 insertions, 0 deletions
diff --git a/console/register.c b/console/register.c
new file mode 100644
index 00000000..e0db7d37
--- /dev/null
+++ b/console/register.c
@@ -0,0 +1,3040 @@
+/* $Id$ */
+#include "bbs.h"
+
+#define FN_REGISTER_LOG "register.log" // global registration history
+#define FN_JUSTIFY "justify"
+#define FN_JUSTIFY_WAIT "justify.wait"
+#define FN_REJECT_NOTIFY "justify.reject"
+
+// New style (Regform2) file names:
+#define FN_REGFORM "regform" // registration form in user home
+#define FN_REGFORM_LOG "regform.log" // regform history in user home
+#define FN_REQLIST "reg.wait" // request list file, in global directory (replacing fn_register)
+
+////////////////////////////////////////////////////////////////////////////
+// Password Hash
+////////////////////////////////////////////////////////////////////////////
+
+// prototype of crypt()
+char *crypt(const char *key, const char *salt);
+
+char *
+genpasswd(char *pw)
+{
+ if (pw[0]) {
+ char saltc[2], c;
+ int i;
+
+ i = 9 * getpid();
+ saltc[0] = i & 077;
+ saltc[1] = (i >> 6) & 077;
+
+ for (i = 0; i < 2; i++) {
+ c = saltc[i] + '.';
+ if (c > '9')
+ c += 7;
+ if (c > 'Z')
+ c += 6;
+ saltc[i] = c;
+ }
+ return crypt(pw, saltc);
+ }
+ return "";
+}
+
+// NOTE it will clean string in "plain"
+int
+checkpasswd(const char *passwd, char *plain)
+{
+ int ok;
+ char *pw;
+
+ ok = 0;
+ pw = crypt(plain, passwd);
+ if(pw && strcmp(pw, passwd)==0)
+ ok = 1;
+ memset(plain, 0, strlen(plain));
+
+ return ok;
+}
+
+////////////////////////////////////////////////////////////////////////////
+// Value Validation
+////////////////////////////////////////////////////////////////////////////
+static int
+HaveRejectStr(const char *s, const char **rej)
+{
+ int i;
+ char *ptr, *rejectstr[] =
+ {"幹", "阿", "不", "你媽", "某", "笨", "呆", "..", "xx",
+ "你管", "管我", "猜", "天才", "超人",
+ "ㄅ", "ㄆ", "ㄇ", "ㄈ", "ㄉ", "ㄊ", "ㄋ", "ㄌ", "ㄍ", "ㄎ", "ㄏ",
+ "ㄐ", "ㄑ", "ㄒ", "ㄓ",/*"ㄔ",*/ "ㄕ", "ㄖ", "ㄗ", "ㄘ", "ㄙ",
+ "ㄧ", "ㄨ", "ㄩ", "ㄚ", "ㄛ", "ㄜ", "ㄝ", "ㄞ", "ㄟ", "ㄠ", "ㄡ",
+ "ㄢ", "ㄣ", "ㄤ", "ㄥ", "ㄦ", NULL};
+
+ if( rej != NULL )
+ for( i = 0 ; rej[i] != NULL ; ++i )
+ if( strstr(s, rej[i]) )
+ return 1;
+
+ for( i = 0 ; rejectstr[i] != NULL ; ++i )
+ if( strstr(s, rejectstr[i]) )
+ return 1;
+
+ if( (ptr = strstr(s, "ㄔ")) != NULL ){
+ if( ptr != s && strncmp(ptr - 1, "都市", 4) == 0 )
+ return 0;
+ return 1;
+ }
+ return 0;
+}
+
+static int
+removespace(char *s)
+{
+ int i, index;
+
+ for (i = 0, index = 0; s[i]; i++) {
+ if (s[i] != ' ')
+ s[index++] = s[i];
+ }
+ s[index] = '\0';
+ return index;
+}
+
+int
+bad_user_id(const char *userid)
+{
+ if(!is_validuserid(userid))
+ return 1;
+
+ if (strcasecmp(userid, str_new) == 0)
+ return 1;
+
+#ifdef NO_GUEST_ACCOUNT_REG
+ if (strcasecmp(userid, STR_GUEST) == 0)
+ return 1;
+#endif
+
+ /* in2: 原本是用strcasestr,
+ 不過有些人中間剛剛好出現這個字應該還算合理吧? */
+ if( strncasecmp(userid, "fuck", 4) == 0 ||
+ strncasecmp(userid, "shit", 4) == 0 )
+ return 1;
+
+ /*
+ * while((ch = *(++userid))) if(not_alnum(ch)) return 1;
+ */
+ return 0;
+}
+
+static char *
+isvalidname(char *rname)
+{
+#ifdef FOREIGN_REG
+ return NULL;
+#else
+ const char *rejectstr[] =
+ {"肥", "胖", "豬頭", "小白", "小明", "路人", "老王", "老李", "寶貝",
+ "先生", "帥哥", "老頭", "小姊", "小姐", "美女", "小妹", "大頭",
+ "公主", "同學", "寶寶", "公子", "大頭", "小小", "小弟", "小妹",
+ "妹妹", "嘿", "嗯", "爺爺", "大哥", "無",
+ NULL};
+ if( removespace(rname) && rname[0] < 0 &&
+ strlen(rname) >= 4 &&
+ !HaveRejectStr(rname, rejectstr) &&
+ strncmp(rname, "小", 2) != 0 && //起頭是「小」
+ strncmp(rname, "我是", 4) != 0 && //起頭是「我是」
+ !(strlen(rname) == 4 && strncmp(&rname[2], "兒", 2) == 0) &&
+ !(strlen(rname) >= 4 && strncmp(&rname[0], &rname[2], 2) == 0))
+ return NULL;
+ return "您的輸入不正確";
+#endif
+
+}
+
+static char *
+isvalidcareer(char *career)
+{
+#ifndef FOREIGN_REG
+ const char *rejectstr[] = {NULL};
+ if (!(removespace(career) && career[0] < 0 && strlen(career) >= 6) ||
+ strcmp(career, "家裡") == 0 || HaveRejectStr(career, rejectstr) )
+ return "您的輸入不正確";
+ if (strcmp(&career[strlen(career) - 2], "大") == 0 ||
+ strcmp(&career[strlen(career) - 4], "大學") == 0 ||
+ strcmp(career, "學生大學") == 0)
+ return "麻煩請加學校系所";
+ if (strcmp(career, "學生高中") == 0)
+ return "麻煩輸入學校名稱";
+#else
+ if( strlen(career) < 6 )
+ return "您的輸入不正確";
+#endif
+ return NULL;
+}
+
+static char *
+isvalidaddr(char *addr)
+{
+ const char *rejectstr[] =
+ {"地球", "銀河", "火星", NULL};
+
+ // addr[0] > 0: check if address is starting by Chinese.
+ if (!removespace(addr) || strlen(addr) < 15)
+ return "這個地址似乎並不完整";
+ if (strstr(addr, "信箱") != NULL || strstr(addr, "郵政") != NULL)
+ return "抱歉我們不接受郵政信箱";
+ if ((strstr(addr, "市") == NULL && strstr(addr, "巿") == NULL &&
+ strstr(addr, "縣") == NULL && strstr(addr, "室") == NULL) ||
+ HaveRejectStr(addr, rejectstr) ||
+ strcmp(&addr[strlen(addr) - 2], "段") == 0 ||
+ strcmp(&addr[strlen(addr) - 2], "路") == 0 ||
+ strcmp(&addr[strlen(addr) - 2], "巷") == 0 ||
+ strcmp(&addr[strlen(addr) - 2], "弄") == 0 ||
+ strcmp(&addr[strlen(addr) - 2], "區") == 0 ||
+ strcmp(&addr[strlen(addr) - 2], "市") == 0 ||
+ strcmp(&addr[strlen(addr) - 2], "街") == 0 )
+ return "這個地址似乎並不完整";
+ return NULL;
+}
+
+static char *
+isvalidphone(char *phone)
+{
+ int i;
+ for( i = 0 ; phone[i] != 0 ; ++i )
+ if( !isdigit((int)phone[i]) )
+ return "請不要加分隔符號";
+ if (!removespace(phone) ||
+ strlen(phone) < 9 ||
+ strstr(phone, "00000000") != NULL ||
+ strstr(phone, "22222222") != NULL ) {
+ return "這個電話號碼並不正確(請含區碼)" ;
+ }
+ return NULL;
+}
+
+
+////////////////////////////////////////////////////////////////////////////
+// Account Expiring
+////////////////////////////////////////////////////////////////////////////
+
+/* -------------------------------- */
+/* New policy for allocate new user */
+/* (a) is the worst user currently */
+/* (b) is the object to be compared */
+/* -------------------------------- */
+static int
+compute_user_value(const userec_t * urec, time4_t clock)
+{
+ int value;
+
+ /* if (urec) has XEMPT permission, don't kick it */
+ if ((urec->userid[0] == '\0') || (urec->userlevel & PERM_XEMPT)
+ /* || (urec->userlevel & PERM_LOGINOK) */
+ || !strcmp(STR_GUEST, urec->userid))
+ return 999999;
+ value = (clock - urec->lastlogin) / 60; /* minutes */
+
+ /* new user should register in 30 mins */
+ // XXX 目前 new acccount 並不會在 utmp 裡放 str_new...
+ if (strcmp(urec->userid, str_new) == 0)
+ return 30 - value;
+
+#if 0
+ if (!urec->numlogins) /* 未 login 成功者,不保留 */
+ return -1;
+ if (urec->numlogins <= 3) /* #login 少於三者,保留 20 天 */
+ return 20 * 24 * 60 - value;
+#endif
+ /* 未完成註冊者,保留 15 天 */
+ /* 一般情況,保留 120 天 */
+ return (urec->userlevel & PERM_LOGINOK ? 120 : 15) * 24 * 60 - value;
+}
+
+int
+check_and_expire_account(int uid, const userec_t * urec, int expireRange)
+{
+ char genbuf[200];
+ int val;
+ if ((val = compute_user_value(urec, now)) < 0) {
+ snprintf(genbuf, sizeof(genbuf), "#%d %-12s %15.15s %d %d %d",
+ uid, urec->userid, ctime4(&(urec->lastlogin)) + 4,
+ urec->numlogins, urec->numposts, val);
+
+ // 若超過 expireRange 則砍人,
+ // 不然就 return 0
+ if (-val > expireRange)
+ {
+ log_usies("DATED", genbuf);
+ // log_usies("CLEAN", genbuf);
+ kill_user(uid, urec->userid);
+ } else val = 0;
+ }
+ return val;
+}
+
+////////////////////////////////////////////////////////////////////////////
+// Regcode Support
+////////////////////////////////////////////////////////////////////////////
+
+#define REGCODE_INITIAL "v6" // always 2 characters
+
+static char *
+getregfile(char *buf)
+{
+ // not in user's home because s/he could zip his/her home
+ snprintf(buf, PATHLEN, "jobspool/.regcode.%s", cuser.userid);
+ return buf;
+}
+
+static char *
+makeregcode(char *buf)
+{
+ char fpath[PATHLEN];
+ int fd, i;
+ // prevent ambigious characters: oOlI
+ const char *alphabet = "qwertyuipasdfghjkzxcvbnmoQWERTYUPASDFGHJKLZXCVBNM";
+
+ /* generate a new regcode */
+ buf[13] = 0;
+ buf[0] = REGCODE_INITIAL[0];
+ buf[1] = REGCODE_INITIAL[1];
+ for( i = 2 ; i < 13 ; ++i )
+ buf[i] = alphabet[random() % strlen(alphabet)];
+
+ getregfile(fpath);
+ if( (fd = open(fpath, O_WRONLY | O_CREAT, 0600)) == -1 ){
+ perror("open");
+ exit(1);
+ }
+ write(fd, buf, 13);
+ close(fd);
+
+ return buf;
+}
+
+static char *
+getregcode(char *buf)
+{
+ int fd;
+ char fpath[PATHLEN];
+
+ getregfile(fpath);
+ if( (fd = open(fpath, O_RDONLY)) == -1 ){
+ buf[0] = 0;
+ return buf;
+ }
+ read(fd, buf, 13);
+ close(fd);
+ buf[13] = 0;
+ return buf;
+}
+
+void
+delregcodefile(void)
+{
+ char fpath[PATHLEN];
+ getregfile(fpath);
+ unlink(fpath);
+}
+
+////////////////////////////////////////////////////////////////////////////
+// Justify Utilities
+////////////////////////////////////////////////////////////////////////////
+
+static void
+justify_wait(char *userid, char *phone, char *career,
+ char *rname, char *addr, char *mobile)
+{
+ char buf[PATHLEN];
+ sethomefile(buf, userid, FN_JUSTIFY_WAIT);
+ if (phone[0] != 0) {
+ FILE* fn = fopen(buf, "w");
+ assert(fn);
+ fprintf(fn, "%s\n%s\ndummy\n%s\n%s\n%s\n",
+ phone, career, rname, addr, mobile);
+ fclose(fn);
+ }
+}
+
+static void
+email_justify(const userec_t *muser)
+{
+ char tmp[IDLEN + 1], buf[256], genbuf[256];
+ /*
+ * It is intended to use BBSENAME instead of BBSNAME here.
+ * Because recently many poor users with poor mail clients
+ * (or evil mail servers) cannot handle/decode Chinese
+ * subjects (BBSNAME) correctly, so we'd like to use
+ * BBSENAME here to prevent subject being messed up.
+ * And please keep BBSENAME short or it may be truncated
+ * by evil mail servers.
+ */
+ snprintf(buf, sizeof(buf),
+ " " BBSENAME " - [ %s ]", makeregcode(genbuf));
+
+ strlcpy(tmp, cuser.userid, sizeof(tmp));
+ // XXX dirty, set userid=SYSOP
+ strlcpy(cuser.userid, str_sysop, sizeof(cuser.userid));
+#ifdef HAVEMOBILE
+ if (strcmp(muser->email, "m") == 0 || strcmp(muser->email, "M") == 0)
+ mobile_message(mobile, buf);
+ else
+#endif
+ bsmtp("etc/registermail", buf, muser->email);
+ strlcpy(cuser.userid, tmp, sizeof(cuser.userid));
+ move(20,0);
+ clrtobot();
+ outs("我們即將寄出認證信 (您應該會在 10 分鐘內收到)\n"
+ "收到後您可以根據認證信標題的認證碼\n"
+ "輸入到 (U)ser -> (R)egister 後就可以完成註冊");
+ pressanykey();
+ return;
+}
+
+
+/* 使用者填寫註冊表格 */
+static void
+getfield(int line, const char *info, const char *desc, char *buf, int len)
+{
+ char prompt[STRLEN];
+ char genbuf[200];
+
+ // clear first
+ move(line+1, 0); clrtoeol();
+ move(line, 0); clrtoeol();
+ prints(" 原先設定:%-30.30s (%s)", buf, info);
+ snprintf(prompt, sizeof(prompt), " %s:", desc);
+ if (getdata_str(line + 1, 0, prompt, genbuf, len, DOECHO, buf))
+ strcpy(buf, genbuf);
+ move(line+1, 0); clrtoeol();
+ move(line, 0); clrtoeol();
+ prints(" %s:%s", desc, buf);
+}
+
+
+int
+setupnewuser(const userec_t *user)
+{
+ char genbuf[50];
+ char *fn_fresh = ".fresh";
+ userec_t utmp;
+ time_t clock;
+ struct stat st;
+ int fd, uid;
+
+ clock = now;
+
+ // XXX race condition...
+ if (dosearchuser(user->userid, NULL))
+ {
+ vmsg("手腳不夠快,別人已經搶走了!");
+ exit(1);
+ }
+
+ /* Lazy method : 先找尋已經清除的過期帳號 */
+ if ((uid = dosearchuser("", NULL)) == 0) {
+ /* 每 1 個小時,清理 user 帳號一次 */
+ if ((stat(fn_fresh, &st) == -1) || (st.st_mtime < clock - 3600)) {
+ if ((fd = open(fn_fresh, O_RDWR | O_CREAT, 0600)) == -1)
+ return -1;
+ write(fd, ctime(&clock), 25);
+ close(fd);
+ log_usies("CLEAN", "dated users");
+
+ fprintf(stdout, "尋找新帳號中, 請稍待片刻...\n\r");
+
+ if ((fd = open(fn_passwd, O_RDWR | O_CREAT, 0600)) == -1)
+ return -1;
+
+ /* 不曉得為什麼要從 2 開始... Ptt:因為SYSOP在1 */
+ for (uid = 2; uid <= MAX_USERS; uid++) {
+ passwd_query(uid, &utmp);
+ // tolerate for one year.
+ check_and_expire_account(uid, &utmp, 365*12*60);
+ }
+ }
+ }
+
+ /* initialize passwd semaphores */
+ if (passwd_init())
+ exit(1);
+
+ passwd_lock();
+
+ uid = dosearchuser("", NULL);
+ if ((uid <= 0) || (uid > MAX_USERS)) {
+ passwd_unlock();
+ vmsg("抱歉,使用者帳號已經滿了,無法註冊新的帳號");
+ exit(1);
+ }
+
+ setuserid(uid, user->userid);
+ snprintf(genbuf, sizeof(genbuf), "uid %d", uid);
+ log_usies("APPLY", genbuf);
+
+ SHM->money[uid - 1] = user->money;
+
+ if (passwd_update(uid, (userec_t *)user) == -1) {
+ passwd_unlock();
+ vmsg("客滿了,再見!");
+ exit(1);
+ }
+
+ passwd_unlock();
+
+ return uid;
+}
+
+/////////////////////////////////////////////////////////////////////////////
+// New Registration (Phase 1)
+/////////////////////////////////////////////////////////////////////////////
+
+void
+new_register(void)
+{
+ userec_t newuser;
+ char passbuf[STRLEN];
+ int try, id, uid;
+ char *errmsg = NULL;
+
+#ifdef HAVE_USERAGREEMENT
+ more(HAVE_USERAGREEMENT, YEA);
+ while( 1 ){
+ getdata(b_lines, 0, "請問您接受這份使用者條款嗎? (yes/no) ",
+ passbuf, 4, LCECHO);
+ if( passbuf[0] == 'y' )
+ break;
+ if( passbuf[0] == 'n' ){
+ vmsg("抱歉, 您須要接受使用者條款才能註冊帳號享受我們的服務唷!");
+ exit(1);
+ }
+ vmsg("請輸入 y表示接受, n表示不接受");
+ }
+#endif
+
+ // setup newuser
+ memset(&newuser, 0, sizeof(newuser));
+ newuser.version = PASSWD_VERSION;
+ newuser.userlevel = PERM_DEFAULT;
+ newuser.uflag = BRDSORT_FLAG | MOVIE_FLAG;
+ newuser.uflag2 = 0;
+ newuser.firstlogin = newuser.lastlogin = now;
+ newuser.money = 0;
+ newuser.pager = PAGER_ON;
+ strlcpy(newuser.lasthost, fromhost, sizeof(newuser.lasthost));
+
+#ifdef DBCSAWARE
+ if(u_detectDBCSAwareEvilClient())
+ newuser.uflag &= ~DBCSAWARE_FLAG;
+ else
+ newuser.uflag |= DBCSAWARE_FLAG;
+#endif
+
+ more("etc/register", NA);
+ try = 0;
+ while (1) {
+ userec_t xuser;
+ int minute;
+
+ if (++try >= 6) {
+ vmsg("您嘗試錯誤的輸入太多,請下次再來吧");
+ exit(1);
+ }
+ getdata(17, 0, msg_uid, newuser.userid,
+ sizeof(newuser.userid), DOECHO);
+ strcpy(passbuf, newuser.userid);
+
+ if (bad_user_id(passbuf))
+ outs("無法接受這個代號,請使用英文字母,並且不要包含空格\n");
+ else if ((id = getuser(passbuf, &xuser)) &&
+ // >=: see check_and_expire_account definition
+ (minute = check_and_expire_account(id, &xuser, 0)) >= 0)
+ {
+ if (minute == 999999) // XXX magic number. It should be greater than MAX_USERS at least.
+ outs("此代號已經有人使用 是不死之身");
+ else {
+ prints("此代號已經有人使用 還有 %d 天才過期 \n",
+ minute / (60 * 24) + 1);
+ }
+ } else
+ break;
+ }
+
+ // XXX 記得最後 create account 前還要再檢查一次 acc
+
+ try = 0;
+ while (1) {
+ if (++try >= 6) {
+ vmsg("您嘗試錯誤的輸入太多,請下次再來吧");
+ exit(1);
+ }
+ move(19, 0); clrtoeol();
+ outs(ANSI_COLOR(1;33)
+ "為避免被偷看,您的密碼並不會顯示在畫面上,直接輸入完後按 Enter 鍵即可。\n"
+ "另外請注意密碼只有前八個字元有效,超過的將自動忽略。"
+ ANSI_RESET);
+ if ((getdata(18, 0, "請設定密碼:", passbuf,
+ sizeof(passbuf), NOECHO) < 3) ||
+ !strcmp(passbuf, newuser.userid)) {
+ outs("密碼太簡單,易遭入侵,至少要 4 個字,請重新輸入\n");
+ continue;
+ }
+ strlcpy(newuser.passwd, passbuf, PASSLEN);
+ getdata(19, 0, "請檢查密碼:", passbuf, sizeof(passbuf), NOECHO);
+ if (strncmp(passbuf, newuser.passwd, PASSLEN)) {
+ outs("密碼輸入錯誤, 請重新輸入密碼.\n");
+ continue;
+ }
+ passbuf[8] = '\0';
+ strlcpy(newuser.passwd, genpasswd(passbuf), PASSLEN);
+ break;
+ }
+ // set-up more information.
+
+ // warning: because currutmp=NULL, we can simply pass newuser.* to getdata.
+ // DON'T DO THIS IF YOUR currutmp != NULL.
+ try = 0;
+ while (strlen(newuser.nickname) < 2)
+ {
+ if (++try > 10) {
+ vmsg("您嘗試錯誤的輸入太多,請下次再來吧");
+ exit(1);
+ }
+ getdata(19, 0, "綽號暱稱:", newuser.nickname,
+ sizeof(newuser.nickname), DOECHO);
+ }
+
+ try = 0;
+ while (strlen(newuser.realname) < 4)
+ {
+ if (++try > 10) {
+ vmsg("您嘗試錯誤的輸入太多,請下次再來吧");
+ exit(1);
+ }
+ getdata(20, 0, "真實姓名:", newuser.realname,
+ sizeof(newuser.realname), DOECHO);
+
+ if ((errmsg = isvalidname(newuser.realname)))
+ {
+ memset(newuser.realname, 0, sizeof(newuser.realname));
+ vmsg(errmsg);
+ }
+ }
+
+ try = 0;
+ while (strlen(newuser.address) < 8)
+ {
+ // do not use isvalidaddr to check,
+ // because that requires foreign info.
+ if (++try > 10) {
+ vmsg("您嘗試錯誤的輸入太多,請下次再來吧");
+ exit(1);
+ }
+ getdata(21, 0, "聯絡地址:", newuser.address,
+ sizeof(newuser.address), DOECHO);
+ }
+
+ try = 0;
+ while (newuser.year < 40) // magic number 40: see user.c
+ {
+ char birthday[sizeof("mmmm/yy/dd ")];
+ int y, m, d;
+
+ if (++try > 20) {
+ vmsg("您嘗試錯誤的輸入太多,請下次再來吧");
+ exit(1);
+ }
+ getdata(22, 0, "生日 (西元年/月/日, 如 1984/02/29):", birthday,
+ sizeof(birthday), DOECHO);
+
+ if (ParseDate(birthday, &y, &m, &d)) {
+ vmsg("日期格式不正確");
+ continue;
+ } else if (y < 1940) {
+ vmsg("你真的有那麼老嗎?");
+ continue;
+ }
+ newuser.year = (unsigned char)(y-1900);
+ newuser.month = (unsigned char)m;
+ newuser.day = (unsigned char)d;
+ }
+
+ setupnewuser(&newuser);
+
+ if( (uid = initcuser(newuser.userid)) < 0) {
+ vmsg("無法建立帳號");
+ exit(1);
+ }
+ log_usies("REGISTER", fromhost);
+}
+
+void
+check_birthday(void)
+{
+ // check birthday
+ int changed = 0;
+ time_t t = (time_t)now;
+ struct tm tm;
+
+ localtime_r(&t, &tm);
+ while ( cuser.year < 40 || // magic number 40: see user.c
+ cuser.year+3 > tm.tm_year)
+ {
+ char birthday[sizeof("mmmm/yy/dd ")];
+ int y, m, d;
+
+ clear();
+ stand_title("輸入生日");
+ move(2,0);
+ outs("本站為配合實行內容分級制度,請您輸入正確的生日資訊。");
+
+ getdata(5, 0, "生日 (西元年/月/日, 如 1984/02/29):", birthday,
+ sizeof(birthday), DOECHO);
+
+ if (ParseDate(birthday, &y, &m, &d)) {
+ vmsg("日期格式不正確");
+ continue;
+ } else if (y < 1940) {
+ vmsg("你真的有那麼老嗎?");
+ continue;
+ } else if (y+3 > tm.tm_year+1900) {
+ vmsg("嬰兒/未出生應該無法使用 BBS...");
+ continue;
+ }
+ cuser.year = (unsigned char)(y-1900);
+ cuser.month = (unsigned char)m;
+ cuser.day = (unsigned char)d;
+ changed = 1;
+ }
+
+ if (changed) {
+ clear();
+ resolve_over18();
+ }
+}
+
+void
+check_register(void)
+{
+ char fn[PATHLEN];
+
+ check_birthday();
+
+ if (HasUserPerm(PERM_LOGINOK))
+ return;
+
+ setuserfile(fn, FN_REJECT_NOTIFY);
+
+ /*
+ * 避免使用者被退回註冊單後,在知道退回的原因之前,
+ * 又送出一次註冊單。
+ */
+ if (dashf(fn))
+ {
+ more(fn, NA);
+ move(b_lines-3, 0);
+ outs("上次註冊單審查失敗。\n"
+ "請重新申請並照上面指示正確填寫註冊單。");
+ while(getans("請輸入 y 繼續: ") != 'y');
+ unlink(fn);
+ } else
+ if (ISNEWMAIL(currutmp))
+ m_read();
+
+ if (!HasUserPerm(PERM_SYSOP)) {
+ /* 回覆過身份認證信函,或曾經 E-mail post 過 */
+ clear();
+ move(9, 3);
+ outs("請詳填寫" ANSI_COLOR(32) "註冊申請單" ANSI_RESET ","
+ "通告站長以獲得進階使用權力。\n\n\n\n");
+ u_register();
+
+#ifdef NEWUSER_LIMIT
+ if (cuser.lastlogin - cuser->firstlogin < 3 * 86400)
+ cuser.userlevel &= ~PERM_POST;
+ more("etc/newuser", YEA);
+#endif
+ }
+}
+
+/////////////////////////////////////////////////////////////////////////////
+// User Registration (Phase 2)
+/////////////////////////////////////////////////////////////////////////////
+
+static void
+toregister(char *email, char *phone, char *career,
+ char *rname, char *addr, char *mobile)
+{
+ FILE *fn = NULL;
+
+ justify_wait(cuser.userid, phone, career, rname, addr, mobile);
+
+ clear();
+ stand_title("認證設定");
+ if (cuser.userlevel & PERM_NOREGCODE){
+ strcpy(email, "x");
+ goto REGFORM2;
+ }
+ move(1, 0);
+ outs("您好, 本站認證認證的方式有:\n"
+ " 1.若您有 E-Mail (本站不接受 yahoo, kimo等免費的 E-Mail)\n"
+ " 請輸入您的 E-Mail , 我們會寄發含有認證碼的信件給您\n"
+ " 收到後請到 (U)ser => (R)egister 輸入認證碼, 即可通過認證\n"
+ "\n"
+ " 2.若您沒有 E-Mail 或是一直無法收到認證信, 請輸入 x \n"
+ " 會有站長親自人工審核註冊資料," ANSI_COLOR(1;33)
+ "但注意這可能會花上數週或更多時間。" ANSI_RESET "\n"
+ "**********************************************************\n"
+ "* 注意! *\n"
+ "* 通常應該會在輸入完成後十分鐘內收到認證信, 若過久未收到 *\n"
+ "* 請到郵件垃圾桶檢查是否被當作垃圾信(SPAM)了,另外若是 *\n"
+ "* 輸入後發生認證碼錯誤請重填一次 E-Mail *\n"
+ "**********************************************************\n");
+
+#ifdef HAVEMOBILE
+ outs(" 3.若您有手機門號且想採取手機簡訊認證的方式 , 請輸入 m \n"
+ " 我們將會寄發含有認證碼的簡訊給您 \n"
+ " 收到後請到(U)ser => (R)egister 輸入認證碼, 即可通過認證\n");
+#endif
+
+ while (1) {
+ email[0] = 0;
+ getfield(15, "身分認證用", "E-Mail Address", email, 50);
+ if (strcmp(email, "x") == 0 || strcmp(email, "X") == 0)
+ break;
+#ifdef HAVEMOBILE
+ else if (strcmp(email, "m") == 0 || strcmp(email, "M") == 0) {
+ if (isvalidmobile(mobile)) {
+ char yn[3];
+ getdata(16, 0, "請再次確認您輸入的手機號碼正確嘛? [y/N]",
+ yn, sizeof(yn), LCECHO);
+ if (yn[0] == 'Y' || yn[0] == 'y')
+ break;
+ } else {
+ move(15, 0); clrtobot();
+ move(17, 0);
+ outs("指定的手機號碼不正確,"
+ "若您無手機門號請選擇其他方式認證");
+ }
+
+ }
+#endif
+ else if (isvalidemail(email)) {
+ char yn[3];
+#ifdef USE_EMAILDB
+ int email_count = emaildb_check_email(email, strlen(email));
+
+ if (email_count < 0) {
+ move(15, 0); clrtobot();
+ move(17, 0);
+ outs("暫時不允許\ email 認證註冊, 請稍後再試\n");
+ pressanykey();
+ return;
+ } else if (email_count >= EMAILDB_LIMIT) {
+ move(15, 0); clrtobot();
+ move(17, 0);
+ outs("指定的 E-Mail 已註冊過多帳號, 請使用其他 E-Mail, 或輸入 x 採手動認證\n");
+ outs("但注意手動認證通常會花上數週以上的時間。\n");
+ } else {
+#endif
+ getdata(16, 0, "請再次確認您輸入的 E-Mail 位置正確嘛? [y/N]",
+ yn, sizeof(yn), LCECHO);
+ if (yn[0] == 'Y' || yn[0] == 'y')
+ break;
+#ifdef USE_EMAILDB
+ }
+#endif
+ } else {
+ move(15, 0); clrtobot();
+ move(17, 0);
+ outs("指定的 E-Mail 不正確, 若您無 E-Mail 請輸入 x 由站長手動認證\n");
+ outs("但注意手動認證通常會花上數週以上的時間。\n");
+ }
+ }
+#ifdef USE_EMAILDB
+ if (emaildb_update_email(cuser.userid, strlen(cuser.userid),
+ email, strlen(email)) < 0) {
+ move(15, 0); clrtobot();
+ move(17, 0);
+ outs("暫時不允許\ email 認證註冊, 請稍後再試\n");
+ pressanykey();
+ return;
+ }
+#endif
+ strlcpy(cuser.email, email, sizeof(cuser.email));
+ REGFORM2:
+ if (strcasecmp(email, "x") == 0) { /* 手動認證 */
+ if ((fn = fopen(fn_register, "a"))) {
+ fprintf(fn, "num: %d, %s", usernum, ctime4(&now));
+ fprintf(fn, "uid: %s\n", cuser.userid);
+ fprintf(fn, "name: %s\n", rname);
+ fprintf(fn, "career: %s\n", career);
+ fprintf(fn, "addr: %s\n", addr);
+ fprintf(fn, "phone: %s\n", phone);
+ fprintf(fn, "mobile: %s\n", mobile);
+ fprintf(fn, "email: %s\n", email);
+ fprintf(fn, "----\n");
+ fclose(fn);
+ // save justify information
+ snprintf(cuser.justify, sizeof(cuser.justify),
+ "%s:%s:<Manual>", phone, career);
+ }
+ // XXX what if we cannot open register form?
+ } else {
+ // register by mail of phone
+ snprintf(cuser.justify, sizeof(cuser.justify),
+ "%s:%s:<Email>", phone, career);
+#ifdef HAVEMOBILE
+ if (phone != NULL && email[1] == 0 && tolower(email[0]) == 'm')
+ sprintf(cuser.justify, sizeof(cuser.justify),
+ "%s:%s:<Mobile>", phone, career);
+#endif
+ email_justify(&cuser);
+ }
+}
+
+
+int
+u_register(void)
+{
+ char rname[20], addr[50], mobile[16];
+#ifdef FOREIGN_REG
+ char fore[2];
+#endif
+ char phone[20], career[40], email[50], birthday[11], sex_is[2];
+ unsigned char year, mon, day;
+ char inregcode[14], regcode[50];
+ char ans[3], *ptr, *errcode;
+ char genbuf[200];
+ FILE *fn;
+
+ if (cuser.userlevel & PERM_LOGINOK) {
+ outs("您的身份確認已經完成,不需填寫申請表");
+ return XEASY;
+ }
+ if ((fn = fopen(fn_register, "r"))) {
+ int i =0;
+ while (fgets(genbuf, STRLEN, fn)) {
+ if ((ptr = strchr(genbuf, '\n')))
+ *ptr = '\0';
+ if (strncmp(genbuf, "uid: ", 5) != 0)
+ continue;
+ i++;
+ if(strcmp(genbuf + 5, cuser.userid) != 0)
+ continue;
+ fclose(fn);
+ /* idiots complain about this, so bug them */
+ clear();
+ move(3, 0);
+ prints(" 您的註冊申請單尚在處理中(處理順位: %d),請耐心等候\n\n", i);
+ outs(" 如果您已收到註冊碼卻看到這個畫面,那代表您在使用 Email 註冊後\n");
+ outs(" " ANSI_COLOR(1;31) "又另外申請了站長直接人工審核的註冊申請單。"
+ ANSI_RESET "\n\n");
+ // outs("該死,都不看說明的...\n");
+ outs(" 進入人工審核程序後 Email 註冊自動失效,有註冊碼也沒用,\n");
+ outs(" 要等到審核完成 (會多花很多時間,通常起碼數天) ,所以請耐心等候。\n\n");
+
+ /* 下面是國王的 code 所需要的 message */
+#if 0
+ outs(" 另外請注意,若站長審註冊單時您正在站上則會無法審核、自動跳過。\n");
+ outs(" 所以等候審核時請勿掛站。若超過兩三天仍未被審到,通常就是這個原因。\n");
+#endif
+
+ vmsg("您的註冊申請單尚在處理中");
+ return FULLUPDATE;
+ }
+ fclose(fn);
+ }
+ strlcpy(rname, cuser.realname, sizeof(rname));
+ strlcpy(addr, cuser.address, sizeof(addr));
+ strlcpy(email, cuser.email, sizeof(email));
+ if (cuser.mobile)
+ snprintf(mobile, sizeof(mobile), "0%09d", cuser.mobile);
+ else
+ mobile[0] = 0;
+ if (cuser.month == 0 && cuser.day == 0 && cuser.year == 0)
+ birthday[0] = 0;
+ else
+ snprintf(birthday, sizeof(birthday), "%04i/%02i/%02i",
+ 1900 + cuser.year, cuser.month, cuser.day);
+ sex_is[0] = (cuser.sex % 8) + '1';
+ sex_is[1] = 0;
+ career[0] = phone[0] = '\0';
+ sethomefile(genbuf, cuser.userid, FN_JUSTIFY_WAIT);
+ if ((fn = fopen(genbuf, "r"))) {
+ fgets(genbuf, sizeof(genbuf), fn);
+ chomp(genbuf);
+ strlcpy(phone, genbuf, sizeof(phone));
+
+ fgets(genbuf, sizeof(genbuf), fn);
+ chomp(genbuf);
+ strlcpy(career, genbuf, sizeof(career));
+
+ fgets(genbuf, sizeof(genbuf), fn); // old version compatible
+
+ fgets(genbuf, sizeof(genbuf), fn);
+ chomp(genbuf);
+ strlcpy(rname, genbuf, sizeof(rname));
+
+ fgets(genbuf, sizeof(genbuf), fn);
+ chomp(genbuf);
+ strlcpy(addr, genbuf, sizeof(addr));
+
+ fgets(genbuf, sizeof(genbuf), fn);
+ chomp(genbuf);
+ strlcpy(mobile, genbuf, sizeof(mobile));
+
+ fclose(fn);
+ }
+
+ if (cuser.userlevel & PERM_NOREGCODE) {
+ vmsg("您不被允許\使用認證碼認證。請填寫註冊申請單");
+ goto REGFORM;
+ }
+
+ // getregcode(regcode);
+
+ // XXX why check by year?
+ // birthday is moved to earlier, so let's check email instead.
+ if (cuser.email[0] && // cuser.year != 0 && /* 已經第一次填過了~ ^^" */
+ strcmp(cuser.email, "x") != 0 && /* 上次手動認證失敗 */
+ strcmp(cuser.email, "X") != 0)
+ {
+ clear();
+ stand_title("EMail認證");
+ move(2, 0);
+
+ prints("請輸入您的認證碼。(由 %s 開頭無空白的十三碼)\n"
+ "或輸入 x 來重新填寫 E-Mail 或改由站長手動認證\n", REGCODE_INITIAL);
+ inregcode[0] = 0;
+
+ do{
+ getdata(10, 0, "您的認證碼:",
+ inregcode, sizeof(inregcode), DOECHO);
+ if( strcmp(inregcode, "x") == 0 || strcmp(inregcode, "X") == 0 )
+ break;
+ if( strlen(inregcode) != 13 || inregcode[0] == ' ')
+ vmsg("認證碼輸入不完整,總共應有十三碼,沒有空白字元。");
+ else if( inregcode[0] != REGCODE_INITIAL[0] || inregcode[1] != REGCODE_INITIAL[1] ) {
+ /* old regcode */
+ vmsg("輸入的認證碼錯誤," // "或因系統昇級已失效,"
+ "請輸入 x 重填一次 E-Mail");
+ }
+ else
+ break;
+ } while( 1 );
+
+ // make it case insensitive.
+ if (strcasecmp(inregcode, getregcode(regcode)) == 0) {
+ int unum;
+ delregcodefile();
+ if ((unum = searchuser(cuser.userid, NULL)) == 0) {
+ vmsg("系統錯誤,查無此人!");
+ u_exit("getuser error");
+ exit(0);
+ }
+ mail_muser(cuser, "[註冊成功\囉]", "etc/registeredmail");
+#if FOREIGN_REG_DAY > 0
+ if(cuser.uflag2 & FOREIGN)
+ mail_muser(cuser, "[出入境管理局]", "etc/foreign_welcome");
+#endif
+ cuser.userlevel |= (PERM_LOGINOK | PERM_POST);
+ outs("\n註冊成功\, 重新上站後將取得完整權限\n"
+ "請按下任一鍵跳離後重新上站~ :)");
+ sethomefile(genbuf, cuser.userid, FN_JUSTIFY_WAIT);
+ unlink(genbuf);
+ snprintf(cuser.justify, sizeof(cuser.justify),
+ "%s:%s:email", phone, career);
+ sethomefile(genbuf, cuser.userid, FN_JUSTIFY);
+ log_file(genbuf, LOG_CREAT, cuser.justify);
+ pressanykey();
+ u_exit("registed");
+ exit(0);
+ return QUIT;
+ } else if (strcasecmp(inregcode, "x") != 0) {
+ if (regcode[0])
+ {
+ vmsg("認證碼錯誤!");
+ return FULLUPDATE;
+ }
+ else
+ {
+ vmsg("認證碼已過期,請重新註冊。");
+ toregister(email, phone, career, rname, addr, mobile);
+ return FULLUPDATE;
+ }
+ } else {
+ toregister(email, phone, career, rname, addr, mobile);
+ return FULLUPDATE;
+ }
+ }
+
+ REGFORM:
+ getdata(b_lines - 1, 0, "您確定要填寫註冊單嗎(Y/N)?[N] ",
+ ans, 3, LCECHO);
+ if (ans[0] != 'y')
+ return FULLUPDATE;
+
+ move(2, 0);
+ clrtobot();
+ while (1) {
+ clear();
+ move(1, 0);
+ prints("%s(%s) 您好,請據實填寫以下的資料:",
+ cuser.userid, cuser.nickname);
+#ifdef FOREIGN_REG
+ fore[0] = 'y';
+ fore[1] = 0;
+ getfield(2, "Y/n", "是否現在住在台灣", fore, 2);
+ if (fore[0] == 'n')
+ fore[0] |= FOREIGN;
+ else
+ fore[0] = 0;
+#endif
+ while (1) {
+ getfield(8,
+#ifdef FOREIGN_REG
+ "請用本名",
+#else
+ "請用中文",
+#endif
+ "真實姓名", rname, 20);
+ if( (errcode = isvalidname(rname)) == NULL )
+ break;
+ else
+ vmsg(errcode);
+ }
+
+ move(11, 0);
+ outs(" 請盡量詳細的填寫您的服務單位,大專院校請麻煩"
+ "加" ANSI_COLOR(1;33) "系所" ANSI_RESET ",公司單位請加" ANSI_COLOR(1;33) "職稱" ANSI_RESET ",\n"
+ " 暫無工作請麻煩填寫" ANSI_COLOR(1;33) "畢業學校" ANSI_RESET "。\n");
+ while (1) {
+ getfield(9, "(畢業)學校(含" ANSI_COLOR(1;33) "系所年級" ANSI_RESET ")或單位職稱",
+ "服務單位", career, 40);
+ if( (errcode = isvalidcareer(career)) == NULL )
+ break;
+ else
+ vmsg(errcode);
+ }
+ move(10, 0); clrtobot();
+ while (1) {
+ getfield(10, "含" ANSI_COLOR(1;33) "縣市" ANSI_RESET "及門寢號碼"
+ "(台北請加" ANSI_COLOR(1;33) "行政區" ANSI_RESET ")",
+ "目前住址", addr, sizeof(addr));
+ if( (errcode = isvalidaddr(addr)) == NULL
+#ifdef FOREIGN_REG
+ || fore[0]
+#endif
+ )
+ break;
+ else
+ vmsg(errcode);
+ }
+ while (1) {
+ getfield(11, "不加-(), 包括長途區號", "連絡電話", phone, 11);
+ if( (errcode = isvalidphone(phone)) == NULL )
+ break;
+ else
+ vmsg(errcode);
+ }
+ getfield(12, "只輸入數字 如:0912345678 (可不填)",
+ "手機號碼", mobile, 20);
+ while (1) {
+ getfield(13, "西元/月月/日日 如:1984/02/29", "生日", birthday, sizeof(birthday));
+ if (birthday[0] == 0) {
+ snprintf(birthday, sizeof(birthday), "%04i/%02i/%02i",
+ 1900 + cuser.year, cuser.month, cuser.day);
+ mon = cuser.month;
+ day = cuser.day;
+ year = cuser.year;
+ } else {
+ int y, m, d;
+ if (ParseDate(birthday, &y, &m, &d)) {
+ vmsg("您的輸入不正確");
+ continue;
+ }
+ mon = (unsigned char)m;
+ day = (unsigned char)d;
+ year = (unsigned char)(y - 1900);
+ }
+ if (year < 40) {
+ vmsg("您的輸入不正確");
+ continue;
+ }
+ break;
+ }
+ getfield(14, "1.葛格 2.姐接 ", "性別", sex_is, 2);
+ getdata(20, 0, "以上資料是否正確(Y/N)?(Q)取消註冊 [N] ",
+ ans, 3, LCECHO);
+ if (ans[0] == 'q')
+ return 0;
+ if (ans[0] == 'y')
+ break;
+ }
+ strlcpy(cuser.realname, rname, sizeof(cuser.realname));
+ strlcpy(cuser.address, addr, sizeof(cuser.address));
+ strlcpy(cuser.email, email, sizeof(cuser.email));
+ cuser.mobile = atoi(mobile);
+ cuser.sex = (sex_is[0] - '1') % 8;
+ cuser.month = mon;
+ cuser.day = day;
+ cuser.year = year;
+#ifdef FOREIGN_REG
+ if (fore[0])
+ cuser.uflag2 |= FOREIGN;
+ else
+ cuser.uflag2 &= ~FOREIGN;
+#endif
+ trim(career);
+ trim(addr);
+ trim(phone);
+
+ toregister(email, phone, career, rname, addr, mobile);
+
+ // update cuser
+ passwd_update(usernum, &cuser);
+
+ return FULLUPDATE;
+}
+
+/////////////////////////////////////////////////////////////////////////////
+// Administration (SYSOP Validation)
+/////////////////////////////////////////////////////////////////////////////
+
+#define REJECT_REASONS (6)
+#define REASON_LEN (60)
+static const char *reasonstr[REJECT_REASONS] = {
+ "輸入真實姓名",
+ "詳填(畢業)學校『系』『級』或服務單位(含所屬縣市及職稱)",
+ "填寫完整的住址資料 (含縣市名稱, 台北市請含行政區域)",
+ "詳填連絡電話 (含區碼, 中間不加 '-', '(', ')' 等符號)",
+ "精確並完整填寫註冊申請表",
+ "用中文填寫申請單",
+};
+
+#define REASON_FIRSTABBREV '0'
+#define REASON_IN_ABBREV(x) \
+ ((x) >= REASON_FIRSTABBREV && (x) - REASON_FIRSTABBREV < REJECT_REASONS)
+#define REASON_EXPANDABBREV(x) reasonstr[(x) - REASON_FIRSTABBREV]
+
+void
+regform_accept(const char *userid, const char *justify)
+{
+ char buf[PATHLEN];
+ int unum = 0;
+ userec_t muser;
+
+ unum = getuser(userid, &muser);
+ if (unum == 0)
+ return; // invalid user
+
+ muser.userlevel |= (PERM_LOGINOK | PERM_POST);
+ strlcpy(muser.justify, justify, sizeof(muser.justify));
+ // manual accept sets email to 'x'
+ strlcpy(muser.email, "x", sizeof(muser.email));
+
+ // handle files
+ sethomefile(buf, muser.userid, FN_JUSTIFY_WAIT);
+ unlink(buf);
+ sethomefile(buf, muser.userid, FN_REJECT_NOTIFY);
+ unlink(buf);
+ sethomefile(buf, muser.userid, FN_JUSTIFY);
+ log_filef(buf, LOG_CREAT, "%s\n", muser.justify);
+
+ // update password file
+ passwd_update(unum, &muser);
+
+ // alert online users?
+ sendalert(muser.userid, ALERT_PWD_PERM|ALERT_PWD_JUSTIFY); // force to reload perm
+
+#if FOREIGN_REG_DAY > 0
+ if(muser.uflag2 & FOREIGN)
+ mail_muser(muser, "[出入境管理局]", "etc/foreign_welcome");
+ else
+#endif
+ // last: send notification mail
+ mail_muser(muser, "[註冊成功\囉]", "etc/registered");
+}
+
+void
+regform_reject(const char *userid, const char *reason)
+{
+ char buf[PATHLEN];
+ FILE *fp = NULL;
+ int unum = 0;
+ userec_t muser;
+
+ unum = getuser(userid, &muser);
+ if (unum == 0)
+ return; // invalid user
+
+ muser.userlevel &= ~(PERM_LOGINOK | PERM_POST);
+
+ // handle files
+ sethomefile(buf, muser.userid, FN_JUSTIFY_WAIT);
+ unlink(buf);
+
+ // update password file
+ passwd_update(unum, &muser);
+
+ // alert notify users?
+ sendalert(muser.userid, ALERT_PWD_PERM); // force to reload perm
+
+ // last: send notification
+ mkuserdir(muser.userid);
+ sethomefile(buf, muser.userid, FN_REJECT_NOTIFY);
+ fp = fopen(buf, "wt");
+ assert(fp);
+ syncnow();
+ fprintf(fp, "%s 註冊失敗。\n", Cdate(&now));
+
+ // multiple abbrev loop
+ if (REASON_IN_ABBREV(reason[0]))
+ {
+ int i = 0;
+ for (i = 0; i < REASON_LEN && REASON_IN_ABBREV(reason[i]); i++)
+ fprintf(fp, "[退回原因] 請%s\n", REASON_EXPANDABBREV(reason[i]));
+ } else {
+ fprintf(fp, "[退回原因] %s\n", reason);
+ }
+ fclose(fp);
+ mail_muser(muser, "[註冊失敗]", buf);
+}
+
+// Regform v1 API
+// read count entries from regsrc to a temp buffer
+FILE *
+pull_regform(const char *regfile, char *workfn, int count)
+{
+ FILE *fp = NULL;
+
+ snprintf(workfn, PATHLEN, "%s.tmp", regfile);
+ if (dashf(workfn)) {
+ vmsg("其他 SYSOP 也在審核註冊申請單");
+ return NULL;
+ }
+
+ // count < 0 means unlimited pulling
+ Rename(regfile, workfn);
+ if ((fp = fopen(workfn, "r")) == NULL) {
+ vmsgf("系統錯誤,無法讀取註冊資料檔: %s", workfn);
+ return NULL;
+ }
+ return fp;
+}
+
+// write all left in "remains" to regfn.
+void
+pump_regform(const char *regfn, FILE *remains)
+{
+ // restore trailing tickets
+ char buf[PATHLEN];
+ FILE *fout = fopen(regfn, "at");
+ if (!fout)
+ return;
+
+ while (fgets(buf, sizeof(buf), remains))
+ fputs(buf, fout);
+ fclose(fout);
+}
+
+// New Regform UI
+static void
+prompt_regform_ui()
+{
+ move(b_lines, 0);
+ outs(ANSI_COLOR(30;47) " "
+ ANSI_COLOR(31) "y" ANSI_COLOR(30) "接受 "
+ ANSI_COLOR(31) "n" ANSI_COLOR(30) "拒絕 "
+ ANSI_COLOR(31) "d" ANSI_COLOR(30) "刪除 "
+ ANSI_COLOR(31) "s" ANSI_COLOR(30) "跳過 "
+ ANSI_COLOR(31) "u" ANSI_COLOR(30) "復原 "
+ " "
+ ANSI_COLOR(31) "0-9jk↑↓" ANSI_COLOR(30) "移動 "
+ ANSI_COLOR(31) "空白/PgDn" ANSI_COLOR(30) "儲存+下頁 "
+ " "
+ ANSI_COLOR(31) "q/END" ANSI_COLOR(30) "結束 "
+ ANSI_RESET);
+}
+
+static void
+resolve_reason(char *s, int y)
+{
+ // should start with REASON_FIRSTABBREV
+ const char *reason_prompt =
+ " (0)真實姓名 (1)詳填系級 (2)完整住址"
+ " (3)詳填電話 (4)確實填寫 (5)中文填寫";
+
+ s[0] = 0;
+ move(y, 0);
+ outs(reason_prompt); outs("\n");
+
+ do {
+ getdata(y+1, 0,
+ "退回原因: ", s, REASON_LEN, DOECHO);
+
+ // convert abbrev reasons (format: single digit, or multiple digites)
+ if (REASON_IN_ABBREV(s[0]))
+ {
+ if (s[1] == 0) // simple replace ment
+ {
+ strlcpy(s+2, REASON_EXPANDABBREV(s[0]),
+ REASON_LEN-2);
+ s[0] = 0xbd; // '請'[0];
+ s[1] = 0xd0; // '請'[1];
+ } else {
+ // strip until all digites
+ char *p = s;
+ while (*p)
+ {
+ if (!REASON_IN_ABBREV(*p))
+ *p = ' ';
+ p++;
+ }
+ strip_blank(s, s);
+ strlcat(s, " [多重原因]", REASON_LEN);
+ }
+ }
+
+ if (strlen(s) < 4)
+ {
+ if (vmsg("原因太短。 要取消退回嗎? (y/N): ") == 'y')
+ {
+ *s = 0;
+ return;
+ }
+ }
+ } while (strlen(s) < 4);
+}
+
+////////////////////////////////////////////////////////////////////////////
+// Regform Utilities
+////////////////////////////////////////////////////////////////////////////
+
+// TODO define and use structure instead, even in reg request file.
+typedef struct {
+ // current format:
+ // (optional) num: unum, date
+ // [0] uid: xxxxx (IDLEN=12)
+ // [1] name: RRRRRR (20)
+ // [2] career: YYYYYYYYYYYYYYYYYYYYYYYYYY (40)
+ // [3] addr: TTTTTTTTT (50)
+ // [4] phone: 02DDDDDDDD (20)
+ // [5] email: x (50) (deprecated)
+ // [6] mobile: (deprecated)
+ // [7] ----
+ // lasthost: 16
+ char userid[IDLEN+1];
+
+ char exist;
+ char online;
+ char pad [ 5]; // IDLEN(12)+1+1+1+5=20
+
+ char name [20];
+ char career[40];
+ char addr [50];
+ char phone [20];
+} RegformEntry;
+
+// regform format utilities
+int
+load_regform_entry(RegformEntry *pre, FILE *fp)
+{
+ char buf[STRLEN];
+ char *v;
+
+ memset(pre, 0, sizeof(RegformEntry));
+ while (fgets(buf, sizeof(buf), fp))
+ {
+ if (buf[0] == '-')
+ break;
+ buf[sizeof(buf)-1] = 0;
+ v = strchr(buf, ':');
+ if (v == NULL)
+ continue;
+ *v++ = 0;
+ if (*v == ' ') v++;
+ chomp(v);
+
+ if (strcmp(buf, "uid") == 0)
+ strlcpy(pre->userid, v, sizeof(pre->userid));
+ else if (strcmp(buf, "name") == 0)
+ strlcpy(pre->name, v, sizeof(pre->name));
+ else if (strcmp(buf, "career") == 0)
+ strlcpy(pre->career, v, sizeof(pre->career));
+ else if (strcmp(buf, "addr") == 0)
+ strlcpy(pre->addr, v, sizeof(pre->addr));
+ else if (strcmp(buf, "phone") == 0)
+ strlcpy(pre->phone, v, sizeof(pre->phone));
+ }
+ return pre->userid[0] ? 1 : 0;
+}
+
+int
+print_regform_entry(const RegformEntry *pre, FILE *fp, int close)
+{
+ fprintf(fp, "uid: %s\n", pre->userid);
+ fprintf(fp, "name: %s\n", pre->name);
+ fprintf(fp, "career: %s\n", pre->career);
+ fprintf(fp, "addr: %s\n", pre->addr);
+ fprintf(fp, "phone: %s\n", pre->phone);
+ if (close)
+ fprintf(fp, "----\n");
+ return 1;
+}
+
+int
+append_regform(const RegformEntry *pre, const char *logfn,
+ const char *varname, const char *varval1, const char *varval2)
+{
+ FILE *fout = fopen(logfn, "at");
+ if (!fout)
+ return 0;
+
+ print_regform_entry(pre, fout, 0);
+ if (varname && *varname)
+ {
+ syncnow();
+ fprintf(fout, "Date: %s\n", Cdate(&now));
+ if (!varval1) varval1 = "";
+ fprintf(fout, "%s: %s", varname, varval1);
+ if (varval2) fprintf(fout, " %s", varval2);
+ fprintf(fout, "\n");
+ }
+ // close it
+ fprintf(fout, "----\n");
+ fclose(fout);
+ return 1;
+}
+
+////////////////////////////////////////////////////////////////////////////
+// Regform2 API
+////////////////////////////////////////////////////////////////////////////
+
+// registration queue
+int
+regq_append(const char *userid)
+{
+ if (file_append_record(FN_REQLIST, userid) < 0)
+ return 0;
+ return 1;
+}
+
+int
+regq_find(const char *userid)
+{
+ return file_find_record(FN_REQLIST, userid);
+}
+
+int
+regq_delete(const char *userid)
+{
+ return file_delete_record(FN_REQLIST, userid, 0);
+}
+
+// user home regform operation
+int
+regfrm_exist(const char *userid)
+{
+ char fn[PATHLEN];
+ sethomefile(fn, userid, FN_REGFORM);
+ return dashf(fn) ? 1 : 0;
+}
+
+int
+regfrm_load(const char *userid, RegformEntry *pre)
+{
+ FILE *fp = NULL;
+ char fn[PATHLEN];
+ int ret = 0;
+ sethomefile(fn, userid, FN_REGFORM);
+ if (!dashf(fn))
+ return 0;
+
+ fp = fopen(fn, "rt");
+ if (!fp)
+ return 0;
+ ret = load_regform_entry(pre, fp);
+ fclose(fp);
+ return ret;
+}
+
+int
+regfrm_save(const char *userid, const RegformEntry *pre)
+{
+ FILE *fp = NULL;
+ char fn[PATHLEN];
+ int ret = 0;
+ sethomefile(fn, userid, FN_REGFORM);
+
+ fp = fopen(fn, "wt");
+ if (!fp)
+ return 0;
+ ret = print_regform_entry(pre, fp, 1);
+ fclose(fp);
+ return ret;
+}
+
+int
+regfrm_trylock(const char *userid)
+{
+ int fd = 0;
+ char fn[PATHLEN];
+ sethomefile(fn, userid, FN_REGFORM);
+ if (!dashf(fn)) return 0;
+ fd = open(fn, O_RDONLY);
+ if (fd < 0) return 0;
+ if (flock(fd, LOCK_EX|LOCK_NB) == 0)
+ return fd;
+ close(fd);
+ return 0;
+}
+
+int
+regfrm_unlock(int lockfd)
+{
+ int fd = lockfd;
+ if (lockfd <= 0)
+ return 0;
+ lockfd = flock(fd, LOCK_UN) == 0 ? 1 : 0;
+ close(fd);
+ return lockfd;
+}
+
+// regform processors
+int
+regfrm_accept(RegformEntry *pre)
+{
+ char justify[REGLEN+1], buf[STRLEN*2];
+ char fn[PATHLEN], fnlog[PATHLEN];
+
+ // dry run!
+ vmsg("regfrm_accept");
+ return 1;
+
+ sethomefile(fn, pre->userid, FN_REGFORM);
+
+ // build justify string
+ removespace(pre->phone);
+ removespace(pre->career);
+ snprintf(justify, sizeof(justify),
+ "%s:%s:%s", pre->phone, pre->career, cuser.userid);
+
+ // call handler
+ regform_accept(pre->userid, justify);
+
+ // append current form to history.
+ sethomefile(fnlog, pre->userid, FN_REGFORM_LOG);
+ AppendTail(fn, fnlog, 0);
+ // global history
+ snprintf(buf, sizeof(buf), "Approved: %s -> %s\nDate: %s\n",
+ cuser.userid, pre->userid, Cdate(&now));
+ file_append_line(FN_REGISTER_LOG, buf);
+ AppendTail(fn, FN_REGISTER_LOG, 0);
+
+ // remove from queue
+ unlink(fn);
+ regq_delete(pre->userid);
+ return 1;
+}
+
+int
+regfrm_reject(RegformEntry *pre, const char *reason)
+{
+ char buf[STRLEN*2];
+ char fn[PATHLEN];
+
+ // dry run!
+ vmsg("regfrm_reject");
+ return 1;
+
+ sethomefile(fn, pre->userid, FN_REGFORM);
+
+ // call handler
+ regform_reject(pre->userid, reason);
+
+ // log it
+ snprintf(buf, sizeof(buf), "Rejected: %s -> %s [%s]\nDate: %s\n",
+ cuser.userid, pre->userid, reason, Cdate(&now));
+ file_append_line(FN_REGISTER_LOG, buf);
+ AppendTail(fn, FN_REGISTER_LOG, 0);
+
+ // remove from queue
+ unlink(fn);
+ regq_delete(pre->userid);
+ return 1;
+}
+
+int
+regfrm_delete(const char *userid)
+{
+ char fn[PATHLEN];
+ sethomefile(fn, userid, FN_REGFORM);
+
+ // dry run!
+ vmsgf("regfrm_delete (%s)", userid);
+ return 1;
+
+ // directly delete.
+ unlink(fn);
+
+ // remove from queue
+ regq_delete(userid);
+ return 1;
+}
+
+// working queue
+FILE *
+regq_init_pull()
+{
+ FILE *fp = tmpfile(), *src =NULL;
+ char buf[STRLEN];
+ if (!fp) return NULL;
+ src = fopen(FN_REQLIST, "rt");
+ if (!src) { fclose(fp); return NULL; }
+ while (fgets(buf, sizeof(buf), src))
+ fputs(buf, fp);
+ fclose(src);
+ rewind(fp);
+ return fp;
+}
+
+int
+regq_pull(FILE *fp, char *uid)
+{
+ char buf[STRLEN];
+ size_t idlen = 0;
+ uid[0] = 0;
+ if (fgets(buf, sizeof(buf), fp) == NULL)
+ return 0;
+ idlen = strcspn(buf, str_space);
+ if (idlen < 1) return 0;
+ if (idlen > IDLEN) idlen = IDLEN;
+ strlcpy(uid, buf, idlen+1);
+ return 1;
+}
+
+int
+regq_end_pull(FILE *fp)
+{
+ // no need to unlink because fp is a tmpfile.
+ if (!fp) return 0;
+ fclose(fp);
+ return 1;
+}
+
+// UI part
+int
+ui_display_regform_single(
+ const userec_t *xuser,
+ const RegformEntry *pre,
+ int tid, char *reason)
+{
+ int c;
+
+ while (1)
+ {
+ move(1, 0);
+ user_display(xuser, 1);
+ move(14, 0);
+ prints(ANSI_COLOR(1;32)
+ "--------------- 這是第 %2d 份註冊單 ------------------"
+ ANSI_RESET "\n", tid);
+ prints(" %-12s: %s\n", "帳號", pre->userid);
+ prints("0.%-12s: %s%s\n", "真實姓名", pre->name,
+ xuser->uflag2 & FOREIGN ? " (外籍)" :
+ "");
+ prints("1.%-12s: %s\n", "服務單位", pre->career);
+ prints("2.%-12s: %s\n", "目前住址", pre->addr);
+ prints("3.%-12s: %s\n", "連絡電話", pre->phone);
+
+ move(b_lines, 0);
+ outs("是否接受此資料(Y/N/Q/Del/Skip)?[S] ");
+
+ c = tolower(igetch() & 0xFF); // round to ASCII
+ if (c == 'y' || c == 'q' || c == 'd' || c == 's')
+ return c;
+ if (c == 'n')
+ {
+ int n = 0;
+ move(3, 0);
+ outs("\n" ANSI_COLOR(1;31)
+ "請提出退回申請表原因,按 <Enter> 取消:\n" ANSI_RESET);
+ for (n = 0; n < REJECT_REASONS; n++)
+ prints("%d) 請%s\n", n, reasonstr[n]);
+ outs("\n\n\n"); // preserved for prompt
+
+ getdata(3+2+REJECT_REASONS+1, 0,"退回原因: ",
+ reason, REASON_LEN, DOECHO);
+ if (reason[0] == 0)
+ continue;
+ // interprete reason
+ return 'n';
+ }
+ else if (REASON_IN_ABBREV(c))
+ {
+ // quick set
+ sprintf(reason, "%c", c);
+ return 'n';
+ }
+ return 's';
+ }
+ // shall never reach here
+ return 's';
+}
+
+// sample validator
+void
+regform2_validate_single()
+{
+ int lfd = 0;
+ int tid = 0;
+ char uid[IDLEN+1];
+ char rsn[REASON_LEN];
+ FILE *fpregq = regq_init_pull();
+ RegformEntry re;
+
+ if (!fpregq)
+ return;
+
+ while (regq_pull(fpregq, uid))
+ {
+ userec_t muser;
+ int unum = 0;
+ int abort = 0;
+
+ // check if user exists.
+ memset(&muser, 0, sizeof(muser));
+ unum = getuser(uid, &muser);
+
+ if (unum < 1)
+ {
+ regq_delete(uid);
+ continue;
+ }
+
+ // check if regform exists.
+ if (!regfrm_exist(uid))
+ {
+ // TODO delete here?
+ regq_delete(uid);
+ continue;
+ }
+
+ // TODO check if user is already registered
+#if 0
+ if (muser.userlevel & PERM_LOGINOK)
+ {
+ regfrm_delete(uid);
+ continue;
+ }
+#endif
+
+ // try to lock
+ lfd = regfrm_trylock(uid);
+ if (lfd <= 0)
+ continue;
+
+ // load it
+ if (!regfrm_load(uid, &re))
+ {
+ regfrm_delete(uid);
+ regfrm_unlock(lfd);
+ // regq_delete(uid); // done in regfrm_delete
+ continue;
+ }
+
+ tid ++;
+ // display regform and process
+ switch(ui_display_regform_single(&muser, &re, tid, rsn))
+ {
+ case 'a': // accept
+ regfrm_accept(&re);
+ break;
+
+ case 'd': // delete
+ regfrm_delete(uid);
+ break;
+
+ case 'q': // quit
+ abort = 1;
+ break;
+
+ case 'n': // reject
+ regfrm_reject(&re, rsn);
+ break;
+
+ case 's': // skip
+ // do nothing.
+ break;
+
+ default: // shall never reach here
+ assert(0);
+ break;
+ }
+
+ // final processing
+ regfrm_unlock(lfd);
+
+ if (abort)
+ break;
+ }
+ regq_end_pull(fpregq);
+
+ // finishing
+ clear(); move(5, 0);
+ prints("您審了 %d 份註冊單份。", tid);
+ pressanykey();
+}
+
+#define FORMS_IN_PAGE (10)
+
+int
+regform2_validate_page(int dryrun)
+{
+ int unum = 0;
+ int yMsg = FORMS_IN_PAGE*2+1;
+ userec_t muser;
+ RegformEntry forms [FORMS_IN_PAGE];
+ char ans [FORMS_IN_PAGE];
+ int lfds [FORMS_IN_PAGE];
+ char rejects[FORMS_IN_PAGE][REASON_LEN]; // reject reason length
+ char rsn [REASON_LEN];
+ int cforms = 0, // current loaded forms
+ ci = 0, // cursor index
+ ch = 0, // input key
+ i;
+ int tid = 0;
+ char uid[IDLEN+1];
+ FILE *fpregq = regq_init_pull();
+
+ if (!fpregq)
+ return 0;
+
+ while (ch != 'q')
+ {
+ // initialize and prepare
+ memset(ans, 0, sizeof(ans));
+ memset(rejects, 0, sizeof(rejects));
+ memset(forms, 0, sizeof(forms));
+ memset(lfds, 0, sizeof(lfds));
+ cforms = 0;
+ clear();
+
+ // load forms
+ while (cforms < FORMS_IN_PAGE)
+ {
+ if (!regq_pull(fpregq, uid))
+ break;
+ i = cforms; // align index
+
+ // check if user exists.
+ memset(&muser, 0, sizeof(muser));
+ unum = getuser(uid, &muser);
+ if (unum < 1)
+ {
+ regq_delete(uid);
+ continue;
+ }
+
+ // check if regform exists.
+ if (!regfrm_exist(uid))
+ {
+ // TODO delete here?
+ regq_delete(uid);
+ continue;
+ }
+ // try to lock
+ lfds[i] = regfrm_trylock(uid);
+ if (lfds[i] <= 0)
+ continue;
+
+ // load it
+ if (!regfrm_load(uid, &forms[i]))
+ {
+ regfrm_delete(uid);
+ regfrm_unlock(lfds[i]);
+ // regq_delete(uid); // done in regfrm_delete
+ continue;
+ }
+
+ forms[i].exist = 1;
+ forms[i].online = search_ulist(unum) ? 1 : 0;
+
+ // assign default answers
+ if (muser.userlevel & PERM_LOGINOK)
+ ans[i] = 'd';
+#ifdef REGFORM_DISABLE_ONLINE_USER
+ else if (forms[i].online)
+ ans[i] = 's';
+#endif // REGFORM_DISABLE_ONLINE_USER
+
+
+ // display
+ move(i*2, 0);
+ prints(" %2d%s %s%-12s " ANSI_RESET,
+ i+1,
+ (unum == 0) ? ANSI_COLOR(1;31) "D" :
+ ( (muser.userlevel & PERM_LOGINOK) ?
+ ANSI_COLOR(1;33) "Y" :
+#ifdef REGFORM_DISABLE_ONLINE_USER
+ forms[i].online ? "s" :
+#endif
+ "."),
+ forms[i].online ? ANSI_COLOR(1;35) : ANSI_COLOR(1),
+ forms[i].userid);
+
+ prints( ANSI_COLOR(1;31) "%19s "
+ ANSI_COLOR(1;32) "%-40s" ANSI_RESET"\n",
+ forms[i].name, forms[i].career);
+
+ move(i*2+1, 0);
+ prints(" %s %-50s%20s\n",
+ (muser.userlevel & PERM_NOREGCODE) ?
+ ANSI_COLOR(1;31) "T" ANSI_RESET : " ",
+ forms[i].addr, forms[i].phone);
+
+ cforms++, tid ++;
+ }
+
+ // if no more forms then leave.
+ if (cforms < 1)
+ break;
+
+ // adjust cursor if required
+ if (ci >= cforms)
+ ci = cforms-1;
+
+ // display page info
+ {
+ char msg[STRLEN];
+ snprintf(msg, sizeof(msg),
+ "%s 已顯示 %d 份註冊單 ", // "(%2d%%) ",
+ dryrun? "(測試模式)" : "",
+ tid);
+ prints(ANSI_COLOR(7) "\n%78s" ANSI_RESET "\n", msg);
+ }
+
+ // handle user input
+ prompt_regform_ui();
+ ch = 0;
+ while (ch != 'q' && ch != ' ') {
+ ch = cursor_key(ci*2, 0);
+ switch (ch)
+ {
+ // nav keys
+ case KEY_UP:
+ case 'k':
+ if (ci > 0) ci--;
+ break;
+
+ case KEY_DOWN:
+ case 'j':
+ ch = 'j'; // go next
+ break;
+
+ // quick nav (assuming to FORMS_IN_PAGE=10)
+ case '1': case '2': case '3': case '4': case '5':
+ case '6': case '7': case '8': case '9':
+ ci = ch - '1';
+ if (ci >= cforms) ci = cforms-1;
+ break;
+ case '0':
+ ci = 10-1;
+ if (ci >= cforms) ci = cforms-1;
+ break;
+
+ /*
+ case KEY_HOME: ci = 0; break;
+ case KEY_END: ci = cforms-1; break;
+ */
+
+ // abort
+ case KEY_END:
+ case 'q':
+ ch = 'q';
+ if (getans("確定要離開了嗎? (本頁變更將不會儲存) [y/N]: ") != 'y')
+ {
+ prompt_regform_ui();
+ ch = 0;
+ continue;
+ }
+ break;
+
+ // prepare to go next page
+ case KEY_PGDN:
+ case ' ':
+ ch = ' ';
+
+ {
+ int blanks = 0;
+ // solving blank (undecided entries)
+ for (i = 0, blanks = 0; i < cforms; i++)
+ if (ans[i] == 0) blanks ++;
+
+ if (!blanks)
+ break;
+
+ // have more blanks
+ ch = getans("尚未指定的 %d 個項目要: (S跳過/y通過/n拒絕/e繼續編輯): ",
+ blanks);
+ }
+
+ if (ch == 'e')
+ {
+ prompt_regform_ui();
+ ch = 0;
+ continue;
+ }
+ if (ch == 'y') {
+ // do nothing.
+ } else if (ch == 'n') {
+ // query reject reason
+ resolve_reason(rsn, yMsg);
+ if (*rsn == 0)
+ ch = 's';
+ } else ch = 's';
+
+ // filling answers
+ for (i = 0; i < cforms; i++)
+ {
+ if (ans[i] != 0)
+ continue;
+ ans[i] = ch;
+ if (ch != 'n')
+ continue;
+ strlcpy(rejects[i], rsn, REASON_LEN);
+ }
+
+ ch = ' '; // go to page mode!
+ break;
+
+ // function keys
+ case 'y': // accept
+#ifdef REGFORM_DISABLE_ONLINE_USER
+ if (forms[ci].online)
+ {
+ vmsg("暫不開放審核在線上使用者。");
+ break;
+ }
+#endif
+ case 's': // skip
+ case 'd': // delete
+ case KEY_DEL: //delete
+ if (ch == KEY_DEL) ch = 'd';
+
+ grayout(ci*2, ci*2+1, GRAYOUT_DARK);
+ move_ansi(ci*2, 4); outc(ch);
+ ans[ci] = ch;
+ ch = 'j'; // go next
+ break;
+
+ case 'u': // undo
+#ifdef REGFORM_DISABLE_ONLINE_USER
+ if (forms[ci].online)
+ {
+ vmsg("暫不開放審核在線上使用者。");
+ break;
+ }
+#endif
+ grayout(ci*2, ci*2+1, GRAYOUT_NORM);
+ move_ansi(ci*2, 4); outc('.');
+ ans[ci] = 0;
+ ch = 'j'; // go next
+ break;
+
+ case 'n': // reject
+#ifdef REGFORM_DISABLE_ONLINE_USER
+ if (forms[ci].online)
+ {
+ vmsg("暫不開放審核在線上使用者。");
+ break;
+ }
+#endif
+ // query for reason
+ resolve_reason(rejects[ci], yMsg);
+ prompt_regform_ui();
+
+ if (!rejects[ci][0])
+ break;
+
+ move(yMsg, 0);
+ prints("退回 %s 註冊單原因:\n %s\n", forms[ci].userid, rejects[ci]);
+
+ // do reject
+ grayout(ci*2, ci*2+1, GRAYOUT_DARK);
+ move_ansi(ci*2, 4); outc(ch);
+ ans[ci] = ch;
+ ch = 'j'; // go next
+
+ break;
+ } // switch(ch)
+
+ // change cursor
+ if (ch == 'j' && ++ci >= cforms)
+ ci = cforms -1;
+ } // while(ch != QUIT/SAVE)
+
+ // if exit, we still need to skip all read forms
+ if (ch == 'q')
+ {
+ for (i = 0; i < cforms; i++)
+ ans[i] = 's';
+ }
+
+ // page complete (save).
+ assert(ch == ' ' || ch == 'q');
+
+ // save/commit if required.
+ if (dryrun)
+ {
+ // prmopt for debug
+ clear();
+ stand_title("測試模式");
+ outs("您正在執行測試模式,所以剛審的註冊單並不會生效。\n"
+ "下面列出的是剛才您審完的結果:\n\n");
+
+ for (i = 0; i < cforms; i++)
+ {
+ char justify[REGLEN];
+ if (ans[i] == 'y')
+ snprintf(justify, sizeof(justify), // build justify string
+ "%s:%s:%s", forms[i].phone, forms[i].career, cuser.userid);
+
+ prints("%2d. %-12s - %c %s\n", i+1, forms[i].userid, ans[i],
+ ans[i] == 'n' ? rejects[i] :
+ ans[i] == 'y' ? justify : "");
+ }
+ if (ch != 'q')
+ pressanykey();
+ }
+ else
+ {
+ // real functionality
+ for (i = 0; i < cforms; i++)
+ {
+ switch(ans[i])
+ {
+ case 'a': // accept
+ regfrm_accept(&forms[i]);
+ break;
+
+ case 'd': // delete
+ regfrm_delete(uid);
+ break;
+
+ case 'n': // reject
+ regfrm_reject(&forms[i], rsn);
+ break;
+
+ case 's': // skip
+ // do nothing.
+ break;
+
+ default:
+ assert(0);
+ break;
+ }
+ }
+ } // !dryrun
+
+ // unlock all forms
+ for (i = 0; i < cforms; i++)
+ regfrm_unlock(lfds[i]);
+
+ } // while (ch != 'q')
+
+ regq_end_pull(fpregq);
+
+ // finishing
+ clear(); move(5, 0);
+ prints("您審了 %d 份註冊單份。", tid);
+ pressanykey();
+ return 0;
+}
+
+/////////////////////////////////////////////////////////////////////////////
+// Regform UI
+// 處理 Register Form
+/////////////////////////////////////////////////////////////////////////////
+
+/* Auto-Regform-Scan
+ * FIXME 真是一團垃圾
+ *
+ * fdata 用了太多 magic number
+ * return value 應該是指 reason (return index + 1)
+ * ans[0] 指的是帳管選擇的「錯誤的欄位」 (Register 選單裡看到的那些)
+ */
+static int
+auto_scan(char fdata[][STRLEN], char ans[])
+{
+ int good = 0;
+ int count = 0;
+ int i;
+ char temp[10];
+
+ if (!strncmp(fdata[1], "小", 2) || strstr(fdata[1], "丫")
+ || strstr(fdata[1], "誰") || strstr(fdata[1], "不")) {
+ ans[0] = '0';
+ return 1;
+ }
+ strlcpy(temp, fdata[1], 3);
+
+ /* 疊字 */
+ if (!strncmp(temp, &(fdata[1][2]), 2)) {
+ ans[0] = '0';
+ return 1;
+ }
+ if (strlen(fdata[1]) >= 6) {
+ if (strstr(fdata[1], "陳水扁")) {
+ ans[0] = '0';
+ return 1;
+ }
+ if (strstr("趙錢孫李周吳鄭王", temp))
+ good++;
+ else if (strstr("杜顏黃林陳官余辛劉", temp))
+ good++;
+ else if (strstr("蘇方吳呂李邵張廖應蘇", temp))
+ good++;
+ else if (strstr("徐謝石盧施戴翁唐", temp))
+ good++;
+ }
+ if (!good)
+ return 0;
+
+ if (!strcmp(fdata[2], fdata[3]) ||
+ !strcmp(fdata[2], fdata[4]) ||
+ !strcmp(fdata[3], fdata[4])) {
+ ans[0] = '4';
+ return 5;
+ }
+ if (strstr(fdata[2], "大")) {
+ if (strstr(fdata[2], "台") || strstr(fdata[2], "淡") ||
+ strstr(fdata[2], "交") || strstr(fdata[2], "政") ||
+ strstr(fdata[2], "清") || strstr(fdata[2], "警") ||
+ strstr(fdata[2], "師") || strstr(fdata[2], "銘傳") ||
+ strstr(fdata[2], "中央") || strstr(fdata[2], "成") ||
+ strstr(fdata[2], "輔") || strstr(fdata[2], "東吳"))
+ good++;
+ } else if (strstr(fdata[2], "女中"))
+ good++;
+
+ if (strstr(fdata[3], "地球") || strstr(fdata[3], "宇宙") ||
+ strstr(fdata[3], "信箱")) {
+ ans[0] = '2';
+ return 3;
+ }
+ if (strstr(fdata[3], "市") || strstr(fdata[3], "縣")) {
+ if (strstr(fdata[3], "路") || strstr(fdata[3], "街")) {
+ if (strstr(fdata[3], "號"))
+ good++;
+ }
+ }
+ for (i = 0; fdata[4][i]; i++) {
+ if (isdigit((int)fdata[4][i]))
+ count++;
+ }
+
+ if (count <= 4) {
+ ans[0] = '3';
+ return 4;
+ } else if (count >= 7)
+ good++;
+
+ if (good >= 3) {
+ ans[0] = 'y';
+ return -1;
+ } else
+ return 0;
+}
+
+/////////////////////////////////////////////////////////////////////////////
+// Traditional Regform UI
+/////////////////////////////////////////////////////////////////////////////
+// TODO XXX process someone directly, according to target_uid.
+int
+scan_register_form(const char *regfile, int automode, const char *target_uid)
+{
+ char genbuf[200];
+ char *field[] = {
+ "uid", "name", "career", "addr", "phone", "email", NULL
+ };
+ char *finfo[] = {
+ "帳號", "真實姓名", "服務單位", "目前住址",
+ "連絡電話", "電子郵件信箱", NULL
+ };
+ char *reason[REJECT_REASONS+1] = {
+ "輸入真實姓名",
+ "詳填「(畢業)學校及『系』『級』」或「服務單位(含所屬縣市及職稱)」",
+ "填寫完整的住址資料 (含縣市名稱, 台北市請含行政區域)",
+ "詳填連絡電話 (含區域碼, 中間不用加 '-', '(', ')'等符號",
+ "精確並完整填寫註冊申請表",
+ "用中文填寫申請單",
+ NULL
+ };
+ char *autoid = "AutoScan";
+ userec_t muser;
+ FILE *fn, *fout, *freg;
+ char fdata[6][STRLEN];
+ char fname[STRLEN] = "", buf[STRLEN];
+ char ans[4], *ptr, *uid;
+ int n = 0, unum = 0, tid = 0;
+ int nSelf = 0, nAuto = 0;
+
+ uid = cuser.userid;
+ move(2, 0);
+
+ fn = pull_regform(regfile, fname, -1);
+ if (!fn)
+ return -1;
+
+ while( fgets(genbuf, STRLEN, fn) ){
+ memset(fdata, 0, sizeof(fdata));
+ do {
+ if( genbuf[0] == '-' )
+ break;
+ if ((ptr = (char *)strstr(genbuf, ": "))) {
+ *ptr = '\0';
+ for (n = 0; field[n]; n++) {
+ if (strcmp(genbuf, field[n]) == 0) {
+ strlcpy(fdata[n], ptr + 2, sizeof(fdata[n]));
+ if ((ptr = (char *)strchr(fdata[n], '\n')))
+ *ptr = '\0';
+ }
+ }
+ }
+ } while( fgets(genbuf, STRLEN, fn) );
+ tid ++;
+
+ if ((unum = getuser(fdata[0], &muser)) == 0) {
+ move(2, 0);
+ clrtobot();
+ outs("系統錯誤,查無此人\n\n");
+ for (n = 0; field[n]; n++)
+ prints("%s : %s\n", finfo[n], fdata[n]);
+ pressanykey();
+ } else {
+ if (automode)
+ uid = autoid;
+
+ if ((!automode || !auto_scan(fdata, ans))) {
+ uid = cuser.userid;
+
+ move(1, 0);
+ clrtobot();
+ prints("帳號位置 : %d\n", unum);
+ user_display(&muser, 1);
+ move(14, 0);
+ prints(ANSI_COLOR(1;32) "------------- "
+ "請站長嚴格審核使用者資料,這是第 %d 份"
+ "------------" ANSI_RESET "\n", tid);
+ prints(" %-12s: %s\n", finfo[0], fdata[0]);
+#ifdef FOREIGN_REG
+ prints("0.%-12s: %s%s\n", finfo[1], fdata[1],
+ muser.uflag2 & FOREIGN ? " (外籍)" : "");
+#else
+ prints("0.%-12s: %s\n", finfo[1], fdata[1]);
+#endif
+ for (n = 2; field[n]; n++) {
+ prints("%d.%-12s: %s\n", n - 1, finfo[n], fdata[n]);
+ }
+ if (muser.userlevel & PERM_LOGINOK) {
+ ans[0] = getkey("此帳號已經完成註冊, "
+ "更新(Y/N/Skip)?[N] ");
+ if (ans[0] != 'y' && ans[0] != 's')
+ ans[0] = 'd';
+ } else {
+ if (search_ulist(unum) == NULL)
+ {
+ move(b_lines, 0); clrtoeol();
+ outs("是否接受此資料(Y/N/Q/Del/Skip)?[S] ");
+ // FIXME if the user got online here
+ ans[0] = igetch();
+ }
+ else
+ ans[0] = 's';
+ ans[0] = tolower(ans[0]);
+ if (ans[0] != 'y' && ans[0] != 'n' &&
+ ans[0] != 'q' && ans[0] != 'd' &&
+ !('0' <= ans[0] && ans[0] < ('0' + REJECT_REASONS)))
+ ans[0] = 's';
+ ans[1] = 0;
+ }
+ nSelf++;
+ } else
+ nAuto++;
+
+ switch (ans[0]) {
+ case 'q':
+ if ((freg = fopen(regfile, "a"))) {
+ for (n = 0; field[n]; n++)
+ fprintf(freg, "%s: %s\n", field[n], fdata[n]);
+ fprintf(freg, "----\n");
+ while (fgets(genbuf, STRLEN, fn))
+ fputs(genbuf, freg);
+ fclose(freg);
+ }
+ case 'd':
+ break;
+
+ case '0': case '1': case '2':
+ case '3': case '4': case '5':
+ /* please confirm match REJECT_REASONS here */
+ case 'n':
+ if (ans[0] == 'n') {
+ int nf = 0;
+ move(8, 0);
+ clrtobot();
+ outs("請提出退回申請表原因,按 <enter> 取消\n");
+ for (n = 0; n < REJECT_REASONS; n++)
+ prints("%d) 請%s\n", n, reason[n]);
+ outs("\n"); // preserved for prompt
+ for (nf = 0; field[nf]; nf++)
+ prints("%s: %s\n", finfo[nf], fdata[nf]);
+ } else
+ buf[0] = ans[0];
+
+ if (ans[0] != 'n' ||
+ getdata(9 + n, 0, "退回原因: ", buf, 60, DOECHO))
+ if ((buf[0] - '0') >= 0 && (buf[0] - '0') < n) {
+ int i;
+ fileheader_t mhdr;
+ char title[128], buf1[80];
+ FILE *fp;
+
+ sethomepath(buf1, muser.userid);
+ stampfile(buf1, &mhdr);
+ strlcpy(mhdr.owner, cuser.userid, sizeof(mhdr.owner));
+ strlcpy(mhdr.title, "[註冊失敗]", TTLEN);
+ mhdr.filemode = 0;
+ sethomedir(title, muser.userid);
+ if (append_record(title, &mhdr, sizeof(mhdr)) != -1) {
+ char rejfn[PATHLEN];
+ fp = fopen(buf1, "w");
+
+ for(i = 0; buf[i] && i < sizeof(buf); i++){
+ if (buf[i] >= '0' && buf[i] < '0'+n)
+ {
+ fprintf(fp, "[退回原因] 請%s\n",
+ reason[buf[i] - '0']);
+ }
+ }
+
+ fclose(fp);
+
+ // build reject file
+ sethomefile(rejfn, muser.userid, FN_REJECT_NOTIFY);
+ Copy(buf1, rejfn);
+ }
+ if ((fout = fopen(FN_REGISTER_LOG, "a"))) {
+ for (n = 0; field[n]; n++)
+ fprintf(fout, "%s: %s\n", field[n], fdata[n]);
+ fprintf(fout, "Date: %s\n", Cdate(&now));
+ fprintf(fout, "Rejected: %s [%s]\n----\n",
+ uid, buf);
+ fclose(fout);
+ }
+ break;
+ }
+ move(10, 0);
+ clrtobot();
+ outs("取消退回此註冊申請表");
+ /* no break? */
+
+ case 's':
+ if ((freg = fopen(regfile, "a"))) {
+ for (n = 0; field[n]; n++)
+ fprintf(freg, "%s: %s\n", field[n], fdata[n]);
+ fprintf(freg, "----\n");
+ fclose(freg);
+ }
+ break;
+
+ default:
+ outs("以下使用者資料已經更新:\n");
+ mail_muser(muser, "[註冊成功\囉]", "etc/registered");
+
+#if FOREIGN_REG_DAY > 0
+ if(muser.uflag2 & FOREIGN)
+ mail_muser(muser, "[出入境管理局]", "etc/foreign_welcome");
+#endif
+
+ muser.userlevel |= (PERM_LOGINOK | PERM_POST);
+ strlcpy(muser.realname, fdata[1], sizeof(muser.realname));
+ strlcpy(muser.address, fdata[3], sizeof(muser.address));
+ strlcpy(muser.email, fdata[5], sizeof(muser.email));
+ snprintf(genbuf, sizeof(genbuf), "%s:%s:%s",
+ fdata[4], fdata[2], uid);
+ strlcpy(muser.justify, genbuf, sizeof(muser.justify));
+
+ passwd_update(unum, &muser);
+ // XXX TODO notify users?
+ sendalert(muser.userid, ALERT_PWD_PERM); // force to reload perm
+
+ sethomefile(buf, muser.userid, FN_JUSTIFY);
+ log_file(buf, LOG_CREAT, genbuf);
+
+ if ((fout = fopen(FN_REGISTER_LOG, "a"))) {
+ for (n = 0; field[n]; n++)
+ fprintf(fout, "%s: %s\n", field[n], fdata[n]);
+ fprintf(fout, "Date: %s\n", Cdate(&now));
+ fprintf(fout, "Approved: %s\n", uid);
+ fprintf(fout, "----\n");
+ fclose(fout);
+ }
+ sethomefile(genbuf, muser.userid, FN_JUSTIFY_WAIT);
+ unlink(genbuf);
+ break;
+ }
+ }
+ }
+
+ fclose(fn);
+ unlink(fname);
+
+ clear(); move(5, 0);
+ prints("您審了 %d 份註冊單,AutoScan 審了 %d 份", nSelf, nAuto);
+ pressanykey();
+ return (0);
+}
+
+/////////////////////////////////////////////////////////////////////////////
+// New Regform UI
+/////////////////////////////////////////////////////////////////////////////
+
+// #define REGFORM_DISABLE_ONLINE_USER
+// #define FORMS_IN_PAGE (10)
+
+int
+handle_register_form(const char *regfile, int dryrun)
+{
+ int unum = 0;
+ int yMsg = FORMS_IN_PAGE*2+1;
+ FILE *fp = NULL;
+ userec_t muser;
+ RegformEntry forms [FORMS_IN_PAGE];
+ char ans [FORMS_IN_PAGE];
+ char rejects[FORMS_IN_PAGE][REASON_LEN]; // reject reason length
+ char fname [PATHLEN] = "";
+ char justify[REGLEN+1];
+ char rsn [REASON_LEN];
+ int cforms = 0, // current loaded forms
+ parsed = 0, // total parsed forms
+ ci = 0, // cursor index
+ ch = 0, // input key
+ i, blanks;
+ long fsz = 0, fpos = 0;
+
+ // prepare reg tickets
+ if (dryrun)
+ {
+ // directly open regfile to try
+ fp = fopen(regfile, "rt");
+ } else {
+ fp = pull_regform(regfile, fname, -1);
+ }
+
+ if (!fp)
+ return 0;
+
+ // retreieve file info
+ fpos = ftell(fp);
+ fseek(fp, 0, SEEK_END);
+ fsz = ftell(fp);
+ fseek(fp, fpos, SEEK_SET);
+ if (!fsz) fsz = 1;
+
+ while (ch != 'q')
+ {
+ // initialize and prepare
+ memset(ans, 0, sizeof(ans));
+ memset(rejects, 0, sizeof(rejects));
+ memset(forms, 0, sizeof(forms));
+ cforms = 0;
+
+ // load forms
+ while (cforms < FORMS_IN_PAGE && load_regform_entry(&forms[cforms], fp))
+ cforms++, parsed ++;
+
+ // if no more forms then leave.
+ // TODO what if regform error?
+ if (cforms < 1)
+ break;
+
+ // adjust cursor if required
+ if (ci >= cforms)
+ ci = cforms-1;
+
+ // display them all.
+ clear();
+ for (i = 0; i < cforms; i++)
+ {
+ // fetch user information
+ memset(&muser, 0, sizeof(muser));
+ unum = getuser(forms[i].userid, &muser);
+ forms[i].exist = unum ? 1 : 0;
+ if (unum) forms[i].online = search_ulist(unum) ? 1 : 0;
+
+ // if already got login level, delete by default.
+ if (!unum)
+ ans[i] = 'd';
+ else {
+ if (muser.userlevel & PERM_LOGINOK)
+ ans[i] = 'd';
+#ifdef REGFORM_DISABLE_ONLINE_USER
+ else if (forms[i].online)
+ ans[i] = 's';
+#endif // REGFORM_DISABLE_ONLINE_USER
+ }
+
+ // print
+ move(i*2, 0);
+ prints(" %2d%s %s%-12s " ANSI_RESET,
+ i+1,
+ (unum == 0) ? ANSI_COLOR(1;31) "D" :
+ ( (muser.userlevel & PERM_LOGINOK) ?
+ ANSI_COLOR(1;33) "Y" :
+#ifdef REGFORM_DISABLE_ONLINE_USER
+ forms[i].online ? "s" :
+#endif
+ "."),
+ forms[i].online ? ANSI_COLOR(1;35) : ANSI_COLOR(1),
+ forms[i].userid);
+
+ prints( ANSI_COLOR(1;31) "%19s "
+ ANSI_COLOR(1;32) "%-40s" ANSI_RESET"\n",
+ forms[i].name, forms[i].career);
+
+ move(i*2+1, 0);
+ prints(" %s %-50s%20s\n",
+ (muser.userlevel & PERM_NOREGCODE) ?
+ ANSI_COLOR(1;31) "T" ANSI_RESET : " ",
+ forms[i].addr, forms[i].phone);
+ }
+
+ // display page info
+ {
+ char msg[STRLEN];
+ fpos = ftell(fp);
+ if (fpos > fsz) fsz = fpos*10;
+ snprintf(msg, sizeof(msg),
+ "%s 已顯示 %d 份註冊單 (%2d%%) ",
+ dryrun? "(測試模式)" : "",
+ parsed, (int)(fpos*100/fsz));
+ prints(ANSI_COLOR(7) "\n%78s" ANSI_RESET "\n", msg);
+ }
+
+ // handle user input
+ prompt_regform_ui();
+ ch = 0;
+ while (ch != 'q' && ch != ' ') {
+ ch = cursor_key(ci*2, 0);
+ switch (ch)
+ {
+ // nav keys
+ case KEY_UP:
+ case 'k':
+ if (ci > 0) ci--;
+ break;
+
+ case KEY_DOWN:
+ case 'j':
+ ch = 'j'; // go next
+ break;
+
+ // quick nav (assuming to FORMS_IN_PAGE=10)
+ case '1': case '2': case '3': case '4': case '5':
+ case '6': case '7': case '8': case '9':
+ ci = ch - '1';
+ if (ci >= cforms) ci = cforms-1;
+ break;
+ case '0':
+ ci = 10-1;
+ if (ci >= cforms) ci = cforms-1;
+ break;
+
+ /*
+ case KEY_HOME: ci = 0; break;
+ case KEY_END: ci = cforms-1; break;
+ */
+
+ // abort
+ case KEY_END:
+ case 'q':
+ ch = 'q';
+ if (getans("確定要離開了嗎? (本頁變更將不會儲存) [y/N]: ") != 'y')
+ {
+ prompt_regform_ui();
+ ch = 0;
+ continue;
+ }
+ break;
+
+ // prepare to go next page
+ case KEY_PGDN:
+ case ' ':
+ ch = ' ';
+
+ // solving blank (undecided entries)
+ for (i = 0, blanks = 0; i < cforms; i++)
+ if (ans[i] == 0) blanks ++;
+
+ if (!blanks)
+ break;
+
+ // have more blanks
+ ch = getans("尚未指定的 %d 個項目要: (S跳過/y通過/n拒絕/e繼續編輯): ",
+ blanks);
+
+ if (ch == 'e')
+ {
+ prompt_regform_ui();
+ ch = 0;
+ continue;
+ }
+ if (ch == 'y') {
+ // do nothing.
+ } else if (ch == 'n') {
+ // query reject reason
+ resolve_reason(rsn, yMsg);
+ if (*rsn == 0)
+ ch = 's';
+ } else ch = 's';
+
+ // filling answers
+ for (i = 0; i < cforms; i++)
+ {
+ if (ans[i] != 0)
+ continue;
+ ans[i] = ch;
+ if (ch != 'n')
+ continue;
+ strlcpy(rejects[i], rsn, REASON_LEN);
+ }
+
+ ch = ' '; // go to page mode!
+ break;
+
+ // function keys
+ case 'y': // accept
+#ifdef REGFORM_DISABLE_ONLINE_USER
+ if (forms[ci].online)
+ {
+ vmsg("暫不開放審核在線上使用者。");
+ break;
+ }
+#endif
+ case 's': // skip
+ case 'd': // delete
+ case KEY_DEL: //delete
+ if (ch == KEY_DEL) ch = 'd';
+
+ grayout(ci*2, ci*2+1, GRAYOUT_DARK);
+ move_ansi(ci*2, 4); outc(ch);
+ ans[ci] = ch;
+ ch = 'j'; // go next
+ break;
+
+ case 'u': // undo
+#ifdef REGFORM_DISABLE_ONLINE_USER
+ if (forms[ci].online)
+ {
+ vmsg("暫不開放審核在線上使用者。");
+ break;
+ }
+#endif
+ grayout(ci*2, ci*2+1, GRAYOUT_NORM);
+ move_ansi(ci*2, 4); outc('.');
+ ans[ci] = 0;
+ ch = 'j'; // go next
+ break;
+
+ case 'n': // reject
+#ifdef REGFORM_DISABLE_ONLINE_USER
+ if (forms[ci].online)
+ {
+ vmsg("暫不開放審核在線上使用者。");
+ break;
+ }
+#endif
+ // query for reason
+ resolve_reason(rejects[ci], yMsg);
+ prompt_regform_ui();
+
+ if (!rejects[ci][0])
+ break;
+
+ move(yMsg, 0);
+ prints("退回 %s 註冊單原因:\n %s\n", forms[ci].userid, rejects[ci]);
+
+ // do reject
+ grayout(ci*2, ci*2+1, GRAYOUT_DARK);
+ move_ansi(ci*2, 4); outc(ch);
+ ans[ci] = ch;
+ ch = 'j'; // go next
+
+ break;
+ } // switch(ch)
+
+ // change cursor
+ if (ch == 'j' && ++ci >= cforms)
+ ci = cforms -1;
+ } // while(ch != QUIT/SAVE)
+
+ // if exit, we still need to skip all read forms
+ if (ch == 'q')
+ {
+ for (i = 0; i < cforms; i++)
+ ans[i] = 's';
+ }
+
+ // page complete (save).
+ assert(ch == ' ' || ch == 'q');
+
+ // save/commit if required.
+ if (dryrun)
+ {
+ // prmopt for debug
+ clear();
+ stand_title("測試模式");
+ outs("您正在執行測試模式,所以剛審的註冊單並不會生效。\n"
+ "下面列出的是剛才您審完的結果:\n\n");
+
+ for (i = 0; i < cforms; i++)
+ {
+ if (ans[i] == 'y')
+ snprintf(justify, sizeof(justify), // build justify string
+ "%s:%s:%s", forms[i].phone, forms[i].career, cuser.userid);
+
+ prints("%2d. %-12s - %c %s\n", i+1, forms[i].userid, ans[i],
+ ans[i] == 'n' ? rejects[i] :
+ ans[i] == 'y' ? justify : "");
+ }
+ if (ch != 'q')
+ pressanykey();
+ }
+ else
+ {
+ // real functionality
+ for (i = 0; i < cforms; i++)
+ {
+ if (ans[i] == 'y')
+ {
+ // build justify string
+ snprintf(justify, sizeof(justify),
+ "%s:%s:%s", forms[i].phone, forms[i].career, cuser.userid);
+
+ regform_accept(forms[i].userid, justify);
+ // log form to FN_REGISTER_LOG
+ append_regform(&forms[i], FN_REGISTER_LOG,
+ "Approved", cuser.userid, NULL);
+ }
+ else if (ans[i] == 'n')
+ {
+ regform_reject(forms[i].userid, rejects[i]);
+ // log form to FN_REGISTER_LOG
+ append_regform(&forms[i], FN_REGISTER_LOG,
+ "Rejected", cuser.userid, rejects[i]);
+ }
+ else if (ans[i] == 's')
+ {
+ // append form back to fn_register
+ append_regform(&forms[i], fn_register,
+ NULL, NULL, NULL);
+ }
+ }
+ } // !dryrun
+
+ } // while (ch != 'q')
+
+ // cleaning left regforms
+ if (!dryrun)
+ {
+ pump_regform(regfile, fp);
+ fclose(fp);
+ unlink(fname);
+ } else {
+ // directly close file should be OK.
+ fclose(fp);
+ }
+
+ return 0;
+}
+
+int
+m_register(void)
+{
+ FILE *fn;
+ int x, y, wid, len;
+ char ans[4];
+ char genbuf[200];
+
+ if ((fn = fopen(fn_register, "r")) == NULL) {
+ outs("目前並無新註冊資料");
+ return XEASY;
+ }
+ stand_title("審核使用者註冊資料");
+ y = 2;
+ x = wid = 0;
+
+ while (fgets(genbuf, STRLEN, fn) && x < 65) {
+ if (strncmp(genbuf, "uid: ", 5) == 0) {
+ move(y++, x);
+ outs(genbuf + 5);
+ len = strlen(genbuf + 5);
+ if (len > wid)
+ wid = len;
+ if (y >= t_lines - 3) {
+ y = 2;
+ x += wid + 2;
+ }
+ }
+ }
+ fclose(fn);
+ getdata(b_lines - 1, 0,
+ "開始審核嗎(Auto自動/Yes手動/No不審/Exp新界面)?[N] ",
+ ans, sizeof(ans), LCECHO);
+ if (ans[0] == 'a')
+ scan_register_form(fn_register, 1, NULL);
+ else if (ans[0] == 'y')
+ scan_register_form(fn_register, 0, NULL);
+ else if (ans[0] == 'e')
+ {
+#ifdef EXP_ADMIN_REGFORM_DRYRUN
+ int dryrun = 0;
+ if (getans("你要進行純測試(T)還是真的執行審核(y)?") == 'y')
+ {
+ vmsg("進入實際執行模式,所有審核動作都是真的。");
+ dryrun = 0;
+ } else {
+ vmsg("測試模式。");
+ dryrun = 1;
+ }
+ handle_register_form(fn_register, dryrun);
+#else
+ // run directly.
+ handle_register_form(fn_register, 0);
+#endif
+ }
+
+ return 0;
+}
+
+int
+cat_register(void)
+{
+ if (system("cat register.new.tmp >> register.new") == 0 &&
+ unlink("register.new.tmp") == 0)
+ vmsg("OK 嚕~~ 繼續去奮鬥吧!!");
+ else
+ vmsg("沒辦法CAT過去呢 去檢查一下系統吧!!");
+ return 0;
+}
+
+/* vim:sw=4
+ */