summaryrefslogtreecommitdiffstats
path: root/src/test_window.c
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--src/test_window.c349
1 files changed, 349 insertions, 0 deletions
diff --git a/src/test_window.c b/src/test_window.c
new file mode 100644
index 0000000..48ab022
--- /dev/null
+++ b/src/test_window.c
@@ -0,0 +1,349 @@
+/*
+** 2018 June 17
+**
+** The author disclaims copyright to this source code. In place of
+** a legal notice, here is a blessing:
+**
+** May you do good and not evil.
+** May you find forgiveness for yourself and forgive others.
+** May you share freely, never taking more than you give.
+**
+*************************************************************************
+*/
+
+#include "sqlite3.h"
+
+#ifdef SQLITE_TEST
+
+#include "sqliteInt.h"
+#include <tcl.h>
+
+extern int getDbPointer(Tcl_Interp *interp, const char *zA, sqlite3 **ppDb);
+extern const char *sqlite3ErrName(int);
+
+typedef struct TestWindow TestWindow;
+struct TestWindow {
+ Tcl_Obj *xStep;
+ Tcl_Obj *xFinal;
+ Tcl_Obj *xValue;
+ Tcl_Obj *xInverse;
+ Tcl_Interp *interp;
+};
+
+typedef struct TestWindowCtx TestWindowCtx;
+struct TestWindowCtx {
+ Tcl_Obj *pVal;
+};
+
+static void doTestWindowStep(
+ int bInverse,
+ sqlite3_context *ctx,
+ int nArg,
+ sqlite3_value **apArg
+){
+ int i;
+ TestWindow *p = (TestWindow*)sqlite3_user_data(ctx);
+ Tcl_Obj *pEval = Tcl_DuplicateObj(bInverse ? p->xInverse : p->xStep);
+ TestWindowCtx *pCtx = sqlite3_aggregate_context(ctx, sizeof(TestWindowCtx));
+
+ Tcl_IncrRefCount(pEval);
+ if( pCtx ){
+ const char *zResult;
+ int rc;
+ if( pCtx->pVal ){
+ Tcl_ListObjAppendElement(p->interp, pEval, Tcl_DuplicateObj(pCtx->pVal));
+ }else{
+ Tcl_ListObjAppendElement(p->interp, pEval, Tcl_NewStringObj("", -1));
+ }
+ for(i=0; i<nArg; i++){
+ Tcl_Obj *pArg;
+ pArg = Tcl_NewStringObj((const char*)sqlite3_value_text(apArg[i]), -1);
+ Tcl_ListObjAppendElement(p->interp, pEval, pArg);
+ }
+ rc = Tcl_EvalObjEx(p->interp, pEval, TCL_EVAL_GLOBAL);
+ if( rc!=TCL_OK ){
+ zResult = Tcl_GetStringResult(p->interp);
+ sqlite3_result_error(ctx, zResult, -1);
+ }else{
+ if( pCtx->pVal ) Tcl_DecrRefCount(pCtx->pVal);
+ pCtx->pVal = Tcl_DuplicateObj(Tcl_GetObjResult(p->interp));
+ Tcl_IncrRefCount(pCtx->pVal);
+ }
+ }
+ Tcl_DecrRefCount(pEval);
+}
+
+static void doTestWindowFinalize(int bValue, sqlite3_context *ctx){
+ TestWindow *p = (TestWindow*)sqlite3_user_data(ctx);
+ Tcl_Obj *pEval = Tcl_DuplicateObj(bValue ? p->xValue : p->xFinal);
+ TestWindowCtx *pCtx = sqlite3_aggregate_context(ctx, sizeof(TestWindowCtx));
+
+ Tcl_IncrRefCount(pEval);
+ if( pCtx ){
+ const char *zResult;
+ int rc;
+ if( pCtx->pVal ){
+ Tcl_ListObjAppendElement(p->interp, pEval, Tcl_DuplicateObj(pCtx->pVal));
+ }else{
+ Tcl_ListObjAppendElement(p->interp, pEval, Tcl_NewStringObj("", -1));
+ }
+
+ rc = Tcl_EvalObjEx(p->interp, pEval, TCL_EVAL_GLOBAL);
+ zResult = Tcl_GetStringResult(p->interp);
+ if( rc!=TCL_OK ){
+ sqlite3_result_error(ctx, zResult, -1);
+ }else{
+ sqlite3_result_text(ctx, zResult, -1, SQLITE_TRANSIENT);
+ }
+
+ if( bValue==0 ){
+ if( pCtx->pVal ) Tcl_DecrRefCount(pCtx->pVal);
+ pCtx->pVal = 0;
+ }
+ }
+ Tcl_DecrRefCount(pEval);
+}
+
+static void testWindowStep(
+ sqlite3_context *ctx,
+ int nArg,
+ sqlite3_value **apArg
+){
+ doTestWindowStep(0, ctx, nArg, apArg);
+}
+static void testWindowInverse(
+ sqlite3_context *ctx,
+ int nArg,
+ sqlite3_value **apArg
+){
+ doTestWindowStep(1, ctx, nArg, apArg);
+}
+
+static void testWindowFinal(sqlite3_context *ctx){
+ doTestWindowFinalize(0, ctx);
+}
+static void testWindowValue(sqlite3_context *ctx){
+ doTestWindowFinalize(1, ctx);
+}
+
+static void testWindowDestroy(void *pCtx){
+ ckfree(pCtx);
+}
+
+/*
+** Usage: sqlite3_create_window_function DB NAME XSTEP XFINAL XVALUE XINVERSE
+*/
+static int SQLITE_TCLAPI test_create_window(
+ void * clientData,
+ Tcl_Interp *interp,
+ int objc,
+ Tcl_Obj *CONST objv[]
+){
+ TestWindow *pNew;
+ sqlite3 *db;
+ const char *zName;
+ int rc;
+
+ if( objc!=7 ){
+ Tcl_WrongNumArgs(interp, 1, objv, "DB NAME XSTEP XFINAL XVALUE XINVERSE");
+ return TCL_ERROR;
+ }
+
+ if( getDbPointer(interp, Tcl_GetString(objv[1]), &db) ) return TCL_ERROR;
+ zName = Tcl_GetString(objv[2]);
+ pNew = (TestWindow*)ckalloc(sizeof(TestWindow));
+ memset(pNew, 0, sizeof(TestWindow));
+ pNew->xStep = Tcl_DuplicateObj(objv[3]);
+ pNew->xFinal = Tcl_DuplicateObj(objv[4]);
+ pNew->xValue = Tcl_DuplicateObj(objv[5]);
+ pNew->xInverse = Tcl_DuplicateObj(objv[6]);
+ pNew->interp = interp;
+
+ Tcl_IncrRefCount(pNew->xStep);
+ Tcl_IncrRefCount(pNew->xFinal);
+ Tcl_IncrRefCount(pNew->xValue);
+ Tcl_IncrRefCount(pNew->xInverse);
+
+ rc = sqlite3_create_window_function(db, zName, -1, SQLITE_UTF8, (void*)pNew,
+ testWindowStep, testWindowFinal, testWindowValue, testWindowInverse,
+ testWindowDestroy
+ );
+ if( rc!=SQLITE_OK ){
+ Tcl_SetObjResult(interp, Tcl_NewStringObj(sqlite3ErrName(rc), -1));
+ return TCL_ERROR;
+ }
+
+ return TCL_OK;
+}
+
+static int SQLITE_TCLAPI test_create_window_misuse(
+ void * clientData,
+ Tcl_Interp *interp,
+ int objc,
+ Tcl_Obj *CONST objv[]
+){
+ sqlite3 *db;
+ int rc;
+
+ if( objc!=2 ){
+ Tcl_WrongNumArgs(interp, 1, objv, "DB");
+ return TCL_ERROR;
+ }
+ if( getDbPointer(interp, Tcl_GetString(objv[1]), &db) ) return TCL_ERROR;
+
+ rc = sqlite3_create_window_function(db, "fff", -1, SQLITE_UTF8, 0,
+ 0, testWindowFinal, testWindowValue, testWindowInverse,
+ 0
+ );
+ if( rc!=SQLITE_MISUSE ) goto error;
+ rc = sqlite3_create_window_function(db, "fff", -1, SQLITE_UTF8, 0,
+ testWindowStep, 0, testWindowValue, testWindowInverse,
+ 0
+ );
+ if( rc!=SQLITE_MISUSE ) goto error;
+ rc = sqlite3_create_window_function(db, "fff", -1, SQLITE_UTF8, 0,
+ testWindowStep, testWindowFinal, 0, testWindowInverse,
+ 0
+ );
+ if( rc!=SQLITE_MISUSE ) goto error;
+ rc = sqlite3_create_window_function(db, "fff", -1, SQLITE_UTF8, 0,
+ testWindowStep, testWindowFinal, testWindowValue, 0,
+ 0
+ );
+ if( rc!=SQLITE_MISUSE ) goto error;
+
+ return TCL_OK;
+
+ error:
+ Tcl_SetObjResult(interp, Tcl_NewStringObj("misuse test error", -1));
+ return TCL_ERROR;
+}
+
+/*
+** xStep for sumint().
+*/
+static void sumintStep(
+ sqlite3_context *ctx,
+ int nArg,
+ sqlite3_value *apArg[]
+){
+ sqlite3_int64 *pInt;
+
+ assert( nArg==1 );
+ if( sqlite3_value_type(apArg[0])!=SQLITE_INTEGER ){
+ sqlite3_result_error(ctx, "invalid argument", -1);
+ return;
+ }
+ pInt = (sqlite3_int64*)sqlite3_aggregate_context(ctx, sizeof(sqlite3_int64));
+ if( pInt ){
+ *pInt += sqlite3_value_int64(apArg[0]);
+ }
+}
+
+/*
+** xInverse for sumint().
+*/
+static void sumintInverse(
+ sqlite3_context *ctx,
+ int nArg,
+ sqlite3_value *apArg[]
+){
+ sqlite3_int64 *pInt;
+ pInt = (sqlite3_int64*)sqlite3_aggregate_context(ctx, sizeof(sqlite3_int64));
+ *pInt -= sqlite3_value_int64(apArg[0]);
+}
+
+/*
+** xFinal for sumint().
+*/
+static void sumintFinal(sqlite3_context *ctx){
+ sqlite3_int64 res = 0;
+ sqlite3_int64 *pInt;
+ pInt = (sqlite3_int64*)sqlite3_aggregate_context(ctx, 0);
+ if( pInt ) res = *pInt;
+ sqlite3_result_int64(ctx, res);
+}
+
+/*
+** xValue for sumint().
+*/
+static void sumintValue(sqlite3_context *ctx){
+ sqlite3_int64 res = 0;
+ sqlite3_int64 *pInt;
+ pInt = (sqlite3_int64*)sqlite3_aggregate_context(ctx, 0);
+ if( pInt ) res = *pInt;
+ sqlite3_result_int64(ctx, res);
+}
+
+static int SQLITE_TCLAPI test_create_sumint(
+ void * clientData,
+ Tcl_Interp *interp,
+ int objc,
+ Tcl_Obj *CONST objv[]
+){
+ sqlite3 *db;
+ int rc;
+
+ if( objc!=2 ){
+ Tcl_WrongNumArgs(interp, 1, objv, "DB");
+ return TCL_ERROR;
+ }
+ if( getDbPointer(interp, Tcl_GetString(objv[1]), &db) ) return TCL_ERROR;
+
+ rc = sqlite3_create_window_function(db, "sumint", 1, SQLITE_UTF8, 0,
+ sumintStep, sumintFinal, sumintValue, sumintInverse,
+ 0
+ );
+
+ if( rc!=SQLITE_OK ){
+ Tcl_SetObjResult(interp, Tcl_NewStringObj(sqlite3ErrName(rc), -1));
+ return TCL_ERROR;
+ }
+ return TCL_OK;
+}
+
+static int SQLITE_TCLAPI test_override_sum(
+ void * clientData,
+ Tcl_Interp *interp,
+ int objc,
+ Tcl_Obj *CONST objv[]
+){
+ sqlite3 *db;
+ int rc;
+
+ if( objc!=2 ){
+ Tcl_WrongNumArgs(interp, 1, objv, "DB");
+ return TCL_ERROR;
+ }
+ if( getDbPointer(interp, Tcl_GetString(objv[1]), &db) ) return TCL_ERROR;
+
+ rc = sqlite3_create_function(db, "sum", -1, SQLITE_UTF8, 0,
+ 0, sumintStep, sumintFinal
+ );
+
+ if( rc!=SQLITE_OK ){
+ Tcl_SetObjResult(interp, Tcl_NewStringObj(sqlite3ErrName(rc), -1));
+ return TCL_ERROR;
+ }
+ return TCL_OK;
+}
+
+int Sqlitetest_window_Init(Tcl_Interp *interp){
+ static struct {
+ char *zName;
+ Tcl_ObjCmdProc *xProc;
+ int clientData;
+ } aObjCmd[] = {
+ { "sqlite3_create_window_function", test_create_window, 0 },
+ { "test_create_window_function_misuse", test_create_window_misuse, 0 },
+ { "test_create_sumint", test_create_sumint, 0 },
+ { "test_override_sum", test_override_sum, 0 },
+ };
+ int i;
+ for(i=0; i<sizeof(aObjCmd)/sizeof(aObjCmd[0]); i++){
+ ClientData c = (ClientData)SQLITE_INT_TO_PTR(aObjCmd[i].clientData);
+ Tcl_CreateObjCommand(interp, aObjCmd[i].zName, aObjCmd[i].xProc, c, 0);
+ }
+ return TCL_OK;
+}
+#endif