mirror of
				https://github.com/gnh1201/welsonjs.git
				synced 2025-10-26 10:31:17 +00:00 
			
		
		
		
	Add the package ManagedEsent (microsoft/ManagedEsent@d358c07), and WelsonJS.Esent
		
			
				
	
	
		
			1173 lines
		
	
	
		
			57 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			1173 lines
		
	
	
		
			57 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| //-----------------------------------------------------------------------
 | |
| // <copyright file="RetrieveColumnHelpers.cs" company="Microsoft Corporation">
 | |
| //     Copyright (c) Microsoft Corporation.
 | |
| // </copyright>
 | |
| //-----------------------------------------------------------------------
 | |
| 
 | |
| namespace Microsoft.Isam.Esent.Interop
 | |
| {
 | |
|     using System;
 | |
|     using System.Collections.Generic;
 | |
|     using System.Diagnostics;
 | |
|     using System.Diagnostics.CodeAnalysis;
 | |
|     using System.Globalization;
 | |
| #if MANAGEDESENT_SUPPORTS_SERIALIZATION
 | |
|     using System.Runtime.Serialization.Formatters.Binary;
 | |
| #endif
 | |
|     using System.Text;
 | |
| 
 | |
|     /// <summary>
 | |
|     /// Helper methods for the ESENT API. These aren't interop versions
 | |
|     /// of the API, but encapsulate very common uses of the functions.
 | |
|     /// </summary>
 | |
|     public static partial class Api
 | |
|     {
 | |
|         /// <summary>
 | |
|         /// Encoding to use to decode ASCII text. We use this because
 | |
|         /// UTF8.GetString is faster than ASCII.GetString.
 | |
|         /// </summary>
 | |
|         private static readonly Encoding AsciiDecoder = new UTF8Encoding(false, true);
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Retrieves the bookmark for the record that is associated with the index entry
 | |
|         /// at the current position of a cursor. This bookmark can then be used to
 | |
|         /// reposition that cursor back to the same record using JetGotoBookmark.
 | |
|         /// </summary>
 | |
|         /// <param name="sesid">The session to use.</param>
 | |
|         /// <param name="tableid">The cursor to retrieve the bookmark from.</param>
 | |
|         /// <returns>The bookmark of the record.</returns>
 | |
|         public static byte[] GetBookmark(JET_SESID sesid, JET_TABLEID tableid)
 | |
|         {
 | |
|             byte[] buffer = null;
 | |
|             byte[] bookmark;
 | |
| 
 | |
|             try
 | |
|             {
 | |
|                 buffer = Caches.BookmarkCache.Allocate();
 | |
|                 int bookmarkSize;
 | |
|                 Api.JetGetBookmark(sesid, tableid, buffer, buffer.Length, out bookmarkSize);
 | |
|                 bookmark = MemoryCache.Duplicate(buffer, bookmarkSize);
 | |
|             }
 | |
|             finally
 | |
|             {
 | |
|                 if (buffer != null)
 | |
|                 {
 | |
|                     Caches.BookmarkCache.Free(ref buffer);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             return bookmark;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Retrieves the bookmark for the record that is associated with the index entry
 | |
|         /// at the current position of a cursor. This bookmark can then be used to
 | |
|         /// reposition that cursor back to the same record using JetGotoBookmark.
 | |
|         /// </summary>
 | |
|         /// <param name="sesid">The session to use.</param>
 | |
|         /// <param name="tableid">The cursor to retrieve the bookmark from.</param>
 | |
|         /// <param name="primaryBookmark">Returns the primary bookmark.</param>
 | |
|         /// <returns>The secondary bookmark of the record.</returns>
 | |
|         public static byte[] GetSecondaryBookmark(
 | |
|             JET_SESID sesid,
 | |
|             JET_TABLEID tableid,
 | |
|             out byte[] primaryBookmark)
 | |
|         {
 | |
|             byte[] bufferPrimary = null;
 | |
|             byte[] bufferSecondary = null;
 | |
|             byte[] secondaryBookmark;
 | |
|             primaryBookmark = null;
 | |
| 
 | |
|             try
 | |
|             {
 | |
|                 bufferPrimary = Caches.BookmarkCache.Allocate();
 | |
|                 bufferSecondary = Caches.SecondaryBookmarkCache.Allocate();
 | |
|                 int bookmarkSizePrimary;
 | |
|                 int bookmarkSizeSecondary;
 | |
| 
 | |
|                 Api.JetGetSecondaryIndexBookmark(
 | |
|                     sesid,
 | |
|                     tableid,
 | |
|                     bufferSecondary,
 | |
|                     bufferSecondary.Length,
 | |
|                     out bookmarkSizeSecondary,
 | |
|                     bufferPrimary,
 | |
|                     bufferPrimary.Length,
 | |
|                     out bookmarkSizePrimary,
 | |
|                     GetSecondaryIndexBookmarkGrbit.None);
 | |
| 
 | |
|                 primaryBookmark = MemoryCache.Duplicate(bufferPrimary, bookmarkSizePrimary);
 | |
|                 secondaryBookmark = MemoryCache.Duplicate(bufferSecondary, bookmarkSizeSecondary);
 | |
|             }
 | |
|             finally
 | |
|             {
 | |
|                 if (bufferPrimary != null)
 | |
|                 {
 | |
|                     Caches.BookmarkCache.Free(ref bufferPrimary);
 | |
|                 }
 | |
| 
 | |
|                 if (bufferSecondary != null)
 | |
|                 {
 | |
|                     Caches.BookmarkCache.Free(ref bufferSecondary);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             return secondaryBookmark;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Retrieves the key for the index entry at the current position of a cursor.
 | |
|         /// </summary>
 | |
|         /// <param name="sesid">The session to use.</param>
 | |
|         /// <param name="tableid">The cursor to retrieve the key from.</param>
 | |
|         /// <param name="grbit">Retrieve key options.</param>
 | |
|         /// <returns>The retrieved key.</returns>
 | |
|         public static byte[] RetrieveKey(JET_SESID sesid, JET_TABLEID tableid, RetrieveKeyGrbit grbit)
 | |
|         {
 | |
|             byte[] buffer = null;
 | |
|             byte[] key;
 | |
| 
 | |
|             try
 | |
|             {
 | |
|                 buffer = Caches.BookmarkCache.Allocate();
 | |
|                 int keySize;
 | |
|                 Api.JetRetrieveKey(sesid, tableid, buffer, buffer.Length, out keySize, grbit);
 | |
|                 key = MemoryCache.Duplicate(buffer, keySize);
 | |
|             }
 | |
|             finally
 | |
|             {
 | |
|                 if (buffer != null)
 | |
|                 {
 | |
|                     Caches.BookmarkCache.Free(ref buffer);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             return key;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Retrieves the size of a single column value from the current record.
 | |
|         /// The record is that record associated with the index entry at the
 | |
|         /// current position of the cursor. Alternatively, this function can
 | |
|         /// retrieve a column from a record being created in the cursor copy
 | |
|         /// buffer. This function can also retrieve column data from an index
 | |
|         /// entry that references the current record.
 | |
|         /// </summary>
 | |
|         /// <param name="sesid">The session to use.</param>
 | |
|         /// <param name="tableid">The cursor to retrieve the column from.</param>
 | |
|         /// <param name="columnid">The columnid to retrieve.</param>
 | |
|         /// <returns>The size of the column. 0 if the column is null.</returns>
 | |
|         public static int? RetrieveColumnSize(JET_SESID sesid, JET_TABLEID tableid, JET_COLUMNID columnid)
 | |
|         {
 | |
|             return RetrieveColumnSize(sesid, tableid, columnid, 1, RetrieveColumnGrbit.None);
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Retrieves the size of a single column value from the current record.
 | |
|         /// The record is that record associated with the index entry at the
 | |
|         /// current position of the cursor. Alternatively, this function can
 | |
|         /// retrieve a column from a record being created in the cursor copy
 | |
|         /// buffer. This function can also retrieve column data from an index
 | |
|         /// entry that references the current record.
 | |
|         /// </summary>
 | |
|         /// <param name="sesid">The session to use.</param>
 | |
|         /// <param name="tableid">The cursor to retrieve the column from.</param>
 | |
|         /// <param name="columnid">The columnid to retrieve.</param>
 | |
|         /// <param name="itagSequence">
 | |
|         /// The sequence number of value in a multi-valued column.
 | |
|         /// The array of values is one-based. The first value is
 | |
|         /// sequence 1, not 0. If the record column has only one value then
 | |
|         /// 1 should be passed as the itagSequence.
 | |
|         /// </param>
 | |
|         /// <param name="grbit">Retrieve column options.</param>
 | |
|         /// <returns>The size of the column. 0 if the column is null.</returns>
 | |
|         public static int? RetrieveColumnSize(
 | |
|             JET_SESID sesid, JET_TABLEID tableid, JET_COLUMNID columnid, int itagSequence, RetrieveColumnGrbit grbit)
 | |
|         {
 | |
|             // We cannot support this request when there is no way to indicate that a column reference is returned.
 | |
|             if ((grbit & (RetrieveColumnGrbit)0x00020000) != 0)  // UnpublishedGrbits.RetrieveAsRefIfNotInRecord
 | |
|             {
 | |
|                 throw new EsentInvalidGrbitException();
 | |
|             }
 | |
| 
 | |
|             var retinfo = new JET_RETINFO { itagSequence = itagSequence };
 | |
|             int dataSize;
 | |
|             JET_wrn wrn = JetRetrieveColumn(
 | |
|                 sesid, tableid, columnid, null, 0, out dataSize, grbit, retinfo);
 | |
|             if (JET_wrn.ColumnNull == wrn)
 | |
|             {
 | |
|                 return null;
 | |
|             }
 | |
| 
 | |
|             return dataSize;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Retrieves a single column value from the current record. The record is that
 | |
|         /// record associated with the index entry at the current position of the cursor.
 | |
|         /// Alternatively, this function can retrieve a column from a record being created
 | |
|         /// in the cursor copy buffer. This function can also retrieve column data from an
 | |
|         /// index entry that references the current record. In addition to retrieving the
 | |
|         /// actual column value, JetRetrieveColumn can also be used to retrieve the size
 | |
|         /// of a column, before retrieving the column data itself so that application
 | |
|         /// buffers can be sized appropriately.
 | |
|         /// </summary>
 | |
|         /// <param name="sesid">The session to use.</param>
 | |
|         /// <param name="tableid">The cursor to retrieve the column from.</param>
 | |
|         /// <param name="columnid">The columnid to retrieve.</param>
 | |
|         /// <param name="grbit">Retrieve column options.</param>
 | |
|         /// <param name="retinfo">
 | |
|         /// If pretinfo is give as NULL then the function behaves as though an itagSequence
 | |
|         /// of 1 and an ibLongValue of 0 (zero) were given. This causes column retrieval to
 | |
|         /// retrieve the first value of a multi-valued column, and to retrieve long data at
 | |
|         /// offset 0 (zero).
 | |
|         /// </param>
 | |
|         /// <returns>The data retrieved from the column. Null if the column is null.</returns>
 | |
|         public static byte[] RetrieveColumn(
 | |
|             JET_SESID sesid, JET_TABLEID tableid, JET_COLUMNID columnid, RetrieveColumnGrbit grbit, JET_RETINFO retinfo)
 | |
|         {
 | |
|             // We cannot support this request when there is no way to indicate that a column reference is returned.
 | |
|             if ((grbit & (RetrieveColumnGrbit)0x00020000) != 0)  // UnpublishedGrbits.RetrieveAsRefIfNotInRecord
 | |
|             {
 | |
|                 throw new EsentInvalidGrbitException();
 | |
|             }
 | |
| 
 | |
|             // We expect most column values retrieved this way to be small (retrieving a 1GB LV as one
 | |
|             // chunk is a bit extreme!). Allocate a cached buffer and use that, allocating a larger one
 | |
|             // if needed.
 | |
|             byte[] cache = null;
 | |
|             byte[] data;
 | |
| 
 | |
|             try
 | |
|             {
 | |
|                 cache = Caches.ColumnCache.Allocate();
 | |
|                 data = cache;
 | |
|                 int dataSize;
 | |
|                 JET_wrn wrn = JetRetrieveColumn(
 | |
|                     sesid, tableid, columnid, data, data.Length, out dataSize, grbit, retinfo);
 | |
| 
 | |
|                 if (JET_wrn.ColumnNull == wrn)
 | |
|                 {
 | |
|                     // null column
 | |
|                     data = null;
 | |
|                 }
 | |
|                 else if (JET_wrn.Success == wrn)
 | |
|                 {
 | |
|                     data = MemoryCache.Duplicate(data, dataSize);
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     // there is more data to retrieve
 | |
|                     Debug.Assert(JET_wrn.BufferTruncated == wrn, "Unexpected warning: " + wrn.ToString());
 | |
|                     data = new byte[dataSize];
 | |
|                     wrn = JetRetrieveColumn(
 | |
|                         sesid, tableid, columnid, data, data.Length, out dataSize, grbit, retinfo);
 | |
|                     if (JET_wrn.BufferTruncated == wrn)
 | |
|                     {
 | |
|                         string error = string.Format(
 | |
|                             CultureInfo.CurrentCulture,
 | |
|                             "Column size changed from {0} to {1}. The record was probably updated by another thread.",
 | |
|                             data.Length,
 | |
|                             dataSize);
 | |
|                         Trace.TraceError(error);
 | |
|                         throw new InvalidOperationException(error);
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|             finally
 | |
|             {
 | |
|                 if (cache != null)
 | |
|                 {
 | |
|                     Caches.ColumnCache.Free(ref cache);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             return data;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Retrieves a single column value from the current record. The record is that
 | |
|         /// record associated with the index entry at the current position of the cursor.
 | |
|         /// Alternatively, this function can retrieve a column from a record being created
 | |
|         /// in the cursor copy buffer. This function can also retrieve column data from an
 | |
|         /// index entry that references the current record. In addition to retrieving the
 | |
|         /// actual column value, JetRetrieveColumn can also be used to retrieve the size
 | |
|         /// of a column, before retrieving the column data itself so that application
 | |
|         /// buffers can be sized appropriately.
 | |
|         /// </summary>
 | |
|         /// <param name="sesid">The session to use.</param>
 | |
|         /// <param name="tableid">The cursor to retrieve the column from.</param>
 | |
|         /// <param name="columnid">The columnid to retrieve.</param>
 | |
|         /// <param name="data">The data buffer to be retrieved into.</param>
 | |
|         /// <param name="dataSize">The size of the data buffer.</param>
 | |
|         /// <param name="actualDataSize">Returns the actual size of the data buffer.</param>         
 | |
|         /// <param name="grbit">Retrieve column options.</param>
 | |
|         /// <param name="retinfo">
 | |
|         /// If pretinfo is give as NULL then the function behaves as though an itagSequence
 | |
|         /// of 1 and an ibLongValue of 0 (zero) were given. This causes column retrieval to
 | |
|         /// retrieve the first value of a multi-valued column, and to retrieve long data at
 | |
|         /// offset 0 (zero).
 | |
|         /// </param>
 | |
|         /// <returns>The data retrieved from the column. Null if the column is null.</returns>
 | |
|         public static JET_wrn RetrieveColumn(
 | |
|             JET_SESID sesid, JET_TABLEID tableid, JET_COLUMNID columnid, byte[] data, int dataSize, out int actualDataSize, RetrieveColumnGrbit grbit, JET_RETINFO retinfo)
 | |
|         {
 | |
|             // We cannot support this request when there is no way to indicate that a column reference is returned.
 | |
|             if ((grbit & (RetrieveColumnGrbit)0x00020000) != 0)  // UnpublishedGrbits.RetrieveAsRefIfNotInRecord
 | |
|             {
 | |
|                 throw new EsentInvalidGrbitException();
 | |
|             }
 | |
|             
 | |
|             return JetRetrieveColumn(sesid, tableid, columnid, data, dataSize, out actualDataSize, grbit, retinfo);
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Retrieves a single column value from the current record. The record is that
 | |
|         /// record associated with the index entry at the current position of the cursor.
 | |
|         /// </summary>
 | |
|         /// <param name="sesid">The session to use.</param>
 | |
|         /// <param name="tableid">The cursor to retrieve the column from.</param>
 | |
|         /// <param name="columnid">The columnid to retrieve.</param>
 | |
|         /// <returns>The data retrieved from the column. Null if the column is null.</returns>
 | |
|         public static byte[] RetrieveColumn(JET_SESID sesid, JET_TABLEID tableid, JET_COLUMNID columnid)
 | |
|         {
 | |
|             return RetrieveColumn(sesid, tableid, columnid, RetrieveColumnGrbit.None, null);
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Retrieves a single column value from the current record. The record is that
 | |
|         /// record associated with the index entry at the current position of the cursor.
 | |
|         /// The Unicode encoding is used.
 | |
|         /// </summary>
 | |
|         /// <param name="sesid">The session to use.</param>
 | |
|         /// <param name="tableid">The cursor to retrieve the column from.</param>
 | |
|         /// <param name="columnid">The columnid to retrieve.</param>
 | |
|         /// <returns>The data retrieved from the column as a string. Null if the column is null.</returns>
 | |
|         public static string RetrieveColumnAsString(JET_SESID sesid, JET_TABLEID tableid, JET_COLUMNID columnid)
 | |
|         {
 | |
|             return RetrieveColumnAsString(sesid, tableid, columnid, Encoding.Unicode, RetrieveColumnGrbit.None);
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Retrieves a string column value from the current record. The record is that
 | |
|         /// record associated with the index entry at the current position of the cursor.
 | |
|         /// </summary>
 | |
|         /// <param name="sesid">The session to use.</param>
 | |
|         /// <param name="tableid">The cursor to retrieve the column from.</param>
 | |
|         /// <param name="columnid">The columnid to retrieve.</param>
 | |
|         /// <param name="encoding">The string encoding to use when converting data.</param>
 | |
|         /// <returns>The data retrieved from the column as a string. Null if the column is null.</returns>
 | |
|         public static string RetrieveColumnAsString(
 | |
|             JET_SESID sesid, JET_TABLEID tableid, JET_COLUMNID columnid, Encoding encoding)
 | |
|         {
 | |
|             return RetrieveColumnAsString(sesid, tableid, columnid, encoding, RetrieveColumnGrbit.None);
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Retrieves a string column value from the current record. The record is that
 | |
|         /// record associated with the index entry at the current position of the cursor.
 | |
|         /// </summary>
 | |
|         /// <param name="sesid">The session to use.</param>
 | |
|         /// <param name="tableid">The cursor to retrieve the column from.</param>
 | |
|         /// <param name="columnid">The columnid to retrieve.</param>
 | |
|         /// <param name="encoding">The string encoding to use when converting data.</param>
 | |
|         /// <param name="grbit">Retrieval options.</param>
 | |
|         /// <returns>The data retrieved from the column as a string. Null if the column is null.</returns>
 | |
|         public static string RetrieveColumnAsString(
 | |
|             JET_SESID sesid, JET_TABLEID tableid, JET_COLUMNID columnid, Encoding encoding, RetrieveColumnGrbit grbit)
 | |
|         {
 | |
|             // We cannot support this request when there is no way to indicate that a column reference is returned.
 | |
|             if ((grbit & (RetrieveColumnGrbit)0x00020000) != 0)  // UnpublishedGrbits.RetrieveAsRefIfNotInRecord
 | |
|             {
 | |
|                 throw new EsentInvalidGrbitException();
 | |
|             }
 | |
| 
 | |
|             // This is an optimization for retrieving Unicode strings
 | |
|             if (Encoding.Unicode == encoding)
 | |
|             {
 | |
|                 return RetrieveUnicodeString(sesid, tableid, columnid, grbit);
 | |
|             }
 | |
| 
 | |
|             // Retrieving a string happens in two stages: first the data is retrieved into a data
 | |
|             // buffer and then the buffer is converted to a string. The buffer isn't needed for
 | |
|             // very long so we try to use a cached buffer.
 | |
|             byte[] cachedBuffer = null;
 | |
|             string s;
 | |
| 
 | |
|             try
 | |
|             {
 | |
|                 cachedBuffer = Caches.ColumnCache.Allocate();
 | |
|                 byte[] data = cachedBuffer;
 | |
| 
 | |
|                 int dataSize;
 | |
|                 JET_wrn wrn = JetRetrieveColumn(sesid, tableid, columnid, data, data.Length, out dataSize, grbit, null);
 | |
|                 if (JET_wrn.ColumnNull == wrn)
 | |
|                 {
 | |
|                     return null;
 | |
|                 }
 | |
| 
 | |
|                 if (JET_wrn.BufferTruncated == wrn)
 | |
|                 {
 | |
|                     Debug.Assert(dataSize > data.Length, "Expected column to be bigger than buffer");
 | |
|                     data = new byte[dataSize];
 | |
|                     wrn = JetRetrieveColumn(sesid, tableid, columnid, data, data.Length, out dataSize, grbit, null);
 | |
|                     if (JET_wrn.BufferTruncated == wrn)
 | |
|                     {
 | |
|                         string error = string.Format(
 | |
|                             CultureInfo.CurrentCulture,
 | |
|                             "Column size changed from {0} to {1}. The record was probably updated by another thread.",
 | |
|                             data.Length,
 | |
|                             dataSize);
 | |
|                         Trace.TraceError(error);
 | |
|                         throw new InvalidOperationException(error);
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
| #if MANAGEDESENT_SUPPORTS_ANSI
 | |
|                 // If we are about to decode ASCII data then use the UTF8 decoder instead. This
 | |
|                 // is done because the UTF8 decoder is faster and will produce the same results
 | |
|                 // on ASCII data. Different results will be produced on invalid data, but that
 | |
|                 // behaviour can be considered undefined.
 | |
|                 Encoding decoder = (encoding is ASCIIEncoding) ? AsciiDecoder : encoding;
 | |
| #else
 | |
|                 Encoding decoder = encoding;
 | |
| #endif
 | |
|                 s = decoder.GetString(data, 0, dataSize);
 | |
|             }
 | |
|             finally
 | |
|             {
 | |
|                 if (cachedBuffer != null)
 | |
|                 {
 | |
|                     // Now we have extracted the string from the buffer we can free (cache) the buffer.
 | |
|                     Caches.ColumnCache.Free(ref cachedBuffer);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             return s;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Retrieves a single column value from the current record. The record is that
 | |
|         /// record associated with the index entry at the current position of the cursor.
 | |
|         /// </summary>
 | |
|         /// <param name="sesid">The session to use.</param>
 | |
|         /// <param name="tableid">The cursor to retrieve the column from.</param>
 | |
|         /// <param name="columnid">The columnid to retrieve.</param>
 | |
|         /// <returns>The data retrieved from the column as a short. Null if the column is null.</returns>
 | |
|         public static short? RetrieveColumnAsInt16(JET_SESID sesid, JET_TABLEID tableid, JET_COLUMNID columnid)
 | |
|         {
 | |
|             return RetrieveColumnAsInt16(sesid, tableid, columnid, RetrieveColumnGrbit.None);
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Retrieves an int16 column value from the current record. The record is that
 | |
|         /// record associated with the index entry at the current position of the cursor.
 | |
|         /// </summary>
 | |
|         /// <param name="sesid">The session to use.</param>
 | |
|         /// <param name="tableid">The cursor to retrieve the column from.</param>
 | |
|         /// <param name="columnid">The columnid to retrieve.</param>
 | |
|         /// <param name="grbit">Retrieval options.</param>
 | |
|         /// <returns>The data retrieved from the column as a short. Null if the column is null.</returns>
 | |
|         public static short? RetrieveColumnAsInt16(
 | |
|             JET_SESID sesid, JET_TABLEID tableid, JET_COLUMNID columnid, RetrieveColumnGrbit grbit)
 | |
|         {
 | |
|             // We cannot support this request when there is no way to indicate that a column reference is returned.
 | |
|             if ((grbit & (RetrieveColumnGrbit)0x00020000) != 0)  // UnpublishedGrbits.RetrieveAsRefIfNotInRecord
 | |
|             {
 | |
|                 throw new EsentInvalidGrbitException();
 | |
|             }
 | |
| 
 | |
|             unsafe
 | |
|             {
 | |
|                 const int DataSize = sizeof(short);
 | |
|                 short data;
 | |
|                 var pointer = new IntPtr(&data);
 | |
|                 int actualDataSize;
 | |
|                 JET_wrn wrn = JetRetrieveColumn(
 | |
|                     sesid, tableid, columnid, pointer, DataSize, out actualDataSize, grbit, null);
 | |
|                 return CreateReturnValue(data, DataSize, wrn, actualDataSize);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Retrieves a single column value from the current record. The record is that
 | |
|         /// record associated with the index entry at the current position of the cursor.
 | |
|         /// </summary>
 | |
|         /// <param name="sesid">The session to use.</param>
 | |
|         /// <param name="tableid">The cursor to retrieve the column from.</param>
 | |
|         /// <param name="columnid">The columnid to retrieve.</param>
 | |
|         /// <returns>The data retrieved from the column as an int. Null if the column is null.</returns>
 | |
|         public static int? RetrieveColumnAsInt32(JET_SESID sesid, JET_TABLEID tableid, JET_COLUMNID columnid)
 | |
|         {
 | |
|             return RetrieveColumnAsInt32(sesid, tableid, columnid, RetrieveColumnGrbit.None);
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Retrieves an int32 column value from the current record. The record is that
 | |
|         /// record associated with the index entry at the current position of the cursor.
 | |
|         /// </summary>
 | |
|         /// <param name="sesid">The session to use.</param>
 | |
|         /// <param name="tableid">The cursor to retrieve the column from.</param>
 | |
|         /// <param name="columnid">The columnid to retrieve.</param>
 | |
|         /// <param name="grbit">Retrieval options.</param>
 | |
|         /// <returns>The data retrieved from the column as an int. Null if the column is null.</returns>
 | |
|         public static int? RetrieveColumnAsInt32(
 | |
|             JET_SESID sesid, JET_TABLEID tableid, JET_COLUMNID columnid, RetrieveColumnGrbit grbit)
 | |
|         {
 | |
|             // We cannot support this request when there is no way to indicate that a column reference is returned.
 | |
|             if ((grbit & (RetrieveColumnGrbit)0x00020000) != 0)  // UnpublishedGrbits.RetrieveAsRefIfNotInRecord
 | |
|             {
 | |
|                 throw new EsentInvalidGrbitException();
 | |
|             }
 | |
| 
 | |
|             unsafe
 | |
|             {
 | |
|                 const int DataSize = sizeof(int);
 | |
|                 int data;
 | |
|                 var pointer = new IntPtr(&data);
 | |
|                 int actualDataSize;
 | |
|                 JET_wrn wrn = JetRetrieveColumn(
 | |
|                     sesid, tableid, columnid, pointer, DataSize, out actualDataSize, grbit, null);
 | |
|                 return CreateReturnValue(data, DataSize, wrn, actualDataSize);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Retrieves a single column value from the current record. The record is that
 | |
|         /// record associated with the index entry at the current position of the cursor.
 | |
|         /// </summary>
 | |
|         /// <param name="sesid">The session to use.</param>
 | |
|         /// <param name="tableid">The cursor to retrieve the column from.</param>
 | |
|         /// <param name="columnid">The columnid to retrieve.</param>
 | |
|         /// <returns>The data retrieved from the column as a long. Null if the column is null.</returns>
 | |
|         public static long? RetrieveColumnAsInt64(JET_SESID sesid, JET_TABLEID tableid, JET_COLUMNID columnid)
 | |
|         {
 | |
|             return RetrieveColumnAsInt64(sesid, tableid, columnid, RetrieveColumnGrbit.None);
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Retrieves a single column value from the current record. The record is that
 | |
|         /// record associated with the index entry at the current position of the cursor.
 | |
|         /// </summary>
 | |
|         /// <param name="sesid">The session to use.</param>
 | |
|         /// <param name="tableid">The cursor to retrieve the column from.</param>
 | |
|         /// <param name="columnid">The columnid to retrieve.</param>
 | |
|         /// <param name="grbit">Retrieval options.</param>
 | |
|         /// <returns>The data retrieved from the column as a long. Null if the column is null.</returns>
 | |
|         public static long? RetrieveColumnAsInt64(
 | |
|             JET_SESID sesid, JET_TABLEID tableid, JET_COLUMNID columnid, RetrieveColumnGrbit grbit)
 | |
|         {
 | |
|             // We cannot support this request when there is no way to indicate that a column reference is returned.
 | |
|             if ((grbit & (RetrieveColumnGrbit)0x00020000) != 0)  // UnpublishedGrbits.RetrieveAsRefIfNotInRecord
 | |
|             {
 | |
|                 throw new EsentInvalidGrbitException();
 | |
|             }
 | |
| 
 | |
|             unsafe
 | |
|             {
 | |
|                 const int DataSize = sizeof(long);
 | |
|                 long data;
 | |
|                 var pointer = new IntPtr(&data);
 | |
|                 int actualDataSize;
 | |
|                 JET_wrn wrn = JetRetrieveColumn(
 | |
|                     sesid, tableid, columnid, pointer, DataSize, out actualDataSize, grbit, null);
 | |
|                 return CreateReturnValue(data, DataSize, wrn, actualDataSize);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Retrieves a float column value from the current record. The record is that
 | |
|         /// record associated with the index entry at the current position of the cursor.
 | |
|         /// </summary>
 | |
|         /// <param name="sesid">The session to use.</param>
 | |
|         /// <param name="tableid">The cursor to retrieve the column from.</param>
 | |
|         /// <param name="columnid">The columnid to retrieve.</param>
 | |
|         /// <returns>The data retrieved from the column as a float. Null if the column is null.</returns>
 | |
|         public static float? RetrieveColumnAsFloat(JET_SESID sesid, JET_TABLEID tableid, JET_COLUMNID columnid)
 | |
|         {
 | |
|             return RetrieveColumnAsFloat(sesid, tableid, columnid, RetrieveColumnGrbit.None);
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Retrieves a float column value from the current record. The record is that
 | |
|         /// record associated with the index entry at the current position of the cursor.
 | |
|         /// </summary>
 | |
|         /// <param name="sesid">The session to use.</param>
 | |
|         /// <param name="tableid">The cursor to retrieve the column from.</param>
 | |
|         /// <param name="columnid">The columnid to retrieve.</param>
 | |
|         /// <param name="grbit">Retrieval options.</param>
 | |
|         /// <returns>The data retrieved from the column as a float. Null if the column is null.</returns>
 | |
|         public static float? RetrieveColumnAsFloat(
 | |
|             JET_SESID sesid, JET_TABLEID tableid, JET_COLUMNID columnid, RetrieveColumnGrbit grbit)
 | |
|         {
 | |
|             // We cannot support this request when there is no way to indicate that a column reference is returned.
 | |
|             if ((grbit & (RetrieveColumnGrbit)0x00020000) != 0)  // UnpublishedGrbits.RetrieveAsRefIfNotInRecord
 | |
|             {
 | |
|                 throw new EsentInvalidGrbitException();
 | |
|             }
 | |
| 
 | |
|             unsafe
 | |
|             {
 | |
|                 const int DataSize = sizeof(float);
 | |
|                 float data;
 | |
|                 var pointer = new IntPtr(&data);
 | |
|                 int actualDataSize;
 | |
|                 JET_wrn wrn = JetRetrieveColumn(
 | |
|                     sesid, tableid, columnid, pointer, DataSize, out actualDataSize, grbit, null);
 | |
|                 return CreateReturnValue(data, DataSize, wrn, actualDataSize);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Retrieves a double column value from the current record. The record is that
 | |
|         /// record associated with the index entry at the current position of the cursor.
 | |
|         /// </summary>
 | |
|         /// <param name="sesid">The session to use.</param>
 | |
|         /// <param name="tableid">The cursor to retrieve the column from.</param>
 | |
|         /// <param name="columnid">The columnid to retrieve.</param>
 | |
|         /// <returns>The data retrieved from the column as a double. Null if the column is null.</returns>
 | |
|         public static double? RetrieveColumnAsDouble(JET_SESID sesid, JET_TABLEID tableid, JET_COLUMNID columnid)
 | |
|         {
 | |
|             return RetrieveColumnAsDouble(sesid, tableid, columnid, RetrieveColumnGrbit.None);
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Retrieves a double column value from the current record. The record is that
 | |
|         /// record associated with the index entry at the current position of the cursor.
 | |
|         /// </summary>
 | |
|         /// <param name="sesid">The session to use.</param>
 | |
|         /// <param name="tableid">The cursor to retrieve the column from.</param>
 | |
|         /// <param name="columnid">The columnid to retrieve.</param>
 | |
|         /// <param name="grbit">Retrieval options.</param>
 | |
|         /// <returns>The data retrieved from the column as a double. Null if the column is null.</returns>
 | |
|         public static double? RetrieveColumnAsDouble(
 | |
|             JET_SESID sesid, JET_TABLEID tableid, JET_COLUMNID columnid, RetrieveColumnGrbit grbit)
 | |
|         {
 | |
|             // We cannot support this request when there is no way to indicate that a column reference is returned.
 | |
|             if ((grbit & (RetrieveColumnGrbit)0x00020000) != 0)  // UnpublishedGrbits.RetrieveAsRefIfNotInRecord
 | |
|             {
 | |
|                 throw new EsentInvalidGrbitException();
 | |
|             }
 | |
| 
 | |
|             unsafe
 | |
|             {
 | |
|                 const int DataSize = sizeof(double);
 | |
|                 double data;
 | |
|                 var pointer = new IntPtr(&data);
 | |
|                 int actualDataSize;
 | |
|                 JET_wrn wrn = JetRetrieveColumn(
 | |
|                     sesid, tableid, columnid, pointer, DataSize, out actualDataSize, grbit, null);
 | |
|                 return CreateReturnValue(data, DataSize, wrn, actualDataSize);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Retrieves a boolean column value from the current record. The record is that
 | |
|         /// record associated with the index entry at the current position of the cursor.
 | |
|         /// </summary>
 | |
|         /// <param name="sesid">The session to use.</param>
 | |
|         /// <param name="tableid">The cursor to retrieve the column from.</param>
 | |
|         /// <param name="columnid">The columnid to retrieve.</param>
 | |
|         /// <returns>The data retrieved from the column as a boolean. Null if the column is null.</returns>
 | |
|         public static bool? RetrieveColumnAsBoolean(JET_SESID sesid, JET_TABLEID tableid, JET_COLUMNID columnid)
 | |
|         {
 | |
|             return RetrieveColumnAsBoolean(sesid, tableid, columnid, RetrieveColumnGrbit.None);
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Retrieves a boolean column value from the current record. The record is that
 | |
|         /// record associated with the index entry at the current position of the cursor.
 | |
|         /// </summary>
 | |
|         /// <param name="sesid">The session to use.</param>
 | |
|         /// <param name="tableid">The cursor to retrieve the column from.</param>
 | |
|         /// <param name="columnid">The columnid to retrieve.</param>
 | |
|         /// <param name="grbit">Retrieval options.</param>
 | |
|         /// <returns>The data retrieved from the column as a boolean. Null if the column is null.</returns>
 | |
|         public static bool? RetrieveColumnAsBoolean(
 | |
|             JET_SESID sesid, JET_TABLEID tableid, JET_COLUMNID columnid, RetrieveColumnGrbit grbit)
 | |
|         {
 | |
|             byte? b = RetrieveColumnAsByte(sesid, tableid, columnid, grbit);
 | |
|             if (b.HasValue)
 | |
|             {
 | |
|                 return 0 != b.Value;
 | |
|             }
 | |
| 
 | |
|             return new bool?();
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Retrieves a byte column value from the current record. The record is that
 | |
|         /// record associated with the index entry at the current position of the cursor.
 | |
|         /// </summary>
 | |
|         /// <param name="sesid">The session to use.</param>
 | |
|         /// <param name="tableid">The cursor to retrieve the column from.</param>
 | |
|         /// <param name="columnid">The columnid to retrieve.</param>
 | |
|         /// <returns>The data retrieved from the column as a byte. Null if the column is null.</returns>
 | |
|         public static byte? RetrieveColumnAsByte(JET_SESID sesid, JET_TABLEID tableid, JET_COLUMNID columnid)
 | |
|         {
 | |
|             return RetrieveColumnAsByte(sesid, tableid, columnid, RetrieveColumnGrbit.None);
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Retrieves a byte column value from the current record. The record is that
 | |
|         /// record associated with the index entry at the current position of the cursor.
 | |
|         /// </summary>
 | |
|         /// <param name="sesid">The session to use.</param>
 | |
|         /// <param name="tableid">The cursor to retrieve the column from.</param>
 | |
|         /// <param name="columnid">The columnid to retrieve.</param>
 | |
|         /// <param name="grbit">Retrieval options.</param>
 | |
|         /// <returns>The data retrieved from the column as a byte. Null if the column is null.</returns>
 | |
|         public static byte? RetrieveColumnAsByte(
 | |
|             JET_SESID sesid, JET_TABLEID tableid, JET_COLUMNID columnid, RetrieveColumnGrbit grbit)
 | |
|         {
 | |
|             // We cannot support this request when there is no way to indicate that a column reference is returned.
 | |
|             if ((grbit & (RetrieveColumnGrbit)0x00020000) != 0)  // UnpublishedGrbits.RetrieveAsRefIfNotInRecord
 | |
|             {
 | |
|                 throw new EsentInvalidGrbitException();
 | |
|             }
 | |
| 
 | |
|             unsafe
 | |
|             {
 | |
|                 const int DataSize = sizeof(byte);
 | |
|                 byte data;
 | |
|                 var pointer = new IntPtr(&data);
 | |
|                 int actualDataSize;
 | |
|                 JET_wrn wrn = JetRetrieveColumn(
 | |
|                     sesid, tableid, columnid, pointer, DataSize, out actualDataSize, grbit, null);
 | |
|                 return CreateReturnValue(data, DataSize, wrn, actualDataSize);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Retrieves a guid column value from the current record. The record is that
 | |
|         /// record associated with the index entry at the current position of the cursor.
 | |
|         /// </summary>
 | |
|         /// <param name="sesid">The session to use.</param>
 | |
|         /// <param name="tableid">The cursor to retrieve the column from.</param>
 | |
|         /// <param name="columnid">The columnid to retrieve.</param>
 | |
|         /// <returns>The data retrieved from the column as a guid. Null if the column is null.</returns>
 | |
|         public static Guid? RetrieveColumnAsGuid(JET_SESID sesid, JET_TABLEID tableid, JET_COLUMNID columnid)
 | |
|         {
 | |
|             return RetrieveColumnAsGuid(sesid, tableid, columnid, RetrieveColumnGrbit.None);
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Retrieves a guid column value from the current record. The record is that
 | |
|         /// record associated with the index entry at the current position of the cursor.
 | |
|         /// </summary>
 | |
|         /// <param name="sesid">The session to use.</param>
 | |
|         /// <param name="tableid">The cursor to retrieve the column from.</param>
 | |
|         /// <param name="columnid">The columnid to retrieve.</param>
 | |
|         /// <param name="grbit">Retrieval options.</param>
 | |
|         /// <returns>The data retrieved from the column as a guid. Null if the column is null.</returns>
 | |
|         public static Guid? RetrieveColumnAsGuid(
 | |
|             JET_SESID sesid, JET_TABLEID tableid, JET_COLUMNID columnid, RetrieveColumnGrbit grbit)
 | |
|         {
 | |
|             // We cannot support this request when there is no way to indicate that a column reference is returned.
 | |
|             if ((grbit & (RetrieveColumnGrbit)0x00020000) != 0)  // UnpublishedGrbits.RetrieveAsRefIfNotInRecord
 | |
|             {
 | |
|                 throw new EsentInvalidGrbitException();
 | |
|             }
 | |
| 
 | |
|             unsafe
 | |
|             {
 | |
|                 const int DataSize = 16;
 | |
|                 Guid data;
 | |
|                 var pointer = new IntPtr(&data);
 | |
|                 int actualDataSize;
 | |
|                 JET_wrn wrn = JetRetrieveColumn(
 | |
|                     sesid, tableid, columnid, pointer, DataSize, out actualDataSize, grbit, null);
 | |
|                 return CreateReturnValue(data, DataSize, wrn, actualDataSize);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Retrieves a datetime column value from the current record. The record is that
 | |
|         /// record associated with the index entry at the current position of the cursor.
 | |
|         /// </summary>
 | |
|         /// <param name="sesid">The session to use.</param>
 | |
|         /// <param name="tableid">The cursor to retrieve the column from.</param>
 | |
|         /// <param name="columnid">The columnid to retrieve.</param>
 | |
|         /// <returns>The data retrieved from the column as a <see cref="DateTime"/>. Null if the column is null.</returns>
 | |
|         public static DateTime? RetrieveColumnAsDateTime(JET_SESID sesid, JET_TABLEID tableid, JET_COLUMNID columnid)
 | |
|         {
 | |
|             return RetrieveColumnAsDateTime(sesid, tableid, columnid, RetrieveColumnGrbit.None);
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Retrieves a datetime column value from the current record. The record is that
 | |
|         /// record associated with the index entry at the current position of the cursor.
 | |
|         /// </summary>
 | |
|         /// <param name="sesid">The session to use.</param>
 | |
|         /// <param name="tableid">The cursor to retrieve the column from.</param>
 | |
|         /// <param name="columnid">The columnid to retrieve.</param>
 | |
|         /// <param name="grbit">Retrieval options.</param>
 | |
|         /// <returns>The data retrieved from the column as a <see cref="DateTime"/>. Null if the column is null.</returns>
 | |
|         public static DateTime? RetrieveColumnAsDateTime(
 | |
|             JET_SESID sesid, JET_TABLEID tableid, JET_COLUMNID columnid, RetrieveColumnGrbit grbit)
 | |
|         {
 | |
|             // Internally DateTime is stored in OLE Automation format
 | |
|             double? oadate = RetrieveColumnAsDouble(sesid, tableid, columnid, grbit);
 | |
|             if (oadate.HasValue)
 | |
|             {
 | |
|                 return Conversions.ConvertDoubleToDateTime(oadate.Value);
 | |
|             }
 | |
| 
 | |
|             return new DateTime?();
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Retrieves a uint16 column value from the current record. The record is that
 | |
|         /// record associated with the index entry at the current position of the cursor.
 | |
|         /// </summary>
 | |
|         /// <param name="sesid">The session to use.</param>
 | |
|         /// <param name="tableid">The cursor to retrieve the column from.</param>
 | |
|         /// <param name="columnid">The columnid to retrieve.</param>
 | |
|         /// <returns>The data retrieved from the column as an UInt16. Null if the column is null.</returns>
 | |
|         [CLSCompliant(false)]
 | |
|         public static ushort? RetrieveColumnAsUInt16(JET_SESID sesid, JET_TABLEID tableid, JET_COLUMNID columnid)
 | |
|         {
 | |
|             return RetrieveColumnAsUInt16(sesid, tableid, columnid, RetrieveColumnGrbit.None);
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Retrieves a uint16 column value from the current record. The record is that
 | |
|         /// record associated with the index entry at the current position of the cursor.
 | |
|         /// </summary>
 | |
|         /// <param name="sesid">The session to use.</param>
 | |
|         /// <param name="tableid">The cursor to retrieve the column from.</param>
 | |
|         /// <param name="columnid">The columnid to retrieve.</param>
 | |
|         /// <param name="grbit">Retrieval options.</param>
 | |
|         /// <returns>The data retrieved from the column as an UInt16. Null if the column is null.</returns>
 | |
|         [CLSCompliant(false)]
 | |
|         public static ushort? RetrieveColumnAsUInt16(
 | |
|             JET_SESID sesid, JET_TABLEID tableid, JET_COLUMNID columnid, RetrieveColumnGrbit grbit)
 | |
|         {
 | |
|             // We cannot support this request when there is no way to indicate that a column reference is returned.
 | |
|             if ((grbit & (RetrieveColumnGrbit)0x00020000) != 0)  // UnpublishedGrbits.RetrieveAsRefIfNotInRecord
 | |
|             {
 | |
|                 throw new EsentInvalidGrbitException();
 | |
|             }
 | |
| 
 | |
|             unsafe
 | |
|             {
 | |
|                 const int DataSize = sizeof(ushort);
 | |
|                 ushort data;
 | |
|                 var pointer = new IntPtr(&data);
 | |
|                 int actualDataSize;
 | |
|                 JET_wrn wrn = JetRetrieveColumn(
 | |
|                     sesid, tableid, columnid, pointer, DataSize, out actualDataSize, grbit, null);
 | |
|                 return CreateReturnValue(data, DataSize, wrn, actualDataSize);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Retrieves a uint32 column value from the current record. The record is that
 | |
|         /// record associated with the index entry at the current position of the cursor.
 | |
|         /// </summary>
 | |
|         /// <param name="sesid">The session to use.</param>
 | |
|         /// <param name="tableid">The cursor to retrieve the column from.</param>
 | |
|         /// <param name="columnid">The columnid to retrieve.</param>
 | |
|         /// <returns>The data retrieved from the column as an UInt32. Null if the column is null.</returns>
 | |
|         [CLSCompliant(false)]
 | |
|         public static uint? RetrieveColumnAsUInt32(JET_SESID sesid, JET_TABLEID tableid, JET_COLUMNID columnid)
 | |
|         {
 | |
|             return RetrieveColumnAsUInt32(sesid, tableid, columnid, RetrieveColumnGrbit.None);
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Retrieves a uint32 column value from the current record. The record is that
 | |
|         /// record associated with the index entry at the current position of the cursor.
 | |
|         /// </summary>
 | |
|         /// <param name="sesid">The session to use.</param>
 | |
|         /// <param name="tableid">The cursor to retrieve the column from.</param>
 | |
|         /// <param name="columnid">The columnid to retrieve.</param>
 | |
|         /// <param name="grbit">Retrieval options.</param>
 | |
|         /// <returns>The data retrieved from the column as an UInt32. Null if the column is null.</returns>
 | |
|         [CLSCompliant(false)]
 | |
|         public static uint? RetrieveColumnAsUInt32(
 | |
|             JET_SESID sesid, JET_TABLEID tableid, JET_COLUMNID columnid, RetrieveColumnGrbit grbit)
 | |
|         {
 | |
|             // We cannot support this request when there is no way to indicate that a column reference is returned.
 | |
|             if ((grbit & (RetrieveColumnGrbit)0x00020000) != 0)  // UnpublishedGrbits.RetrieveAsRefIfNotInRecord
 | |
|             {
 | |
|                 throw new EsentInvalidGrbitException();
 | |
|             }
 | |
| 
 | |
|             unsafe
 | |
|             {
 | |
|                 const int DataSize = sizeof(uint);
 | |
|                 uint data;
 | |
|                 var pointer = new IntPtr(&data);
 | |
|                 int actualDataSize;
 | |
|                 JET_wrn wrn = JetRetrieveColumn(
 | |
|                     sesid, tableid, columnid, pointer, DataSize, out actualDataSize, grbit, null);
 | |
|                 return CreateReturnValue(data, DataSize, wrn, actualDataSize);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Retrieves a uint64 column value from the current record. The record is that
 | |
|         /// record associated with the index entry at the current position of the cursor.
 | |
|         /// </summary>
 | |
|         /// <param name="sesid">The session to use.</param>
 | |
|         /// <param name="tableid">The cursor to retrieve the column from.</param>
 | |
|         /// <param name="columnid">The columnid to retrieve.</param>
 | |
|         /// <returns>The data retrieved from the column as an UInt64. Null if the column is null.</returns>
 | |
|         [CLSCompliant(false)]
 | |
|         public static ulong? RetrieveColumnAsUInt64(JET_SESID sesid, JET_TABLEID tableid, JET_COLUMNID columnid)
 | |
|         {
 | |
|             return RetrieveColumnAsUInt64(sesid, tableid, columnid, RetrieveColumnGrbit.None);
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Retrieves a uint64 column value from the current record. The record is that
 | |
|         /// record associated with the index entry at the current position of the cursor.
 | |
|         /// </summary>
 | |
|         /// <param name="sesid">The session to use.</param>
 | |
|         /// <param name="tableid">The cursor to retrieve the column from.</param>
 | |
|         /// <param name="columnid">The columnid to retrieve.</param>
 | |
|         /// <param name="grbit">Retrieval options.</param>
 | |
|         /// <returns>The data retrieved from the column as an UInt64. Null if the column is null.</returns>
 | |
|         [CLSCompliant(false)]
 | |
|         public static ulong? RetrieveColumnAsUInt64(
 | |
|             JET_SESID sesid, JET_TABLEID tableid, JET_COLUMNID columnid, RetrieveColumnGrbit grbit)
 | |
|         {
 | |
|             // We cannot support this request when there is no way to indicate that a column reference is returned.
 | |
|             if ((grbit & (RetrieveColumnGrbit)0x00020000) != 0)  // UnpublishedGrbits.RetrieveAsRefIfNotInRecord
 | |
|             {
 | |
|                 throw new EsentInvalidGrbitException();
 | |
|             }
 | |
| 
 | |
|             unsafe
 | |
|             {
 | |
|                 const int DataSize = sizeof(ulong);
 | |
|                 ulong data;
 | |
|                 var pointer = new IntPtr(&data);
 | |
|                 int actualDataSize;
 | |
|                 JET_wrn wrn = JetRetrieveColumn(
 | |
|                     sesid, tableid, columnid, pointer, DataSize, out actualDataSize, grbit, null);
 | |
|                 return CreateReturnValue(data, DataSize, wrn, actualDataSize);
 | |
|             }
 | |
|         }
 | |
| 
 | |
| #if MANAGEDESENT_SUPPORTS_SERIALIZATION
 | |
|         /// <summary>
 | |
|         /// Deserialize an object from a column.
 | |
|         /// </summary>
 | |
|         /// <param name="sesid">The session to use.</param>
 | |
|         /// <param name="tableid">The table to read from.</param>
 | |
|         /// <param name="columnid">The column to read from.</param>
 | |
|         /// <returns>The deserialized object. Null if the column was null.</returns>
 | |
|         public static object DeserializeObjectFromColumn(JET_SESID sesid, JET_TABLEID tableid, JET_COLUMNID columnid)
 | |
|         {
 | |
|             return DeserializeObjectFromColumn(sesid, tableid, columnid, RetrieveColumnGrbit.None);
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Deserialize an object from a column.
 | |
|         /// </summary>
 | |
|         /// <param name="sesid">The session to use.</param>
 | |
|         /// <param name="tableid">The table to read from.</param>
 | |
|         /// <param name="columnid">The column to read from.</param>
 | |
|         /// <param name="grbit">The retrieval options to use.</param>
 | |
|         /// <returns>The deserialized object. Null if the column was null.</returns>
 | |
|         [SuppressMessage("Exchange.Security", "EX0043:DoNotUseBinarySoapFormatter", Justification = "Suppress warning in current code base.The usage has already been verified.")]
 | |
|         public static object DeserializeObjectFromColumn(JET_SESID sesid, JET_TABLEID tableid, JET_COLUMNID columnid, RetrieveColumnGrbit grbit)
 | |
|         {
 | |
|             // We cannot support this request when there is no way to indicate that a column reference is returned.
 | |
|             if ((grbit & (RetrieveColumnGrbit)0x00020000) != 0)  // UnpublishedGrbits.RetrieveAsRefIfNotInRecord
 | |
|             {
 | |
|                 throw new EsentInvalidGrbitException();
 | |
|             }
 | |
| 
 | |
|             int actualSize;
 | |
|             if (JET_wrn.ColumnNull == Api.JetRetrieveColumn(sesid, tableid, columnid, null, 0, out actualSize, grbit, null))
 | |
|             {
 | |
|                 return null;
 | |
|             }
 | |
| 
 | |
|             using (var stream = new ColumnStream(sesid, tableid, columnid))
 | |
|             {
 | |
|                 var deseriaizer = new BinaryFormatter();
 | |
|                 return deseriaizer.Deserialize(stream);
 | |
|             }
 | |
|         }
 | |
| #endif
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Retrieves columns into ColumnValue objects.
 | |
|         /// </summary>
 | |
|         /// <param name="sesid">The session to use.</param>
 | |
|         /// <param name="tableid">The cursor retrieve the data from. The cursor should be positioned on a record.</param>
 | |
|         /// <param name="values">The values to retrieve.</param>
 | |
|         public static void RetrieveColumns(JET_SESID sesid, JET_TABLEID tableid, params ColumnValue[] values)
 | |
|         {
 | |
|             if (null == values)
 | |
|             {
 | |
|                 throw new ArgumentNullException("values");
 | |
|             }
 | |
| 
 | |
|             if (0 == values.Length)
 | |
|             {
 | |
|                 throw new ArgumentOutOfRangeException("values", values.Length, "must have at least one value");
 | |
|             }
 | |
| 
 | |
|             ColumnValue.RetrieveColumns(sesid, tableid, values);
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Efficiently retrieves a set of columns and their values from the
 | |
|         /// current record of a cursor or the copy buffer of that cursor.
 | |
|         /// </summary>
 | |
|         /// <param name="sesid">The session to use.</param>
 | |
|         /// <param name="tableid">The cursor to retrieve data from.</param>
 | |
|         /// <param name="grbit">Enumerate options.</param>
 | |
|         /// <returns>The discovered columns and their values.</returns>
 | |
|         public static IEnumerable<EnumeratedColumn> EnumerateColumns(
 | |
|             JET_SESID sesid,
 | |
|             JET_TABLEID tableid,
 | |
|             EnumerateColumnsGrbit grbit)
 | |
|         {
 | |
|             IEnumerable<EnumeratedColumn> enumeratedColumns;
 | |
|             Api.JetEnumerateColumns(sesid, tableid, grbit, out enumeratedColumns);
 | |
|             return enumeratedColumns;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Create the nullable return value.
 | |
|         /// </summary>
 | |
|         /// <typeparam name="T">The (struct) type to return.</typeparam>
 | |
|         /// <param name="data">The data retrieved from the column.</param>
 | |
|         /// <param name="dataSize">The size of the data.</param>
 | |
|         /// <param name="wrn">The warning code from esent.</param>
 | |
|         /// <param name="actualDataSize">The actual size of the data retireved fomr esent.</param>
 | |
|         /// <returns>A nullable struct of type T.</returns>
 | |
|         private static T? CreateReturnValue<T>(T data, int dataSize, JET_wrn wrn, int actualDataSize) where T : struct
 | |
|         {
 | |
|             if (JET_wrn.ColumnNull == wrn)
 | |
|             {
 | |
|                 return new T?();
 | |
|             }
 | |
| 
 | |
|             CheckDataSize(dataSize, actualDataSize);
 | |
|             return data;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Make sure the retrieved data size is at least as long as the expected size.
 | |
|         /// An exception is thrown if the data isn't long enough.
 | |
|         /// </summary>
 | |
|         /// <param name="expectedDataSize">The expected data size.</param>
 | |
|         /// <param name="actualDataSize">The size of the retrieved data.</param>
 | |
|         private static void CheckDataSize(int expectedDataSize, int actualDataSize)
 | |
|         {
 | |
|             if (actualDataSize != expectedDataSize)
 | |
|             {
 | |
|                 throw new EsentInvalidColumnException();
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Recursively pin the retrieve buffers in the JET_RETRIEVECOLUMN
 | |
|         /// structures and then retrieve the columns. This is done to avoid
 | |
|         /// creating GCHandles, which are expensive. This function pins
 | |
|         /// the current retrievecolumn structure (indicated by i) and then
 | |
|         /// recursively calls itself until all structures are pinned. This
 | |
|         /// is done because it isn't possible to create an arbitrary number
 | |
|         /// of pinned variables in a method.
 | |
|         /// </summary>
 | |
|         /// <param name="sesid">
 | |
|         /// The session to use.
 | |
|         /// </param>
 | |
|         /// <param name="tableid">
 | |
|         /// The table to retrieve from.
 | |
|         /// </param>
 | |
|         /// <param name="nativeretrievecolumns">
 | |
|         /// The nativeretrievecolumns structure.</param>
 | |
|         /// <param name="retrievecolumns">
 | |
|         /// The managed retrieve columns structure.
 | |
|         /// </param>
 | |
|         /// <param name="numColumns">The number of columns.</param>
 | |
|         /// <param name="i">The column currently being processed.</param>
 | |
|         /// <returns>An error code from JetRetrieveColumns.</returns>
 | |
|         private static unsafe int PinColumnsAndRetrieve(
 | |
|             JET_SESID sesid,
 | |
|             JET_TABLEID tableid,
 | |
|             NATIVE_RETRIEVECOLUMN* nativeretrievecolumns,
 | |
|             IList<JET_RETRIEVECOLUMN> retrievecolumns,
 | |
|             int numColumns,
 | |
|             int i)
 | |
|         {
 | |
|             // If consecutive JET_RETRIEVECOLUMN structures are using the same buffer then only pin it once.
 | |
|             fixed (byte* pinnedBuffer = retrievecolumns[i].pvData)
 | |
|             {
 | |
|                 do
 | |
|                 {
 | |
|                     retrievecolumns[i].CheckDataSize();
 | |
|                     retrievecolumns[i].GetNativeRetrievecolumn(ref nativeretrievecolumns[i]);
 | |
|                     nativeretrievecolumns[i].pvData = new IntPtr(pinnedBuffer + retrievecolumns[i].ibData);
 | |
|                     i++;
 | |
|                 }
 | |
|                 while (i < numColumns && retrievecolumns[i].pvData == retrievecolumns[i - 1].pvData);
 | |
| 
 | |
|                 return i == numColumns ?
 | |
|                     Impl.JetRetrieveColumns(sesid, tableid, nativeretrievecolumns, numColumns)
 | |
|                     : PinColumnsAndRetrieve(sesid, tableid, nativeretrievecolumns, retrievecolumns, numColumns, i);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Retrieve a Unicode (UTF16) string. This is optimized to take advantage of the fact
 | |
|         /// that no conversion is needed.
 | |
|         /// </summary>
 | |
|         /// <param name="sesid">The session to use.</param>
 | |
|         /// <param name="tableid">The table to retrieve from.</param>
 | |
|         /// <param name="columnid">The column to retrieve.</param>
 | |
|         /// <param name="grbit">Retrieve options.</param>
 | |
|         /// <returns>The string retrieved from the column.</returns>
 | |
|         private static unsafe string RetrieveUnicodeString(JET_SESID sesid, JET_TABLEID tableid, JET_COLUMNID columnid, RetrieveColumnGrbit grbit)
 | |
|         {
 | |
|             Debug.Assert((grbit & (RetrieveColumnGrbit)0x00020000) == 0, "UnpublishedGrbits.RetrieveAsRefIfNotInRecord is not supported.");
 | |
| 
 | |
|             // 512 Unicode characters (1kb on stack)
 | |
|             const int BufferSize = 512;
 | |
|             char* buffer = stackalloc char[BufferSize];
 | |
|             int actualDataSize;
 | |
|             JET_wrn wrn = JetRetrieveColumn(sesid, tableid, columnid, new IntPtr(buffer), BufferSize * sizeof(char), out actualDataSize, grbit, null);
 | |
|             if (JET_wrn.ColumnNull == wrn)
 | |
|             {
 | |
|                 return null;
 | |
|             }
 | |
| 
 | |
|             if (JET_wrn.Success == wrn)
 | |
|             {
 | |
|                 ////return StringCache.GetString(buffer, 0, actualDataSize);
 | |
|                 return new string(buffer, 0, actualDataSize / sizeof(char));
 | |
|             }
 | |
| 
 | |
|             Debug.Assert(JET_wrn.BufferTruncated == wrn, "Unexpected warning code");
 | |
| 
 | |
|             // Create a fake string of the appropriate size and then fill it in.
 | |
|             var s = new string('\0', actualDataSize / sizeof(char));
 | |
|             fixed (char* p = s)
 | |
|             {
 | |
|                 int newDataSize;
 | |
|                 wrn = JetRetrieveColumn(sesid, tableid, columnid, new IntPtr(p), actualDataSize, out newDataSize, grbit, null);
 | |
|                 if (JET_wrn.BufferTruncated == wrn)
 | |
|                 {
 | |
|                     string error = string.Format(
 | |
|                         CultureInfo.CurrentCulture,
 | |
|                         "Column size changed from {0} to {1}. The record was probably updated by another thread.",
 | |
|                         actualDataSize,
 | |
|                         newDataSize);
 | |
|                     Trace.TraceError(error);
 | |
|                     throw new InvalidOperationException(error);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             return s;
 | |
|         }
 | |
|     }
 | |
| }
 |