os/persistentdata/persistentstorage/sqlite3api/TEST/TclScript/crashtest1.c
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
sl@0
     1
/*
sl@0
     2
** This program tests the ability of SQLite database to recover from a crash.
sl@0
     3
** This program runs under Unix only, but the results are applicable to all
sl@0
     4
** systems.
sl@0
     5
**
sl@0
     6
** The main process first constructs a test database, then starts creating
sl@0
     7
** subprocesses that write to that database.  Each subprocess is killed off,
sl@0
     8
** without a chance to clean up its database connection, after a random
sl@0
     9
** delay.  This killing of the subprocesses simulates a crash or power
sl@0
    10
** failure.  The next subprocess to open the database should rollback
sl@0
    11
** whatever operation was in process at the time of the simulated crash.
sl@0
    12
**
sl@0
    13
** If any problems are encountered, an error is reported and the test stops.
sl@0
    14
** If no problems are seen after a large number of tests, we assume that
sl@0
    15
** the rollback mechanism is working.
sl@0
    16
*/
sl@0
    17
#include <stdio.h>
sl@0
    18
#include <unistd.h>
sl@0
    19
#include <sys/types.h>
sl@0
    20
#include <sys/wait.h>
sl@0
    21
#include <signal.h>
sl@0
    22
#include <stdlib.h>
sl@0
    23
#include <string.h>
sl@0
    24
#include <sched.h>
sl@0
    25
#include "sqlite.h"
sl@0
    26
sl@0
    27
static void do_some_sql(int parent){
sl@0
    28
  char *zErr;
sl@0
    29
  int rc = SQLITE_OK;
sl@0
    30
  sqlite *db;
sl@0
    31
  int cnt = 0;
sl@0
    32
  static char zBig[] = 
sl@0
    33
    "-abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ"
sl@0
    34
    "-abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ";
sl@0
    35
sl@0
    36
  if( access("./test.db-journal",0)==0 ){
sl@0
    37
    /*printf("pid %d: journal exists.  rollback will be required\n",getpid());*/    unlink("test.db-saved");
sl@0
    38
    system("cp test.db test.db-saved");
sl@0
    39
    unlink("test.db-journal-saved");
sl@0
    40
    system("cp test.db-journal test.db-journal-saved");
sl@0
    41
  }
sl@0
    42
  db = sqlite_open("./test.db", 0, &zErr);
sl@0
    43
  if( db==0 ){
sl@0
    44
    printf("ERROR: %s\n", zErr);
sl@0
    45
    if( strcmp(zErr,"database disk image is malformed")==0 ){
sl@0
    46
      kill(parent, SIGKILL);
sl@0
    47
    }
sl@0
    48
    exit(1);
sl@0
    49
  }
sl@0
    50
  srand(getpid());
sl@0
    51
  while( rc==SQLITE_OK ){
sl@0
    52
    cnt++;
sl@0
    53
    rc = sqlite_exec_printf(db, 
sl@0
    54
       "INSERT INTO t1 VALUES(%d,'%d%s')", 0, 0, &zErr,
sl@0
    55
       rand(), rand(), zBig);
sl@0
    56
  }
sl@0
    57
  if( rc!=SQLITE_OK ){
sl@0
    58
    printf("ERROR #%d: %s\n", rc, zErr);
sl@0
    59
    if( rc==SQLITE_CORRUPT ){
sl@0
    60
      kill(parent, SIGKILL);
sl@0
    61
    }
sl@0
    62
  }
sl@0
    63
  printf("pid %d: cnt=%d\n", getpid(), cnt);
sl@0
    64
}
sl@0
    65
sl@0
    66
sl@0
    67
int main(int argc, char **argv){
sl@0
    68
  int i;
sl@0
    69
  sqlite *db;
sl@0
    70
  char *zErr;
sl@0
    71
  int status;
sl@0
    72
  int parent = getpid();
sl@0
    73
sl@0
    74
  unlink("test.db");
sl@0
    75
  unlink("test.db-journal");
sl@0
    76
  db = sqlite_open("test.db", 0, &zErr);
sl@0
    77
  if( db==0 ){
sl@0
    78
    printf("Cannot initialize: %s\n", zErr);
sl@0
    79
    return 1;
sl@0
    80
  }
sl@0
    81
  sqlite_exec(db, "CREATE TABLE t1(a,b)", 0, 0, 0);
sl@0
    82
  sqlite_close(db);
sl@0
    83
  for(i=0; i<10000; i++){
sl@0
    84
    int pid = fork();
sl@0
    85
    if( pid==0 ){
sl@0
    86
      sched_yield();
sl@0
    87
      do_some_sql(parent);
sl@0
    88
      return 0;
sl@0
    89
    }
sl@0
    90
    printf("test %d, pid=%d\n", i, pid);
sl@0
    91
    usleep(rand()%10000 + 1000);
sl@0
    92
    kill(pid, SIGKILL);
sl@0
    93
    waitpid(pid, &status, 0);
sl@0
    94
  }
sl@0
    95
  return 0;
sl@0
    96
}