mirror of
				https://github.com/sqlite/sqlite.git
				synced 2025-10-25 20:58:26 +03:00 
			
		
		
		
	
		
			
				
	
	
		
			1133 lines
		
	
	
		
			32 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			1133 lines
		
	
	
		
			32 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
| ** 2014-06-13
 | |
| **
 | |
| ** 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.
 | |
| **
 | |
| ******************************************************************************
 | |
| **
 | |
| ** This SQLite extension implements SQL functions readfile() and
 | |
| ** writefile(), and eponymous virtual type "fsdir".
 | |
| **
 | |
| ** WRITEFILE(FILE, DATA [, MODE [, MTIME]]):
 | |
| **
 | |
| **   If neither of the optional arguments is present, then this UDF
 | |
| **   function writes blob DATA to file FILE. If successful, the number
 | |
| **   of bytes written is returned. If an error occurs, NULL is returned.
 | |
| **
 | |
| **   If the first option argument - MODE - is present, then it must
 | |
| **   be passed an integer value that corresponds to a POSIX mode
 | |
| **   value (file type + permissions, as returned in the stat.st_mode
 | |
| **   field by the stat() system call). Three types of files may
 | |
| **   be written/created:
 | |
| **
 | |
| **     regular files:  (mode & 0170000)==0100000
 | |
| **     symbolic links: (mode & 0170000)==0120000
 | |
| **     directories:    (mode & 0170000)==0040000
 | |
| **
 | |
| **   For a directory, the DATA is ignored. For a symbolic link, it is
 | |
| **   interpreted as text and used as the target of the link. For a
 | |
| **   regular file, it is interpreted as a blob and written into the
 | |
| **   named file. Regardless of the type of file, its permissions are
 | |
| **   set to (mode & 0777) before returning.
 | |
| **
 | |
| **   If the optional MTIME argument is present, then it is interpreted
 | |
| **   as an integer - the number of seconds since the unix epoch. The
 | |
| **   modification-time of the target file is set to this value before
 | |
| **   returning.
 | |
| **
 | |
| **   If five or more arguments are passed to this function and an
 | |
| **   error is encountered, an exception is raised.
 | |
| **
 | |
| ** READFILE(FILE):
 | |
| **
 | |
| **   Read and return the contents of file FILE (type blob) from disk.
 | |
| **
 | |
| ** FSDIR:
 | |
| **
 | |
| **   Used as follows:
 | |
| **
 | |
| **     SELECT * FROM fsdir($path [, $dir]);
 | |
| **
 | |
| **   Parameter $path is an absolute or relative pathname. If the file that it
 | |
| **   refers to does not exist, it is an error. If the path refers to a regular
 | |
| **   file or symbolic link, it returns a single row. Or, if the path refers
 | |
| **   to a directory, it returns one row for the directory, and one row for each
 | |
| **   file within the hierarchy rooted at $path.
 | |
| **
 | |
| **   Each row has the following columns:
 | |
| **
 | |
| **     name:  Path to file or directory (text value).
 | |
| **     mode:  Value of stat.st_mode for directory entry (an integer).
 | |
| **     mtime: Value of stat.st_mtime for directory entry (an integer).
 | |
| **     data:  For a regular file, a blob containing the file data. For a
 | |
| **            symlink, a text value containing the text of the link. For a
 | |
| **            directory, NULL.
 | |
| **     level: Directory hierarchy level.  Topmost is 1.
 | |
| **
 | |
| **   If a non-NULL value is specified for the optional $dir parameter and
 | |
| **   $path is a relative path, then $path is interpreted relative to $dir. 
 | |
| **   And the paths returned in the "name" column of the table are also 
 | |
| **   relative to directory $dir.
 | |
| **
 | |
| ** Notes on building this extension for Windows:
 | |
| **   Unless linked statically with the SQLite library, a preprocessor
 | |
| **   symbol, FILEIO_WIN32_DLL, must be #define'd to create a stand-alone
 | |
| **   DLL form of this extension for WIN32. See its use below for details.
 | |
| */
 | |
| #include "sqlite3ext.h"
 | |
| SQLITE_EXTENSION_INIT1
 | |
| #include <stdio.h>
 | |
| #include <string.h>
 | |
| #include <assert.h>
 | |
| 
 | |
| #include <sys/types.h>
 | |
| #include <sys/stat.h>
 | |
| #include <fcntl.h>
 | |
| #if !defined(_WIN32) && !defined(WIN32)
 | |
| #  include <unistd.h>
 | |
| #  include <dirent.h>
 | |
| #  include <utime.h>
 | |
| #  include <sys/time.h>
 | |
| #  define STRUCT_STAT struct stat
 | |
| #else
 | |
| #  include "windirent.h"
 | |
| #  include <direct.h>
 | |
| #  define STRUCT_STAT struct _stat
 | |
| #  define chmod(path,mode) fileio_chmod(path,mode)
 | |
| #  define mkdir(path,mode) fileio_mkdir(path)
 | |
| #endif
 | |
| #include <time.h>
 | |
| #include <errno.h>
 | |
| 
 | |
| /* When used as part of the CLI, the sqlite3_stdio.h module will have
 | |
| ** been included before this one. In that case use the sqlite3_stdio.h
 | |
| ** #defines.  If not, create our own for fopen().
 | |
| */
 | |
| #ifndef _SQLITE3_STDIO_H_
 | |
| # define sqlite3_fopen fopen
 | |
| #endif
 | |
| 
 | |
| /*
 | |
| ** Structure of the fsdir() table-valued function
 | |
| */
 | |
|                  /*    0    1    2     3    4     5           6          */
 | |
| #define FSDIR_SCHEMA "(name,mode,mtime,data,level,path HIDDEN,dir HIDDEN)"
 | |
| 
 | |
| #define FSDIR_COLUMN_NAME     0     /* Name of the file */
 | |
| #define FSDIR_COLUMN_MODE     1     /* Access mode */
 | |
| #define FSDIR_COLUMN_MTIME    2     /* Last modification time */
 | |
| #define FSDIR_COLUMN_DATA     3     /* File content */
 | |
| #define FSDIR_COLUMN_LEVEL    4     /* Level.  Topmost is 1 */
 | |
| #define FSDIR_COLUMN_PATH     5     /* Path to top of search */
 | |
| #define FSDIR_COLUMN_DIR      6     /* Path is relative to this directory */
 | |
| 
 | |
| /*
 | |
| ** UTF8 chmod() function for Windows
 | |
| */
 | |
| #if defined(_WIN32) || defined(WIN32)
 | |
