]> jfr.im git - irc/quakenet/newserv.git/blobdiff - dbapi2/dbapi2.c
CHANSERV: authtracker now keeps 240 days history
[irc/quakenet/newserv.git] / dbapi2 / dbapi2.c
index d33d54654c0e422a1519ff7781edbfd7cae3ce46..a960b74c9e0fd1b5cfe778497d8c9541e7ff6faa 100644 (file)
@@ -2,18 +2,23 @@
 #define PROVIDER_NAME_LEN 100
 #define QUERYBUFLEN 8192*2
 
-#define DBAPI_SNPRINTF_MAX_ARGS       20
-#define DBAPI_SNPRINTF_MAX_ARG_LENGTH 2048
+#define VSNPF_MAXARGS   20
+#define VSNPF_MAXARGLEN 2048
 
 #include <string.h>
 #include <stdlib.h>
 #include <stdarg.h>
+#include <time.h>
+#include <stdint.h>
 
 #include "../core/error.h"
 #include "../lib/strlfunc.h"
 #include "../lib/stringbuf.h"
+#include "../lib/version.h"
 #include "dbapi2.h"
 
+MODULE_VERSION("");
+
 struct DBAPIProviderData {
   char name[PROVIDER_NAME_LEN+1];
 };
@@ -64,7 +69,7 @@ static void dbclose(DBAPIConn *db) {
   free((DBAPIConn *)db);
 }
 
