3cc9601fd9
Former-commit-id: ff34202749e8df2aa83f2578b16260b444f50987
278 lines
13 KiB
C#
278 lines
13 KiB
C#
/********************************************************
|
|
* ADO.NET 2.0 Data Provider for SQLite Version 3.X
|
|
* Written by Robert Simpson (robert@blackcastlesoft.com)
|
|
*
|
|
* Released to the public domain, use at your own risk!
|
|
********************************************************/
|
|
|
|
namespace Mono.Data.Sqlite
|
|
{
|
|
using System;
|
|
using System.Data;
|
|
using System.Runtime.InteropServices;
|
|
using System.Collections.Generic;
|
|
|
|
/// <summary>
|
|
/// This internal class provides the foundation of SQLite support. It defines all the abstract members needed to implement
|
|
/// a SQLite data provider, and inherits from SqliteConvert which allows for simple translations of string to and from SQLite.
|
|
/// </summary>
|
|
internal abstract class SQLiteBase : SqliteConvert, IDisposable
|
|
{
|
|
internal SQLiteBase(SQLiteDateFormats fmt)
|
|
: base(fmt) { }
|
|
|
|
static internal object _lock = new object();
|
|
|
|
/// <summary>
|
|
/// Returns a string representing the active version of SQLite
|
|
/// </summary>
|
|
internal abstract string Version { get; }
|
|
/// <summary>
|
|
/// Returns the number of changes the last executing insert/update caused.
|
|
/// </summary>
|
|
internal abstract int Changes { get; }
|
|
/// <summary>
|
|
/// Opens a database.
|
|
/// </summary>
|
|
/// <remarks>
|
|
/// Implementers should call SqliteFunction.BindFunctions() and save the array after opening a connection
|
|
/// to bind all attributed user-defined functions and collating sequences to the new connection.
|
|
/// </remarks>
|
|
/// <param name="strFilename">The filename of the database to open. SQLite automatically creates it if it doesn't exist.</param>
|
|
/// <param name="flags">The open flags to use when creating the connection</param>
|
|
/// <param name="maxPoolSize">The maximum size of the pool for the given filename</param>
|
|
/// <param name="usePool">If true, the connection can be pulled from the connection pool</param>
|
|
internal abstract void Open(string strFilename, SQLiteOpenFlagsEnum flags, int maxPoolSize, bool usePool);
|
|
/// <summary>
|
|
/// Closes the currently-open database.
|
|
/// </summary>
|
|
/// <remarks>
|
|
/// After the database has been closed implemeters should call SqliteFunction.UnbindFunctions() to deallocate all interop allocated
|
|
/// memory associated with the user-defined functions and collating sequences tied to the closed connection.
|
|
/// </remarks>
|
|
internal abstract void Close();
|
|
/// <summary>
|
|
/// Sets the busy timeout on the connection. SqliteCommand will call this before executing any command.
|
|
/// </summary>
|
|
/// <param name="nTimeoutMS">The number of milliseconds to wait before returning SQLITE_BUSY</param>
|
|
internal abstract void SetTimeout(int nTimeoutMS);
|
|
/// <summary>
|
|
/// Returns the text of the last error issued by SQLite
|
|
/// </summary>
|
|
/// <returns></returns>
|
|
internal abstract string SQLiteLastError();
|
|
|
|
/// <summary>
|
|
/// When pooling is enabled, force this connection to be disposed rather than returned to the pool
|
|
/// </summary>
|
|
internal abstract void ClearPool();
|
|
|
|
/// <summary>
|
|
/// Prepares a SQL statement for execution.
|
|
/// </summary>
|
|
/// <param name="cnn">The source connection preparing the command. Can be null for any caller except LINQ</param>
|
|
/// <param name="strSql">The SQL command text to prepare</param>
|
|
/// <param name="previous">The previous statement in a multi-statement command, or null if no previous statement exists</param>
|
|
/// <param name="timeoutMS">The timeout to wait before aborting the prepare</param>
|
|
/// <param name="strRemain">The remainder of the statement that was not processed. Each call to prepare parses the
|
|
/// SQL up to to either the end of the text or to the first semi-colon delimiter. The remaining text is returned
|
|
/// here for a subsequent call to Prepare() until all the text has been processed.</param>
|
|
/// <returns>Returns an initialized SqliteStatement.</returns>
|
|
internal abstract SqliteStatement Prepare(SqliteConnection cnn, string strSql, SqliteStatement previous, uint timeoutMS, out string strRemain);
|
|
/// <summary>
|
|
/// Steps through a prepared statement.
|
|
/// </summary>
|
|
/// <param name="stmt">The SqliteStatement to step through</param>
|
|
/// <returns>True if a row was returned, False if not.</returns>
|
|
internal abstract bool Step(SqliteStatement stmt);
|
|
/// <summary>
|
|
/// Resets a prepared statement so it can be executed again. If the error returned is SQLITE_SCHEMA,
|
|
/// transparently attempt to rebuild the SQL statement and throw an error if that was not possible.
|
|
/// </summary>
|
|
/// <param name="stmt">The statement to reset</param>
|
|
/// <returns>Returns -1 if the schema changed while resetting, 0 if the reset was sucessful or 6 (SQLITE_LOCKED) if the reset failed due to a lock</returns>
|
|
internal abstract int Reset(SqliteStatement stmt);
|
|
internal abstract void Cancel();
|
|
|
|
internal abstract void Bind_Double(SqliteStatement stmt, int index, double value);
|
|
internal abstract void Bind_Int32(SqliteStatement stmt, int index, Int32 value);
|
|
internal abstract void Bind_Int64(SqliteStatement stmt, int index, Int64 value);
|
|
internal abstract void Bind_Text(SqliteStatement stmt, int index, string value);
|
|
internal abstract void Bind_Blob(SqliteStatement stmt, int index, byte[] blobData);
|
|
internal abstract void Bind_DateTime(SqliteStatement stmt, int index, DateTime dt);
|
|
internal abstract void Bind_Null(SqliteStatement stmt, int index);
|
|
|
|
internal abstract int Bind_ParamCount(SqliteStatement stmt);
|
|
internal abstract string Bind_ParamName(SqliteStatement stmt, int index);
|
|
internal abstract int Bind_ParamIndex(SqliteStatement stmt, string paramName);
|
|
|
|
internal abstract int ColumnCount(SqliteStatement stmt);
|
|
internal abstract string ColumnName(SqliteStatement stmt, int index);
|
|
internal abstract TypeAffinity ColumnAffinity(SqliteStatement stmt, int index);
|
|
internal abstract string ColumnType(SqliteStatement stmt, int index, out TypeAffinity nAffinity);
|
|
internal abstract int ColumnIndex(SqliteStatement stmt, string columnName);
|
|
internal abstract string ColumnOriginalName(SqliteStatement stmt, int index);
|
|
internal abstract string ColumnDatabaseName(SqliteStatement stmt, int index);
|
|
internal abstract string ColumnTableName(SqliteStatement stmt, int index);
|
|
internal abstract void ColumnMetaData(string dataBase, string table, string column, out string dataType, out string collateSequence, out bool notNull, out bool primaryKey, out bool autoIncrement);
|
|
internal abstract void GetIndexColumnExtendedInfo(string database, string index, string column, out int sortMode, out int onError, out string collationSequence);
|
|
|
|
internal abstract double GetDouble(SqliteStatement stmt, int index);
|
|
internal abstract Int32 GetInt32(SqliteStatement stmt, int index);
|
|
internal abstract Int64 GetInt64(SqliteStatement stmt, int index);
|
|
internal abstract string GetText(SqliteStatement stmt, int index);
|
|
internal abstract long GetBytes(SqliteStatement stmt, int index, int nDataoffset, byte[] bDest, int nStart, int nLength);
|
|
internal abstract long GetChars(SqliteStatement stmt, int index, int nDataoffset, char[] bDest, int nStart, int nLength);
|
|
internal abstract DateTime GetDateTime(SqliteStatement stmt, int index);
|
|
internal abstract bool IsNull(SqliteStatement stmt, int index);
|
|
|
|
internal abstract void CreateCollation(string strCollation, SQLiteCollation func, SQLiteCollation func16, IntPtr user_data);
|
|
internal abstract void CreateFunction(string strFunction, int nArgs, bool needCollSeq, SQLiteCallback func, SQLiteCallback funcstep, SQLiteFinalCallback funcfinal);
|
|
internal abstract CollationSequence GetCollationSequence(SqliteFunction func, IntPtr context);
|
|
internal abstract int ContextCollateCompare(CollationEncodingEnum enc, IntPtr context, string s1, string s2);
|
|
internal abstract int ContextCollateCompare(CollationEncodingEnum enc, IntPtr context, char[] c1, char[] c2);
|
|
|
|
internal abstract int AggregateCount(IntPtr context);
|
|
internal abstract IntPtr AggregateContext(IntPtr context);
|
|
|
|
internal abstract long GetParamValueBytes(IntPtr ptr, int nDataOffset, byte[] bDest, int nStart, int nLength);
|
|
internal abstract double GetParamValueDouble(IntPtr ptr);
|
|
internal abstract int GetParamValueInt32(IntPtr ptr);
|
|
internal abstract Int64 GetParamValueInt64(IntPtr ptr);
|
|
internal abstract string GetParamValueText(IntPtr ptr);
|
|
internal abstract TypeAffinity GetParamValueType(IntPtr ptr);
|
|
|
|
internal abstract void ReturnBlob(IntPtr context, byte[] value);
|
|
internal abstract void ReturnDouble(IntPtr context, double value);
|
|
internal abstract void ReturnError(IntPtr context, string value);
|
|
internal abstract void ReturnInt32(IntPtr context, Int32 value);
|
|
internal abstract void ReturnInt64(IntPtr context, Int64 value);
|
|
internal abstract void ReturnNull(IntPtr context);
|
|
internal abstract void ReturnText(IntPtr context, string value);
|
|
|
|
internal abstract void SetPassword(byte[] passwordBytes);
|
|
internal abstract void ChangePassword(byte[] newPasswordBytes);
|
|
|
|
internal abstract void SetUpdateHook(SQLiteUpdateCallback func);
|
|
internal abstract void SetCommitHook(SQLiteCommitCallback func);
|
|
internal abstract void SetRollbackHook(SQLiteRollbackCallback func);
|
|
|
|
internal abstract int GetCursorForTable(SqliteStatement stmt, int database, int rootPage);
|
|
internal abstract long GetRowIdForCursor(SqliteStatement stmt, int cursor);
|
|
|
|
internal abstract object GetValue(SqliteStatement stmt, int index, SQLiteType typ);
|
|
|
|
protected virtual void Dispose(bool bDisposing)
|
|
{
|
|
}
|
|
|
|
public void Dispose()
|
|
{
|
|
Dispose(true);
|
|
}
|
|
|
|
// These statics are here for lack of a better place to put them.
|
|
// They exist here because they are called during the finalization of
|
|
// a SqliteStatementHandle, SqliteConnectionHandle, and SqliteFunctionCookieHandle.
|
|
// Therefore these functions have to be static, and have to be low-level.
|
|
|
|
internal static string SQLiteLastError(SqliteConnectionHandle db)
|
|
{
|
|
#if !SQLITE_STANDARD
|
|
int len;
|
|
return UTF8ToString(UnsafeNativeMethods.sqlite3_errmsg_interop(db, out len), len);
|
|
#else
|
|
return UTF8ToString(UnsafeNativeMethods.sqlite3_errmsg(db), -1);
|
|
#endif
|
|
}
|
|
|
|
internal static void FinalizeStatement(SqliteStatementHandle stmt)
|
|
{
|
|
lock (_lock)
|
|
{
|
|
#if !SQLITE_STANDARD
|
|
int n = UnsafeNativeMethods.sqlite3_finalize_interop(stmt);
|
|
#else
|
|
int n = UnsafeNativeMethods.sqlite3_finalize(stmt);
|
|
#endif
|
|
if (n > 0) throw new SqliteException(n, null);
|
|
}
|
|
}
|
|
|
|
internal static void CloseConnection(SqliteConnectionHandle db)
|
|
{
|
|
lock (_lock)
|
|
{
|
|
#if !SQLITE_STANDARD
|
|
int n = UnsafeNativeMethods.sqlite3_close_interop(db);
|
|
#else
|
|
ResetConnection(db);
|
|
int n;
|
|
if (UnsafeNativeMethods.use_sqlite3_close_v2) {
|
|
n = UnsafeNativeMethods.sqlite3_close_v2(db);
|
|
} else {
|
|
n = UnsafeNativeMethods.sqlite3_close(db);
|
|
}
|
|
#endif
|
|
if (n > 0) throw new SqliteException(n, SQLiteLastError(db));
|
|
}
|
|
}
|
|
|
|
internal static void ResetConnection(SqliteConnectionHandle db)
|
|
{
|
|
lock (_lock)
|
|
{
|
|
IntPtr stmt = IntPtr.Zero;
|
|
|
|
do
|
|
{
|
|
stmt = UnsafeNativeMethods.sqlite3_next_stmt(db, stmt);
|
|
if (stmt != IntPtr.Zero)
|
|
{
|
|
#if !SQLITE_STANDARD
|
|
UnsafeNativeMethods.sqlite3_reset_interop(stmt);
|
|
#else
|
|
UnsafeNativeMethods.sqlite3_reset(stmt);
|
|
#endif
|
|
}
|
|
} while (stmt != IntPtr.Zero);
|
|
|
|
// Not overly concerned with the return value from a rollback.
|
|
UnsafeNativeMethods.sqlite3_exec(db, ToUTF8("ROLLBACK"), IntPtr.Zero, IntPtr.Zero, out stmt);
|
|
// but free the error message if any!
|
|
if (stmt != IntPtr.Zero)
|
|
UnsafeNativeMethods.sqlite3_free (stmt);
|
|
}
|
|
}
|
|
}
|
|
|
|
internal interface ISQLiteSchemaExtensions
|
|
{
|
|
void BuildTempSchema(SqliteConnection cnn);
|
|
}
|
|
|
|
[Flags]
|
|
internal enum SQLiteOpenFlagsEnum
|
|
{
|
|
None = 0,
|
|
ReadOnly = 0x01,
|
|
ReadWrite = 0x02,
|
|
Create = 0x04,
|
|
//SharedCache = 0x01000000,
|
|
Default = 0x06,
|
|
|
|
// iOS Specific
|
|
FileProtectionComplete = 0x00100000,
|
|
FileProtectionCompleteUnlessOpen = 0x00200000,
|
|
FileProtectionCompleteUntilFirstUserAuthentication = 0x00300000,
|
|
FileProtectionNone = 0x00400000
|
|
}
|
|
|
|
// subset of the options available in http://www.sqlite.org/c3ref/c_config_getmalloc.html
|
|
public enum SQLiteConfig {
|
|
SingleThread = 1,
|
|
MultiThread = 2,
|
|
Serialized = 3,
|
|
}
|
|
}
|