2017-04-07 23:14:22 +03:00
|
|
|
/*
|
|
|
|
** 2017 April 07
|
|
|
|
**
|
|
|
|
** The author disclaims copyright to this source code. In place of
|
|
|
|
** a legal notice, here is a blessing:
|
|
|
|
**
|
|
|
|
** May you do good and not evil.
|
|
|
|
** May you find forgiveness for yourself and forgive others.
|
|
|
|
** May you share freely, never taking more than you give.
|
|
|
|
**
|
|
|
|
*************************************************************************
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
#include <sqlite3.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include "sqlite3expert.h"
|
|
|
|
|
|
|
|
|
|
|
|
static void option_requires_argument(const char *zOpt){
|
|
|
|
fprintf(stderr, "Option requires an argument: %s\n", zOpt);
|
|
|
|
exit(-3);
|
|
|
|
}
|
|
|
|
|
2017-04-08 21:56:32 +03:00
|
|
|
static int option_integer_arg(const char *zVal){
|
|
|
|
return atoi(zVal);
|
|
|
|
}
|
|
|
|
|
2017-04-07 23:14:22 +03:00
|
|
|
static void usage(char **argv){
|
|
|
|
fprintf(stderr, "\n");
|
|
|
|
fprintf(stderr, "Usage %s ?OPTIONS? DATABASE\n", argv[0]);
|
|
|
|
fprintf(stderr, "\n");
|
|
|
|
fprintf(stderr, "Options are:\n");
|
|
|
|
fprintf(stderr, " -sql SQL (analyze SQL statements passed as argument)\n");
|
|
|
|
fprintf(stderr, " -file FILE (read SQL statements from file FILE)\n");
|
2017-04-08 21:56:32 +03:00
|
|
|
fprintf(stderr, " -verbose LEVEL (integer verbosity level. default 1)\n");
|
2017-04-20 12:54:04 +03:00
|
|
|
fprintf(stderr, " -sample PERCENT (percent of db to sample. default 100)\n");
|
2017-04-07 23:14:22 +03:00
|
|
|
exit(-1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int readSqlFromFile(sqlite3expert *p, const char *zFile, char **pzErr){
|
2017-04-09 11:38:37 +03:00
|
|
|
FILE *in = fopen(zFile, "rb");
|
|
|
|
long nIn;
|
|
|
|
size_t nRead;
|
|
|
|
char *pBuf;
|
|
|
|
int rc;
|
|
|
|
if( in==0 ){
|
|
|
|
*pzErr = sqlite3_mprintf("failed to open file %s\n", zFile);
|
|
|
|
return SQLITE_ERROR;
|
|
|
|
}
|
|
|
|
fseek(in, 0, SEEK_END);
|
|
|
|
nIn = ftell(in);
|
|
|
|
rewind(in);
|
|
|
|
pBuf = sqlite3_malloc64( nIn+1 );
|
|
|
|
nRead = fread(pBuf, nIn, 1, in);
|
|
|
|
fclose(in);
|
|
|
|
if( nRead!=1 ){
|
|
|
|
sqlite3_free(pBuf);
|
|
|
|
*pzErr = sqlite3_mprintf("failed to read file %s\n", zFile);
|
|
|
|
return SQLITE_ERROR;
|
|
|
|
}
|
|
|
|
pBuf[nIn] = 0;
|
|
|
|
rc = sqlite3_expert_sql(p, pBuf, pzErr);
|
|
|
|
sqlite3_free(pBuf);
|
|
|
|
return rc;
|
2017-04-07 23:14:22 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
int main(int argc, char **argv){
|
|
|
|
const char *zDb;
|
|
|
|
int rc = 0;
|
|
|
|
char *zErr = 0;
|
|
|
|
int i;
|
2017-04-08 21:56:32 +03:00
|
|
|
int iVerbose = 1; /* -verbose option */
|
2017-04-07 23:14:22 +03:00
|
|
|
|
|
|
|
sqlite3 *db = 0;
|
|
|
|
sqlite3expert *p = 0;
|
|
|
|
|
|
|
|
if( argc<2 ) usage(argv);
|
|
|
|
zDb = argv[argc-1];
|
2017-05-02 23:42:30 +03:00
|
|
|
if( zDb[0]=='-' ) usage(argv);
|
2017-04-07 23:14:22 +03:00
|
|
|
rc = sqlite3_open(zDb, &db);
|
|
|
|
if( rc!=SQLITE_OK ){
|
|
|
|
fprintf(stderr, "Cannot open db file: %s - %s\n", zDb, sqlite3_errmsg(db));
|
|
|
|
exit(-2);
|
|
|
|
}
|
|
|
|
|
|
|
|
p = sqlite3_expert_new(db, &zErr);
|
|
|
|
if( p==0 ){
|
|
|
|
fprintf(stderr, "Cannot run analysis: %s\n", zErr);
|
|
|
|
rc = 1;
|
|
|
|
}else{
|
|
|
|
for(i=1; i<(argc-1); i++){
|
|
|
|
char *zArg = argv[i];
|
2018-03-07 17:42:17 +03:00
|
|
|
int nArg;
|
2017-05-02 23:42:30 +03:00
|
|
|
if( zArg[0]=='-' && zArg[1]=='-' && zArg[2]!=0 ) zArg++;
|
2018-03-07 17:42:17 +03:00
|
|
|
nArg = (int)strlen(zArg);
|
2017-04-07 23:14:22 +03:00
|
|
|
if( nArg>=2 && 0==sqlite3_strnicmp(zArg, "-file", nArg) ){
|
|
|
|
if( ++i==(argc-1) ) option_requires_argument("-file");
|
|
|
|
rc = readSqlFromFile(p, argv[i], &zErr);
|
|
|
|
}
|
|
|
|
|
2017-04-20 12:54:04 +03:00
|
|
|
else if( nArg>=3 && 0==sqlite3_strnicmp(zArg, "-sql", nArg) ){
|
2017-04-07 23:14:22 +03:00
|
|
|
if( ++i==(argc-1) ) option_requires_argument("-sql");
|
|
|
|
rc = sqlite3_expert_sql(p, argv[i], &zErr);
|
|
|
|
}
|
|
|
|
|
2017-04-20 12:54:04 +03:00
|
|
|
else if( nArg>=3 && 0==sqlite3_strnicmp(zArg, "-sample", nArg) ){
|
|
|
|
int iSample;
|
|
|
|
if( ++i==(argc-1) ) option_requires_argument("-sample");
|
|
|
|
iSample = option_integer_arg(argv[i]);
|
|
|
|
sqlite3_expert_config(p, EXPERT_CONFIG_SAMPLE, iSample);
|
|
|
|
}
|
|
|
|
|
2017-04-08 21:56:32 +03:00
|
|
|
else if( nArg>=2 && 0==sqlite3_strnicmp(zArg, "-verbose", nArg) ){
|
|
|
|
if( ++i==(argc-1) ) option_requires_argument("-verbose");
|
|
|
|
iVerbose = option_integer_arg(argv[i]);
|
|
|
|
}
|
|
|
|
|
2017-04-07 23:14:22 +03:00
|
|
|
else{
|
|
|
|
usage(argv);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if( rc==SQLITE_OK ){
|
|
|
|
rc = sqlite3_expert_analyze(p, &zErr);
|
|
|
|
}
|
|
|
|
|
|
|
|
if( rc==SQLITE_OK ){
|
|
|
|
int nQuery = sqlite3_expert_count(p);
|
2017-04-11 20:43:12 +03:00
|
|
|
if( iVerbose>0 ){
|
|
|
|
const char *zCand = sqlite3_expert_report(p,0,EXPERT_REPORT_CANDIDATES);
|
|
|
|
fprintf(stdout, "-- Candidates -------------------------------\n");
|
|
|
|
fprintf(stdout, "%s\n", zCand);
|
|
|
|
}
|
2017-04-07 23:14:22 +03:00
|
|
|
for(i=0; i<nQuery; i++){
|
|
|
|
const char *zSql = sqlite3_expert_report(p, i, EXPERT_REPORT_SQL);
|
|
|
|
const char *zIdx = sqlite3_expert_report(p, i, EXPERT_REPORT_INDEXES);
|
|
|
|
const char *zEQP = sqlite3_expert_report(p, i, EXPERT_REPORT_PLAN);
|
2017-04-11 21:29:14 +03:00
|
|
|
if( zIdx==0 ) zIdx = "(no new indexes)\n";
|
2017-04-08 21:56:32 +03:00
|
|
|
if( iVerbose>0 ){
|
2017-04-11 20:43:12 +03:00
|
|
|
fprintf(stdout, "-- Query %d ----------------------------------\n",i+1);
|
2017-04-08 21:56:32 +03:00
|
|
|
fprintf(stdout, "%s\n\n", zSql);
|
|
|
|
}
|
|
|
|
fprintf(stdout, "%s\n%s\n", zIdx, zEQP);
|
2017-04-07 23:14:22 +03:00
|
|
|
}
|
2017-04-18 12:04:48 +03:00
|
|
|
}else{
|
|
|
|
fprintf(stderr, "Error: %s\n", zErr ? zErr : "?");
|
2017-04-07 23:14:22 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
sqlite3_expert_destroy(p);
|
|
|
|
sqlite3_free(zErr);
|
|
|
|
return rc;
|
|
|
|
}
|