summaryrefslogtreecommitdiffstats
path: root/pttpi
diff options
context:
space:
mode:
Diffstat (limited to 'pttpi')
-rw-r--r--pttpi/cgi/.cvsignore1
-rw-r--r--pttpi/cgi/Makefile16
-rw-r--r--pttpi/cgi/pttpi.c244
3 files changed, 261 insertions, 0 deletions
diff --git a/pttpi/cgi/.cvsignore b/pttpi/cgi/.cvsignore
new file mode 100644
index 00000000..169dce0c
--- /dev/null
+++ b/pttpi/cgi/.cvsignore
@@ -0,0 +1 @@
+*.cgi
diff --git a/pttpi/cgi/Makefile b/pttpi/cgi/Makefile
new file mode 100644
index 00000000..42613de3
--- /dev/null
+++ b/pttpi/cgi/Makefile
@@ -0,0 +1,16 @@
+# $Id: Makefile,v 1.1 2003/05/19 01:33:00 in2 Exp $
+
+.include "../../pttbbs.mk"
+
+CFLAGS+= -I../../include `xmlrpc-c-config cgi-server --cflags`
+LDFLAGS+= `xmlrpc-c-config cgi-server --libs`
+
+UTIL_OBJS= \
+ ../../util/util_cache.o ../../util/util_record.o ../../util/util_passwd.o
+all: pttpi.cgi
+
+pttpi.cgi: pttpi.c
+ $(CC) $(CFLAGS) $(LDFLAGS) -o pttpi.cgi pttpi.c $(UTIL_OBJS)
+
+clean:
+ rm -f *.cgi *~
diff --git a/pttpi/cgi/pttpi.c b/pttpi/cgi/pttpi.c
new file mode 100644
index 00000000..187c4f63
--- /dev/null
+++ b/pttpi/cgi/pttpi.c
@@ -0,0 +1,244 @@
+/* $Id: pttpi.c,v 1.1 2003/05/19 01:33:00 in2 Exp $ */
+#include "bbs.h"
+#include "pierr.h"
+#include <xmlrpc.h>
+#include <xmlrpc_cgi.h>
+
+extern SHM_t *SHM;
+typedef xmlrpc_int32 int32;
+
+#define errorexit() if( env->fault_occurred ) return NULL
+#define check_board_and_permission(bid) \
+ if( bid < 0 || bid > MAX_BOARD || \
+ !bcache[bid].brdname[0] || \
+ (bcache[bid].brdattr & BRD_HIDE) || \
+ (!(bcache[bid].brdattr & BRD_GROUPBOARD) && \
+ (bcache[bid].brdattr & BRD_POSTMASK)) ) \
+ return xmlrpc_build_value(env, "{s:i}", \
+ "errno", PIERR_NOBRD);
+
+#define errorreturn(returncode) \
+ return xmlrpc_build_value(env, "{s:i}", \
+ "errno", returncode)
+
+xmlrpc_value *
+getBid(xmlrpc_env *env, xmlrpc_value *param_array, void *user_data)
+{
+ /* !!! overwrite util_cache.c !!! */
+ char *brdname;
+ int bid;
+
+ xmlrpc_parse_value(env, param_array, "(s)", &brdname);
+ errorexit();
+
+ bid = getbnum(brdname);
+ return xmlrpc_build_value(env, "{s:i,s:s,s:i}",
+ "errno", (bid == -1 ? PIERR_NOBRD : PIERR_OK),
+ "brdname", brdname,
+ "bid", (int32)bid);
+}
+
+xmlrpc_value *
+getBrdInfo(xmlrpc_env *env, xmlrpc_value *param_array, void *user_data)
+{
+ int32 bid;
+
+ xmlrpc_parse_value(env, param_array, "(i)", &bid);
+ errorexit();
+ check_board_and_permission(bid);
+
+ return xmlrpc_build_value(env, "{s:i,s:s,s:i,s:6,s:s,s:i}",
+ "errno", PIERR_OK,
+ "brdname", bcache[bid].brdname,
+ "bid", (int32)bid,
+ "title", bcache[bid].title, strlen(bcache[bid].title),
+ "BM", bcache[bid].BM,
+ "nuser", bcache[bid].nuser);
+}
+
+char *getpath(int bid, char *fn)
+{
+ static char fpath[MAXPATHLEN];
+ if( fn == NULL )
+ fn = "";
+ snprintf(fpath, sizeof(fpath), "boards/%c/%s/%s",
+ bcache[bid].brdname[0], bcache[bid].brdname, fn);
+ return fpath;
+}
+
+int getfilesize(int bid, char *fn, int *fd)
+{
+ struct stat sb;
+ if( fd == NULL ){
+ if( stat(getpath(bid, fn), &sb) < 0 )
+ return -1;
+ }
+ else {
+ if( (*fd = open(getpath(bid, fn), O_RDONLY)) < 0 )
+ return -1;
+ if( fstat(*fd, &sb) < 0 ){
+ close(*fd);
+ return -1;
+ }
+ }
+ return sb.st_size;
+}
+
+xmlrpc_value *
+getNarticle(xmlrpc_env *env, xmlrpc_value *param_array, void *user_data)
+{
+ int bid, nas;
+
+ xmlrpc_parse_value(env, param_array, "(i)", &bid);
+ errorexit();
+ check_board_and_permission(bid);
+ nas = getfilesize(bid, ".DIR", NULL);
+
+ return xmlrpc_build_value(env, "{s:i,s:i}",
+ "errno", (nas == -1 ? PIERR_INT : PIERR_OK),
+ "narticle", nas / sizeof(fileheader_t));
+}
+
+xmlrpc_value *
+class_list(xmlrpc_env *env, xmlrpc_value *param_array, void *user_data)
+{
+ int bid;
+ boardheader_t *bptr;
+ xmlrpc_value *ret, *t;
+
+ xmlrpc_parse_value(env, param_array, "(i)", &bid);
+ errorexit();
+
+ if( bid != 0 )
+ check_board_and_permission(bid);
+
+ if( bid != 0 && !(bcache[bid].brdattr & BRD_GROUPBOARD) )
+ errorreturn(PIERR_NOTCLASS);
+
+ ret = xmlrpc_build_value(env, "()");
+ for( bptr = bcache[bid].firstchild[0] ;
+ bptr != (boardheader_t*)~0 ;
+ bptr = bptr->next[0] ){
+ if( (bptr->brdattr & BRD_HIDE) ||
+ (bptr->level && !bptr->brdattr & BRD_POSTMASK) )
+ continue;
+ t = xmlrpc_build_value(env, "{s:i,s:s,s:6,s:i,s:6,s:i,s:b}",
+ "bid", (int32)(bptr - bcache),
+ "brdname", bptr->brdname,
+ "title", bptr->title + 5, strlen(bptr->title) - 5,
+ "gid", bptr->gid,
+ "BM", bptr->BM, strlen(bptr->BM),
+ "nuser", (int32)bptr->nuser,
+ "isclass", (xmlrpc_bool)
+ (bptr->brdattr & BRD_GROUPBOARD) ? 1 : 0
+ );
+ xmlrpc_array_append_item(env, ret, t);
+ }
+ return ret;
+}
+
+xmlrpc_value *
+article_list(xmlrpc_env *env, xmlrpc_value *param_array, void *user_data)
+{
+#define MAXGETARTICLES 20
+ xmlrpc_value *ret, *t;
+ fileheader_t art[MAXGETARTICLES];
+ int32 bid, from, nArticles, fd, nGets, i;
+
+ xmlrpc_parse_value(env, param_array, "(ii)", &bid, &from);
+ errorexit();
+ check_board_and_permission(bid);
+
+ if( (nArticles = getfilesize(bid, ".DIR", &fd)) < 0 )
+ errorreturn(PIERR_INT);
+ nArticles /= sizeof(fileheader_t);
+
+ if( from < 0 )
+ from += nArticles;
+ if( from < 0 || from > nArticles ){
+ close(fd);
+ errorreturn(PIERR_NOMORE);
+ }
+
+ nGets = (((from + MAXGETARTICLES) < nArticles) ?
+ MAXGETARTICLES : (nArticles - from));
+ if( (nGets = read(fd, art, nGets * sizeof(fileheader_t))) < 0 ){
+ close(fd);
+ errorreturn(PIERR_INT);
+ }
+ close(fd);
+ nGets /= sizeof(fileheader_t);
+
+ ret = xmlrpc_build_value(env, "()");
+ for( i = 0 ; i < nGets ; ++i ){
+ t = xmlrpc_build_value(env, "{s:i,s:s,s:i,s:s,s:s,s:6}",
+ "articleid", (int32)(from + i),
+ "filename", art[i].filename,
+ "recommend", (int32)art[i].recommend,
+ "owner", art[i].owner,
+ "date", art[i].date,
+ "title", art[i].title, strlen(art[i].title)
+ );
+ xmlrpc_array_append_item(env, ret, t);
+ }
+ return ret;
+}
+
+xmlrpc_value *
+_article_readfn(xmlrpc_env *env, int bid, char *fn)
+{
+ char *content;
+ int fd, size;
+ xmlrpc_value *ret;
+ if( (size = getfilesize(bid, fn, &fd)) < 0 )
+ errorreturn(PIERR_INT);
+
+ content = (char *)malloc(sizeof(char) * size);
+ read(fd, content, size);
+ close(fd);
+ ret = xmlrpc_build_value(env, "{s:6}", "content", content, size);
+ free(content);
+
+ return ret;
+}
+
+xmlrpc_value *
+article_readfn(xmlrpc_env *env, xmlrpc_value *param_array, void *user_data)
+{
+ int bid;
+ char *fn;
+
+ xmlrpc_parse_value(env, param_array, "(is)", &bid, &fn);
+ errorexit();
+
+ if( fn == NULL || fn[0] != 'M' || fn[1] != '.' )
+ errorreturn(PIERR_NOBRD);
+ check_board_and_permission(bid);
+
+ return _article_readfn(env, bid, fn);
+}
+
+int main(int argc, char **argv)
+{
+ attach_SHM();
+ chdir(BBSHOME);
+ xmlrpc_cgi_init(XMLRPC_CGI_NO_FLAGS);
+ xmlrpc_cgi_add_method_w_doc("board.getBid", &getBid, NULL, "?",
+ "get bid from brdname");
+ xmlrpc_cgi_add_method_w_doc("board.getBrdInfo", &getBrdInfo, NULL, "?",
+ "get board information");
+ xmlrpc_cgi_add_method_w_doc("board.getNarticle", &getNarticle, NULL, "?",
+ "get # articles in the board");
+
+ xmlrpc_cgi_add_method_w_doc("class.list", &class_list, NULL, "?",
+ "list (C)lass");
+
+ xmlrpc_cgi_add_method_w_doc("article.list", &article_list, NULL, "?",
+ "article list");
+ xmlrpc_cgi_add_method_w_doc("article.readfn", &article_readfn, NULL, "?",
+ "read the article by filename");
+
+ xmlrpc_cgi_process_call();
+ xmlrpc_cgi_cleanup();
+ return 0;
+}