forked from nhibernate/nhibernate-core
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathOracleDataClientBatchingBatcher.cs
125 lines (106 loc) · 3.52 KB
/
OracleDataClientBatchingBatcher.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
using System;
using System.Collections;
using System.Data;
using System.Reflection;
namespace NHibernate.AdoNet
{
/// <summary>
/// Summary description for OracleDataClientBatchingBatcher.
/// By Tomer Avissar
/// </summary>
internal class OracleDataClientBatchingBatcher : BatcherImpl
{
private int batchSize;
private int countOfCommands = 0;
private int totalExpectedRowsAffected;
private IDbCommand currentBatch;
private Hashtable parameterValueArrayHashTable;
private Hashtable parameterIsAllNullsHashTable;
public OracleDataClientBatchingBatcher(ConnectionManager connectionManager)
: base(connectionManager)
{
batchSize = Factory.BatchSize;
}
public override void AddToBatch(IExpectation expectation)
{
bool firstOnBatch = true;
totalExpectedRowsAffected += expectation.ExpectedRowCount;
log.Info("Adding to batch");
LogCommand(CurrentCommand);
if (currentBatch == null)
{
// use first command as the batching command
currentBatch = CurrentCommand;
parameterValueArrayHashTable = new Hashtable();
//oracle does not allow array containing all null values
// so this HashTable is keeping track if all values are null or not
parameterIsAllNullsHashTable = new Hashtable();
}
else
{
firstOnBatch = false;
}
ArrayList parameterValueArray;
foreach (IDataParameter currentParameter in CurrentCommand.Parameters)
{
if (firstOnBatch)
{
parameterValueArray = new ArrayList();
parameterValueArrayHashTable.Add(currentParameter.ParameterName, parameterValueArray);
parameterIsAllNullsHashTable.Add(currentParameter.ParameterName, true);
}
else
{
parameterValueArray = parameterValueArrayHashTable[currentParameter.ParameterName] as ArrayList;
}
if (currentParameter.Value != System.DBNull.Value)
{
parameterIsAllNullsHashTable[currentParameter.ParameterName] = false;
}
parameterValueArray.Add(currentParameter.Value);
}
countOfCommands++;
if (countOfCommands >= batchSize)
{
DoExecuteBatch(currentBatch);
}
}
protected override void DoExecuteBatch(IDbCommand ps)
{
if (currentBatch != null)
{
int arraySize = 0;
countOfCommands = 0;
log.Info("Executing batch");
CheckReaders();
Prepare(currentBatch);
foreach (IDataParameter currentParameter in currentBatch.Parameters)
{
ArrayList parameterValueArray = parameterValueArrayHashTable[currentParameter.ParameterName] as ArrayList;
currentParameter.Value = parameterValueArray.ToArray();
arraySize = parameterValueArray.Count;
}
// setting the ArrayBindCount on the OracleCommand
// this value is not a part of the ADO.NET API.
// It's and ODP implementation, so it is being set by reflection
SetObjectParam(currentBatch, "ArrayBindCount", arraySize);
int rowsAffected = currentBatch.ExecuteNonQuery();
Expectations.VerifyOutcomeBatched(totalExpectedRowsAffected, rowsAffected);
totalExpectedRowsAffected = 0;
currentBatch = null;
parameterValueArrayHashTable = null;
}
}
private void SetObjectParam(Object obj, string paramName, object paramValue)
{
System.Type objType = obj.GetType();
PropertyInfo propInfo = objType.GetProperty(paramName);
propInfo.SetValue(obj, paramValue, null);
}
public override int BatchSize
{
get { return batchSize; }
set { batchSize = value; }
}
}
}