亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频

? 歡迎來到蟲蟲下載站! | ?? 資源下載 ?? 資源專輯 ?? 關于我們
? 蟲蟲下載站

?? trigger.c

?? 調用sqlite開源數據的小程序
?? C
?? 第 1 頁 / 共 2 頁
字號:
/***** 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 "sqliteInt.h"#ifndef SQLITE_OMIT_TRIGGER/*** Delete a linked list of TriggerStep structures.*/void sqlite3DeleteTriggerStep(TriggerStep *pTriggerStep){  while( pTriggerStep ){    TriggerStep * pTmp = pTriggerStep;    pTriggerStep = pTriggerStep->pNext;    if( pTmp->target.dyn ) sqliteFree((char*)pTmp->target.z);    sqlite3ExprDelete(pTmp->pWhere);    sqlite3ExprListDelete(pTmp->pExprList);    sqlite3SelectDelete(pTmp->pSelect);    sqlite3IdListDelete(pTmp->pIdList);    sqliteFree(pTmp);  }}/*** This is called by the parser when it sees a CREATE TRIGGER statement** up to the point of the BEGIN before the trigger actions.  A Trigger** structure is generated based on the information available and stored** in pParse->pNewTrigger.  After the trigger actions have been parsed, the** sqlite3FinishTrigger() function is called to complete the trigger** construction process.*/void sqlite3BeginTrigger(  Parse *pParse,      /* The parse context of the CREATE TRIGGER statement */  Token *pName1,      /* The name of the trigger */  Token *pName2,      /* The name of the trigger */  int tr_tm,          /* One of TK_BEFORE, TK_AFTER, TK_INSTEAD */  int op,             /* One of TK_INSERT, TK_UPDATE, TK_DELETE */  IdList *pColumns,   /* column list if this is an UPDATE OF trigger */  SrcList *pTableName,/* The name of the table/view the trigger applies to */  int foreach,        /* One of TK_ROW or TK_STATEMENT */  Expr *pWhen,        /* WHEN clause */  int isTemp          /* True if the TEMPORARY keyword is present */){  Trigger *pTrigger = 0;  Table *pTab;  char *zName = 0;        /* Name of the trigger */  sqlite3 *db = pParse->db;  int iDb;                /* The database to store the trigger in */  Token *pName;           /* The unqualified db name */  DbFixer sFix;  if( isTemp ){    /* If TEMP was specified, then the trigger name may not be qualified. */    if( pName2 && pName2->n>0 ){      sqlite3ErrorMsg(pParse, "temporary trigger may not have qualified name");      goto trigger_cleanup;    }    iDb = 1;    pName = pName1;  }else{    /* Figure out the db that the the trigger will be created in */    iDb = sqlite3TwoPartName(pParse, pName1, pName2, &pName);    if( iDb<0 ){      goto trigger_cleanup;    }  }  /* If the trigger name was unqualified, and the table is a temp table,  ** then set iDb to 1 to create the trigger in the temporary database.  ** If sqlite3SrcListLookup() returns 0, indicating the table does not  ** exist, the error is caught by the block below.  */  if( !pTableName || sqlite3_malloc_failed ) goto trigger_cleanup;  pTab = sqlite3SrcListLookup(pParse, pTableName);  if( pName2->n==0 && pTab && pTab->iDb==1 ){    iDb = 1;  }  /* Ensure the table name matches database name and that the table exists */  if( sqlite3_malloc_failed ) goto trigger_cleanup;  assert( pTableName->nSrc==1 );  if( sqlite3FixInit(&sFix, pParse, iDb, "trigger", pName) &&       sqlite3FixSrcList(&sFix, pTableName) ){    goto trigger_cleanup;  }  pTab = sqlite3SrcListLookup(pParse, pTableName);  if( !pTab ){    /* The table does not exist. */    goto trigger_cleanup;  }  /* Check that the trigger name is not reserved and that no trigger of the  ** specified name exists */  zName = sqlite3NameFromToken(pName);  if( !zName || SQLITE_OK!=sqlite3CheckObjectName(pParse, zName) ){    goto trigger_cleanup;  }  if( sqlite3HashFind(&(db->aDb[iDb].trigHash), zName,pName->n+1) ){    sqlite3ErrorMsg(pParse, "trigger %T already exists", pName);    goto trigger_cleanup;  }  /* Do not create a trigger on a system table */  if( sqlite3StrNICmp(pTab->zName, "sqlite_", 7)==0 ){    sqlite3ErrorMsg(pParse, "cannot create trigger on system table");    pParse->nErr++;    goto trigger_cleanup;  }  /* INSTEAD of triggers are only for views and views only support INSTEAD  ** of triggers.  */  if( pTab->pSelect && tr_tm!=TK_INSTEAD ){    sqlite3ErrorMsg(pParse, "cannot create %s trigger on view: %S",         (tr_tm == TK_BEFORE)?"BEFORE":"AFTER", pTableName, 0);    goto trigger_cleanup;  }  if( !pTab->pSelect && tr_tm==TK_INSTEAD ){    sqlite3ErrorMsg(pParse, "cannot create INSTEAD OF"        " trigger on table: %S", pTableName, 0);    goto trigger_cleanup;  }#ifndef SQLITE_OMIT_AUTHORIZATION  {    int code = SQLITE_CREATE_TRIGGER;    const char *zDb = db->aDb[pTab->iDb].zName;    const char *zDbTrig = isTemp ? db->aDb[1].zName : zDb;    if( pTab->iDb==1 || isTemp ) code = SQLITE_CREATE_TEMP_TRIGGER;    if( sqlite3AuthCheck(pParse, code, zName, pTab->zName, zDbTrig) ){      goto trigger_cleanup;    }    if( sqlite3AuthCheck(pParse, SQLITE_INSERT, SCHEMA_TABLE(pTab->iDb),0,zDb)){      goto trigger_cleanup;    }  }#endif  /* INSTEAD OF triggers can only appear on views and BEFORE triggers  ** cannot appear on views.  So we might as well translate every  ** INSTEAD OF trigger into a BEFORE trigger.  It simplifies code  ** elsewhere.  */  if (tr_tm == TK_INSTEAD){    tr_tm = TK_BEFORE;  }  /* Build the Trigger object */  pTrigger = (Trigger*)sqliteMalloc(sizeof(Trigger));  if( pTrigger==0 ) goto trigger_cleanup;  pTrigger->name = zName;  zName = 0;  pTrigger->table = sqliteStrDup(pTableName->a[0].zName);  pTrigger->iDb = iDb;  pTrigger->iTabDb = pTab->iDb;  pTrigger->op = op;  pTrigger->tr_tm = tr_tm==TK_BEFORE ? TRIGGER_BEFORE : TRIGGER_AFTER;  pTrigger->pWhen = sqlite3ExprDup(pWhen);  pTrigger->pColumns = sqlite3IdListDup(pColumns);  pTrigger->foreach = foreach;  sqlite3TokenCopy(&pTrigger->nameToken,pName);  assert( pParse->pNewTrigger==0 );  pParse->pNewTrigger = pTrigger;trigger_cleanup:  sqliteFree(zName);  sqlite3SrcListDelete(pTableName);  sqlite3IdListDelete(pColumns);  sqlite3ExprDelete(pWhen);  if( !pParse->pNewTrigger ){    sqlite3DeleteTrigger(pTrigger);  }else{    assert( pParse->pNewTrigger==pTrigger );  }}/*** This routine is called after all of the trigger actions have been parsed** in order to complete the process of building the trigger.*/void sqlite3FinishTrigger(  Parse *pParse,          /* Parser context */  TriggerStep *pStepList, /* The triggered program */  Token *pAll             /* Token that describes the complete CREATE TRIGGER */){  Trigger *pTrig = 0;     /* The trigger whose construction is finishing up */  sqlite3 *db = pParse->db;  /* The database */  DbFixer sFix;  pTrig = pParse->pNewTrigger;  pParse->pNewTrigger = 0;  if( pParse->nErr || pTrig==0 ) goto triggerfinish_cleanup;  pTrig->step_list = pStepList;  while( pStepList ){    pStepList->pTrig = pTrig;    pStepList = pStepList->pNext;  }  if( sqlite3FixInit(&sFix, pParse, pTrig->iDb, "trigger", &pTrig->nameToken)           && sqlite3FixTriggerStep(&sFix, pTrig->step_list) ){    goto triggerfinish_cleanup;  }  /* if we are not initializing, and this trigger is not on a TEMP table,   ** build the sqlite_master entry  */  if( !db->init.busy ){    static const VdbeOpList insertTrig[] = {      { OP_NewRowid,   0, 0,  0          },      { OP_String8,    0, 0,  "trigger"  },      { OP_String8,    0, 0,  0          },  /* 2: trigger name */      { OP_String8,    0, 0,  0          },  /* 3: table name */      { OP_Integer,    0, 0,  0          },      { OP_String8,    0, 0,  "CREATE TRIGGER "},      { OP_String8,    0, 0,  0          },  /* 6: SQL */      { OP_Concat,     0, 0,  0          },       { OP_MakeRecord, 5, 0,  "tttit"    },      { OP_Insert,     0, 0,  0          },    };    int addr;    Vdbe *v;    /* Make an entry in the sqlite_master table */    v = sqlite3GetVdbe(pParse);    if( v==0 ) goto triggerfinish_cleanup;    sqlite3BeginWriteOperation(pParse, 0, pTrig->iDb);    sqlite3OpenMasterTable(v, pTrig->iDb);    addr = sqlite3VdbeAddOpList(v, ArraySize(insertTrig), insertTrig);    sqlite3VdbeChangeP3(v, addr+2, pTrig->name, 0);     sqlite3VdbeChangeP3(v, addr+3, pTrig->table, 0);     sqlite3VdbeChangeP3(v, addr+6, pAll->z, pAll->n);    sqlite3ChangeCookie(db, v, pTrig->iDb);    sqlite3VdbeAddOp(v, OP_Close, 0, 0);    sqlite3VdbeOp3(v, OP_ParseSchema, pTrig->iDb, 0,        sqlite3MPrintf("type='trigger' AND name='%q'", pTrig->name), P3_DYNAMIC);  }  if( db->init.busy ){    Table *pTab;    Trigger *pDel;    pDel = sqlite3HashInsert(&db->aDb[pTrig->iDb].trigHash,                      pTrig->name, strlen(pTrig->name)+1, pTrig);    if( pDel ){      assert( sqlite3_malloc_failed && pDel==pTrig );      goto triggerfinish_cleanup;    }    pTab = sqlite3LocateTable(pParse,pTrig->table,db->aDb[pTrig->iTabDb].zName);    assert( pTab!=0 );    pTrig->pNext = pTab->pTrigger;    pTab->pTrigger = pTrig;    pTrig = 0;  }triggerfinish_cleanup:  sqlite3DeleteTrigger(pTrig);  assert( !pParse->pNewTrigger );  sqlite3DeleteTriggerStep(pStepList);}/*** Make a copy of all components of the given trigger step.  This has** the effect of copying all Expr.token.z values into memory obtained** from sqliteMalloc().  As initially created, the Expr.token.z values** all point to the input string that was fed to the parser.  But that** string is ephemeral - it will go away as soon as the sqlite3_exec()** call that started the parser exits.  This routine makes a persistent** copy of all the Expr.token.z strings so that the TriggerStep structure** will be valid even after the sqlite3_exec() call returns.*/static void sqlitePersistTriggerStep(TriggerStep *p){  if( p->target.z ){    p->target.z = sqliteStrNDup(p->target.z, p->target.n);    p->target.dyn = 1;  }  if( p->pSelect ){    Select *pNew = sqlite3SelectDup(p->pSelect);    sqlite3SelectDelete(p->pSelect);    p->pSelect = pNew;  }  if( p->pWhere ){    Expr *pNew = sqlite3ExprDup(p->pWhere);    sqlite3ExprDelete(p->pWhere);    p->pWhere = pNew;  }  if( p->pExprList ){    ExprList *pNew = sqlite3ExprListDup(p->pExprList);    sqlite3ExprListDelete(p->pExprList);    p->pExprList = pNew;  }  if( p->pIdList ){    IdList *pNew = sqlite3IdListDup(p->pIdList);    sqlite3IdListDelete(p->pIdList);    p->pIdList = pNew;  }}/*** Turn a SELECT statement (that the pSelect parameter points to) into** a trigger step.  Return a pointer to a TriggerStep structure.**** The parser calls this routine when it finds a SELECT statement in** body of a TRIGGER.  */TriggerStep *sqlite3TriggerSelectStep(Select *pSelect){  TriggerStep *pTriggerStep = sqliteMalloc(sizeof(TriggerStep));  if( pTriggerStep==0 ) return 0;  pTriggerStep->op = TK_SELECT;  pTriggerStep->pSelect = pSelect;  pTriggerStep->orconf = OE_Default;  sqlitePersistTriggerStep(pTriggerStep);  return pTriggerStep;}/*** Build a trigger step out of an INSERT statement.  Return a pointer** to the new trigger step.**** The parser calls this routine when it sees an INSERT inside the** body of a trigger.*/TriggerStep *sqlite3TriggerInsertStep(  Token *pTableName,  /* Name of the table into which we insert */  IdList *pColumn,    /* List of columns in pTableName to insert into */  ExprList *pEList,   /* The VALUE clause: a list of values to be inserted */  Select *pSelect,    /* A SELECT statement that supplies values */  int orconf          /* The conflict algorithm (OE_Abort, OE_Replace, etc.) */){  TriggerStep *pTriggerStep = sqliteMalloc(sizeof(TriggerStep));  assert(pEList == 0 || pSelect == 0);  assert(pEList != 0 || pSelect != 0);  if( pTriggerStep ){    pTriggerStep->op = TK_INSERT;    pTriggerStep->pSelect = pSelect;    pTriggerStep->target  = *pTableName;    pTriggerStep->pIdList = pColumn;    pTriggerStep->pExprList = pEList;    pTriggerStep->orconf = orconf;    sqlitePersistTriggerStep(pTriggerStep);  }else{    sqlite3IdListDelete(pColumn);    sqlite3ExprListDelete(pEList);    sqlite3SelectDup(pSelect);  }  return pTriggerStep;}/*** Construct a trigger step that implements an UPDATE statement and return** a pointer to that trigger step.  The parser calls this routine when it** sees an UPDATE statement inside the body of a CREATE TRIGGER.*/TriggerStep *sqlite3TriggerUpdateStep(  Token *pTableName,   /* Name of the table to be updated */  ExprList *pEList,    /* The SET clause: list of column and new values */  Expr *pWhere,        /* The WHERE clause */  int orconf           /* The conflict algorithm. (OE_Abort, OE_Ignore, etc) */){  TriggerStep *pTriggerStep = sqliteMalloc(sizeof(TriggerStep));  if( pTriggerStep==0 ) return 0;  pTriggerStep->op = TK_UPDATE;  pTriggerStep->target  = *pTableName;  pTriggerStep->pExprList = pEList;  pTriggerStep->pWhere = pWhere;  pTriggerStep->orconf = orconf;  sqlitePersistTriggerStep(pTriggerStep);  return pTriggerStep;}/*** Construct a trigger step that implements a DELETE statement and return** a pointer to that trigger step.  The parser calls this routine when it** sees a DELETE statement inside the body of a CREATE TRIGGER.*/TriggerStep *sqlite3TriggerDeleteStep(Token *pTableName, Expr *pWhere){  TriggerStep *pTriggerStep = sqliteMalloc(sizeof(TriggerStep));  if( pTriggerStep==0 ) return 0;  pTriggerStep->op = TK_DELETE;  pTriggerStep->target  = *pTableName;  pTriggerStep->pWhere = pWhere;  pTriggerStep->orconf = OE_Default;  sqlitePersistTriggerStep(pTriggerStep);  return pTriggerStep;}

?? 快捷鍵說明

復制代碼 Ctrl + C
搜索代碼 Ctrl + F
全屏模式 F11
切換主題 Ctrl + Shift + D
顯示快捷鍵 ?
增大字號 Ctrl + =
減小字號 Ctrl + -
亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频
麻豆精品新av中文字幕| 国产精品国产馆在线真实露脸| 一区二区三区中文字幕| 97久久精品人人爽人人爽蜜臀| 中国色在线观看另类| 不卡影院免费观看| 亚洲精品日韩专区silk| 欧美视频在线一区| 色综合网站在线| 一区二区三区四区视频精品免费| 欧美日韩午夜在线视频| 美腿丝袜亚洲三区| 国产清纯白嫩初高生在线观看91| 不卡av在线免费观看| 亚洲国产精品一区二区久久| 欧美一区二区三区啪啪| 国产成人精品亚洲午夜麻豆| 中文字幕字幕中文在线中不卡视频| 欧美午夜寂寞影院| 激情成人综合网| 亚洲手机成人高清视频| 欧美一区二区久久| 99re这里只有精品首页| 婷婷久久综合九色综合绿巨人| 精品少妇一区二区三区免费观看| www.色综合.com| 手机精品视频在线观看| 国产女同互慰高潮91漫画| 在线看国产一区| 极品少妇xxxx精品少妇偷拍| 亚洲女人的天堂| 久久亚区不卡日本| 欧美在线|欧美| 国产精一区二区三区| 亚洲在线中文字幕| 久久精品在这里| 欧美日韩高清不卡| 99热这里都是精品| 极品美女销魂一区二区三区| 一区二区三区精品视频在线| 国产丝袜在线精品| 欧美成人性战久久| 欧美综合视频在线观看| 懂色中文一区二区在线播放| 天天色天天操综合| 一区二区三区四区蜜桃| 亚洲国产精品高清| 欧美v国产在线一区二区三区| 色8久久精品久久久久久蜜| 国产在线视频一区二区| 石原莉奈一区二区三区在线观看| 亚洲色图.com| 中文字幕精品在线不卡| www成人在线观看| 在线不卡免费av| 欧美亚洲日本国产| 972aa.com艺术欧美| 国产成人午夜视频| 久久99久久99小草精品免视看| 五月天国产精品| 亚洲伊人色欲综合网| 中文字幕在线一区二区三区| 久久先锋影音av鲁色资源网| 日韩欧美aaaaaa| 欧美一区日本一区韩国一区| 欧美午夜不卡视频| 在线视频一区二区三区| 日本韩国欧美国产| 91农村精品一区二区在线| 成人国产精品免费观看动漫| 粉嫩av亚洲一区二区图片| 成人午夜伦理影院| 成人丝袜视频网| 99视频超级精品| 一本到不卡精品视频在线观看| av亚洲精华国产精华| 91免费版在线看| 久久伊99综合婷婷久久伊| 久久影音资源网| 欧美激情中文字幕| 综合av第一页| 亚洲综合免费观看高清在线观看| 亚洲一二三四区不卡| 亚洲国产综合91精品麻豆| 午夜精品久久久久久久久| 日韩不卡手机在线v区| 日韩av一区二区在线影视| 久久激情五月婷婷| 国产精一品亚洲二区在线视频| 国产成人综合自拍| 成人国产亚洲欧美成人综合网| 色综合久久久久久久久| 欧美日韩中字一区| 欧美一区日本一区韩国一区| www国产亚洲精品久久麻豆| 中文字幕乱码亚洲精品一区| 亚洲欧美日韩在线不卡| 亚洲高清视频在线| 国产一区二区三区美女| av在线一区二区三区| 欧美在线视频全部完| 538在线一区二区精品国产| 日韩一区二区电影在线| 欧美国产日产图区| 亚洲国产精品久久久久秋霞影院| 久久精品国产色蜜蜜麻豆| 成人午夜精品在线| 欧美日韩一区二区在线视频| 欧美哺乳videos| 国产精品国产三级国产aⅴ无密码| 一区二区三区免费看视频| 蜜桃视频在线观看一区| 国产成人亚洲综合a∨猫咪| 欧美在线一二三四区| 精品久久人人做人人爰| 亚洲激情av在线| 久久精品国产一区二区三区免费看| www.欧美色图| 日韩欧美一区二区不卡| 亚洲区小说区图片区qvod| 久久激情五月婷婷| 99久久精品国产一区| 欧美一区二区三区日韩| 亚洲天堂久久久久久久| 激情综合色综合久久| 欧美日韩在线播放三区| 国产片一区二区三区| 日日夜夜免费精品| 91麻豆福利精品推荐| 久久久另类综合| 五月综合激情婷婷六月色窝| 成人免费视频国产在线观看| 欧美一区二区成人6969| 亚洲黄色av一区| 高清国产午夜精品久久久久久| 在线播放中文字幕一区| 一区二区三区精品视频| 成人午夜视频免费看| 2019国产精品| 三级欧美在线一区| 日本高清成人免费播放| 国产人妖乱国产精品人妖| 美女视频一区在线观看| 欧美在线免费视屏| 自拍偷拍亚洲综合| 粉嫩aⅴ一区二区三区四区五区 | 国产一区二区三区电影在线观看 | 精品视频资源站| 国产精品免费视频网站| 国内精品久久久久影院色 | 色综合久久六月婷婷中文字幕| 国产日韩欧美综合一区| 理论片日本一区| 欧美一二三在线| 日韩精品成人一区二区三区 | 日韩一区二区三区高清免费看看| 亚洲三级视频在线观看| 国产不卡在线视频| 久久久久国产成人精品亚洲午夜| 日产国产欧美视频一区精品| 欧美三级视频在线| 亚洲成人午夜影院| 在线观看日韩毛片| 亚洲一区二区三区中文字幕| 日本二三区不卡| 亚洲日本在线a| 色哟哟精品一区| 亚洲精品久久嫩草网站秘色| 91丝袜美女网| 亚洲伦理在线精品| 91福利小视频| 亚洲成在线观看| 欧美喷潮久久久xxxxx| 视频一区视频二区中文| 欧美一区二区视频观看视频 | 一区二区成人在线| 在线观看不卡视频| 亚洲国产欧美一区二区三区丁香婷| 色综合咪咪久久| 亚洲愉拍自拍另类高清精品| 欧美性受xxxx黑人xyx| 五月激情丁香一区二区三区| 日韩一级在线观看| 国产一区二区三区蝌蚪| 国产精品久久久久一区二区三区 | 精品成人一区二区| 国产精品资源网站| 中文字幕一区二区不卡| 色八戒一区二区三区| 日本中文一区二区三区| 久久日韩粉嫩一区二区三区| 成人av网站免费观看| 亚洲综合999| 欧美大白屁股肥臀xxxxxx| 国产69精品久久99不卡| 一区二区三区欧美久久| 欧美一级xxx| 99在线精品视频| 日韩av不卡一区二区| 中文幕一区二区三区久久蜜桃|