1c3f9bad2Sdanielk1977 /* 2633ed08dSdanielk1977 ** 3633ed08dSdanielk1977 ** The author disclaims copyright to this source code. In place of 4633ed08dSdanielk1977 ** a legal notice, here is a blessing: 5633ed08dSdanielk1977 ** 6633ed08dSdanielk1977 ** May you do good and not evil. 7633ed08dSdanielk1977 ** May you find forgiveness for yourself and forgive others. 8633ed08dSdanielk1977 ** May you share freely, never taking more than you give. 9633ed08dSdanielk1977 ** 10633ed08dSdanielk1977 ************************************************************************* 11633ed08dSdanielk1977 * 12c3f9bad2Sdanielk1977 */ 13c3f9bad2Sdanielk1977 #include "sqliteInt.h" 149adf9ac4Sdrh 15c3f9bad2Sdanielk1977 /* 164b59ab5eSdrh ** Delete a linked list of TriggerStep structures. 174b59ab5eSdrh */ 184adee20fSdanielk1977 void sqlite3DeleteTriggerStep(TriggerStep *pTriggerStep){ 194b59ab5eSdrh while( pTriggerStep ){ 204b59ab5eSdrh TriggerStep * pTmp = pTriggerStep; 214b59ab5eSdrh pTriggerStep = pTriggerStep->pNext; 224b59ab5eSdrh 238c74a8caSdrh if( pTmp->target.dyn ) sqliteFree((char*)pTmp->target.z); 244adee20fSdanielk1977 sqlite3ExprDelete(pTmp->pWhere); 254adee20fSdanielk1977 sqlite3ExprListDelete(pTmp->pExprList); 264adee20fSdanielk1977 sqlite3SelectDelete(pTmp->pSelect); 274adee20fSdanielk1977 sqlite3IdListDelete(pTmp->pIdList); 284b59ab5eSdrh 294b59ab5eSdrh sqliteFree(pTmp); 304b59ab5eSdrh } 314b59ab5eSdrh } 324b59ab5eSdrh 334b59ab5eSdrh /* 34f0f258b1Sdrh ** This is called by the parser when it sees a CREATE TRIGGER statement 35f0f258b1Sdrh ** up to the point of the BEGIN before the trigger actions. A Trigger 36f0f258b1Sdrh ** structure is generated based on the information available and stored 37f0f258b1Sdrh ** in pParse->pNewTrigger. After the trigger actions have been parsed, the 384adee20fSdanielk1977 ** sqlite3FinishTrigger() function is called to complete the trigger 39f0f258b1Sdrh ** construction process. 40c3f9bad2Sdanielk1977 */ 414adee20fSdanielk1977 void sqlite3BeginTrigger( 42c3f9bad2Sdanielk1977 Parse *pParse, /* The parse context of the CREATE TRIGGER statement */ 43633ed08dSdanielk1977 Token *pName, /* The name of the trigger */ 44d702fccbSdanielk1977 int tr_tm, /* One of TK_BEFORE, TK_AFTER, TK_INSTEAD */ 45c3f9bad2Sdanielk1977 int op, /* One of TK_INSERT, TK_UPDATE, TK_DELETE */ 46633ed08dSdanielk1977 IdList *pColumns, /* column list if this is an UPDATE OF trigger */ 47d24cc427Sdrh SrcList *pTableName,/* The name of the table/view the trigger applies to */ 48c3f9bad2Sdanielk1977 int foreach, /* One of TK_ROW or TK_STATEMENT */ 49c3f9bad2Sdanielk1977 Expr *pWhen, /* WHEN clause */ 50f0f258b1Sdrh int isTemp /* True if the TEMPORARY keyword is present */ 519adf9ac4Sdrh ){ 52c3f9bad2Sdanielk1977 Trigger *nt; 53c3f9bad2Sdanielk1977 Table *tab; 54e5f9c644Sdrh char *zName = 0; /* Name of the trigger */ 55d24cc427Sdrh sqlite *db = pParse->db; 56f0f258b1Sdrh int iDb; /* When database to store the trigger in */ 574312db55Sdrh DbFixer sFix; 58ed6c8671Sdrh 59c3f9bad2Sdanielk1977 /* Check that: 609adf9ac4Sdrh ** 1. the trigger name does not already exist. 61f0f258b1Sdrh ** 2. the table (or view) does exist in the same database as the trigger. 62d702fccbSdanielk1977 ** 3. that we are not trying to create a trigger on the sqlite_master table 63d702fccbSdanielk1977 ** 4. That we are not trying to create an INSTEAD OF trigger on a table. 64d702fccbSdanielk1977 ** 5. That we are not trying to create a BEFORE or AFTER trigger on a view. 65c3f9bad2Sdanielk1977 */ 66*6f8a503dSdanielk1977 if( sqlite3_malloc_failed ) goto trigger_cleanup; 67d24cc427Sdrh assert( pTableName->nSrc==1 ); 681d85d931Sdrh if( db->init.busy 694adee20fSdanielk1977 && sqlite3FixInit(&sFix, pParse, db->init.iDb, "trigger", pName) 704adee20fSdanielk1977 && sqlite3FixSrcList(&sFix, pTableName) 714312db55Sdrh ){ 724312db55Sdrh goto trigger_cleanup; 73f26e09c8Sdrh } 744adee20fSdanielk1977 tab = sqlite3SrcListLookup(pParse, pTableName); 75d24cc427Sdrh if( !tab ){ 76d24cc427Sdrh goto trigger_cleanup; 77d24cc427Sdrh } 78f0f258b1Sdrh iDb = isTemp ? 1 : tab->iDb; 791d85d931Sdrh if( iDb>=2 && !db->init.busy ){ 804adee20fSdanielk1977 sqlite3ErrorMsg(pParse, "triggers may not be added to auxiliary " 81da93d238Sdrh "database %s", db->aDb[tab->iDb].zName); 82d24cc427Sdrh goto trigger_cleanup; 83d24cc427Sdrh } 84d24cc427Sdrh 85e5f9c644Sdrh zName = sqliteStrNDup(pName->z, pName->n); 864adee20fSdanielk1977 sqlite3Dequote(zName); 874adee20fSdanielk1977 if( sqlite3HashFind(&(db->aDb[iDb].trigHash), zName,pName->n+1) ){ 884adee20fSdanielk1977 sqlite3ErrorMsg(pParse, "trigger %T already exists", pName); 89c3f9bad2Sdanielk1977 goto trigger_cleanup; 90c3f9bad2Sdanielk1977 } 914adee20fSdanielk1977 if( sqlite3StrNICmp(tab->zName, "sqlite_", 7)==0 ){ 924adee20fSdanielk1977 sqlite3ErrorMsg(pParse, "cannot create trigger on system table"); 93e0bc4048Sdrh pParse->nErr++; 94e0bc4048Sdrh goto trigger_cleanup; 95e0bc4048Sdrh } 96d702fccbSdanielk1977 if( tab->pSelect && tr_tm != TK_INSTEAD ){ 974adee20fSdanielk1977 sqlite3ErrorMsg(pParse, "cannot create %s trigger on view: %S", 98da93d238Sdrh (tr_tm == TK_BEFORE)?"BEFORE":"AFTER", pTableName, 0); 99d702fccbSdanielk1977 goto trigger_cleanup; 100d702fccbSdanielk1977 } 101d702fccbSdanielk1977 if( !tab->pSelect && tr_tm == TK_INSTEAD ){ 1024adee20fSdanielk1977 sqlite3ErrorMsg(pParse, "cannot create INSTEAD OF" 103da93d238Sdrh " trigger on table: %S", pTableName, 0); 104d702fccbSdanielk1977 goto trigger_cleanup; 105d702fccbSdanielk1977 } 106e5f9c644Sdrh #ifndef SQLITE_OMIT_AUTHORIZATION 107e5f9c644Sdrh { 108e5f9c644Sdrh int code = SQLITE_CREATE_TRIGGER; 109e22a334bSdrh const char *zDb = db->aDb[tab->iDb].zName; 110e22a334bSdrh const char *zDbTrig = isTemp ? db->aDb[1].zName : zDb; 111f0f258b1Sdrh if( tab->iDb==1 || isTemp ) code = SQLITE_CREATE_TEMP_TRIGGER; 1124adee20fSdanielk1977 if( sqlite3AuthCheck(pParse, code, zName, tab->zName, zDbTrig) ){ 113ed6c8671Sdrh goto trigger_cleanup; 114ed6c8671Sdrh } 1154adee20fSdanielk1977 if( sqlite3AuthCheck(pParse, SQLITE_INSERT, SCHEMA_TABLE(tab->iDb), 0, zDb)){ 11677ad4e41Sdrh goto trigger_cleanup; 11777ad4e41Sdrh } 118d702fccbSdanielk1977 } 119e5f9c644Sdrh #endif 120d702fccbSdanielk1977 1215cf590c1Sdrh /* INSTEAD OF triggers can only appear on views and BEGIN triggers 1225cf590c1Sdrh ** cannot appear on views. So we might as well translate every 1235cf590c1Sdrh ** INSTEAD OF trigger into a BEFORE trigger. It simplifies code 1245cf590c1Sdrh ** elsewhere. 1255cf590c1Sdrh */ 126d702fccbSdanielk1977 if (tr_tm == TK_INSTEAD){ 127d702fccbSdanielk1977 tr_tm = TK_BEFORE; 128c3f9bad2Sdanielk1977 } 129c3f9bad2Sdanielk1977 130c3f9bad2Sdanielk1977 /* Build the Trigger object */ 131c3f9bad2Sdanielk1977 nt = (Trigger*)sqliteMalloc(sizeof(Trigger)); 132e4697f5eSdrh if( nt==0 ) goto trigger_cleanup; 133e5f9c644Sdrh nt->name = zName; 134e5f9c644Sdrh zName = 0; 135d24cc427Sdrh nt->table = sqliteStrDup(pTableName->a[0].zName); 136*6f8a503dSdanielk1977 if( sqlite3_malloc_failed ) goto trigger_cleanup; 137f0f258b1Sdrh nt->iDb = iDb; 13879a519c0Sdrh nt->iTabDb = tab->iDb; 139c3f9bad2Sdanielk1977 nt->op = op; 140c3f9bad2Sdanielk1977 nt->tr_tm = tr_tm; 1414adee20fSdanielk1977 nt->pWhen = sqlite3ExprDup(pWhen); 1424adee20fSdanielk1977 nt->pColumns = sqlite3IdListDup(pColumns); 143c3f9bad2Sdanielk1977 nt->foreach = foreach; 1444adee20fSdanielk1977 sqlite3TokenCopy(&nt->nameToken,pName); 145f0f258b1Sdrh assert( pParse->pNewTrigger==0 ); 146f0f258b1Sdrh pParse->pNewTrigger = nt; 147f0f258b1Sdrh 148f0f258b1Sdrh trigger_cleanup: 149f0f258b1Sdrh sqliteFree(zName); 1504adee20fSdanielk1977 sqlite3SrcListDelete(pTableName); 1514adee20fSdanielk1977 sqlite3IdListDelete(pColumns); 1524adee20fSdanielk1977 sqlite3ExprDelete(pWhen); 153f0f258b1Sdrh } 154f0f258b1Sdrh 155f0f258b1Sdrh /* 156f0f258b1Sdrh ** This routine is called after all of the trigger actions have been parsed 157f0f258b1Sdrh ** in order to complete the process of building the trigger. 158f0f258b1Sdrh */ 1594adee20fSdanielk1977 void sqlite3FinishTrigger( 160f0f258b1Sdrh Parse *pParse, /* Parser context */ 161f0f258b1Sdrh TriggerStep *pStepList, /* The triggered program */ 162f0f258b1Sdrh Token *pAll /* Token that describes the complete CREATE TRIGGER */ 163f0f258b1Sdrh ){ 164f26e09c8Sdrh Trigger *nt = 0; /* The trigger whose construction is finishing up */ 165f0f258b1Sdrh sqlite *db = pParse->db; /* The database */ 166f26e09c8Sdrh DbFixer sFix; 167f0f258b1Sdrh 168f0f258b1Sdrh if( pParse->nErr || pParse->pNewTrigger==0 ) goto triggerfinish_cleanup; 169f0f258b1Sdrh nt = pParse->pNewTrigger; 170f0f258b1Sdrh pParse->pNewTrigger = 0; 171633ed08dSdanielk1977 nt->step_list = pStepList; 172a69d9168Sdrh while( pStepList ){ 173a69d9168Sdrh pStepList->pTrig = nt; 174a69d9168Sdrh pStepList = pStepList->pNext; 175a69d9168Sdrh } 1764adee20fSdanielk1977 if( sqlite3FixInit(&sFix, pParse, nt->iDb, "trigger", &nt->nameToken) 1774adee20fSdanielk1977 && sqlite3FixTriggerStep(&sFix, nt->step_list) ){ 178f26e09c8Sdrh goto triggerfinish_cleanup; 179f26e09c8Sdrh } 180c3f9bad2Sdanielk1977 181c3f9bad2Sdanielk1977 /* if we are not initializing, and this trigger is not on a TEMP table, 1829adf9ac4Sdrh ** build the sqlite_master entry 1839adf9ac4Sdrh */ 1841d85d931Sdrh if( !db->init.busy ){ 185905793e2Sdrh static VdbeOpList insertTrig[] = { 186c977f7f5Sdrh { OP_NewRecno, 0, 0, 0 }, 187c977f7f5Sdrh { OP_String, 0, 0, "trigger" }, 188e0bc4048Sdrh { OP_String, 0, 0, 0 }, /* 2: trigger name */ 189e0bc4048Sdrh { OP_String, 0, 0, 0 }, /* 3: table name */ 190c977f7f5Sdrh { OP_Integer, 0, 0, 0 }, 191e0bc4048Sdrh { OP_String, 0, 0, 0 }, /* 5: SQL */ 192c977f7f5Sdrh { OP_MakeRecord, 5, 0, 0 }, 193c977f7f5Sdrh { OP_PutIntKey, 0, 0, 0 }, 194c977f7f5Sdrh }; 195c977f7f5Sdrh int addr; 196c977f7f5Sdrh Vdbe *v; 197c3f9bad2Sdanielk1977 198c3f9bad2Sdanielk1977 /* Make an entry in the sqlite_master table */ 1994adee20fSdanielk1977 v = sqlite3GetVdbe(pParse); 200f0f258b1Sdrh if( v==0 ) goto triggerfinish_cleanup; 2014adee20fSdanielk1977 sqlite3BeginWriteOperation(pParse, 0, 0); 2024adee20fSdanielk1977 sqlite3OpenMasterTable(v, nt->iDb); 2034adee20fSdanielk1977 addr = sqlite3VdbeAddOpList(v, ArraySize(insertTrig), insertTrig); 2044adee20fSdanielk1977 sqlite3VdbeChangeP3(v, addr+2, nt->name, 0); 2054adee20fSdanielk1977 sqlite3VdbeChangeP3(v, addr+3, nt->table, 0); 2064adee20fSdanielk1977 sqlite3VdbeChangeP3(v, addr+5, pAll->z, pAll->n); 207f0f258b1Sdrh if( nt->iDb==0 ){ 2084adee20fSdanielk1977 sqlite3ChangeCookie(db, v); 209e0bc4048Sdrh } 2104adee20fSdanielk1977 sqlite3VdbeAddOp(v, OP_Close, 0, 0); 2114adee20fSdanielk1977 sqlite3EndWriteOperation(pParse); 212c3f9bad2Sdanielk1977 } 213c3f9bad2Sdanielk1977 214c3f9bad2Sdanielk1977 if( !pParse->explain ){ 215f0f258b1Sdrh Table *pTab; 2164adee20fSdanielk1977 sqlite3HashInsert(&db->aDb[nt->iDb].trigHash, 217f0f258b1Sdrh nt->name, strlen(nt->name)+1, nt); 2184adee20fSdanielk1977 pTab = sqlite3LocateTable(pParse, nt->table, db->aDb[nt->iTabDb].zName); 219f0f258b1Sdrh assert( pTab!=0 ); 220f0f258b1Sdrh nt->pNext = pTab->pTrigger; 221f0f258b1Sdrh pTab->pTrigger = nt; 222f26e09c8Sdrh nt = 0; 223c3f9bad2Sdanielk1977 } 224c3f9bad2Sdanielk1977 225f0f258b1Sdrh triggerfinish_cleanup: 2264adee20fSdanielk1977 sqlite3DeleteTrigger(nt); 2274adee20fSdanielk1977 sqlite3DeleteTrigger(pParse->pNewTrigger); 228f0f258b1Sdrh pParse->pNewTrigger = 0; 2294adee20fSdanielk1977 sqlite3DeleteTriggerStep(pStepList); 230c3f9bad2Sdanielk1977 } 2314b59ab5eSdrh 2324b59ab5eSdrh /* 2334b59ab5eSdrh ** Make a copy of all components of the given trigger step. This has 2344b59ab5eSdrh ** the effect of copying all Expr.token.z values into memory obtained 2354b59ab5eSdrh ** from sqliteMalloc(). As initially created, the Expr.token.z values 2364b59ab5eSdrh ** all point to the input string that was fed to the parser. But that 237*6f8a503dSdanielk1977 ** string is ephemeral - it will go away as soon as the sqlite3_exec() 2384b59ab5eSdrh ** call that started the parser exits. This routine makes a persistent 2394b59ab5eSdrh ** copy of all the Expr.token.z strings so that the TriggerStep structure 240*6f8a503dSdanielk1977 ** will be valid even after the sqlite3_exec() call returns. 2414b59ab5eSdrh */ 2424b59ab5eSdrh static void sqlitePersistTriggerStep(TriggerStep *p){ 2434b59ab5eSdrh if( p->target.z ){ 2444b59ab5eSdrh p->target.z = sqliteStrNDup(p->target.z, p->target.n); 2454b59ab5eSdrh p->target.dyn = 1; 2464b59ab5eSdrh } 2474b59ab5eSdrh if( p->pSelect ){ 2484adee20fSdanielk1977 Select *pNew = sqlite3SelectDup(p->pSelect); 2494adee20fSdanielk1977 sqlite3SelectDelete(p->pSelect); 2504b59ab5eSdrh p->pSelect = pNew; 2514b59ab5eSdrh } 2524b59ab5eSdrh if( p->pWhere ){ 2534adee20fSdanielk1977 Expr *pNew = sqlite3ExprDup(p->pWhere); 2544adee20fSdanielk1977 sqlite3ExprDelete(p->pWhere); 2554b59ab5eSdrh p->pWhere = pNew; 2564b59ab5eSdrh } 2574b59ab5eSdrh if( p->pExprList ){ 2584adee20fSdanielk1977 ExprList *pNew = sqlite3ExprListDup(p->pExprList); 2594adee20fSdanielk1977 sqlite3ExprListDelete(p->pExprList); 2604b59ab5eSdrh p->pExprList = pNew; 2614b59ab5eSdrh } 2624b59ab5eSdrh if( p->pIdList ){ 2634adee20fSdanielk1977 IdList *pNew = sqlite3IdListDup(p->pIdList); 2644adee20fSdanielk1977 sqlite3IdListDelete(p->pIdList); 2654b59ab5eSdrh p->pIdList = pNew; 266c3f9bad2Sdanielk1977 } 267c3f9bad2Sdanielk1977 } 268c3f9bad2Sdanielk1977 269c977f7f5Sdrh /* 270c977f7f5Sdrh ** Turn a SELECT statement (that the pSelect parameter points to) into 271c977f7f5Sdrh ** a trigger step. Return a pointer to a TriggerStep structure. 272c977f7f5Sdrh ** 273c977f7f5Sdrh ** The parser calls this routine when it finds a SELECT statement in 274c977f7f5Sdrh ** body of a TRIGGER. 275c977f7f5Sdrh */ 2764adee20fSdanielk1977 TriggerStep *sqlite3TriggerSelectStep(Select *pSelect){ 277633ed08dSdanielk1977 TriggerStep *pTriggerStep = sqliteMalloc(sizeof(TriggerStep)); 278e4697f5eSdrh if( pTriggerStep==0 ) return 0; 279c3f9bad2Sdanielk1977 280633ed08dSdanielk1977 pTriggerStep->op = TK_SELECT; 281633ed08dSdanielk1977 pTriggerStep->pSelect = pSelect; 282633ed08dSdanielk1977 pTriggerStep->orconf = OE_Default; 2834b59ab5eSdrh sqlitePersistTriggerStep(pTriggerStep); 284c3f9bad2Sdanielk1977 285633ed08dSdanielk1977 return pTriggerStep; 286c3f9bad2Sdanielk1977 } 287c3f9bad2Sdanielk1977 288c977f7f5Sdrh /* 289c977f7f5Sdrh ** Build a trigger step out of an INSERT statement. Return a pointer 290c977f7f5Sdrh ** to the new trigger step. 291c977f7f5Sdrh ** 292c977f7f5Sdrh ** The parser calls this routine when it sees an INSERT inside the 293c977f7f5Sdrh ** body of a trigger. 294c977f7f5Sdrh */ 2954adee20fSdanielk1977 TriggerStep *sqlite3TriggerInsertStep( 296c977f7f5Sdrh Token *pTableName, /* Name of the table into which we insert */ 297c977f7f5Sdrh IdList *pColumn, /* List of columns in pTableName to insert into */ 298c977f7f5Sdrh ExprList *pEList, /* The VALUE clause: a list of values to be inserted */ 299c977f7f5Sdrh Select *pSelect, /* A SELECT statement that supplies values */ 300c977f7f5Sdrh int orconf /* The conflict algorithm (OE_Abort, OE_Replace, etc.) */ 301633ed08dSdanielk1977 ){ 302633ed08dSdanielk1977 TriggerStep *pTriggerStep = sqliteMalloc(sizeof(TriggerStep)); 303e4697f5eSdrh if( pTriggerStep==0 ) return 0; 304c3f9bad2Sdanielk1977 305633ed08dSdanielk1977 assert(pEList == 0 || pSelect == 0); 306633ed08dSdanielk1977 assert(pEList != 0 || pSelect != 0); 307c3f9bad2Sdanielk1977 308633ed08dSdanielk1977 pTriggerStep->op = TK_INSERT; 309633ed08dSdanielk1977 pTriggerStep->pSelect = pSelect; 310633ed08dSdanielk1977 pTriggerStep->target = *pTableName; 311633ed08dSdanielk1977 pTriggerStep->pIdList = pColumn; 312633ed08dSdanielk1977 pTriggerStep->pExprList = pEList; 313633ed08dSdanielk1977 pTriggerStep->orconf = orconf; 3144b59ab5eSdrh sqlitePersistTriggerStep(pTriggerStep); 315c3f9bad2Sdanielk1977 316633ed08dSdanielk1977 return pTriggerStep; 317c3f9bad2Sdanielk1977 } 318c3f9bad2Sdanielk1977 319c977f7f5Sdrh /* 320c977f7f5Sdrh ** Construct a trigger step that implements an UPDATE statement and return 321c977f7f5Sdrh ** a pointer to that trigger step. The parser calls this routine when it 322c977f7f5Sdrh ** sees an UPDATE statement inside the body of a CREATE TRIGGER. 323c977f7f5Sdrh */ 3244adee20fSdanielk1977 TriggerStep *sqlite3TriggerUpdateStep( 325c977f7f5Sdrh Token *pTableName, /* Name of the table to be updated */ 326c977f7f5Sdrh ExprList *pEList, /* The SET clause: list of column and new values */ 327c977f7f5Sdrh Expr *pWhere, /* The WHERE clause */ 328c977f7f5Sdrh int orconf /* The conflict algorithm. (OE_Abort, OE_Ignore, etc) */ 329c977f7f5Sdrh ){ 330633ed08dSdanielk1977 TriggerStep *pTriggerStep = sqliteMalloc(sizeof(TriggerStep)); 331e4697f5eSdrh if( pTriggerStep==0 ) return 0; 332c3f9bad2Sdanielk1977 333633ed08dSdanielk1977 pTriggerStep->op = TK_UPDATE; 334633ed08dSdanielk1977 pTriggerStep->target = *pTableName; 335633ed08dSdanielk1977 pTriggerStep->pExprList = pEList; 336633ed08dSdanielk1977 pTriggerStep->pWhere = pWhere; 337633ed08dSdanielk1977 pTriggerStep->orconf = orconf; 3384b59ab5eSdrh sqlitePersistTriggerStep(pTriggerStep); 339c3f9bad2Sdanielk1977 340633ed08dSdanielk1977 return pTriggerStep; 341c3f9bad2Sdanielk1977 } 342c3f9bad2Sdanielk1977 343c977f7f5Sdrh /* 344c977f7f5Sdrh ** Construct a trigger step that implements a DELETE statement and return 345c977f7f5Sdrh ** a pointer to that trigger step. The parser calls this routine when it 346c977f7f5Sdrh ** sees a DELETE statement inside the body of a CREATE TRIGGER. 347c977f7f5Sdrh */ 3484adee20fSdanielk1977 TriggerStep *sqlite3TriggerDeleteStep(Token *pTableName, Expr *pWhere){ 349633ed08dSdanielk1977 TriggerStep *pTriggerStep = sqliteMalloc(sizeof(TriggerStep)); 350e4697f5eSdrh if( pTriggerStep==0 ) return 0; 351c3f9bad2Sdanielk1977 352633ed08dSdanielk1977 pTriggerStep->op = TK_DELETE; 353633ed08dSdanielk1977 pTriggerStep->target = *pTableName; 354633ed08dSdanielk1977 pTriggerStep->pWhere = pWhere; 355633ed08dSdanielk1977 pTriggerStep->orconf = OE_Default; 3564b59ab5eSdrh sqlitePersistTriggerStep(pTriggerStep); 357c3f9bad2Sdanielk1977 358633ed08dSdanielk1977 return pTriggerStep; 359c3f9bad2Sdanielk1977 } 360c3f9bad2Sdanielk1977 361c3f9bad2Sdanielk1977 /* 362633ed08dSdanielk1977 ** Recursively delete a Trigger structure 363c3f9bad2Sdanielk1977 */ 3644adee20fSdanielk1977 void sqlite3DeleteTrigger(Trigger *pTrigger){ 365f0f258b1Sdrh if( pTrigger==0 ) return; 3664adee20fSdanielk1977 sqlite3DeleteTriggerStep(pTrigger->step_list); 367633ed08dSdanielk1977 sqliteFree(pTrigger->name); 368633ed08dSdanielk1977 sqliteFree(pTrigger->table); 3694adee20fSdanielk1977 sqlite3ExprDelete(pTrigger->pWhen); 3704adee20fSdanielk1977 sqlite3IdListDelete(pTrigger->pColumns); 3714312db55Sdrh if( pTrigger->nameToken.dyn ) sqliteFree((char*)pTrigger->nameToken.z); 372633ed08dSdanielk1977 sqliteFree(pTrigger); 373633ed08dSdanielk1977 } 374633ed08dSdanielk1977 375633ed08dSdanielk1977 /* 376633ed08dSdanielk1977 * This function is called to drop a trigger from the database schema. 377633ed08dSdanielk1977 * 37879a519c0Sdrh * This may be called directly from the parser and therefore identifies 3794adee20fSdanielk1977 * the trigger by name. The sqlite3DropTriggerPtr() routine does the 38079a519c0Sdrh * same job as this routine except it take a spointer to the trigger 38179a519c0Sdrh * instead of the trigger name. 382633ed08dSdanielk1977 * 383633ed08dSdanielk1977 * Note that this function does not delete the trigger entirely. Instead it 384633ed08dSdanielk1977 * removes it from the internal schema and places it in the trigDrop hash 385633ed08dSdanielk1977 * table. This is so that the trigger can be restored into the database schema 386633ed08dSdanielk1977 * if the transaction is rolled back. 387633ed08dSdanielk1977 */ 3884adee20fSdanielk1977 void sqlite3DropTrigger(Parse *pParse, SrcList *pName){ 389633ed08dSdanielk1977 Trigger *pTrigger; 390d24cc427Sdrh int i; 391d24cc427Sdrh const char *zDb; 392d24cc427Sdrh const char *zName; 393d24cc427Sdrh int nName; 394d24cc427Sdrh sqlite *db = pParse->db; 395633ed08dSdanielk1977 396*6f8a503dSdanielk1977 if( sqlite3_malloc_failed ) goto drop_trigger_cleanup; 397d24cc427Sdrh assert( pName->nSrc==1 ); 398d24cc427Sdrh zDb = pName->a[0].zDatabase; 399d24cc427Sdrh zName = pName->a[0].zName; 400d24cc427Sdrh nName = strlen(zName); 401d24cc427Sdrh for(i=0; i<db->nDb; i++){ 402812d7a21Sdrh int j = (i<2) ? i^1 : i; /* Search TEMP before MAIN */ 4034adee20fSdanielk1977 if( zDb && sqlite3StrICmp(db->aDb[j].zName, zDb) ) continue; 4044adee20fSdanielk1977 pTrigger = sqlite3HashFind(&(db->aDb[j].trigHash), zName, nName+1); 405d24cc427Sdrh if( pTrigger ) break; 406c3f9bad2Sdanielk1977 } 407d24cc427Sdrh if( !pTrigger ){ 4084adee20fSdanielk1977 sqlite3ErrorMsg(pParse, "no such trigger: %S", pName, 0); 409d24cc427Sdrh goto drop_trigger_cleanup; 410d24cc427Sdrh } 4114adee20fSdanielk1977 sqlite3DropTriggerPtr(pParse, pTrigger, 0); 41279a519c0Sdrh 41379a519c0Sdrh drop_trigger_cleanup: 4144adee20fSdanielk1977 sqlite3SrcListDelete(pName); 41579a519c0Sdrh } 41679a519c0Sdrh 41779a519c0Sdrh /* 41879a519c0Sdrh ** Drop a trigger given a pointer to that trigger. If nested is false, 41979a519c0Sdrh ** then also generate code to remove the trigger from the SQLITE_MASTER 42079a519c0Sdrh ** table. 42179a519c0Sdrh */ 4224adee20fSdanielk1977 void sqlite3DropTriggerPtr(Parse *pParse, Trigger *pTrigger, int nested){ 42379a519c0Sdrh Table *pTable; 42479a519c0Sdrh Vdbe *v; 42579a519c0Sdrh sqlite *db = pParse->db; 42679a519c0Sdrh 427f0f258b1Sdrh assert( pTrigger->iDb<db->nDb ); 428d24cc427Sdrh if( pTrigger->iDb>=2 ){ 4294adee20fSdanielk1977 sqlite3ErrorMsg(pParse, "triggers may not be removed from " 430da93d238Sdrh "auxiliary database %s", db->aDb[pTrigger->iDb].zName); 43179a519c0Sdrh return; 432d24cc427Sdrh } 4334adee20fSdanielk1977 pTable = sqlite3FindTable(db, pTrigger->table,db->aDb[pTrigger->iTabDb].zName); 434ed6c8671Sdrh assert(pTable); 435f0f258b1Sdrh assert( pTable->iDb==pTrigger->iDb || pTrigger->iDb==1 ); 436e5f9c644Sdrh #ifndef SQLITE_OMIT_AUTHORIZATION 437e5f9c644Sdrh { 438e5f9c644Sdrh int code = SQLITE_DROP_TRIGGER; 439e22a334bSdrh const char *zDb = db->aDb[pTrigger->iDb].zName; 440e22a334bSdrh const char *zTab = SCHEMA_TABLE(pTrigger->iDb); 441f0f258b1Sdrh if( pTrigger->iDb ) code = SQLITE_DROP_TEMP_TRIGGER; 4424adee20fSdanielk1977 if( sqlite3AuthCheck(pParse, code, pTrigger->name, pTable->zName, zDb) || 4434adee20fSdanielk1977 sqlite3AuthCheck(pParse, SQLITE_DELETE, zTab, 0, zDb) ){ 44479a519c0Sdrh return; 445ed6c8671Sdrh } 446e5f9c644Sdrh } 447e5f9c644Sdrh #endif 448c3f9bad2Sdanielk1977 449f0f258b1Sdrh /* Generate code to destroy the database record of the trigger. 450f0f258b1Sdrh */ 4514adee20fSdanielk1977 if( pTable!=0 && !nested && (v = sqlite3GetVdbe(pParse))!=0 ){ 452f0f258b1Sdrh int base; 453905793e2Sdrh static VdbeOpList dropTrigger[] = { 4549b1b01bbSdrh { OP_Rewind, 0, ADDR(9), 0}, 455f0f258b1Sdrh { OP_String, 0, 0, 0}, /* 1 */ 456f0f258b1Sdrh { OP_Column, 0, 1, 0}, 4579b1b01bbSdrh { OP_Ne, 0, ADDR(8), 0}, 4589b1b01bbSdrh { OP_String, 0, 0, "trigger"}, 4599b1b01bbSdrh { OP_Column, 0, 0, 0}, 4609b1b01bbSdrh { OP_Ne, 0, ADDR(8), 0}, 461f0f258b1Sdrh { OP_Delete, 0, 0, 0}, 4629b1b01bbSdrh { OP_Next, 0, ADDR(1), 0}, /* 8 */ 463f0f258b1Sdrh }; 464f0f258b1Sdrh 4654adee20fSdanielk1977 sqlite3BeginWriteOperation(pParse, 0, 0); 4664adee20fSdanielk1977 sqlite3OpenMasterTable(v, pTrigger->iDb); 4674adee20fSdanielk1977 base = sqlite3VdbeAddOpList(v, ArraySize(dropTrigger), dropTrigger); 4684adee20fSdanielk1977 sqlite3VdbeChangeP3(v, base+1, pTrigger->name, 0); 469f0f258b1Sdrh if( pTrigger->iDb==0 ){ 4704adee20fSdanielk1977 sqlite3ChangeCookie(db, v); 471f0f258b1Sdrh } 4724adee20fSdanielk1977 sqlite3VdbeAddOp(v, OP_Close, 0, 0); 4734adee20fSdanielk1977 sqlite3EndWriteOperation(pParse); 474f0f258b1Sdrh } 475f0f258b1Sdrh 476c3f9bad2Sdanielk1977 /* 477e0bc4048Sdrh * If this is not an "explain", then delete the trigger structure. 478c3f9bad2Sdanielk1977 */ 479c3f9bad2Sdanielk1977 if( !pParse->explain ){ 48079a519c0Sdrh const char *zName = pTrigger->name; 48179a519c0Sdrh int nName = strlen(zName); 482633ed08dSdanielk1977 if( pTable->pTrigger == pTrigger ){ 483633ed08dSdanielk1977 pTable->pTrigger = pTrigger->pNext; 484633ed08dSdanielk1977 }else{ 485633ed08dSdanielk1977 Trigger *cc = pTable->pTrigger; 486c3f9bad2Sdanielk1977 while( cc ){ 487633ed08dSdanielk1977 if( cc->pNext == pTrigger ){ 488c3f9bad2Sdanielk1977 cc->pNext = cc->pNext->pNext; 489c3f9bad2Sdanielk1977 break; 490c3f9bad2Sdanielk1977 } 491c3f9bad2Sdanielk1977 cc = cc->pNext; 492c3f9bad2Sdanielk1977 } 493c3f9bad2Sdanielk1977 assert(cc); 494c3f9bad2Sdanielk1977 } 4954adee20fSdanielk1977 sqlite3HashInsert(&(db->aDb[pTrigger->iDb].trigHash), zName, nName+1, 0); 4964adee20fSdanielk1977 sqlite3DeleteTrigger(pTrigger); 497c3f9bad2Sdanielk1977 } 498c3f9bad2Sdanielk1977 } 499c3f9bad2Sdanielk1977 500c977f7f5Sdrh /* 501c977f7f5Sdrh ** pEList is the SET clause of an UPDATE statement. Each entry 502c977f7f5Sdrh ** in pEList is of the format <id>=<expr>. If any of the entries 503c977f7f5Sdrh ** in pEList have an <id> which matches an identifier in pIdList, 504c977f7f5Sdrh ** then return TRUE. If pIdList==NULL, then it is considered a 505c977f7f5Sdrh ** wildcard that matches anything. Likewise if pEList==NULL then 506c977f7f5Sdrh ** it matches anything so always return true. Return false only 507c977f7f5Sdrh ** if there is no match. 508c977f7f5Sdrh */ 509c977f7f5Sdrh static int checkColumnOverLap(IdList *pIdList, ExprList *pEList){ 510ad2d8307Sdrh int e; 511ad2d8307Sdrh if( !pIdList || !pEList ) return 1; 512f29ce559Sdanielk1977 for(e=0; e<pEList->nExpr; e++){ 5134adee20fSdanielk1977 if( sqlite3IdListIndex(pIdList, pEList->a[e].zName)>=0 ) return 1; 514f29ce559Sdanielk1977 } 515c3f9bad2Sdanielk1977 return 0; 516c3f9bad2Sdanielk1977 } 517c3f9bad2Sdanielk1977 518c3f9bad2Sdanielk1977 /* A global variable that is TRUE if we should always set up temp tables for 519c3f9bad2Sdanielk1977 * for triggers, even if there are no triggers to code. This is used to test 520c3f9bad2Sdanielk1977 * how much overhead the triggers algorithm is causing. 521c3f9bad2Sdanielk1977 * 522c3f9bad2Sdanielk1977 * This flag can be set or cleared using the "trigger_overhead_test" pragma. 523c3f9bad2Sdanielk1977 * The pragma is not documented since it is not really part of the interface 524c3f9bad2Sdanielk1977 * to SQLite, just the test procedure. 525c3f9bad2Sdanielk1977 */ 526c3f9bad2Sdanielk1977 int always_code_trigger_setup = 0; 527c3f9bad2Sdanielk1977 528c3f9bad2Sdanielk1977 /* 529c3f9bad2Sdanielk1977 * Returns true if a trigger matching op, tr_tm and foreach that is NOT already 530c3f9bad2Sdanielk1977 * on the Parse objects trigger-stack (to prevent recursive trigger firing) is 531c3f9bad2Sdanielk1977 * found in the list specified as pTrigger. 532c3f9bad2Sdanielk1977 */ 5334adee20fSdanielk1977 int sqlite3TriggersExist( 534c977f7f5Sdrh Parse *pParse, /* Used to check for recursive triggers */ 535c977f7f5Sdrh Trigger *pTrigger, /* A list of triggers associated with a table */ 536c3f9bad2Sdanielk1977 int op, /* one of TK_DELETE, TK_INSERT, TK_UPDATE */ 537c3f9bad2Sdanielk1977 int tr_tm, /* one of TK_BEFORE, TK_AFTER */ 538c3f9bad2Sdanielk1977 int foreach, /* one of TK_ROW or TK_STATEMENT */ 539c977f7f5Sdrh ExprList *pChanges /* Columns that change in an UPDATE statement */ 540c977f7f5Sdrh ){ 541633ed08dSdanielk1977 Trigger * pTriggerCursor; 542c3f9bad2Sdanielk1977 543633ed08dSdanielk1977 if( always_code_trigger_setup ){ 544633ed08dSdanielk1977 return 1; 545633ed08dSdanielk1977 } 546c3f9bad2Sdanielk1977 547633ed08dSdanielk1977 pTriggerCursor = pTrigger; 548633ed08dSdanielk1977 while( pTriggerCursor ){ 549633ed08dSdanielk1977 if( pTriggerCursor->op == op && 550633ed08dSdanielk1977 pTriggerCursor->tr_tm == tr_tm && 551633ed08dSdanielk1977 pTriggerCursor->foreach == foreach && 552633ed08dSdanielk1977 checkColumnOverLap(pTriggerCursor->pColumns, pChanges) ){ 553c3f9bad2Sdanielk1977 TriggerStack * ss; 554c3f9bad2Sdanielk1977 ss = pParse->trigStack; 555f29ce559Sdanielk1977 while( ss && ss->pTrigger != pTrigger ){ 556f29ce559Sdanielk1977 ss = ss->pNext; 557f29ce559Sdanielk1977 } 558c3f9bad2Sdanielk1977 if( !ss )return 1; 559c3f9bad2Sdanielk1977 } 560633ed08dSdanielk1977 pTriggerCursor = pTriggerCursor->pNext; 561c3f9bad2Sdanielk1977 } 562c3f9bad2Sdanielk1977 563c3f9bad2Sdanielk1977 return 0; 564c3f9bad2Sdanielk1977 } 565c3f9bad2Sdanielk1977 566c977f7f5Sdrh /* 567f26e09c8Sdrh ** Convert the pStep->target token into a SrcList and return a pointer 568f26e09c8Sdrh ** to that SrcList. 569f26e09c8Sdrh ** 570f26e09c8Sdrh ** This routine adds a specific database name, if needed, to the target when 571f26e09c8Sdrh ** forming the SrcList. This prevents a trigger in one database from 572f26e09c8Sdrh ** referring to a target in another database. An exception is when the 573f26e09c8Sdrh ** trigger is in TEMP in which case it can refer to any other database it 574f26e09c8Sdrh ** wants. 575f26e09c8Sdrh */ 576f26e09c8Sdrh static SrcList *targetSrcList( 577f26e09c8Sdrh Parse *pParse, /* The parsing context */ 578f26e09c8Sdrh TriggerStep *pStep /* The trigger containing the target token */ 579f26e09c8Sdrh ){ 580f26e09c8Sdrh Token sDb; /* Dummy database name token */ 581f26e09c8Sdrh int iDb; /* Index of the database to use */ 582f26e09c8Sdrh SrcList *pSrc; /* SrcList to be returned */ 583f26e09c8Sdrh 584f26e09c8Sdrh iDb = pStep->pTrig->iDb; 585f26e09c8Sdrh if( iDb==0 || iDb>=2 ){ 586f26e09c8Sdrh assert( iDb<pParse->db->nDb ); 587f26e09c8Sdrh sDb.z = pParse->db->aDb[iDb].zName; 588f26e09c8Sdrh sDb.n = strlen(sDb.z); 5894adee20fSdanielk1977 pSrc = sqlite3SrcListAppend(0, &sDb, &pStep->target); 590f26e09c8Sdrh } else { 5914adee20fSdanielk1977 pSrc = sqlite3SrcListAppend(0, &pStep->target, 0); 592f26e09c8Sdrh } 593f26e09c8Sdrh return pSrc; 594f26e09c8Sdrh } 595f26e09c8Sdrh 596f26e09c8Sdrh /* 597c977f7f5Sdrh ** Generate VDBE code for zero or more statements inside the body of a 598c977f7f5Sdrh ** trigger. 599c977f7f5Sdrh */ 600c3f9bad2Sdanielk1977 static int codeTriggerProgram( 601c977f7f5Sdrh Parse *pParse, /* The parser context */ 602c977f7f5Sdrh TriggerStep *pStepList, /* List of statements inside the trigger body */ 603c977f7f5Sdrh int orconfin /* Conflict algorithm. (OE_Abort, etc) */ 604633ed08dSdanielk1977 ){ 605633ed08dSdanielk1977 TriggerStep * pTriggerStep = pStepList; 606c3f9bad2Sdanielk1977 int orconf; 607c3f9bad2Sdanielk1977 608633ed08dSdanielk1977 while( pTriggerStep ){ 609c3f9bad2Sdanielk1977 int saveNTab = pParse->nTab; 610f26e09c8Sdrh 611633ed08dSdanielk1977 orconf = (orconfin == OE_Default)?pTriggerStep->orconf:orconfin; 612c3f9bad2Sdanielk1977 pParse->trigStack->orconf = orconf; 613633ed08dSdanielk1977 switch( pTriggerStep->op ){ 614c3f9bad2Sdanielk1977 case TK_SELECT: { 6154adee20fSdanielk1977 Select * ss = sqlite3SelectDup(pTriggerStep->pSelect); 6166f34903eSdanielk1977 assert(ss); 6176f34903eSdanielk1977 assert(ss->pSrc); 6184adee20fSdanielk1977 sqlite3Select(pParse, ss, SRT_Discard, 0, 0, 0, 0); 6194adee20fSdanielk1977 sqlite3SelectDelete(ss); 620c3f9bad2Sdanielk1977 break; 621c3f9bad2Sdanielk1977 } 622c3f9bad2Sdanielk1977 case TK_UPDATE: { 623113088ecSdrh SrcList *pSrc; 624f26e09c8Sdrh pSrc = targetSrcList(pParse, pTriggerStep); 6254adee20fSdanielk1977 sqlite3VdbeAddOp(pParse->pVdbe, OP_ListPush, 0, 0); 6264adee20fSdanielk1977 sqlite3Update(pParse, pSrc, 6274adee20fSdanielk1977 sqlite3ExprListDup(pTriggerStep->pExprList), 6284adee20fSdanielk1977 sqlite3ExprDup(pTriggerStep->pWhere), orconf); 6294adee20fSdanielk1977 sqlite3VdbeAddOp(pParse->pVdbe, OP_ListPop, 0, 0); 630c3f9bad2Sdanielk1977 break; 631c3f9bad2Sdanielk1977 } 632c3f9bad2Sdanielk1977 case TK_INSERT: { 633113088ecSdrh SrcList *pSrc; 634f26e09c8Sdrh pSrc = targetSrcList(pParse, pTriggerStep); 6354adee20fSdanielk1977 sqlite3Insert(pParse, pSrc, 6364adee20fSdanielk1977 sqlite3ExprListDup(pTriggerStep->pExprList), 6374adee20fSdanielk1977 sqlite3SelectDup(pTriggerStep->pSelect), 6384adee20fSdanielk1977 sqlite3IdListDup(pTriggerStep->pIdList), orconf); 639c3f9bad2Sdanielk1977 break; 640c3f9bad2Sdanielk1977 } 641c3f9bad2Sdanielk1977 case TK_DELETE: { 642113088ecSdrh SrcList *pSrc; 6434adee20fSdanielk1977 sqlite3VdbeAddOp(pParse->pVdbe, OP_ListPush, 0, 0); 644f26e09c8Sdrh pSrc = targetSrcList(pParse, pTriggerStep); 6454adee20fSdanielk1977 sqlite3DeleteFrom(pParse, pSrc, sqlite3ExprDup(pTriggerStep->pWhere)); 6464adee20fSdanielk1977 sqlite3VdbeAddOp(pParse->pVdbe, OP_ListPop, 0, 0); 647c3f9bad2Sdanielk1977 break; 648c3f9bad2Sdanielk1977 } 649c3f9bad2Sdanielk1977 default: 650c3f9bad2Sdanielk1977 assert(0); 651c3f9bad2Sdanielk1977 } 652c3f9bad2Sdanielk1977 pParse->nTab = saveNTab; 653633ed08dSdanielk1977 pTriggerStep = pTriggerStep->pNext; 654c3f9bad2Sdanielk1977 } 655c3f9bad2Sdanielk1977 656c3f9bad2Sdanielk1977 return 0; 657c3f9bad2Sdanielk1977 } 658c3f9bad2Sdanielk1977 659633ed08dSdanielk1977 /* 660633ed08dSdanielk1977 ** This is called to code FOR EACH ROW triggers. 661633ed08dSdanielk1977 ** 662633ed08dSdanielk1977 ** When the code that this function generates is executed, the following 663633ed08dSdanielk1977 ** must be true: 664c977f7f5Sdrh ** 665c977f7f5Sdrh ** 1. No cursors may be open in the main database. (But newIdx and oldIdx 666c977f7f5Sdrh ** can be indices of cursors in temporary tables. See below.) 667c977f7f5Sdrh ** 668633ed08dSdanielk1977 ** 2. If the triggers being coded are ON INSERT or ON UPDATE triggers, then 669633ed08dSdanielk1977 ** a temporary vdbe cursor (index newIdx) must be open and pointing at 670633ed08dSdanielk1977 ** a row containing values to be substituted for new.* expressions in the 671633ed08dSdanielk1977 ** trigger program(s). 672c977f7f5Sdrh ** 673633ed08dSdanielk1977 ** 3. If the triggers being coded are ON DELETE or ON UPDATE triggers, then 674633ed08dSdanielk1977 ** a temporary vdbe cursor (index oldIdx) must be open and pointing at 675633ed08dSdanielk1977 ** a row containing values to be substituted for old.* expressions in the 676633ed08dSdanielk1977 ** trigger program(s). 677633ed08dSdanielk1977 ** 678633ed08dSdanielk1977 */ 6794adee20fSdanielk1977 int sqlite3CodeRowTrigger( 680c3f9bad2Sdanielk1977 Parse *pParse, /* Parse context */ 681c3f9bad2Sdanielk1977 int op, /* One of TK_UPDATE, TK_INSERT, TK_DELETE */ 682633ed08dSdanielk1977 ExprList *pChanges, /* Changes list for any UPDATE OF triggers */ 683c3f9bad2Sdanielk1977 int tr_tm, /* One of TK_BEFORE, TK_AFTER */ 684633ed08dSdanielk1977 Table *pTab, /* The table to code triggers from */ 685633ed08dSdanielk1977 int newIdx, /* The indice of the "new" row to access */ 686633ed08dSdanielk1977 int oldIdx, /* The indice of the "old" row to access */ 6876f34903eSdanielk1977 int orconf, /* ON CONFLICT policy */ 6886f34903eSdanielk1977 int ignoreJump /* Instruction to jump to for RAISE(IGNORE) */ 689c977f7f5Sdrh ){ 690c3f9bad2Sdanielk1977 Trigger * pTrigger; 691c3f9bad2Sdanielk1977 TriggerStack * pTriggerStack; 692c3f9bad2Sdanielk1977 693c3f9bad2Sdanielk1977 assert(op == TK_UPDATE || op == TK_INSERT || op == TK_DELETE); 694c3f9bad2Sdanielk1977 assert(tr_tm == TK_BEFORE || tr_tm == TK_AFTER ); 695c3f9bad2Sdanielk1977 696633ed08dSdanielk1977 assert(newIdx != -1 || oldIdx != -1); 697c3f9bad2Sdanielk1977 698633ed08dSdanielk1977 pTrigger = pTab->pTrigger; 699c3f9bad2Sdanielk1977 while( pTrigger ){ 700c3f9bad2Sdanielk1977 int fire_this = 0; 701c3f9bad2Sdanielk1977 702c3f9bad2Sdanielk1977 /* determine whether we should code this trigger */ 703c3f9bad2Sdanielk1977 if( pTrigger->op == op && pTrigger->tr_tm == tr_tm && 704c3f9bad2Sdanielk1977 pTrigger->foreach == TK_ROW ){ 705c3f9bad2Sdanielk1977 fire_this = 1; 706c3f9bad2Sdanielk1977 pTriggerStack = pParse->trigStack; 707c3f9bad2Sdanielk1977 while( pTriggerStack ){ 708f29ce559Sdanielk1977 if( pTriggerStack->pTrigger == pTrigger ){ 709f29ce559Sdanielk1977 fire_this = 0; 710f29ce559Sdanielk1977 } 711c3f9bad2Sdanielk1977 pTriggerStack = pTriggerStack->pNext; 712c3f9bad2Sdanielk1977 } 713c3f9bad2Sdanielk1977 if( op == TK_UPDATE && pTrigger->pColumns && 714f29ce559Sdanielk1977 !checkColumnOverLap(pTrigger->pColumns, pChanges) ){ 715c3f9bad2Sdanielk1977 fire_this = 0; 716c3f9bad2Sdanielk1977 } 717f29ce559Sdanielk1977 } 718c3f9bad2Sdanielk1977 719e4697f5eSdrh if( fire_this && (pTriggerStack = sqliteMalloc(sizeof(TriggerStack)))!=0 ){ 720c3f9bad2Sdanielk1977 int endTrigger; 721ad3cab52Sdrh SrcList dummyTablist; 722c3f9bad2Sdanielk1977 Expr * whenExpr; 72385e2096fSdrh AuthContext sContext; 724c3f9bad2Sdanielk1977 725ad3cab52Sdrh dummyTablist.nSrc = 0; 726c3f9bad2Sdanielk1977 727c3f9bad2Sdanielk1977 /* Push an entry on to the trigger stack */ 728c3f9bad2Sdanielk1977 pTriggerStack->pTrigger = pTrigger; 729633ed08dSdanielk1977 pTriggerStack->newIdx = newIdx; 730633ed08dSdanielk1977 pTriggerStack->oldIdx = oldIdx; 731633ed08dSdanielk1977 pTriggerStack->pTab = pTab; 732c3f9bad2Sdanielk1977 pTriggerStack->pNext = pParse->trigStack; 7336f34903eSdanielk1977 pTriggerStack->ignoreJump = ignoreJump; 734c3f9bad2Sdanielk1977 pParse->trigStack = pTriggerStack; 7354adee20fSdanielk1977 sqlite3AuthContextPush(pParse, &sContext, pTrigger->name); 736c3f9bad2Sdanielk1977 737c3f9bad2Sdanielk1977 /* code the WHEN clause */ 7384adee20fSdanielk1977 endTrigger = sqlite3VdbeMakeLabel(pParse->pVdbe); 7394adee20fSdanielk1977 whenExpr = sqlite3ExprDup(pTrigger->pWhen); 7404adee20fSdanielk1977 if( sqlite3ExprResolveIds(pParse, &dummyTablist, 0, whenExpr) ){ 741c3f9bad2Sdanielk1977 pParse->trigStack = pParse->trigStack->pNext; 742c3f9bad2Sdanielk1977 sqliteFree(pTriggerStack); 7434adee20fSdanielk1977 sqlite3ExprDelete(whenExpr); 744c3f9bad2Sdanielk1977 return 1; 745c3f9bad2Sdanielk1977 } 7464adee20fSdanielk1977 sqlite3ExprIfFalse(pParse, whenExpr, endTrigger, 1); 7474adee20fSdanielk1977 sqlite3ExprDelete(whenExpr); 748c3f9bad2Sdanielk1977 7494adee20fSdanielk1977 sqlite3VdbeAddOp(pParse->pVdbe, OP_ContextPush, 0, 0); 750633ed08dSdanielk1977 codeTriggerProgram(pParse, pTrigger->step_list, orconf); 7514adee20fSdanielk1977 sqlite3VdbeAddOp(pParse->pVdbe, OP_ContextPop, 0, 0); 752c3f9bad2Sdanielk1977 753c3f9bad2Sdanielk1977 /* Pop the entry off the trigger stack */ 754c3f9bad2Sdanielk1977 pParse->trigStack = pParse->trigStack->pNext; 7554adee20fSdanielk1977 sqlite3AuthContextPop(&sContext); 756c3f9bad2Sdanielk1977 sqliteFree(pTriggerStack); 757c3f9bad2Sdanielk1977 7584adee20fSdanielk1977 sqlite3VdbeResolveLabel(pParse->pVdbe, endTrigger); 759c3f9bad2Sdanielk1977 } 760c3f9bad2Sdanielk1977 pTrigger = pTrigger->pNext; 761c3f9bad2Sdanielk1977 } 762c3f9bad2Sdanielk1977 763c3f9bad2Sdanielk1977 return 0; 764c3f9bad2Sdanielk1977 } 7654adee20fSdanielk1977 7664adee20fSdanielk1977 7674adee20fSdanielk1977 768