-static void dbquery(const DBAPIConn *db, DBAPIQueryCallback cb, DBAPIUserData data, const char *format, ...) {
+static void dbunsafequery(const DBAPIConn *db, DBAPIQueryCallback cb, DBAPIUserData data, const char *format, ...) {
   va_list ap;
   char buf[QUERYBUFLEN];
   size_t ret;
@@ -74,12 +79,12 @@ static void dbquery(const DBAPIConn *db, DBAPIQueryCallback cb, DBAPIUserData da
   va_end(ap);
 
   if(ret >= sizeof(buf))
-    Error("dbapi2", ERR_STOP, "Query truncated in dbquery, format: '%s', database: %s", format, db->name);
+    Error("dbapi2", ERR_STOP, "Query truncated in dbunsafequery, format: '%s', database: %s", format, db->name);
 
   db->__query(db, cb, data, buf);
 }
 
-static void dbcreatetable(const DBAPIConn *db, DBAPIQueryCallback cb, DBAPIUserData data, const char *format, ...) {
+static void dbunsafecreatetable(const DBAPIConn *db, DBAPIQueryCallback cb, DBAPIUserData data, const char *format, ...) {
   va_list ap;
   char buf[QUERYBUFLEN];
   size_t ret;
@@ -89,17 +94,24 @@ static void dbcreatetable(const DBAPIConn *db, DBAPIQueryCallback cb, DBAPIUserD
   va_end(ap);
 
   if(ret >= sizeof(buf))
-    Error("dbapi2", ERR_STOP, "Query truncated in dbcreatetable, format: '%s', database: %s", format, db->name);
+    Error("dbapi2", ERR_STOP, "Query truncated in dbunsafecreatetable, format: '%s', database: %s", format, db->name);
 
   db->__createtable(db, cb, data, buf);
 }
 
-static void dbsimplequery(const DBAPIConn *db, const char *format, ...) {
+static void dbunsafesimplequery(const DBAPIConn *db, const char *format, ...) {
   va_list ap;
+  char buf[QUERYBUFLEN];
+  size_t ret;
 
   va_start(ap, format);
-  dbquery(db, NULL, NULL, format, ap);
+  ret = vsnprintf(buf, sizeof(buf), format, ap);
   va_end(ap);
+
+  if(ret >= sizeof(buf))
+    Error("dbapi2", ERR_STOP, "Query truncated in dbunsafequery, format: '%s', database: %s", format, db->name);
+
+  db->__query(db, NULL, NULL, buf);
 }
 
 static void dbsafequery(const DBAPIConn *db, DBAPIQueryCallback cb, DBAPIUserData data, const char *format, const char *types, ...) {
@@ -126,10 +138,39 @@ static void dbsafecreatetable(const DBAPIConn *db, DBAPIQueryCallback cb, DBAPIU
 
 static void dbsafesimplequery(const DBAPIConn *db, const char *format, const char *types, ...) {
   va_list ap;
+  char buf[QUERYBUFLEN];
+
+  va_start(ap, types);
+  dbvsnprintf(db, buf, sizeof(buf), format, types, ap);
+  va_end(ap);
+
+  db->__query(db, NULL, NULL, buf);
+}
+
+static void dbloadtable(const DBAPIConn *db, DBAPIQueryCallback init, DBAPIQueryCallback data, DBAPIQueryCallback fini, DBAPIUserData tag, const char *tablename) {
+  db->__loadtable(db, init, data, fini, tag, db->tablename(db, tablename));
+}
+
+static void dbcall(const DBAPIConn *db, DBAPIQueryCallback cb, DBAPIUserData data, const char *function, const char *format, const char *types, ...) {
+  va_list ap;
+  char buf[QUERYBUFLEN];
+
+  va_start(ap, types);
+  dbvsnprintf(db, buf, sizeof(buf), format, types, ap);
+  va_end(ap);
+
+  db->__call(db, cb, data, function, buf);
+}
+
+static void dbsimplecall(const DBAPIConn *db, const char *function, const char *format, const char *types, ...) {
+  va_list ap;
+  char buf[QUERYBUFLEN];
 
   va_start(ap, types);
-  dbsafequery(db, NULL, NULL, format, types, ap);
+  dbvsnprintf(db, buf, sizeof(buf), format, types, ap);
   va_end(ap);
+
+  db->__call(db, NULL, NULL, function, buf);
 }
 
 DBAPIConn *dbapi2open(const char *provider, const char *database) {
@@ -169,20 +210,24 @@ DBAPIConn *dbapi2open(const char *provider, const char *database) {
     return NULL;
 
   db->close = dbclose;
-  db->query = dbquery;
-  db->createtable = dbcreatetable;
-  db->loadtable = p->loadtable;
+  db->query = dbsafequery;
+  db->createtable = dbsafecreatetable;
+  db->squery = dbsafesimplequery;
+  db->loadtable = dbloadtable;
   db->escapestring = p->escapestring;
-  db->squery = dbsimplequery;
   db->tablename = p->tablename;
-  db->safequery = dbsafequery;
-  db->safesquery = dbsafesimplequery;
-  db->safecreatetable = dbsafecreatetable;
+  db->unsafequery = dbunsafequery;
+  db->unsafesquery = dbunsafesimplequery;
+  db->unsafecreatetable = dbunsafecreatetable;
+  db->call = dbcall;
+  db->scall = dbsimplecall;
 
   db->__query = p->query;
   db->__close = p->close;
   db->__quotestring = p->quotestring;
   db->__createtable = p->createtable;
+  db->__loadtable = p->loadtable;
+  db->__call = p->call;
 
   strlcpy(db->name, database, DBNAME_LEN);
 
@@ -201,7 +246,7 @@ DBAPIConn *dbapi2open(const char *provider, const char *database) {
 static void dbvsnprintf(const DBAPIConn *db, char *buf, size_t size, const char *format, const char *types, va_list ap) {
   StringBuf b;
   const char *p;
-  static char convbuf[DBAPI_SNPRINTF_MAX_ARGS][DBAPI_SNPRINTF_MAX_ARG_LENGTH+10];
+  static char convbuf[VSNPF_MAXARGS][VSNPF_MAXARGLEN+10];
   int arg, argcount;
 
   if(size == 0)
@@ -216,15 +261,18 @@ static void dbvsnprintf(const DBAPIConn *db, char *buf, size_t size, const char
     unsigned int u;
     size_t l;
     int fallthrough;
+    time_t t;
+    unsigned long ul;
+    long _l;
 
-    for(i=0;i<DBAPI_SNPRINTF_MAX_ARGS;i++)
+    for(i=0;i<VSNPF_MAXARGS;i++)
       convbuf[i][0] = '\0';
 
     argcount=0;
     for(;*types;types++) {
       char *cb = convbuf[argcount];
 
-      if(argcount++ >= DBAPI_SNPRINTF_MAX_ARGS) {
+      if(argcount++ >= VSNPF_MAXARGS) {
         /* calls exit(0) */
         Error("dbapi2", ERR_STOP, "Maximum arguments reached in dbvsnprintf, format: '%s', database: %s", format, db->name);
       }
@@ -233,7 +281,8 @@ static void dbvsnprintf(const DBAPIConn *db, char *buf, size_t size, const char
       switch(*types) {
         case 's':
           s = va_arg(ap, char *);
-          l = strlen(s);
+          if(s)
+            l = strlen(s);
           fallthrough = 1;
 
         /* falling through */
@@ -243,24 +292,48 @@ static void dbvsnprintf(const DBAPIConn *db, char *buf, size_t size, const char
             l = va_arg(ap, size_t);
           }
 
-          /* now... this is a guess, but we should catch it most of the time */
-          if((l > (DBAPI_SNPRINTF_MAX_ARG_LENGTH / 2)) || !db->__quotestring(db, cb, sizeof(convbuf[0]), s, l)) {
-            Error("dbapi2", ERR_WARNING, "Long string truncated, format: '%s', database: %s", format, db->name);
-            l = DBAPI_SNPRINTF_MAX_ARG_LENGTH;
+          if(!s) {
+            strlcpy(cb, "NULL", sizeof(convbuf[0]));
+          } else if((l > (VSNPF_MAXARGLEN / 2)) || !db->__quotestring(db, cb, sizeof(convbuf[0]), s, l)) {
+            /* now... this is a guess, but we should catch it most of the time */
+            Error("dbapi2", ERR_STOP, "Long string truncated, format: '%s', database: %s", format, db->name);
+            l = VSNPF_MAXARGLEN;
           }
 
+          break;
+        case 'R':
+          s = va_arg(ap, char *);
+
+          strlcpy(cb, s, sizeof(convbuf[0]));
+          break;
+        case 'T':
+          s = va_arg(ap, char *);
+
+          strlcpy(cb, db->tablename(db, s), sizeof(convbuf[0]));
           break;
         case 'd':
           d = va_arg(ap, int);
-          snprintf(cb, DBAPI_SNPRINTF_MAX_ARG_LENGTH, "%d", d);
+          snprintf(cb, VSNPF_MAXARGLEN, "%d", d);
           break;
         case 'u':
           u = va_arg(ap, unsigned int);
-          snprintf(cb, DBAPI_SNPRINTF_MAX_ARG_LENGTH, "%u", u);
+          snprintf(cb, VSNPF_MAXARGLEN, "%u", u);
+          break;
+        case 't':
+          t = va_arg(ap, time_t);
+          snprintf(cb, VSNPF_MAXARGLEN, "%jd", (intmax_t)t);
+          break;
+        case 'D':
+          _l = va_arg(ap, long);
+          snprintf(cb, VSNPF_MAXARGLEN, "%ld", _l);
+          break;
+        case 'U':
+          ul = va_arg(ap, unsigned long);
+          snprintf(cb, VSNPF_MAXARGLEN, "%lu", ul);
           break;
         case 'g':
           g = va_arg(ap, double);
-          snprintf(cb, DBAPI_SNPRINTF_MAX_ARG_LENGTH, "%.1f", g);
+          snprintf(cb, VSNPF_MAXARGLEN, "%.1f", g);
           break;
         default:
           /* calls exit(0) */
@@ -272,12 +345,15 @@ static void dbvsnprintf(const DBAPIConn *db, char *buf, size_t size, const char
   sbinit(&b, buf, size);
 
   for(arg=0,p=format;*p;p++) {
-    if(*p != '?') {
+    if (*p == '\\' && *(p + 1) == '?')
+      continue;
+
+    if((p != format && *(p - 1) == '\\') || *p != '?') {
       if(!sbaddchar(&b, *p))
         break;
       continue;
     }
-    p++;
+
     if(arg >= argcount)
       Error("dbapi2", ERR_STOP, "Gone over number of arguments in dbvsnprintf, format: '%s', database: %s", format, db->name);