X Tutup
using System; using System.Data; using System.Diagnostics; using System.Runtime.CompilerServices; using System.Threading; using System.Threading.Tasks; using Npgsql.Internal; using NpgsqlTypes; namespace Npgsql; /// /// A generic version of which provides more type safety and /// avoids boxing of value types. Use instead of . /// /// The type of the value that will be stored in the parameter. public sealed class NpgsqlParameter : NpgsqlParameter { T? _typedValue; /// /// Gets or sets the strongly-typed value of the parameter. /// public T? TypedValue { get => _typedValue; set { if (typeof(T) == typeof(object) && ShouldResetObjectTypeInfo(value)) ResetTypeInfo(); else ResetBindingInfo(); _typedValue = value; } } /// /// Gets or sets the value of the parameter. This delegates to . /// public override object? Value { get => TypedValue; set => TypedValue = (T)value!; } private protected override Type StaticValueType => typeof(T); #region Constructors /// /// Initializes a new instance of . /// public NpgsqlParameter() { } /// /// Initializes a new instance of with a parameter name and value. /// public NpgsqlParameter(string parameterName, T value) { ParameterName = parameterName; TypedValue = value; } /// /// Initializes a new instance of with a parameter name and type. /// public NpgsqlParameter(string parameterName, NpgsqlDbType npgsqlDbType) { ParameterName = parameterName; NpgsqlDbType = npgsqlDbType; } /// /// Initializes a new instance of with a parameter name and type. /// public NpgsqlParameter(string parameterName, DbType dbType) { ParameterName = parameterName; DbType = dbType; } #endregion Constructors private protected override void SetOutputValueCore(NpgsqlDataReader reader, int ordinal) => TypedValue = reader.GetFieldValue(ordinal); private protected override PgConverterResolution ResolveConverter(PgTypeInfo typeInfo) { if (typeof(T) == typeof(object) || TypeInfo!.IsBoxing) return base.ResolveConverter(typeInfo); _asObject = false; return typeInfo.GetResolution(TypedValue); } // We ignore allowNullReference, it's just there to control the base implementation. private protected override void BindCore(DataFormat? formatPreference, bool allowNullReference = false) { if (_asObject) { // If we're object typed we should not support null. base.BindCore(formatPreference, typeof(T) != typeof(object)); return; } var value = TypedValue; if (TypeInfo!.Bind(Converter!.UnsafeDowncast(), value, out var size, out _writeState, out var dataFormat, formatPreference) is { } info) { WriteSize = size; _bufferRequirement = info.BufferRequirement; } else { WriteSize = -1; _bufferRequirement = default; } Format = dataFormat; } private protected override ValueTask WriteValue(bool async, PgWriter writer, CancellationToken cancellationToken) { if (_asObject) return base.WriteValue(async, writer, cancellationToken); if (async) return Converter!.UnsafeDowncast().WriteAsync(writer, TypedValue!, cancellationToken); Converter!.UnsafeDowncast().Write(writer, TypedValue!); return new(); } private protected override NpgsqlParameter CloneCore() => // use fields instead of properties // to avoid auto-initializing something like type_info new NpgsqlParameter { _precision = _precision, _scale = _scale, _size = _size, _dbType = _dbType, _npgsqlDbType = _npgsqlDbType, _dataTypeName = _dataTypeName, Direction = Direction, IsNullable = IsNullable, _name = _name, TrimmedName = TrimmedName, SourceColumn = SourceColumn, SourceVersion = SourceVersion, TypedValue = TypedValue, SourceColumnNullMapping = SourceColumnNullMapping, }; }
X Tutup