| static int fileio_chmod(const char *zPath, int pmode){
 | |
|   sqlite3_int64 sz = strlen(zPath);
 | |
|   wchar_t *b1 = sqlite3_malloc64( (sz+1)*sizeof(b1[0]) );
 | |
|   int rc;
 | |
|   if( b1==0 ) return -1;
 | |
|   sz = MultiByteToWideChar(CP_UTF8, 0, zPath, sz, b1, sz);
 | |
|   b1[sz] = 0;
 | |
|   rc = _wchmod(b1, pmode);
 | |
|   sqlite3_free(b1);
 | |
|   return rc;
 | |
| }
 | |
| #endif
 | |
| 
 | |
| /*
 | |
| ** UTF8 mkdir() function for Windows
 | |
| */
 | |
| #if defined(_WIN32) || defined(WIN32)
 | |
| static int fileio_mkdir(const char *zPath){
 | |
|   sqlite3_int64 sz = strlen(zPath);
 | |
|   wchar_t *b1 = sqlite3_malloc64( (sz+1)*sizeof(b1[0]) );
 | |
|   int rc;
 | |
|   if( b1==0 ) return -1;
 | |
|   sz = MultiByteToWideChar(CP_UTF8, 0, zPath, sz, b1, sz);
 | |
|   b1[sz] = 0;
 | |
|   rc = _wmkdir(b1);
 | |
|   sqlite3_free(b1);
 | |
|   return rc;
 | |
| }
 | |
| #endif
 | |
| 
 | |
| 
 | |
| /*
 | |
| ** Set the result stored by context ctx to a blob containing the 
 | |
| ** contents of file zName.  Or, leave the result unchanged (NULL)
 | |
| ** if the file does not exist or is unreadable.
 | |
| **
 | |
| ** If the file exceeds the SQLite blob size limit, through an
 | |
| ** SQLITE_TOOBIG error.
 | |
| **
 | |
| ** Throw an SQLITE_IOERR if there are difficulties pulling the file
 | |
| ** off of disk.
 | |
| */
 | |
| static void readFileContents(sqlite3_context *ctx, const char *zName){
 | |
|   FILE *in;
 | |
|   sqlite3_int64 nIn;
 | |
|   void *pBuf;
 | |
|   sqlite3 *db;
 | |
|   int mxBlob;
 | |
| 
 | |
|   in = sqlite3_fopen(zName, "rb");
 | |
|   if( in==0 ){
 | |
|     /* File does not exist or is unreadable. Leave the result set to NULL. */
 | |
|     return;
 | |
|   }
 | |
|   fseek(in, 0, SEEK_END);
 | |
|   nIn = ftell(in);
 | |
|   rewind(in);
 | |
|   db = sqlite3_context_db_handle(ctx);
 | |
|   mxBlob = sqlite3_limit(db, SQLITE_LIMIT_LENGTH, -1);
 | |
|   if( nIn>mxBlob ){
 | |
|     sqlite3_result_error_code(ctx, SQLITE_TOOBIG);
 | |
|     fclose(in);
 | |
|     return;
 | |
|   }
 | |
|   pBuf = sqlite3_malloc64( nIn ? nIn : 1 );
 | |
|   if( pBuf==0 ){
 | |
|     sqlite3_result_error_nomem(ctx);
 | |
|     fclose(in);
 | |
|     return;
 | |
|   }
 | |
|   if( nIn==(sqlite3_int64)fread(pBuf, 1, (size_t)nIn, in) ){
 | |
|     sqlite3_result_blob64(ctx, pBuf, nIn, sqlite3_free);
 | |
|   }else{
 | |
|     sqlite3_result_error_code(ctx, SQLITE_IOERR);
 | |
|     sqlite3_free(pBuf);
 | |
|   }
 | |
|   fclose(in);
 | |
| }
 | |
| 
 | |
| /*
 | |
| ** Implementation of the "readfile(X)" SQL function.  The entire content
 | |
| ** of the file named X is read and returned as a BLOB.  NULL is returned
 | |
| ** if the file does not exist or is unreadable.
 | |
| */
 | |
| static void readfileFunc(
 | |
|   sqlite3_context *context,
 | |
|   int argc,
 | |
|   sqlite3_value **argv
 | |
| ){
 | |
|   const char *zName;
 | |
|   (void)(argc);  /* Unused parameter */
 | |
|   zName = (const char*)sqlite3_value_text(argv[0]);
 | |
|   if( zName==0 ) return;
 | |
|   readFileContents(context, zName);
 | |
| }
 | |
| 
 | |
| /*
 | |
| ** Set the error message contained in context ctx to the results of
 | |
| ** vprintf(zFmt, ...).
 | |
| */
 | |
| static void ctxErrorMsg(sqlite3_context *ctx, const char *zFmt, ...){
 | |
|   char *zMsg = 0;
 | |
|   va_list ap;
 | |
|   va_start(ap, zFmt);
 | |
|   zMsg = sqlite3_vmprintf(zFmt, ap);
 | |
|   sqlite3_result_error(ctx, zMsg, -1);
 | |
|   sqlite3_free(zMsg);
 | |
|   va_end(ap);
 | |
| }
 | |
| 
 | |
| #if defined(_WIN32)
 | |
| /*
 | |
| ** This function is designed to convert a Win32 FILETIME structure into the
 | |
| ** number of seconds since the Unix Epoch (1970-01-01 00:00:00 UTC).
 | |
| */
 | |
| static sqlite3_uint64 fileTimeToUnixTime(
 | |
|   LPFILETIME pFileTime
 | |
| ){
 | |
|   SYSTEMTIME epochSystemTime;
 | |
|   ULARGE_INTEGER epochIntervals;
 | |
|   FILETIME epochFileTime;
 | |
|   ULARGE_INTEGER fileIntervals;
 | |
| 
 | |
|   memset(&epochSystemTime, 0, sizeof(SYSTEMTIME));
 | |
|   epochSystemTime.wYear = 1970;
 | |
|   epochSystemTime.wMonth = 1;
 | |
|   epochSystemTime.wDay = 1;
 | |
|   SystemTimeToFileTime(&epochSystemTime, &epochFileTime);
 | |
|   epochIntervals.LowPart = epochFileTime.dwLowDateTime;
 | |
|   epochIntervals.HighPart = epochFileTime.dwHighDateTime;
 | |
| 
 | |
|   fileIntervals.LowPart = pFileTime->dwLowDateTime;
 | |
|   fileIntervals.HighPart = pFileTime->dwHighDateTime;
 | |
| 
 | |
|   return (fileIntervals.QuadPart - epochIntervals.QuadPart) / 10000000;
 | |
| }
 | |
| 
 | |
| 
 | |
| #if defined(FILEIO_WIN32_DLL) && (defined(_WIN32) || defined(WIN32))
 | |
| #  /* To allow a standalone DLL, use this next replacement function: */
 | |
| #  undef sqlite3_win32_utf8_to_unicode
 | |
| #  define sqlite3_win32_utf8_to_unicode utf8_to_utf16
 | |
| #
 | |
