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 */ 18f0f258b1Sdrh void sqliteDeleteTriggerStep(TriggerStep *pTriggerStep){ 194b59ab5eSdrh while( pTriggerStep ){ 204b59ab5eSdrh TriggerStep * pTmp = pTriggerStep; 214b59ab5eSdrh pTriggerStep = pTriggerStep->pNext; 224b59ab5eSdrh 238c74a8caSdrh if( pTmp->target.dyn ) sqliteFree((char*)pTmp->target.z); 244b59ab5eSdrh sqliteExprDelete(pTmp->pWhere); 254b59ab5eSdrh sqliteExprListDelete(pTmp->pExprList); 264b59ab5eSdrh sqliteSelectDelete(pTmp->pSelect); 274b59ab5eSdrh sqliteIdListDelete(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 38f0f258b1Sdrh ** sqliteFinishTrigger() function is called to complete the trigger 39f0f258b1Sdrh ** construction process. 40c3f9bad2Sdanielk1977 */ 41f0f258b1Sdrh void sqliteBeginTrigger( 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 */ 57ed6c8671Sdrh 58c3f9bad2Sdanielk1977 /* Check that: 599adf9ac4Sdrh ** 1. the trigger name does not already exist. 60f0f258b1Sdrh ** 2. the table (or view) does exist in the same database as the trigger. 61d702fccbSdanielk1977 ** 3. that we are not trying to create a trigger on the sqlite_master table 62d702fccbSdanielk1977 ** 4. That we are not trying to create an INSTEAD OF trigger on a table. 63d702fccbSdanielk1977 ** 5. That we are not trying to create a BEFORE or AFTER trigger on a view. 64c3f9bad2Sdanielk1977 */ 65d24cc427Sdrh if( sqlite_malloc_failed ) goto trigger_cleanup; 66d24cc427Sdrh assert( pTableName->nSrc==1 ); 67812d7a21Sdrh tab = sqliteSrcListLookup(pParse, pTableName); 68d24cc427Sdrh if( !tab ){ 69d24cc427Sdrh goto trigger_cleanup; 70d24cc427Sdrh } 71f0f258b1Sdrh iDb = isTemp ? 1 : tab->iDb; 72f0f258b1Sdrh if( iDb>=2 && !pParse->initFlag ){ 73da93d238Sdrh sqliteErrorMsg(pParse, "triggers may not be added to auxiliary " 74da93d238Sdrh "database %s", db->aDb[tab->iDb].zName); 75d24cc427Sdrh goto trigger_cleanup; 76d24cc427Sdrh } 77d24cc427Sdrh 78e5f9c644Sdrh zName = sqliteStrNDup(pName->z, pName->n); 79f0f258b1Sdrh if( sqliteHashFind(&(db->aDb[iDb].trigHash), zName,pName->n+1) ){ 80da93d238Sdrh sqliteErrorMsg(pParse, "trigger %T already exists", pName); 81c3f9bad2Sdanielk1977 goto trigger_cleanup; 82c3f9bad2Sdanielk1977 } 83d24cc427Sdrh if( sqliteStrNICmp(tab->zName, "sqlite_", 7)==0 ){ 84da93d238Sdrh sqliteErrorMsg(pParse, "cannot create trigger on system table"); 85e0bc4048Sdrh pParse->nErr++; 86e0bc4048Sdrh goto trigger_cleanup; 87e0bc4048Sdrh } 88d702fccbSdanielk1977 if( tab->pSelect && tr_tm != TK_INSTEAD ){ 89da93d238Sdrh sqliteErrorMsg(pParse, "cannot create %s trigger on view: %S", 90da93d238Sdrh (tr_tm == TK_BEFORE)?"BEFORE":"AFTER", pTableName, 0); 91d702fccbSdanielk1977 goto trigger_cleanup; 92d702fccbSdanielk1977 } 93d702fccbSdanielk1977 if( !tab->pSelect && tr_tm == TK_INSTEAD ){ 94da93d238Sdrh sqliteErrorMsg(pParse, "cannot create INSTEAD OF" 95da93d238Sdrh " trigger on table: %S", pTableName, 0); 96d702fccbSdanielk1977 goto trigger_cleanup; 97d702fccbSdanielk1977 } 98e5f9c644Sdrh #ifndef SQLITE_OMIT_AUTHORIZATION 99e5f9c644Sdrh { 100e5f9c644Sdrh int code = SQLITE_CREATE_TRIGGER; 101e22a334bSdrh const char *zDb = db->aDb[tab->iDb].zName; 102e22a334bSdrh const char *zDbTrig = isTemp ? db->aDb[1].zName : zDb; 103f0f258b1Sdrh if( tab->iDb==1 || isTemp ) code = SQLITE_CREATE_TEMP_TRIGGER; 104e22a334bSdrh if( sqliteAuthCheck(pParse, code, zName, tab->zName, zDbTrig) ){ 105ed6c8671Sdrh goto trigger_cleanup; 106ed6c8671Sdrh } 107e22a334bSdrh if( sqliteAuthCheck(pParse, SQLITE_INSERT, SCHEMA_TABLE(tab->iDb), 0, zDb)){ 10877ad4e41Sdrh goto trigger_cleanup; 10977ad4e41Sdrh } 110d702fccbSdanielk1977 } 111e5f9c644Sdrh #endif 112d702fccbSdanielk1977 1135cf590c1Sdrh /* INSTEAD OF triggers can only appear on views and BEGIN triggers 1145cf590c1Sdrh ** cannot appear on views. So we might as well translate every 1155cf590c1Sdrh ** INSTEAD OF trigger into a BEFORE trigger. It simplifies code 1165cf590c1Sdrh ** elsewhere. 1175cf590c1Sdrh */ 118d702fccbSdanielk1977 if (tr_tm == TK_INSTEAD){ 119d702fccbSdanielk1977 tr_tm = TK_BEFORE; 120c3f9bad2Sdanielk1977 } 121c3f9bad2Sdanielk1977 122c3f9bad2Sdanielk1977 /* Build the Trigger object */ 123c3f9bad2Sdanielk1977 nt = (Trigger*)sqliteMalloc(sizeof(Trigger)); 124e4697f5eSdrh if( nt==0 ) goto trigger_cleanup; 125e5f9c644Sdrh nt->name = zName; 126e5f9c644Sdrh zName = 0; 127d24cc427Sdrh nt->table = sqliteStrDup(pTableName->a[0].zName); 128e4697f5eSdrh if( sqlite_malloc_failed ) goto trigger_cleanup; 129f0f258b1Sdrh nt->iDb = iDb; 130c3f9bad2Sdanielk1977 nt->op = op; 131c3f9bad2Sdanielk1977 nt->tr_tm = tr_tm; 1324b59ab5eSdrh nt->pWhen = sqliteExprDup(pWhen); 1334b59ab5eSdrh nt->pColumns = sqliteIdListDup(pColumns); 134c3f9bad2Sdanielk1977 nt->foreach = foreach; 135f0f258b1Sdrh assert( pParse->pNewTrigger==0 ); 136f0f258b1Sdrh pParse->pNewTrigger = nt; 137f0f258b1Sdrh 138f0f258b1Sdrh trigger_cleanup: 139f0f258b1Sdrh sqliteFree(zName); 140f0f258b1Sdrh sqliteSrcListDelete(pTableName); 141f0f258b1Sdrh sqliteIdListDelete(pColumns); 142f0f258b1Sdrh sqliteExprDelete(pWhen); 143f0f258b1Sdrh } 144f0f258b1Sdrh 145f0f258b1Sdrh /* 146f0f258b1Sdrh ** This routine is called after all of the trigger actions have been parsed 147f0f258b1Sdrh ** in order to complete the process of building the trigger. 148f0f258b1Sdrh */ 149f0f258b1Sdrh void sqliteFinishTrigger( 150f0f258b1Sdrh Parse *pParse, /* Parser context */ 151f0f258b1Sdrh TriggerStep *pStepList, /* The triggered program */ 152f0f258b1Sdrh Token *pAll /* Token that describes the complete CREATE TRIGGER */ 153f0f258b1Sdrh ){ 154f0f258b1Sdrh Trigger *nt; /* The trigger whose construction is finishing up */ 155f0f258b1Sdrh sqlite *db = pParse->db; /* The database */ 156f0f258b1Sdrh 157f0f258b1Sdrh if( pParse->nErr || pParse->pNewTrigger==0 ) goto triggerfinish_cleanup; 158f0f258b1Sdrh nt = pParse->pNewTrigger; 159f0f258b1Sdrh pParse->pNewTrigger = 0; 160633ed08dSdanielk1977 nt->step_list = pStepList; 161a69d9168Sdrh while( pStepList ){ 162a69d9168Sdrh pStepList->pTrig = nt; 163a69d9168Sdrh pStepList = pStepList->pNext; 164a69d9168Sdrh } 165c3f9bad2Sdanielk1977 166c3f9bad2Sdanielk1977 /* if we are not initializing, and this trigger is not on a TEMP table, 1679adf9ac4Sdrh ** build the sqlite_master entry 1689adf9ac4Sdrh */ 169e0bc4048Sdrh if( !pParse->initFlag ){ 170c977f7f5Sdrh static VdbeOp insertTrig[] = { 171c977f7f5Sdrh { OP_NewRecno, 0, 0, 0 }, 172c977f7f5Sdrh { OP_String, 0, 0, "trigger" }, 173e0bc4048Sdrh { OP_String, 0, 0, 0 }, /* 2: trigger name */ 174e0bc4048Sdrh { OP_String, 0, 0, 0 }, /* 3: table name */ 175c977f7f5Sdrh { OP_Integer, 0, 0, 0 }, 176e0bc4048Sdrh { OP_String, 0, 0, 0 }, /* 5: SQL */ 177c977f7f5Sdrh { OP_MakeRecord, 5, 0, 0 }, 178c977f7f5Sdrh { OP_PutIntKey, 0, 0, 0 }, 179c977f7f5Sdrh }; 180c977f7f5Sdrh int addr; 181c977f7f5Sdrh Vdbe *v; 182c3f9bad2Sdanielk1977 183c3f9bad2Sdanielk1977 /* Make an entry in the sqlite_master table */ 184c977f7f5Sdrh v = sqliteGetVdbe(pParse); 185f0f258b1Sdrh if( v==0 ) goto triggerfinish_cleanup; 186cabb0819Sdrh sqliteBeginWriteOperation(pParse, 0, 0); 187f0f258b1Sdrh sqliteOpenMasterTable(v, nt->iDb==1); 188c977f7f5Sdrh addr = sqliteVdbeAddOpList(v, ArraySize(insertTrig), insertTrig); 189e0bc4048Sdrh sqliteVdbeChangeP3(v, addr+2, nt->name, 0); 190e0bc4048Sdrh sqliteVdbeChangeP3(v, addr+3, nt->table, 0); 1914b59ab5eSdrh sqliteVdbeChangeP3(v, addr+5, pAll->z, pAll->n); 192f0f258b1Sdrh if( nt->iDb==0 ){ 193d24cc427Sdrh sqliteChangeCookie(db, v); 194e0bc4048Sdrh } 195e0bc4048Sdrh sqliteVdbeAddOp(v, OP_Close, 0, 0); 196c3f9bad2Sdanielk1977 sqliteEndWriteOperation(pParse); 197c3f9bad2Sdanielk1977 } 198c3f9bad2Sdanielk1977 199c3f9bad2Sdanielk1977 if( !pParse->explain ){ 200f0f258b1Sdrh Table *pTab; 201f0f258b1Sdrh sqliteHashInsert(&db->aDb[nt->iDb].trigHash, 202f0f258b1Sdrh nt->name, strlen(nt->name)+1, nt); 203f0f258b1Sdrh pTab = sqliteLocateTable(pParse, nt->table, 0); 204f0f258b1Sdrh assert( pTab!=0 ); 205f0f258b1Sdrh nt->pNext = pTab->pTrigger; 206f0f258b1Sdrh pTab->pTrigger = nt; 207c3f9bad2Sdanielk1977 }else{ 208f0f258b1Sdrh sqliteDeleteTrigger(nt); 209c3f9bad2Sdanielk1977 } 210c3f9bad2Sdanielk1977 211f0f258b1Sdrh triggerfinish_cleanup: 212f0f258b1Sdrh sqliteDeleteTrigger(pParse->pNewTrigger); 213f0f258b1Sdrh pParse->pNewTrigger = 0; 2144b59ab5eSdrh sqliteDeleteTriggerStep(pStepList); 215c3f9bad2Sdanielk1977 } 2164b59ab5eSdrh 2174b59ab5eSdrh /* 2184b59ab5eSdrh ** Make a copy of all components of the given trigger step. This has 2194b59ab5eSdrh ** the effect of copying all Expr.token.z values into memory obtained 2204b59ab5eSdrh ** from sqliteMalloc(). As initially created, the Expr.token.z values 2214b59ab5eSdrh ** all point to the input string that was fed to the parser. But that 2224b59ab5eSdrh ** string is ephemeral - it will go away as soon as the sqlite_exec() 2234b59ab5eSdrh ** call that started the parser exits. This routine makes a persistent 2244b59ab5eSdrh ** copy of all the Expr.token.z strings so that the TriggerStep structure 2254b59ab5eSdrh ** will be valid even after the sqlite_exec() call returns. 2264b59ab5eSdrh */ 2274b59ab5eSdrh static void sqlitePersistTriggerStep(TriggerStep *p){ 2284b59ab5eSdrh if( p->target.z ){ 2294b59ab5eSdrh p->target.z = sqliteStrNDup(p->target.z, p->target.n); 2304b59ab5eSdrh p->target.dyn = 1; 2314b59ab5eSdrh } 2324b59ab5eSdrh if( p->pSelect ){ 2334b59ab5eSdrh Select *pNew = sqliteSelectDup(p->pSelect); 2344b59ab5eSdrh sqliteSelectDelete(p->pSelect); 2354b59ab5eSdrh p->pSelect = pNew; 2364b59ab5eSdrh } 2374b59ab5eSdrh if( p->pWhere ){ 2384b59ab5eSdrh Expr *pNew = sqliteExprDup(p->pWhere); 2394b59ab5eSdrh sqliteExprDelete(p->pWhere); 2404b59ab5eSdrh p->pWhere = pNew; 2414b59ab5eSdrh } 2424b59ab5eSdrh if( p->pExprList ){ 2434b59ab5eSdrh ExprList *pNew = sqliteExprListDup(p->pExprList); 2444b59ab5eSdrh sqliteExprListDelete(p->pExprList); 2454b59ab5eSdrh p->pExprList = pNew; 2464b59ab5eSdrh } 2474b59ab5eSdrh if( p->pIdList ){ 2484b59ab5eSdrh IdList *pNew = sqliteIdListDup(p->pIdList); 2494b59ab5eSdrh sqliteIdListDelete(p->pIdList); 2504b59ab5eSdrh p->pIdList = pNew; 251c3f9bad2Sdanielk1977 } 252c3f9bad2Sdanielk1977 } 253c3f9bad2Sdanielk1977 254c977f7f5Sdrh /* 255c977f7f5Sdrh ** Turn a SELECT statement (that the pSelect parameter points to) into 256c977f7f5Sdrh ** a trigger step. Return a pointer to a TriggerStep structure. 257c977f7f5Sdrh ** 258c977f7f5Sdrh ** The parser calls this routine when it finds a SELECT statement in 259c977f7f5Sdrh ** body of a TRIGGER. 260c977f7f5Sdrh */ 261c977f7f5Sdrh TriggerStep *sqliteTriggerSelectStep(Select *pSelect){ 262633ed08dSdanielk1977 TriggerStep *pTriggerStep = sqliteMalloc(sizeof(TriggerStep)); 263e4697f5eSdrh if( pTriggerStep==0 ) return 0; 264c3f9bad2Sdanielk1977 265633ed08dSdanielk1977 pTriggerStep->op = TK_SELECT; 266633ed08dSdanielk1977 pTriggerStep->pSelect = pSelect; 267633ed08dSdanielk1977 pTriggerStep->orconf = OE_Default; 2684b59ab5eSdrh sqlitePersistTriggerStep(pTriggerStep); 269c3f9bad2Sdanielk1977 270633ed08dSdanielk1977 return pTriggerStep; 271c3f9bad2Sdanielk1977 } 272c3f9bad2Sdanielk1977 273c977f7f5Sdrh /* 274c977f7f5Sdrh ** Build a trigger step out of an INSERT statement. Return a pointer 275c977f7f5Sdrh ** to the new trigger step. 276c977f7f5Sdrh ** 277c977f7f5Sdrh ** The parser calls this routine when it sees an INSERT inside the 278c977f7f5Sdrh ** body of a trigger. 279c977f7f5Sdrh */ 280633ed08dSdanielk1977 TriggerStep *sqliteTriggerInsertStep( 281c977f7f5Sdrh Token *pTableName, /* Name of the table into which we insert */ 282c977f7f5Sdrh IdList *pColumn, /* List of columns in pTableName to insert into */ 283c977f7f5Sdrh ExprList *pEList, /* The VALUE clause: a list of values to be inserted */ 284c977f7f5Sdrh Select *pSelect, /* A SELECT statement that supplies values */ 285c977f7f5Sdrh int orconf /* The conflict algorithm (OE_Abort, OE_Replace, etc.) */ 286633ed08dSdanielk1977 ){ 287633ed08dSdanielk1977 TriggerStep *pTriggerStep = sqliteMalloc(sizeof(TriggerStep)); 288e4697f5eSdrh if( pTriggerStep==0 ) return 0; 289c3f9bad2Sdanielk1977 290633ed08dSdanielk1977 assert(pEList == 0 || pSelect == 0); 291633ed08dSdanielk1977 assert(pEList != 0 || pSelect != 0); 292c3f9bad2Sdanielk1977 293633ed08dSdanielk1977 pTriggerStep->op = TK_INSERT; 294633ed08dSdanielk1977 pTriggerStep->pSelect = pSelect; 295633ed08dSdanielk1977 pTriggerStep->target = *pTableName; 296633ed08dSdanielk1977 pTriggerStep->pIdList = pColumn; 297633ed08dSdanielk1977 pTriggerStep->pExprList = pEList; 298633ed08dSdanielk1977 pTriggerStep->orconf = orconf; 2994b59ab5eSdrh sqlitePersistTriggerStep(pTriggerStep); 300c3f9bad2Sdanielk1977 301633ed08dSdanielk1977 return pTriggerStep; 302c3f9bad2Sdanielk1977 } 303c3f9bad2Sdanielk1977 304c977f7f5Sdrh /* 305c977f7f5Sdrh ** Construct a trigger step that implements an UPDATE statement and return 306c977f7f5Sdrh ** a pointer to that trigger step. The parser calls this routine when it 307c977f7f5Sdrh ** sees an UPDATE statement inside the body of a CREATE TRIGGER. 308c977f7f5Sdrh */ 309633ed08dSdanielk1977 TriggerStep *sqliteTriggerUpdateStep( 310c977f7f5Sdrh Token *pTableName, /* Name of the table to be updated */ 311c977f7f5Sdrh ExprList *pEList, /* The SET clause: list of column and new values */ 312c977f7f5Sdrh Expr *pWhere, /* The WHERE clause */ 313c977f7f5Sdrh int orconf /* The conflict algorithm. (OE_Abort, OE_Ignore, etc) */ 314c977f7f5Sdrh ){ 315633ed08dSdanielk1977 TriggerStep *pTriggerStep = sqliteMalloc(sizeof(TriggerStep)); 316e4697f5eSdrh if( pTriggerStep==0 ) return 0; 317c3f9bad2Sdanielk1977 318633ed08dSdanielk1977 pTriggerStep->op = TK_UPDATE; 319633ed08dSdanielk1977 pTriggerStep->target = *pTableName; 320633ed08dSdanielk1977 pTriggerStep->pExprList = pEList; 321633ed08dSdanielk1977 pTriggerStep->pWhere = pWhere; 322633ed08dSdanielk1977 pTriggerStep->orconf = orconf; 3234b59ab5eSdrh sqlitePersistTriggerStep(pTriggerStep); 324c3f9bad2Sdanielk1977 325633ed08dSdanielk1977 return pTriggerStep; 326c3f9bad2Sdanielk1977 } 327c3f9bad2Sdanielk1977 328c977f7f5Sdrh /* 329c977f7f5Sdrh ** Construct a trigger step that implements a DELETE statement and return 330c977f7f5Sdrh ** a pointer to that trigger step. The parser calls this routine when it 331c977f7f5Sdrh ** sees a DELETE statement inside the body of a CREATE TRIGGER. 332c977f7f5Sdrh */ 333c977f7f5Sdrh TriggerStep *sqliteTriggerDeleteStep(Token *pTableName, Expr *pWhere){ 334633ed08dSdanielk1977 TriggerStep *pTriggerStep = sqliteMalloc(sizeof(TriggerStep)); 335e4697f5eSdrh if( pTriggerStep==0 ) return 0; 336c3f9bad2Sdanielk1977 337633ed08dSdanielk1977 pTriggerStep->op = TK_DELETE; 338633ed08dSdanielk1977 pTriggerStep->target = *pTableName; 339633ed08dSdanielk1977 pTriggerStep->pWhere = pWhere; 340633ed08dSdanielk1977 pTriggerStep->orconf = OE_Default; 3414b59ab5eSdrh sqlitePersistTriggerStep(pTriggerStep); 342c3f9bad2Sdanielk1977 343633ed08dSdanielk1977 return pTriggerStep; 344c3f9bad2Sdanielk1977 } 345c3f9bad2Sdanielk1977 346c3f9bad2Sdanielk1977 /* 347633ed08dSdanielk1977 ** Recursively delete a Trigger structure 348c3f9bad2Sdanielk1977 */ 3491d1f3055Sdrh void sqliteDeleteTrigger(Trigger *pTrigger){ 350f0f258b1Sdrh if( pTrigger==0 ) return; 3514b59ab5eSdrh sqliteDeleteTriggerStep(pTrigger->step_list); 352633ed08dSdanielk1977 sqliteFree(pTrigger->name); 353633ed08dSdanielk1977 sqliteFree(pTrigger->table); 354633ed08dSdanielk1977 sqliteExprDelete(pTrigger->pWhen); 355633ed08dSdanielk1977 sqliteIdListDelete(pTrigger->pColumns); 356633ed08dSdanielk1977 sqliteFree(pTrigger); 357633ed08dSdanielk1977 } 358633ed08dSdanielk1977 359633ed08dSdanielk1977 /* 360633ed08dSdanielk1977 * This function is called to drop a trigger from the database schema. 361633ed08dSdanielk1977 * 362633ed08dSdanielk1977 * This may be called directly from the parser, or from within 363633ed08dSdanielk1977 * sqliteDropTable(). In the latter case the "nested" argument is true. 364633ed08dSdanielk1977 * 365633ed08dSdanielk1977 * Note that this function does not delete the trigger entirely. Instead it 366633ed08dSdanielk1977 * removes it from the internal schema and places it in the trigDrop hash 367633ed08dSdanielk1977 * table. This is so that the trigger can be restored into the database schema 368633ed08dSdanielk1977 * if the transaction is rolled back. 369633ed08dSdanielk1977 */ 370d24cc427Sdrh void sqliteDropTrigger(Parse *pParse, SrcList *pName, int nested){ 371633ed08dSdanielk1977 Trigger *pTrigger; 372633ed08dSdanielk1977 Table *pTable; 373e0bc4048Sdrh Vdbe *v; 374d24cc427Sdrh int i; 375d24cc427Sdrh const char *zDb; 376d24cc427Sdrh const char *zName; 377d24cc427Sdrh int nName; 378d24cc427Sdrh sqlite *db = pParse->db; 379633ed08dSdanielk1977 380d24cc427Sdrh if( sqlite_malloc_failed ) goto drop_trigger_cleanup; 381d24cc427Sdrh assert( pName->nSrc==1 ); 382d24cc427Sdrh zDb = pName->a[0].zDatabase; 383d24cc427Sdrh zName = pName->a[0].zName; 384d24cc427Sdrh nName = strlen(zName); 385d24cc427Sdrh for(i=0; i<db->nDb; i++){ 386812d7a21Sdrh int j = (i<2) ? i^1 : i; /* Search TEMP before MAIN */ 387812d7a21Sdrh if( zDb && sqliteStrICmp(db->aDb[j].zName, zDb) ) continue; 388812d7a21Sdrh pTrigger = sqliteHashFind(&(db->aDb[j].trigHash), zName, nName+1); 389d24cc427Sdrh if( pTrigger ) break; 390c3f9bad2Sdanielk1977 } 391d24cc427Sdrh if( !pTrigger ){ 392da93d238Sdrh sqliteErrorMsg(pParse, "no such trigger: %S", pName, 0); 393d24cc427Sdrh goto drop_trigger_cleanup; 394d24cc427Sdrh } 395f0f258b1Sdrh assert( pTrigger->iDb<db->nDb ); 396d24cc427Sdrh if( pTrigger->iDb>=2 ){ 397da93d238Sdrh sqliteErrorMsg(pParse, "triggers may not be removed from " 398da93d238Sdrh "auxiliary database %s", db->aDb[pTrigger->iDb].zName); 399d24cc427Sdrh goto drop_trigger_cleanup; 400d24cc427Sdrh } 401d24cc427Sdrh pTable = sqliteFindTable(db, pTrigger->table, db->aDb[pTrigger->iDb].zName); 402ed6c8671Sdrh assert(pTable); 403f0f258b1Sdrh assert( pTable->iDb==pTrigger->iDb || pTrigger->iDb==1 ); 404e5f9c644Sdrh #ifndef SQLITE_OMIT_AUTHORIZATION 405e5f9c644Sdrh { 406e5f9c644Sdrh int code = SQLITE_DROP_TRIGGER; 407e22a334bSdrh const char *zDb = db->aDb[pTrigger->iDb].zName; 408e22a334bSdrh const char *zTab = SCHEMA_TABLE(pTrigger->iDb); 409f0f258b1Sdrh if( pTrigger->iDb ) code = SQLITE_DROP_TEMP_TRIGGER; 410e22a334bSdrh if( sqliteAuthCheck(pParse, code, pTrigger->name, pTable->zName, zDb) || 411e22a334bSdrh sqliteAuthCheck(pParse, SQLITE_DELETE, zTab, 0, zDb) ){ 4120be9df07Sdrh goto drop_trigger_cleanup; 413ed6c8671Sdrh } 414e5f9c644Sdrh } 415e5f9c644Sdrh #endif 416c3f9bad2Sdanielk1977 417f0f258b1Sdrh /* Generate code to destroy the database record of the trigger. 418f0f258b1Sdrh */ 419f0f258b1Sdrh if( pTable!=0 && !nested && (v = sqliteGetVdbe(pParse))!=0 ){ 420f0f258b1Sdrh int base; 421f0f258b1Sdrh static VdbeOp dropTrigger[] = { 422f0f258b1Sdrh { OP_Rewind, 0, ADDR(8), 0}, 423f0f258b1Sdrh { OP_String, 0, 0, 0}, /* 1 */ 424f0f258b1Sdrh { OP_MemStore, 1, 1, 0}, 425f0f258b1Sdrh { OP_MemLoad, 1, 0, 0}, /* 3 */ 426f0f258b1Sdrh { OP_Column, 0, 1, 0}, 427f0f258b1Sdrh { OP_Ne, 0, ADDR(7), 0}, 428f0f258b1Sdrh { OP_Delete, 0, 0, 0}, 429f0f258b1Sdrh { OP_Next, 0, ADDR(3), 0}, /* 7 */ 430f0f258b1Sdrh }; 431f0f258b1Sdrh 432f0f258b1Sdrh sqliteBeginWriteOperation(pParse, 0, 0); 433f0f258b1Sdrh sqliteOpenMasterTable(v, pTrigger->iDb); 434f0f258b1Sdrh base = sqliteVdbeAddOpList(v, ArraySize(dropTrigger), dropTrigger); 435f0f258b1Sdrh sqliteVdbeChangeP3(v, base+1, zName, 0); 436f0f258b1Sdrh if( pTrigger->iDb==0 ){ 437f0f258b1Sdrh sqliteChangeCookie(db, v); 438f0f258b1Sdrh } 439f0f258b1Sdrh sqliteVdbeAddOp(v, OP_Close, 0, 0); 440f0f258b1Sdrh sqliteEndWriteOperation(pParse); 441f0f258b1Sdrh } 442f0f258b1Sdrh 443c3f9bad2Sdanielk1977 /* 444e0bc4048Sdrh * If this is not an "explain", then delete the trigger structure. 445c3f9bad2Sdanielk1977 */ 446c3f9bad2Sdanielk1977 if( !pParse->explain ){ 447633ed08dSdanielk1977 if( pTable->pTrigger == pTrigger ){ 448633ed08dSdanielk1977 pTable->pTrigger = pTrigger->pNext; 449633ed08dSdanielk1977 }else{ 450633ed08dSdanielk1977 Trigger *cc = pTable->pTrigger; 451c3f9bad2Sdanielk1977 while( cc ){ 452633ed08dSdanielk1977 if( cc->pNext == pTrigger ){ 453c3f9bad2Sdanielk1977 cc->pNext = cc->pNext->pNext; 454c3f9bad2Sdanielk1977 break; 455c3f9bad2Sdanielk1977 } 456c3f9bad2Sdanielk1977 cc = cc->pNext; 457c3f9bad2Sdanielk1977 } 458c3f9bad2Sdanielk1977 assert(cc); 459c3f9bad2Sdanielk1977 } 460d24cc427Sdrh sqliteHashInsert(&(db->aDb[pTrigger->iDb].trigHash), zName, nName+1, 0); 461e0bc4048Sdrh sqliteDeleteTrigger(pTrigger); 462c3f9bad2Sdanielk1977 } 463c3f9bad2Sdanielk1977 464d24cc427Sdrh drop_trigger_cleanup: 465d24cc427Sdrh sqliteSrcListDelete(pName); 466c3f9bad2Sdanielk1977 } 467c3f9bad2Sdanielk1977 468c977f7f5Sdrh /* 469c977f7f5Sdrh ** pEList is the SET clause of an UPDATE statement. Each entry 470c977f7f5Sdrh ** in pEList is of the format <id>=<expr>. If any of the entries 471c977f7f5Sdrh ** in pEList have an <id> which matches an identifier in pIdList, 472c977f7f5Sdrh ** then return TRUE. If pIdList==NULL, then it is considered a 473c977f7f5Sdrh ** wildcard that matches anything. Likewise if pEList==NULL then 474c977f7f5Sdrh ** it matches anything so always return true. Return false only 475c977f7f5Sdrh ** if there is no match. 476c977f7f5Sdrh */ 477c977f7f5Sdrh static int checkColumnOverLap(IdList *pIdList, ExprList *pEList){ 478ad2d8307Sdrh int e; 479ad2d8307Sdrh if( !pIdList || !pEList ) return 1; 480f29ce559Sdanielk1977 for(e=0; e<pEList->nExpr; e++){ 481ad2d8307Sdrh if( sqliteIdListIndex(pIdList, pEList->a[e].zName)>=0 ) return 1; 482f29ce559Sdanielk1977 } 483c3f9bad2Sdanielk1977 return 0; 484c3f9bad2Sdanielk1977 } 485c3f9bad2Sdanielk1977 486c3f9bad2Sdanielk1977 /* A global variable that is TRUE if we should always set up temp tables for 487c3f9bad2Sdanielk1977 * for triggers, even if there are no triggers to code. This is used to test 488c3f9bad2Sdanielk1977 * how much overhead the triggers algorithm is causing. 489c3f9bad2Sdanielk1977 * 490c3f9bad2Sdanielk1977 * This flag can be set or cleared using the "trigger_overhead_test" pragma. 491c3f9bad2Sdanielk1977 * The pragma is not documented since it is not really part of the interface 492c3f9bad2Sdanielk1977 * to SQLite, just the test procedure. 493c3f9bad2Sdanielk1977 */ 494c3f9bad2Sdanielk1977 int always_code_trigger_setup = 0; 495c3f9bad2Sdanielk1977 496c3f9bad2Sdanielk1977 /* 497c3f9bad2Sdanielk1977 * Returns true if a trigger matching op, tr_tm and foreach that is NOT already 498c3f9bad2Sdanielk1977 * on the Parse objects trigger-stack (to prevent recursive trigger firing) is 499c3f9bad2Sdanielk1977 * found in the list specified as pTrigger. 500c3f9bad2Sdanielk1977 */ 501c3f9bad2Sdanielk1977 int sqliteTriggersExist( 502c977f7f5Sdrh Parse *pParse, /* Used to check for recursive triggers */ 503c977f7f5Sdrh Trigger *pTrigger, /* A list of triggers associated with a table */ 504c3f9bad2Sdanielk1977 int op, /* one of TK_DELETE, TK_INSERT, TK_UPDATE */ 505c3f9bad2Sdanielk1977 int tr_tm, /* one of TK_BEFORE, TK_AFTER */ 506c3f9bad2Sdanielk1977 int foreach, /* one of TK_ROW or TK_STATEMENT */ 507c977f7f5Sdrh ExprList *pChanges /* Columns that change in an UPDATE statement */ 508c977f7f5Sdrh ){ 509633ed08dSdanielk1977 Trigger * pTriggerCursor; 510c3f9bad2Sdanielk1977 511633ed08dSdanielk1977 if( always_code_trigger_setup ){ 512633ed08dSdanielk1977 return 1; 513633ed08dSdanielk1977 } 514c3f9bad2Sdanielk1977 515633ed08dSdanielk1977 pTriggerCursor = pTrigger; 516633ed08dSdanielk1977 while( pTriggerCursor ){ 517633ed08dSdanielk1977 if( pTriggerCursor->op == op && 518633ed08dSdanielk1977 pTriggerCursor->tr_tm == tr_tm && 519633ed08dSdanielk1977 pTriggerCursor->foreach == foreach && 520633ed08dSdanielk1977 checkColumnOverLap(pTriggerCursor->pColumns, pChanges) ){ 521c3f9bad2Sdanielk1977 TriggerStack * ss; 522c3f9bad2Sdanielk1977 ss = pParse->trigStack; 523f29ce559Sdanielk1977 while( ss && ss->pTrigger != pTrigger ){ 524f29ce559Sdanielk1977 ss = ss->pNext; 525f29ce559Sdanielk1977 } 526c3f9bad2Sdanielk1977 if( !ss )return 1; 527c3f9bad2Sdanielk1977 } 528633ed08dSdanielk1977 pTriggerCursor = pTriggerCursor->pNext; 529c3f9bad2Sdanielk1977 } 530c3f9bad2Sdanielk1977 531c3f9bad2Sdanielk1977 return 0; 532c3f9bad2Sdanielk1977 } 533c3f9bad2Sdanielk1977 534c977f7f5Sdrh /* 535c977f7f5Sdrh ** Generate VDBE code for zero or more statements inside the body of a 536c977f7f5Sdrh ** trigger. 537c977f7f5Sdrh */ 538c3f9bad2Sdanielk1977 static int codeTriggerProgram( 539c977f7f5Sdrh Parse *pParse, /* The parser context */ 540c977f7f5Sdrh TriggerStep *pStepList, /* List of statements inside the trigger body */ 541c977f7f5Sdrh int orconfin /* Conflict algorithm. (OE_Abort, etc) */ 542633ed08dSdanielk1977 ){ 543633ed08dSdanielk1977 TriggerStep * pTriggerStep = pStepList; 544c3f9bad2Sdanielk1977 int orconf; 545c3f9bad2Sdanielk1977 546633ed08dSdanielk1977 while( pTriggerStep ){ 547c3f9bad2Sdanielk1977 int saveNTab = pParse->nTab; 548a69d9168Sdrh int saveUseDb = pParse->useDb; 549633ed08dSdanielk1977 orconf = (orconfin == OE_Default)?pTriggerStep->orconf:orconfin; 550c3f9bad2Sdanielk1977 pParse->trigStack->orconf = orconf; 551a69d9168Sdrh pParse->useDb = pTriggerStep->pTrig->iDb; 552f0f258b1Sdrh if( pParse->useDb==1 ) pParse->useDb = -1; 553633ed08dSdanielk1977 switch( pTriggerStep->op ){ 554c3f9bad2Sdanielk1977 case TK_SELECT: { 5556f34903eSdanielk1977 Select * ss = sqliteSelectDup(pTriggerStep->pSelect); 5566f34903eSdanielk1977 assert(ss); 5576f34903eSdanielk1977 assert(ss->pSrc); 5586f34903eSdanielk1977 sqliteSelect(pParse, ss, SRT_Discard, 0, 0, 0, 0); 5596f34903eSdanielk1977 sqliteSelectDelete(ss); 560c3f9bad2Sdanielk1977 break; 561c3f9bad2Sdanielk1977 } 562c3f9bad2Sdanielk1977 case TK_UPDATE: { 563113088ecSdrh SrcList *pSrc; 564113088ecSdrh pSrc = sqliteSrcListAppend(0, &pTriggerStep->target, 0); 565bd5a451dSdrh sqliteVdbeAddOp(pParse->pVdbe, OP_ListPush, 0, 0); 566113088ecSdrh sqliteUpdate(pParse, pSrc, 567633ed08dSdanielk1977 sqliteExprListDup(pTriggerStep->pExprList), 568633ed08dSdanielk1977 sqliteExprDup(pTriggerStep->pWhere), orconf); 569bd5a451dSdrh sqliteVdbeAddOp(pParse->pVdbe, OP_ListPop, 0, 0); 570c3f9bad2Sdanielk1977 break; 571c3f9bad2Sdanielk1977 } 572c3f9bad2Sdanielk1977 case TK_INSERT: { 573113088ecSdrh SrcList *pSrc; 574113088ecSdrh pSrc = sqliteSrcListAppend(0, &pTriggerStep->target, 0); 575113088ecSdrh sqliteInsert(pParse, pSrc, 576633ed08dSdanielk1977 sqliteExprListDup(pTriggerStep->pExprList), 577633ed08dSdanielk1977 sqliteSelectDup(pTriggerStep->pSelect), 578633ed08dSdanielk1977 sqliteIdListDup(pTriggerStep->pIdList), orconf); 579c3f9bad2Sdanielk1977 break; 580c3f9bad2Sdanielk1977 } 581c3f9bad2Sdanielk1977 case TK_DELETE: { 582113088ecSdrh SrcList *pSrc; 583bd5a451dSdrh sqliteVdbeAddOp(pParse->pVdbe, OP_ListPush, 0, 0); 584113088ecSdrh pSrc = sqliteSrcListAppend(0, &pTriggerStep->target, 0); 585113088ecSdrh sqliteDeleteFrom(pParse, pSrc, sqliteExprDup(pTriggerStep->pWhere)); 586bd5a451dSdrh sqliteVdbeAddOp(pParse->pVdbe, OP_ListPop, 0, 0); 587c3f9bad2Sdanielk1977 break; 588c3f9bad2Sdanielk1977 } 589c3f9bad2Sdanielk1977 default: 590c3f9bad2Sdanielk1977 assert(0); 591c3f9bad2Sdanielk1977 } 592c3f9bad2Sdanielk1977 pParse->nTab = saveNTab; 593a69d9168Sdrh pParse->useDb = saveUseDb; 594633ed08dSdanielk1977 pTriggerStep = pTriggerStep->pNext; 595c3f9bad2Sdanielk1977 } 596c3f9bad2Sdanielk1977 597c3f9bad2Sdanielk1977 return 0; 598c3f9bad2Sdanielk1977 } 599c3f9bad2Sdanielk1977 600633ed08dSdanielk1977 /* 601633ed08dSdanielk1977 ** This is called to code FOR EACH ROW triggers. 602633ed08dSdanielk1977 ** 603633ed08dSdanielk1977 ** When the code that this function generates is executed, the following 604633ed08dSdanielk1977 ** must be true: 605c977f7f5Sdrh ** 606c977f7f5Sdrh ** 1. No cursors may be open in the main database. (But newIdx and oldIdx 607c977f7f5Sdrh ** can be indices of cursors in temporary tables. See below.) 608c977f7f5Sdrh ** 609633ed08dSdanielk1977 ** 2. If the triggers being coded are ON INSERT or ON UPDATE triggers, then 610633ed08dSdanielk1977 ** a temporary vdbe cursor (index newIdx) must be open and pointing at 611633ed08dSdanielk1977 ** a row containing values to be substituted for new.* expressions in the 612633ed08dSdanielk1977 ** trigger program(s). 613c977f7f5Sdrh ** 614633ed08dSdanielk1977 ** 3. If the triggers being coded are ON DELETE or ON UPDATE triggers, then 615633ed08dSdanielk1977 ** a temporary vdbe cursor (index oldIdx) must be open and pointing at 616633ed08dSdanielk1977 ** a row containing values to be substituted for old.* expressions in the 617633ed08dSdanielk1977 ** trigger program(s). 618633ed08dSdanielk1977 ** 619633ed08dSdanielk1977 */ 620c3f9bad2Sdanielk1977 int sqliteCodeRowTrigger( 621c3f9bad2Sdanielk1977 Parse *pParse, /* Parse context */ 622c3f9bad2Sdanielk1977 int op, /* One of TK_UPDATE, TK_INSERT, TK_DELETE */ 623633ed08dSdanielk1977 ExprList *pChanges, /* Changes list for any UPDATE OF triggers */ 624c3f9bad2Sdanielk1977 int tr_tm, /* One of TK_BEFORE, TK_AFTER */ 625633ed08dSdanielk1977 Table *pTab, /* The table to code triggers from */ 626633ed08dSdanielk1977 int newIdx, /* The indice of the "new" row to access */ 627633ed08dSdanielk1977 int oldIdx, /* The indice of the "old" row to access */ 6286f34903eSdanielk1977 int orconf, /* ON CONFLICT policy */ 6296f34903eSdanielk1977 int ignoreJump /* Instruction to jump to for RAISE(IGNORE) */ 630c977f7f5Sdrh ){ 631c3f9bad2Sdanielk1977 Trigger * pTrigger; 632c3f9bad2Sdanielk1977 TriggerStack * pTriggerStack; 633c3f9bad2Sdanielk1977 634c3f9bad2Sdanielk1977 assert(op == TK_UPDATE || op == TK_INSERT || op == TK_DELETE); 635c3f9bad2Sdanielk1977 assert(tr_tm == TK_BEFORE || tr_tm == TK_AFTER ); 636c3f9bad2Sdanielk1977 637633ed08dSdanielk1977 assert(newIdx != -1 || oldIdx != -1); 638c3f9bad2Sdanielk1977 639633ed08dSdanielk1977 pTrigger = pTab->pTrigger; 640c3f9bad2Sdanielk1977 while( pTrigger ){ 641c3f9bad2Sdanielk1977 int fire_this = 0; 642c3f9bad2Sdanielk1977 643c3f9bad2Sdanielk1977 /* determine whether we should code this trigger */ 644c3f9bad2Sdanielk1977 if( pTrigger->op == op && pTrigger->tr_tm == tr_tm && 645c3f9bad2Sdanielk1977 pTrigger->foreach == TK_ROW ){ 646c3f9bad2Sdanielk1977 fire_this = 1; 647c3f9bad2Sdanielk1977 pTriggerStack = pParse->trigStack; 648c3f9bad2Sdanielk1977 while( pTriggerStack ){ 649f29ce559Sdanielk1977 if( pTriggerStack->pTrigger == pTrigger ){ 650f29ce559Sdanielk1977 fire_this = 0; 651f29ce559Sdanielk1977 } 652c3f9bad2Sdanielk1977 pTriggerStack = pTriggerStack->pNext; 653c3f9bad2Sdanielk1977 } 654c3f9bad2Sdanielk1977 if( op == TK_UPDATE && pTrigger->pColumns && 655f29ce559Sdanielk1977 !checkColumnOverLap(pTrigger->pColumns, pChanges) ){ 656c3f9bad2Sdanielk1977 fire_this = 0; 657c3f9bad2Sdanielk1977 } 658f29ce559Sdanielk1977 } 659c3f9bad2Sdanielk1977 660e4697f5eSdrh if( fire_this && (pTriggerStack = sqliteMalloc(sizeof(TriggerStack)))!=0 ){ 661c3f9bad2Sdanielk1977 int endTrigger; 662ad3cab52Sdrh SrcList dummyTablist; 663c3f9bad2Sdanielk1977 Expr * whenExpr; 66485e2096fSdrh AuthContext sContext; 665c3f9bad2Sdanielk1977 666ad3cab52Sdrh dummyTablist.nSrc = 0; 667c3f9bad2Sdanielk1977 668c3f9bad2Sdanielk1977 /* Push an entry on to the trigger stack */ 669c3f9bad2Sdanielk1977 pTriggerStack->pTrigger = pTrigger; 670633ed08dSdanielk1977 pTriggerStack->newIdx = newIdx; 671633ed08dSdanielk1977 pTriggerStack->oldIdx = oldIdx; 672633ed08dSdanielk1977 pTriggerStack->pTab = pTab; 673c3f9bad2Sdanielk1977 pTriggerStack->pNext = pParse->trigStack; 6746f34903eSdanielk1977 pTriggerStack->ignoreJump = ignoreJump; 675c3f9bad2Sdanielk1977 pParse->trigStack = pTriggerStack; 67685e2096fSdrh sqliteAuthContextPush(pParse, &sContext, pTrigger->name); 677c3f9bad2Sdanielk1977 678c3f9bad2Sdanielk1977 /* code the WHEN clause */ 679c3f9bad2Sdanielk1977 endTrigger = sqliteVdbeMakeLabel(pParse->pVdbe); 680c3f9bad2Sdanielk1977 whenExpr = sqliteExprDup(pTrigger->pWhen); 681*6a3ea0e6Sdrh if( sqliteExprResolveIds(pParse, &dummyTablist, 0, whenExpr) ){ 682c3f9bad2Sdanielk1977 pParse->trigStack = pParse->trigStack->pNext; 683c3f9bad2Sdanielk1977 sqliteFree(pTriggerStack); 684c3f9bad2Sdanielk1977 sqliteExprDelete(whenExpr); 685c3f9bad2Sdanielk1977 return 1; 686c3f9bad2Sdanielk1977 } 687f5905aa7Sdrh sqliteExprIfFalse(pParse, whenExpr, endTrigger, 1); 688c3f9bad2Sdanielk1977 sqliteExprDelete(whenExpr); 689c3f9bad2Sdanielk1977 690633ed08dSdanielk1977 codeTriggerProgram(pParse, pTrigger->step_list, orconf); 691c3f9bad2Sdanielk1977 692c3f9bad2Sdanielk1977 /* Pop the entry off the trigger stack */ 693c3f9bad2Sdanielk1977 pParse->trigStack = pParse->trigStack->pNext; 69485e2096fSdrh sqliteAuthContextPop(&sContext); 695c3f9bad2Sdanielk1977 sqliteFree(pTriggerStack); 696c3f9bad2Sdanielk1977 697c3f9bad2Sdanielk1977 sqliteVdbeResolveLabel(pParse->pVdbe, endTrigger); 698c3f9bad2Sdanielk1977 } 699c3f9bad2Sdanielk1977 pTrigger = pTrigger->pNext; 700c3f9bad2Sdanielk1977 } 701c3f9bad2Sdanielk1977 702c3f9bad2Sdanielk1977 return 0; 703c3f9bad2Sdanielk1977 } 704