Import ezmlm-idx 0.40
[ezmlm] / sub_pgsql / checktag.c
diff --git a/sub_pgsql/checktag.c b/sub_pgsql/checktag.c
new file mode 100644 (file)
index 0000000..330c236
--- /dev/null
@@ -0,0 +1,106 @@
+/*$Id: checktag.c,v 1.3 1999/12/23 02:40:57 lindberg Exp $*/
+/*$Name: ezmlm-idx-040 $*/
+#include "stralloc.h"
+#include "scan.h"
+#include "fmt.h"
+#include "cookie.h"
+#include "makehash.h"
+#include "strerr.h"
+#include "errtxt.h"
+#include "subscribe.h"
+#include <unistd.h>
+#include <libpq-fe.h>
+
+static stralloc key = {0};
+static stralloc line = {0};
+static strnum[FMT_ULONG];
+static newcookie[COOKIE];
+
+char *checktag (dir,num,listno,action,seed,hash)
+/* reads dir/sql. If not present, returns success (NULL). If dir/sql is    */
+/* present, checks hash against the cookie table. If match, returns success*/
+/* (NULL), else returns "". If error, returns error string. */
+
+
+char *dir;                             /* the db base dir */
+unsigned long num;                     /* message number */
+unsigned long listno;                  /* bottom of range => slave */
+char *action;
+char *seed;                            /* cookie base */
+char *hash;                            /* cookie */
+{
+  PGresult *result;
+  /*  int row; */
+  char *table = (char *) 0;
+  char *r;
+
+  if ((r = opensql(dir,&table))) {
+    if (*r) return r;
+    if (!seed) return (char *) 0;              /* no data - accept */
+
+    strnum[fmt_ulong(strnum,num)] = '\0';      /* message nr ->string*/
+
+    switch(slurp("key",&key,32)) {
+    case -1:
+      return ERR_READ_KEY;
+    case 0:
+      return ERR_NOEXIST_KEY;
+    }
+
+    cookie(newcookie,key.s,key.len,strnum,seed,action);
+    if (byte_diff(hash,COOKIE,newcookie)) return "";
+    else return (char *) 0;
+
+  } else {
+
+    /* SELECT msgnum FROM table_cookie WHERE msgnum=num and cookie='hash' */
+    /* succeeds only is everything correct. 'hash' is quoted since it is  */
+    /* potentially hostile. */
+    if (listno) {                      /* only for slaves */
+      if (!stralloc_copys(&line,"SELECT listno FROM ")) return ERR_NOMEM;
+      if (!stralloc_cats(&line,table)) return ERR_NOMEM;
+      if (!stralloc_cats(&line,"_mlog WHERE listno=")) return ERR_NOMEM;
+      if (!stralloc_catb(&line,strnum,fmt_ulong(strnum,listno)))
+       return ERR_NOMEM;
+      if (!stralloc_cats(&line," AND msgnum=")) return ERR_NOMEM;
+      if (!stralloc_catb(&line,strnum,fmt_ulong(strnum,num))) return ERR_NOMEM;
+      if (!stralloc_cats(&line," AND done > 3")) return ERR_NOMEM;
+
+      if (!stralloc_0(&line)) return ERR_NOMEM;
+      result = PQexec( psql, line.s );
+      if(result == NULL)
+       return (PQerrorMessage(psql));
+      if( PQresultStatus(result) != PGRES_TUPLES_OK)
+       return (char *) (PQresultErrorMessage(result));
+      if( PQntuples(result) > 0 ) {
+       PQclear(result);
+       return("");
+      } else
+       PQclear(result);
+    }
+
+    if (!stralloc_copys(&line,"SELECT msgnum FROM ")) return ERR_NOMEM;
+    if (!stralloc_cats(&line,table)) return ERR_NOMEM;
+    if (!stralloc_cats(&line,"_cookie WHERE msgnum=")) return ERR_NOMEM;
+    if (!stralloc_catb(&line,strnum,fmt_ulong(strnum,num))) return ERR_NOMEM;
+    if (!stralloc_cats(&line," and cookie='")) return ERR_NOMEM;
+    if (!stralloc_catb(&line,strnum,fmt_ulong(strnum,hash))) return ERR_NOMEM;
+    if (!stralloc_cats(&line,"'")) return ERR_NOMEM;
+
+    if (!stralloc_0(&line)) return ERR_NOMEM;
+    result = PQexec(psql,line.s);
+    if (result == NULL)
+      return (PQerrorMessage(psql));
+    if (PQresultStatus(result) != PGRES_TUPLES_OK)
+      return (char *) (PQresultErrorMessage(result));
+    if(PQntuples(result) < 0) {
+      PQclear( result );
+      return("");
+    }
+
+    PQclear(result);
+    if (listno)
+      (void) logmsg(dir,num,listno,0L,3);      /* non-ess mysql logging */
+    return (char *)0;
+  }
+}