| LPWSTR utf8_to_utf16(const char *z){
 | |
|   int nAllot = MultiByteToWideChar(CP_UTF8, 0, z, -1, NULL, 0);
 | |
|   LPWSTR rv = sqlite3_malloc(nAllot * sizeof(WCHAR));
 | |
|   if( rv!=0 && 0 < MultiByteToWideChar(CP_UTF8, 0, z, -1, rv, nAllot) )
 | |
|     return rv;
 | |
|   sqlite3_free(rv);
 | |
|   return 0;
 | |
| }
 | |
| #endif
 | |
| 
 | |
| /*
 | |
| ** This function attempts to normalize the time values found in the stat()
 | |
| ** buffer to UTC.  This is necessary on Win32, where the runtime library
 | |
| ** appears to return these values as local times.
 | |
| */
 | |
| static void statTimesToUtc(
 | |
|   const char *zPath,
 | |
|   STRUCT_STAT *pStatBuf
 | |
| ){
 | |
|   HANDLE hFindFile;
 | |
|   WIN32_FIND_DATAW fd;
 | |
|   LPWSTR zUnicodeName;
 | |
|   extern LPWSTR sqlite3_win32_utf8_to_unicode(const char*);
 | |
|   zUnicodeName = sqlite3_win32_utf8_to_unicode(zPath);
 | |
|   if( zUnicodeName ){
 | |
|     memset(&fd, 0, sizeof(WIN32_FIND_DATAW));
 | |
|     hFindFile = FindFirstFileW(zUnicodeName, &fd);
 | |
|     if( hFindFile!=NULL ){
 | |
|       pStatBuf->st_ctime = (time_t)fileTimeToUnixTime(&fd.ftCreationTime);
 | |
|       pStatBuf->st_atime = (time_t)fileTimeToUnixTime(&fd.ftLastAccessTime);
 | |
|       pStatBuf->st_mtime = (time_t)fileTimeToUnixTime(&fd.ftLastWriteTime);
 | |
|       FindClose(hFindFile);
 | |
|     }
 | |
|     sqlite3_free(zUnicodeName);
 | |
|   }
 | |
| }
 | |
| #endif
 | |
| 
 | |
| /*
 | |
| ** This function is used in place of stat().  On Windows, special handling
 | |
| ** is required in order for the included time to be returned as UTC.  On all
 | |
| ** other systems, this function simply calls stat().
 | |
| */
 | |
| static int fileStat(
 | |
|   const char *zPath,
 | |
|   STRUCT_STAT *pStatBuf
 | |
| ){
 | |
| #if defined(_WIN32)
 | |
|   sqlite3_int64 sz = strlen(zPath);
 | |
|   wchar_t *b1 = sqlite3_malloc64( (sz+1)*sizeof(b1[0]) );
 | |
|   int rc;
 | |
|   if( b1==0 ) return 1;
 | |
|   sz = MultiByteToWideChar(CP_UTF8, 0, zPath, sz, b1, sz);
 | |
|   b1[sz] = 0;
 | |
|   rc = _wstat(b1, pStatBuf);
 | |
|   if( rc==0 ) statTimesToUtc(zPath, pStatBuf);
 | |
|   return rc;
 | |
| #else
 | |
|   return stat(zPath, pStatBuf);
 | |
| #endif
 | |
| }
 | |
| 
 | |
| /*
 | |
| ** This function is used in place of lstat().  On Windows, special handling
 | |
| ** is required in order for the included time to be returned as UTC.  On all
 | |
| ** other systems, this function simply calls lstat().
 | |
| */
 | |
| static int fileLinkStat(
 | |
|   const char *zPath,
 | |
|   STRUCT_STAT *pStatBuf
 | |
| ){
 | |
| #if defined(_WIN32)
 | |
|   return fileStat(zPath, pStatBuf);
 | |
| #else
 | |
|   return lstat(zPath, pStatBuf);
 | |
| #endif
 | |
| }
 | |
| 
 | |
| /*
 | |
| ** Argument zFile is the name of a file that will be created and/or written
 | |
| ** by SQL function writefile(). This function ensures that the directory
 | |
| ** zFile will be written to exists, creating it if required. The permissions
 | |
| ** for any path components created by this function are set in accordance
 | |
| ** with the current umask.
 | |
| **
 | |
| ** If an OOM condition is encountered, SQLITE_NOMEM is returned. Otherwise,
 | |
| ** SQLITE_OK is returned if the directory is successfully created, or
 | |
| ** SQLITE_ERROR otherwise.
 | |
| */
 | |
| static int makeDirectory(
 | |
|   const char *zFile
 | |
| ){
 | |
|   char *zCopy = sqlite3_mprintf("%s", zFile);
 | |
|   int rc = SQLITE_OK;
 | |
| 
 | |
|   if( zCopy==0 ){
 | |
|     rc = SQLITE_NOMEM;
 | |
|   }else{
 | |
|     int nCopy = (int)strlen(zCopy);
 | |
|     int i = 1;
 | |
| 
 | |
|     while( rc==SQLITE_OK ){
 | |
|       STRUCT_STAT sStat;
 | |
|       int rc2;
 | |
| 
 | |
|       for(; zCopy[i]!='/' && i<nCopy; i++);
 | |
|       if( i==nCopy ) break;
 | |
|       zCopy[i] = '\0';
 | |
| 
 | |
|       rc2 = fileStat(zCopy, &sStat);
 | |
|       if( rc2!=0 ){
 | |
|         if( mkdir(zCopy, 0777) ) rc = SQLITE_ERROR;
 | |
|       }else{
 | |
|         if( !S_ISDIR(sStat.st_mode) ) rc = SQLITE_ERROR;
 | |
|       }
 | |
|       zCopy[i] = '/';
 | |
|       i++;
 | |
|     }
 | |
| 
 | |
|     sqlite3_free(zCopy);
 | |
|   }
 | |
| 
 | |
|   return rc;
 | |
| }
 | |
| 
 | |
| /*
 | |
| ** This function does the work for the writefile() UDF. Refer to 
 | |
| ** header comments at the top of this file for details.
 | |
| */
 | |
| static int writeFile(
 | |
|   sqlite3_context *pCtx,          /* Context to return bytes written in */
 | |
|   const char *zFile,              /* File to write */
 | |
|   sqlite3_value *pData,           /* Data to write */
 | |
|   mode_t mode,                    /* MODE parameter passed to writefile() */
 | |
|   sqlite3_int64 mtime             /* MTIME parameter (or -1 to not set time) */
 | |
| ){
 | |
|   if( zFile==0 ) return 1;
 | |
| #if !defined(_WIN32) && !defined(WIN32)
 | |
|   if( S_ISLNK(mode) ){
 | |
|     const char *zTo = (const char*)sqlite3_value_text(pData);
 | |
|     if( zTo==0 ) return 1;
 | |
|     unlink(zFile);
 | |
|     if( symlink(zTo, zFile)<0 ) return 1;
 | |
|   }else
 | |
| #endif
 | |
|   {
 | |
|     if( S_ISDIR(mode) ){
 | |
|       if( mkdir(zFile, mode) ){
 | |
|         /* The mkdir() call to create the directory failed. This might not
 | |
|         ** be an error though - if there is already a directory at the same
 | |
|         ** path and either the permissions already match or can be changed
 | |
|         ** to do so using chmod(), it is not an error.  */
 | |
|         STRUCT_STAT sStat;
 | |
|         if( errno!=EEXIST
 | |
|          || 0!=fileStat(zFile, &sStat)
 | |
|          || !S_ISDIR(sStat.st_mode)
 | |
|          || ((sStat.st_mode&0777)!=(mode&0777) && 0!=chmod(zFile, mode&0777))
 | |
|         ){
 | |
|           return 1;
 | |
|         }
 | |
|       }
 | |
|     }else{
 | |
|       sqlite3_int64 nWrite = 0;
 | |
|       const char *z;
 | |
|       int rc = 0;
 | |
|       FILE *out = sqlite3_fopen(zFile, "wb");
 | |
|       if( out==0 ) return 1;
 | |
|       z = (const char*)sqlite3_value_blob(pData);
 | |
|       if( z ){
 | |
|         sqlite3_int64 n = fwrite(z, 1, sqlite3_value_bytes(pData), out);
 | |
|         nWrite = sqlite3_value_bytes(pData);
 | |
|         if( nWrite!=n ){
 | |
|           rc = 1;
 | |
|         }
 | |
|       }
 | |
|       fclose(out);
 | |
|       if( rc==0 && mode && chmod(zFile, mode & 0777) ){
 | |
|         rc = 1;
 | |
|       }
 | |
|       if( rc ) return 2;
 | |
|       sqlite3_result_int64(pCtx, nWrite);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if( mtime>=0 ){
 | |
| #if defined(_WIN32)
 | |
| #if !SQLITE_OS_WINRT
 | |
|     /* Windows */
 | |
|     FILETIME lastAccess;
 | |
|     FILETIME lastWrite;
 | |
|     SYSTEMTIME currentTime;
 | |
|     LONGLONG intervals;
 | |
|     HANDLE hFile;
 | |
|     LPWSTR zUnicodeName;
 | |
|     extern LPWSTR sqlite3_win32_utf8_to_unicode(const char*);
 | |
| 
 | |
|     GetSystemTime(¤tTime);
 | |
|     SystemTimeToFileTime(¤tTime, &lastAccess);
 | |
|     intervals = (mtime*10000000) + 116444736000000000;
 | |
|     lastWrite.dwLowDateTime = (DWORD)intervals;
 | |
|     lastWrite.dwHighDateTime = intervals >> 32;
 | |
|     zUnicodeName = sqlite3_win32_utf8_to_unicode(zFile);
 | |
|     if( zUnicodeName==0 ){
 | |
|       return 1;
 | |
|     }
 | |
|     hFile = CreateFileW(
 | |
|       zUnicodeName, FILE_WRITE_ATTRIBUTES, 0, NULL, OPEN_EXISTING,
 | |
|       FILE_FLAG_BACKUP_SEMANTICS, NULL
 | |
|     );
 | |
|     sqlite3_free(zUnicodeName);
 | |
|     if( hFile!=INVALID_HANDLE_VALUE ){
 | |
|       BOOL bResult = SetFileTime(hFile, NULL, &lastAccess, &lastWrite);
 | |
|       CloseHandle(hFile);
 | |
|       return !bResult;
 | |
|     }else{
 | |
|       return 1;
 | |
|     }
 | |
| #endif
 | |
| #elif defined(AT_FDCWD) && 0 /* utimensat() is not universally available */
 | |
|     /* Recent unix */
 | |
|     struct timespec times[2];
 | |
|     times[0].tv_nsec = times[1].tv_nsec = 0;
 | |
|     times[0].tv_sec = time(0);
 | |
|     times[1].tv_sec = mtime;
 | |
|     if( utimensat(AT_FDCWD, zFile, times, AT_SYMLINK_NOFOLLOW) ){
 | |
|       return 1;
 | |
|     }
 | |
| #else
 | |
|     /* Legacy unix. 
 | |
|     **
 | |
|     ** Do not use utimes() on a symbolic link - it sees through the link and
 | |
|     ** modifies the timestamps on the target. Or fails if the target does 
 | |
|     ** not exist.  */
 | |
|     if( 0==S_ISLNK(mode) ){
 | |
|       struct timeval times[2];
 | |
|       times[0].tv_usec = times[1].tv_usec = 0;
 | |
|       times[0].tv_sec = time(0);
 | |
|       times[1].tv_sec = mtime;
 | |
|       if( utimes(zFile, times) ){
 | |
|         return 1;
 | |
|       }
 | |
|     }
 | |
| #endif
 | |
|   }
 | |
| 
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| /*
 | |
| ** Implementation of the "writefile(W,X[,Y[,Z]]])" SQL function.  
 | |
| ** Refer to header comments at the top of this file for details.
 | |
| */
 | |
| static void writefileFunc(
 | |
|   sqlite3_context *context,
 | |
|   int argc,
 | |
|   sqlite3_value **argv
 | |
| ){
 | |
|   const char *zFile;
 | |
|   mode_t mode = 0;
 | |
|   int res;
 | |
|   sqlite3_int64 mtime = -1;
 | |
| 
 | |
|   if( argc<2 || argc>4 ){
 | |
|     sqlite3_result_error(context, 
 | |
|         "wrong number of arguments to function writefile()", -1
 | |
|     );
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   zFile = (const char*)sqlite3_value_text(argv[0]);
 | |
|   if( zFile==0 ) return;
 | |
|   if( argc>=3 ){
 | |
|     mode = (mode_t)sqlite3_value_int(argv[2]);
 | |
|   }
 | |
|   if( argc==4 ){
 | |
|     mtime = sqlite3_value_int64(argv[3]);
 | |
|   }
 | |
| 
 | |
|   res = writeFile(context, zFile, argv[1], mode, mtime);
 | |
|   if( res==1 && errno==ENOENT ){
 | |
|     if( makeDirectory(zFile)==SQLITE_OK ){
 | |
|       res = writeFile(context, zFile, argv[1], mode, mtime);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if( argc>2 && res!=0 ){
 | |
|     if( S_ISLNK(mode) ){
 | |
|       ctxErrorMsg(context, "failed to create symlink: %s", zFile);
 | |
|     }else if( S_ISDIR(mode) ){
 | |
|       ctxErrorMsg(context, "failed to create directory: %s", zFile);
 | |
|     }else{
 | |
|       ctxErrorMsg(context, "failed to write file: %s", zFile);
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| /*
 | |
| ** SQL function:   lsmode(MODE)
 | |
| **
 | |
| ** Given a numberic st_mode from stat(), convert it into a human-readable
 | |
| ** text string in the style of "ls -l".
 | |
| */
 | |
| static void lsModeFunc(
 | |
|   sqlite3_context *context,
 | |
|   int argc,
 | |
|   sqlite3_value **argv
 | |
| ){
 | |
|   int i;
 | |
|   int iMode = sqlite3_value_int(argv[0]);
 | |
|   char z[16];
 | |
|   (void)argc;
 | |
|   if( S_ISLNK(iMode) ){
 | |
|     z[0] = 'l';
 | |
|   }else if( S_ISREG(iMode) ){
 | |
|     z[0] = '-';
 | |
|   }else if( S_ISDIR(iMode) ){
 | |
|     z[0] = 'd';
 | |
|   }else{
 | |
|     z[0] = '?';
 | |
|   }
 | |
|   for(i=0; i<3; i++){
 | |
|     int m = (iMode >> ((2-i)*3));
 | |
|     char *a = &z[1 + i*3];
 | |
|     a[0] = (m & 0x4) ? 'r' : '-';
 | |
|     a[1] = (m & 0x2) ? 'w' : '-';
 | |
|     a[2] = (m & 0x1) ? 'x' : '-';
 | |
|   }
 | |
|   z[10] = '\0';
 | |
|   sqlite3_result_text(context, z, -1, SQLITE_TRANSIENT);
 | |
| }
 | |
| 
 | |
| #ifndef SQLITE_OMIT_VIRTUALTABLE
 | |
| 
 | |
| /* 
 | |
| ** Cursor type for recursively iterating through a directory structure.
 | |
| */
 | |
| typedef struct fsdir_cursor fsdir_cursor;
 | |
| typedef struct FsdirLevel FsdirLevel;
 | |
| 
 | |
| struct FsdirLevel {
 | |
|   DIR *pDir;                 /* From opendir() */
 | |
|   char *zDir;                /* Name of directory (nul-terminated) */
 | |
| };
 | |
| 
 | |
| struct fsdir_cursor {
 | |
|   sqlite3_vtab_cursor base;  /* Base class - must be first */
 | |
| 
 | |
|   int nLvl;                  /* Number of entries in aLvl[] array */
 | |
|   int mxLvl;                 /* Maximum level */
 | |
|   int iLvl;                  /* Index of current entry */
 | |
|   FsdirLevel *aLvl;          /* Hierarchy of directories being traversed */
 | |
| 
 | |
|   const char *zBase;
 | |
|   int nBase;
 | |
| 
 | |
|   STRUCT_STAT sStat;         /* Current lstat() results */
 | |
|   char *zPath;               /* Path to current entry */
 | |
|   sqlite3_int64 iRowid;      /* Current rowid */
 | |
| };
 | |
| 
 | |
| typedef struct fsdir_tab fsdir_tab;
 | |
| struct fsdir_tab {
 | |
|   sqlite3_vtab base;         /* Base class - must be first */
 | |
| };
 | |
| 
 | |
| /*
 | |
| ** Construct a new fsdir virtual table object.
 | |
| */
 | |
| static int fsdirConnect(
 | |
|   sqlite3 *db,
 | |
|   void *pAux,
 | |
|   int argc, const char *const*argv,
 | |
|   sqlite3_vtab **ppVtab,
 | |
|   char **pzErr
 | |
| ){
 | |
|   fsdir_tab *pNew = 0;
 | |
|   int rc;
 | |
|   (void)pAux;
 | |
|   (void)argc;
 | |
|   (void)argv;
 | |
|   (void)pzErr;
 | |
|   rc = sqlite3_declare_vtab(db, "CREATE TABLE x" FSDIR_SCHEMA);
 | |
|   if( rc==SQLITE_OK ){
 | |
|     pNew = (fsdir_tab*)sqlite3_malloc( sizeof(*pNew) );
 | |
|     if( pNew==0 ) return SQLITE_NOMEM;
 | |
|     memset(pNew, 0, sizeof(*pNew));
 | |
|     sqlite3_vtab_config(db, SQLITE_VTAB_DIRECTONLY);
 | |
|   }
 | |
|   *ppVtab = (sqlite3_vtab*)pNew;
 | |
|   return rc;
 | |
| }
 | |
| 
 | |
| /*
 | |
| ** This method is the destructor for fsdir vtab objects.
 | |
| */
 | |
| static int fsdirDisconnect(sqlite3_vtab *pVtab){
 | |
|   sqlite3_free(pVtab);
 | |
|   return SQLITE_OK;
 | |
| }
 | |
| 
 | |
| /*
 | |
| ** Constructor for a new fsdir_cursor object.
 | |
| */
 | |
| static int fsdirOpen(sqlite3_vtab *p, sqlite3_vtab_cursor **ppCursor){
 | |
|   fsdir_cursor *pCur;
 | |
|   (void)p;
 | |
|   pCur = sqlite3_malloc( sizeof(*pCur) );
 | |
|   if( pCur==0 ) return SQLITE_NOMEM;
 | |
|   memset(pCur, 0, sizeof(*pCur));
 | |
|   pCur->iLvl = -1;
 | |
|   *ppCursor = &pCur->base;
 | |
|   return SQLITE_OK;
 | |
| }
 | |
| 
 | |
| /*
 | |
| ** Reset a cursor back to the state it was in when first returned
 | |
| ** by fsdirOpen().
 | |
| */
 | |
| static void fsdirResetCursor(fsdir_cursor *pCur){
 | |
|   int i;
 | |
|   for(i=0; i<=pCur->iLvl; i++){
 | |
|     FsdirLevel *pLvl = &pCur->aLvl[i];
 | |
|     if( pLvl->pDir ) closedir(pLvl->pDir);
 | |
|     sqlite3_free(pLvl->zDir);
 | |
|   }
 | |
|   sqlite3_free(pCur->zPath);
 | |
|   sqlite3_free(pCur->aLvl);
 | |
|   pCur->aLvl = 0;
 | |
|   pCur->zPath = 0;
 | |
|   pCur->zBase = 0;
 | |
|   pCur->nBase = 0;
 | |
|   pCur->nLvl = 0;
 | |
|   pCur->iLvl = -1;
 | |
|   pCur->iRowid = 1;
 | |
| }
 | |
| 
 | |
| /*
 | |
| ** Destructor for an fsdir_cursor.
 | |
| */
 | |
| static int fsdirClose(sqlite3_vtab_cursor *cur){
 | |
|   fsdir_cursor *pCur = (fsdir_cursor*)cur;
 | |
| 
 | |
|   fsdirResetCursor(pCur);
 | |
|   sqlite3_free(pCur);
 | |
|   return SQLITE_OK;
 | |
| }
 | |
| 
 | |
| /*
 | |
| ** Set the error message for the virtual table associated with cursor
 | |
| ** pCur to the results of vprintf(zFmt, ...).
 | |
| */
 | |
| static void fsdirSetErrmsg(fsdir_cursor *pCur, const char *zFmt, ...){
 | |
|   va_list ap;
 | |
|   va_start(ap, zFmt);
 | |
|   pCur->base.pVtab->zErrMsg = sqlite3_vmprintf(zFmt, ap);
 | |
|   va_end(ap);
 | |
| }
 | |
| 
 | |
| 
 | |
| /*
 | |
| ** Advance an fsdir_cursor to its next row of output.
 | |
| */
 | |
| static int fsdirNext(sqlite3_vtab_cursor *cur){
 | |
|   fsdir_cursor *pCur = (fsdir_cursor*)cur;
 | |
|   mode_t m = pCur->sStat.st_mode;
 | |
| 
 | |
|   pCur->iRowid++;
 | |
|   if( S_ISDIR(m) && pCur->iLvl+3<pCur->mxLvl ){
 | |
|     /* Descend into this directory */
 | |
|     int iNew = pCur->iLvl + 1;
 | |
|     FsdirLevel *pLvl;
 | |
|     if( iNew>=pCur->nLvl ){
 | |
|       int nNew = iNew+1;
 | |
|       sqlite3_int64 nByte = nNew*sizeof(FsdirLevel);
 | |
|       FsdirLevel *aNew = (FsdirLevel*)sqlite3_realloc64(pCur->aLvl, nByte);
 | |
|       if( aNew==0 ) return SQLITE_NOMEM;
 | |
|       memset(&aNew[pCur->nLvl], 0, sizeof(FsdirLevel)*(nNew-pCur->nLvl));
 | |
|       pCur->aLvl = aNew;
 | |
|       pCur->nLvl = nNew;
 | |
|     }
 | |
|     pCur->iLvl = iNew;
 | |
|     pLvl = &pCur->aLvl[iNew];
 | |
|     
 | |
|     pLvl->zDir = pCur->zPath;
 | |
|     pCur->zPath = 0;
 | |
|     pLvl->pDir = opendir(pLvl->zDir);
 | |
|     if( pLvl->pDir==0 ){
 | |
|       fsdirSetErrmsg(pCur, "cannot read directory: %s", pCur->zPath);
 | |
|       return SQLITE_ERROR;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   while( pCur->iLvl>=0 ){
 | |
|     FsdirLevel *pLvl = &pCur->aLvl[pCur->iLvl];
 | |
|     struct dirent *pEntry = readdir(pLvl->pDir);
 | |
|     if( pEntry ){
 | |
|       if( pEntry->d_name[0]=='.' ){
 | |
|        if( pEntry->d_name[1]=='.' && pEntry->d_name[2]=='\0' ) continue;
 | |
|        if( pEntry->d_name[1]=='\0' ) continue;
 | |
|       }
 | |
|       sqlite3_free(pCur->zPath);
 | |
|       pCur->zPath = sqlite3_mprintf("%s/%s", pLvl->zDir, pEntry->d_name);
 | |
|       if( pCur->zPath==0 ) return SQLITE_NOMEM;
 | |
|       if( fileLinkStat(pCur->zPath, &pCur->sStat) ){
 | |
|         fsdirSetErrmsg(pCur, "cannot stat file: %s", pCur->zPath);
 | |
|         return SQLITE_ERROR;
 | |
|       }
 | |
|       return SQLITE_OK;
 | |
|     }
 | |
|     closedir(pLvl->pDir);
 | |
|     sqlite3_free(pLvl->zDir);
 | |
|     pLvl->pDir = 0;
 | |
|     pLvl->zDir = 0;
 | |
|     pCur->iLvl--;
 | |
|   }
 | |
| 
 | |
|   /* EOF */
 | |
|   sqlite3_free(pCur->zPath);
 | |
|   pCur->zPath = 0;
 | |
|   return SQLITE_OK;
 | |
| }
 | |
| 
 | |
| /*
 | |
| ** Return values of columns for the row at which the series_cursor
 | |
| ** is currently pointing.
 | |
| */
 | |
| static int fsdirColumn(
 | |
|   sqlite3_vtab_cursor *cur,   /* The cursor */
 | |
|   sqlite3_context *ctx,       /* First argument to sqlite3_result_...() */
 | |
|   int i                       /* Which column to return */
 | |
| ){
 | |
|   fsdir_cursor *pCur = (fsdir_cursor*)cur;
 | |
|   switch( i ){
 | |
|     case FSDIR_COLUMN_NAME: {
 | |
|       sqlite3_result_text(ctx, &pCur->zPath[pCur->nBase], -1, SQLITE_TRANSIENT);
 | |
|       break;
 | |
|     }
 | |
| 
 | |
|     case FSDIR_COLUMN_MODE:
 | |
|       sqlite3_result_int64(ctx, pCur->sStat.st_mode);
 | |
|       break;
 | |
| 
 | |
|     case FSDIR_COLUMN_MTIME:
 | |
|       sqlite3_result_int64(ctx, pCur->sStat.st_mtime);
 | |
|       break;
 | |
| 
 | |
|     case FSDIR_COLUMN_DATA: {
 | |
|       mode_t m = pCur->sStat.st_mode;
 | |
|       if( S_ISDIR(m) ){
 | |
|         sqlite3_result_null(ctx);
 | |
| #if !defined(_WIN32) && !defined(WIN32)
 | |
|       }else if( S_ISLNK(m) ){
 | |
|         char aStatic[64];
 | |
|         char *aBuf = aStatic;
 | |
|         sqlite3_int64 nBuf = 64;
 | |
|         int n;
 | |
| 
 | |
|         while( 1 ){
 | |
|           n = readlink(pCur->zPath, aBuf, nBuf);
 | |
|           if( n<nBuf ) break;
 | |
|           if( aBuf!=aStatic ) sqlite3_free(aBuf);
 | |
|           nBuf = nBuf*2;
 | |
|           aBuf = sqlite3_malloc64(nBuf);
 | |
|           if( aBuf==0 ){
 | |
|             sqlite3_result_error_nomem(ctx);
 | |
|             return SQLITE_NOMEM;
 | |
|           }
 | |
|         }
 | |
| 
 | |
|         sqlite3_result_text(ctx, aBuf, n, SQLITE_TRANSIENT);
 | |
|         if( aBuf!=aStatic ) sqlite3_free(aBuf);
 | |
| #endif
 | |
|       }else{
 | |
|         readFileContents(ctx, pCur->zPath);
 | |
|       }
 | |
|       break;
 | |
|     }
 | |
|     case FSDIR_COLUMN_LEVEL:
 | |
|       sqlite3_result_int(ctx, pCur->iLvl+2);
 | |
|       break;
 | |
|     case FSDIR_COLUMN_PATH:
 | |
|     default: {
 | |
|       /* The FSDIR_COLUMN_PATH and FSDIR_COLUMN_DIR are input parameters.
 | |
|       ** always return their values as NULL */
 | |
|       break;
 | |
|     }
 | |
|   }
 | |
|   return SQLITE_OK;
 | |
| }
 | |
| 
 | |
| /*
 | |
| ** Return the rowid for the current row. In this implementation, the
 | |
| ** first row returned is assigned rowid value 1, and each subsequent
 | |
| ** row a value 1 more than that of the previous.
 | |
| */
 | |
| static int fsdirRowid(sqlite3_vtab_cursor *cur, sqlite_int64 *pRowid){
 | |
|   fsdir_cursor *pCur = (fsdir_cursor*)cur;
 | |
|   *pRowid = pCur->iRowid;
 | |
|   return SQLITE_OK;
 | |
| }
 | |
| 
 | |
| /*
 | |
| ** Return TRUE if the cursor has been moved off of the last
 | |
| ** row of output.
 | |
| */
 | |
| static int fsdirEof(sqlite3_vtab_cursor *cur){
 | |
|   fsdir_cursor *pCur = (fsdir_cursor*)cur;
 | |
|   return (pCur->zPath==0);
 | |
| }
 | |
| 
 | |
| /*
 | |
| ** xFilter callback.
 | |
| **
 | |
| ** idxNum bit      Meaning
 | |
| **     0x01         PATH=N
 | |
| **     0x02         DIR=N
 | |
| **     0x04         LEVEL<N
 | |
| **     0x08         LEVEL<=N  
 | |
| */
 | |
| static int fsdirFilter(
 | |
|   sqlite3_vtab_cursor *cur, 
 | |
|   int idxNum, const char *idxStr,
 | |
|   int argc, sqlite3_value **argv
 | |
| ){
 | |
|   const char *zDir = 0;
 | |
|   fsdir_cursor *pCur = (fsdir_cursor*)cur;
 | |
|   int i;
 | |
|   (void)idxStr;
 | |
|   fsdirResetCursor(pCur);
 | |
| 
 | |
|   if( idxNum==0 ){
 | |
|     fsdirSetErrmsg(pCur, "table function fsdir requires an argument");
 | |
|     return SQLITE_ERROR;
 | |
|   }
 | |
| 
 | |
|   assert( (idxNum & 0x01)!=0 && argc>0 );
 | |
|   zDir = (const char*)sqlite3_value_text(argv[0]);
 | |
|   if( zDir==0 ){
 | |
|     fsdirSetErrmsg(pCur, "table function fsdir requires a non-NULL argument");
 | |
|     return SQLITE_ERROR;
 | |
|   }
 | |
|   i = 1;
 | |
|   if( (idxNum & 0x02)!=0 ){
 | |
|     assert( argc>i );
 | |
|     pCur->zBase = (const char*)sqlite3_value_text(argv[i++]);
 | |
|   }
 | |
|   if( (idxNum & 0x0c)!=0 ){
 | |
|     assert( argc>i );
 | |
|     pCur->mxLvl = sqlite3_value_int(argv[i++]);
 | |
|     if( idxNum & 0x08 ) pCur->mxLvl++;
 | |
|     if( pCur->mxLvl<=0 ) pCur->mxLvl = 1000000000;
 | |
|   }else{
 | |
|     pCur->mxLvl = 1000000000;
 | |
|   }
 | |
|   if( pCur->zBase ){
 | |
|     pCur->nBase = (int)strlen(pCur->zBase)+1;
 | |
|     pCur->zPath = sqlite3_mprintf("%s/%s", pCur->zBase, zDir);
 | |
|   }else{
 | |
|     pCur->zPath = sqlite3_mprintf("%s", zDir);
 | |
|   }
 | |
| 
 | |
|   if( pCur->zPath==0 ){
 | |
|     return SQLITE_NOMEM;
 | |
|   }
 | |
|   if( fileLinkStat(pCur->zPath, &pCur->sStat) ){
 | |
|     fsdirSetErrmsg(pCur, "cannot stat file: %s", pCur->zPath);
 | |
|     return SQLITE_ERROR;
 | |
|   }
 | |
| 
 | |
|   return SQLITE_OK;
 | |
| }
 | |
| 
 | |
| /*
 | |
| ** SQLite will invoke this method one or more times while planning a query
 | |
| ** that uses the generate_series virtual table.  This routine needs to create
 | |
| ** a query plan for each invocation and compute an estimated cost for that
 | |
| ** plan.
 | |
| **
 | |
| ** In this implementation idxNum is used to represent the
 | |
| ** query plan.  idxStr is unused.
 | |
| **
 | |
| ** The query plan is represented by bits in idxNum:
 | |
| **
 | |
| **  0x01  The path value is supplied by argv[0]
 | |
| **  0x02  dir is in argv[1]
 | |
| **  0x04  maxdepth is in argv[1] or [2]
 | |
| */
 | |
| static int fsdirBestIndex(
 | |
|   sqlite3_vtab *tab,
 | |
|   sqlite3_index_info *pIdxInfo
 | |
| ){
 | |
|   int i;                 /* Loop over constraints */
 | |
|   int idxPath = -1;      /* Index in pIdxInfo->aConstraint of PATH= */
 | |
|   int idxDir = -1;       /* Index in pIdxInfo->aConstraint of DIR= */
 | |
|   int idxLevel = -1;     /* Index in pIdxInfo->aConstraint of LEVEL< or <= */
 | |
|   int idxLevelEQ = 0;    /* 0x08 for LEVEL<= or LEVEL=.  0x04 for LEVEL< */
 | |
|   int omitLevel = 0;     /* omit the LEVEL constraint */
 | |
|   int seenPath = 0;      /* True if an unusable PATH= constraint is seen */
 | |
|   int seenDir = 0;       /* True if an unusable DIR= constraint is seen */
 | |
|   const struct sqlite3_index_constraint *pConstraint;
 | |
| 
 | |
|   (void)tab;
 | |
|   pConstraint = pIdxInfo->aConstraint;
 | |
|   for(i=0; i<pIdxInfo->nConstraint; i++, pConstraint++){
 | |
|     if( pConstraint->op==SQLITE_INDEX_CONSTRAINT_EQ ){
 | |
|       switch( pConstraint->iColumn ){
 | |
|         case FSDIR_COLUMN_PATH: {
 | |
|           if( pConstraint->usable ){
 | |
|             idxPath = i;
 | |
|             seenPath = 0;
 | |
|           }else if( idxPath<0 ){
 | |
|             seenPath = 1;
 | |
|           }
 | |
|           break;
 | |
|         }
 | |
|         case FSDIR_COLUMN_DIR: {
 | |
|           if( pConstraint->usable ){
 | |
|             idxDir = i;
 | |
|             seenDir = 0;
 | |
|           }else if( idxDir<0 ){
 | |
|             seenDir = 1;
 | |
|           }
 | |
|           break;
 | |
|         }
 | |
|         case FSDIR_COLUMN_LEVEL: {
 | |
|           if( pConstraint->usable && idxLevel<0 ){
 | |
|             idxLevel = i;
 | |
|             idxLevelEQ = 0x08;
 | |
|             omitLevel = 0;
 | |
|           }
 | |
|           break;
 | |
|         }
 | |
|       }
 | |
|     }else
 | |
|     if( pConstraint->iColumn==FSDIR_COLUMN_LEVEL
 | |
|      && pConstraint->usable
 | |
|      && idxLevel<0
 | |
|     ){
 | |
|       if( pConstraint->op==SQLITE_INDEX_CONSTRAINT_LE ){
 | |
|         idxLevel = i;
 | |
|         idxLevelEQ = 0x08;
 | |
|         omitLevel = 1;
 | |
|       }else if( pConstraint->op==SQLITE_INDEX_CONSTRAINT_LT ){
 | |
|         idxLevel = i;
 | |
|         idxLevelEQ = 0x04;
 | |
|         omitLevel = 1;
 | |
|       }
 | |
|     } 
 | |
|   }
 | |
|   if( seenPath || seenDir ){
 | |
|     /* If input parameters are unusable, disallow this plan */
 | |
|     return SQLITE_CONSTRAINT;
 | |
|   }
 | |
| 
 | |
|   if( idxPath<0 ){
 | |
|     pIdxInfo->idxNum = 0;
 | |
|     /* The pIdxInfo->estimatedCost should have been initialized to a huge
 | |
|     ** number.  Leave it unchanged. */
 | |
|     pIdxInfo->estimatedRows = 0x7fffffff;
 | |
|   }else{
 | |
|     pIdxInfo->aConstraintUsage[idxPath].omit = 1;
 | |
|     pIdxInfo->aConstraintUsage[idxPath].argvIndex = 1;
 | |
|     pIdxInfo->idxNum = 0x01;
 | |
|     pIdxInfo->estimatedCost = 1.0e9;
 | |
|     i = 2;
 | |
|     if( idxDir>=0 ){
 | |
|       pIdxInfo->aConstraintUsage[idxDir].omit = 1;
 | |
|       pIdxInfo->aConstraintUsage[idxDir].argvIndex = i++;
 | |
|       pIdxInfo->idxNum |= 0x02;
 | |
|       pIdxInfo->estimatedCost /= 1.0e4;
 | |
|     }
 | |
|     if( idxLevel>=0 ){
 | |
|       pIdxInfo->aConstraintUsage[idxLevel].omit = omitLevel;
 | |
|       pIdxInfo->aConstraintUsage[idxLevel].argvIndex = i++;
 | |
|       pIdxInfo->idxNum |= idxLevelEQ;
 | |
|       pIdxInfo->estimatedCost /= 1.0e4;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return SQLITE_OK;
 | |
| }
 | |
| 
 | |
| /*
 | |
| ** Register the "fsdir" virtual table.
 | |
| */
 | |
| static int fsdirRegister(sqlite3 *db){
 | |
|   static sqlite3_module fsdirModule = {
 | |
|     0,                         /* iVersion */
 | |
|     0,                         /* xCreate */
 | |
|     fsdirConnect,              /* xConnect */
 | |
|     fsdirBestIndex,            /* xBestIndex */
 | |
|     fsdirDisconnect,           /* xDisconnect */
 | |
|     0,                         /* xDestroy */
 | |
|     fsdirOpen,                 /* xOpen - open a cursor */
 | |
|     fsdirClose,                /* xClose - close a cursor */
 | |
|     fsdirFilter,               /* xFilter - configure scan constraints */
 | |
|     fsdirNext,                 /* xNext - advance a cursor */
 | |
|     fsdirEof,                  /* xEof - check for end of scan */
 | |
|     fsdirColumn,               /* xColumn - read data */
 | |
|     fsdirRowid,                /* xRowid - read data */
 | |
|     0,                         /* xUpdate */
 | |
|     0,                         /* xBegin */
 | |
|     0,                         /* xSync */
 | |
|     0,                         /* xCommit */
 | |
|     0,                         /* xRollback */
 | |
|     0,                         /* xFindMethod */
 | |
|     0,                         /* xRename */
 | |
|     0,                         /* xSavepoint */
 | |
|     0,                         /* xRelease */
 | |
|     0,                         /* xRollbackTo */
 | |
|     0,                         /* xShadowName */
 | |
|     0                          /* xIntegrity */
 | |
|   };
 | |
| 
 | |
|   int rc = sqlite3_create_module(db, "fsdir", &fsdirModule, 0);
 | |
|   return rc;
 | |
| }
 | |
| #else         /* SQLITE_OMIT_VIRTUALTABLE */
 | |
| # define fsdirRegister(x) SQLITE_OK
 | |
| #endif
 | |
| 
 | |
| #ifdef _WIN32
 | |
| __declspec(dllexport)
 | |
| #endif
 | |
| int sqlite3_fileio_init(
 | |
|   sqlite3 *db, 
 | |
|   char **pzErrMsg, 
 | |
|   const sqlite3_api_routines *pApi
 | |
| ){
 | |
|   int rc = SQLITE_OK;
 | |
|   SQLITE_EXTENSION_INIT2(pApi);
 | |
|   (void)pzErrMsg;  /* Unused parameter */
 | |
|   rc = sqlite3_create_function(db, "readfile", 1, 
 | |
|                                SQLITE_UTF8|SQLITE_DIRECTONLY, 0,
 | |
|                                readfileFunc, 0, 0);
 | |
|   if( rc==SQLITE_OK ){
 | |
|     rc = sqlite3_create_function(db, "writefile", -1,
 | |
|                                  SQLITE_UTF8|SQLITE_DIRECTONLY, 0,
 | |
|                                  writefileFunc, 0, 0);
 | |
|   }
 | |
|   if( rc==SQLITE_OK ){
 | |
|     rc = sqlite3_create_function(db, "lsmode", 1, SQLITE_UTF8, 0,
 | |
|                                  lsModeFunc, 0, 0);
 | |
|   }
 | |
|   if( rc==SQLITE_OK ){
 | |
|     rc = fsdirRegister(db);
 | |
|   }
 | |
|   return rc;
 | |
| }
 | |
| 
 | |
| #if defined(FILEIO_WIN32_DLL) && (defined(_WIN32) || defined(WIN32))
 | |
| /* To allow a standalone DLL, make test_windirent.c use the same
 | |
|  * redefined SQLite API calls as the above extension code does.
 | |
|  * Just pull in this .c to accomplish this. As a beneficial side
 | |
|  * effect, this extension becomes a single translation unit. */
 | |
| #  include "test_windirent.c"
 | |
| #endif
 | 
