markus / ConvertService / ServiceBase / Markus.Service.DataBase.Dapper / MixParameters.cs @ master
이력 | 보기 | 이력해설 | 다운로드 (10.8 KB)
1 |
using Dapper; |
---|---|
2 |
using Npgsql; |
3 |
using NpgsqlTypes; |
4 |
using System; |
5 |
using System.Collections.Generic; |
6 |
using System.Data; |
7 |
using System.Linq; |
8 |
|
9 |
namespace Markus.Service.DataBase |
10 |
{ |
11 |
internal class MixParameters : SqlMapper.IDynamicParameters, SqlMapper.IParameterLookup |
12 |
{ |
13 |
public MixParameters(DBMSType dbms) |
14 |
{ |
15 |
DBMS = dbms; |
16 |
} |
17 |
|
18 |
private DBMSType DBMS = DBMSType.MSSQL; |
19 |
|
20 |
internal const DbType EnumerableMultiParameter = (DbType)(-1); |
21 |
private static readonly Dictionary<SqlMapper.Identity, Action<IDbCommand, object>> paramReaderCache = new Dictionary<SqlMapper.Identity, Action<IDbCommand, object>>(); |
22 |
private readonly Dictionary<string, ParamInfo> _parameters = new Dictionary<string, ParamInfo>(); |
23 |
private List<object> templates; |
24 |
|
25 |
/// <summary> |
26 |
/// All the names of the param in the bag, use Get to yank them out |
27 |
/// </summary> |
28 |
public IEnumerable<string> ParameterNames |
29 |
{ |
30 |
get { return _parameters.Select(p => p.Key); } |
31 |
} |
32 |
|
33 |
public int ParameterCount |
34 |
{ |
35 |
get { return _parameters.Count; } |
36 |
} |
37 |
|
38 |
object SqlMapper.IParameterLookup.this[string name] => |
39 |
_parameters.TryGetValue(name, out ParamInfo param) ? param.Value : null; |
40 |
|
41 |
internal static bool ShouldSetDbType(DbType? dbType) |
42 |
=> dbType.HasValue && dbType.GetValueOrDefault() != EnumerableMultiParameter; |
43 |
|
44 |
internal static bool ShouldSetDbType(DbType dbType) |
45 |
=> dbType != EnumerableMultiParameter; // just in case called with non-nullable |
46 |
|
47 |
void SqlMapper.IDynamicParameters.AddParameters(IDbCommand command, SqlMapper.Identity identity) |
48 |
{ |
49 |
if (templates != null) |
50 |
foreach (var template in templates) |
51 |
{ |
52 |
var newIdent = identity.ForDynamicParameters(template.GetType()); |
53 |
Action<IDbCommand, object> appender; |
54 |
|
55 |
lock (paramReaderCache) |
56 |
{ |
57 |
if (!paramReaderCache.TryGetValue(newIdent, out appender)) |
58 |
{ |
59 |
appender = SqlMapper.CreateParamInfoGenerator(newIdent, false, true); |
60 |
paramReaderCache[newIdent] = appender; |
61 |
} |
62 |
} |
63 |
|
64 |
appender(command, template); |
65 |
} |
66 |
|
67 |
switch (DBMS) |
68 |
{ |
69 |
case DBMSType.POSTGRESQL: |
70 |
case DBMSType.MSSQL: |
71 |
msSqlAddParamaters(command, identity); |
72 |
break; |
73 |
//case DBMSType.POSTGRESQL: |
74 |
// npgAddParamaters(command, identity); |
75 |
//break; |
76 |
default: |
77 |
break; |
78 |
} |
79 |
} |
80 |
|
81 |
private void msSqlAddParamaters(IDbCommand command, SqlMapper.Identity identity) |
82 |
{ |
83 |
foreach (var param in _parameters.Values) |
84 |
{ |
85 |
if (param.CameFromTemplate) continue; |
86 |
|
87 |
var dbType = param.DbType; |
88 |
var val = param.Value; |
89 |
string name = param.Name.ToLower(); |
90 |
|
91 |
if (command.CommandType == CommandType.StoredProcedure) |
92 |
name = Clean(param.Name); |
93 |
|
94 |
var isCustomQueryParameter = val is SqlMapper.ICustomQueryParameter; |
95 |
|
96 |
SqlMapper.ITypeHandler handler = null; |
97 |
if (dbType == null && val != null && !isCustomQueryParameter) |
98 |
{ |
99 |
#pragma warning disable 618 |
100 |
dbType = SqlMapper.LookupDbType(val.GetType(), name, true, out handler); |
101 |
#pragma warning disable 618 |
102 |
} |
103 |
if (isCustomQueryParameter) |
104 |
{ |
105 |
((SqlMapper.ICustomQueryParameter)val).AddParameter(command, name); |
106 |
} |
107 |
else if (dbType == EnumerableMultiParameter) |
108 |
{ |
109 |
#pragma warning disable 612, 618 |
110 |
SqlMapper.PackListParameters(command, name, val); |
111 |
#pragma warning restore 612, 618 |
112 |
} |
113 |
else |
114 |
{ |
115 |
bool add = !command.Parameters.Contains(name); |
116 |
IDbDataParameter p; |
117 |
if (add) |
118 |
{ |
119 |
p = command.CreateParameter(); |
120 |
p.ParameterName = name; |
121 |
} |
122 |
else |
123 |
{ |
124 |
p = (IDbDataParameter)command.Parameters[name]; |
125 |
} |
126 |
|
127 |
p.Direction = param.ParameterDirection; |
128 |
if (handler == null) |
129 |
{ |
130 |
#pragma warning disable 0618 |
131 |
p.Value = SqlMapper.SanitizeParameterValue(val); |
132 |
#pragma warning restore 0618 |
133 |
if (ShouldSetDbType(dbType) && p.DbType != dbType.GetValueOrDefault()) |
134 |
{ |
135 |
p.DbType = dbType.GetValueOrDefault(); |
136 |
} |
137 |
var s = val as string; |
138 |
if (s?.Length <= DbString.DefaultLength) |
139 |
{ |
140 |
p.Size = DbString.DefaultLength; |
141 |
} |
142 |
if (param.Size != null) p.Size = param.Size.Value; |
143 |
if (param.Precision != null) p.Precision = param.Precision.Value; |
144 |
if (param.Scale != null) p.Scale = param.Scale.Value; |
145 |
} |
146 |
else |
147 |
{ |
148 |
if (ShouldSetDbType(dbType)) p.DbType = dbType.GetValueOrDefault(); |
149 |
if (param.Size != null) p.Size = param.Size.Value; |
150 |
if (param.Precision != null) p.Precision = param.Precision.Value; |
151 |
if (param.Scale != null) p.Scale = param.Scale.Value; |
152 |
handler.SetValue(p, val ?? DBNull.Value); |
153 |
} |
154 |
|
155 |
if (add) |
156 |
{ |
157 |
command.Parameters.Add(p); |
158 |
} |
159 |
param.AttachedParam = p; |
160 |
} |
161 |
} |
162 |
|
163 |
// note: most non-privileged implementations would use: this.ReplaceLiterals(command); |
164 |
//if (literals.Count != 0) SqlMapper.ReplaceLiterals(this, command); |
165 |
} |
166 |
|
167 |
private void npgAddParamaters(IDbCommand command, SqlMapper.Identity identity) |
168 |
{ |
169 |
foreach (var param in _parameters.Values) |
170 |
{ |
171 |
if (((NpgsqlCommand)command).Parameters.Contains(param.Name)) |
172 |
{ |
173 |
((NpgsqlCommand)command).Parameters.Remove(param.Name); |
174 |
} |
175 |
|
176 |
var p = ParamInfoToNpgParam(param); |
177 |
p.ParameterName = param.Name; |
178 |
|
179 |
command.Parameters.Add(p); |
180 |
|
181 |
param.AttachedParam = p; |
182 |
} |
183 |
} |
184 |
|
185 |
private NpgsqlParameter ParamInfoToNpgParam(ParamInfo param) |
186 |
{ |
187 |
NpgsqlParameter p = new NpgsqlParameter(); |
188 |
|
189 |
var val = param.Value; |
190 |
p.Value = val ?? DBNull.Value; |
191 |
p.Direction = param.ParameterDirection; |
192 |
if (param.Size != null) p.Size = param.Size.Value; |
193 |
if (param.NpgsqlType != null) |
194 |
{ |
195 |
p.NpgsqlDbType = param.NpgsqlType.Value; |
196 |
} |
197 |
else if (param.Value != null) |
198 |
{ |
199 |
var npgType = dbTypeConvert.GetNpgsqlDbType(param.Value.GetType()).Value; |
200 |
|
201 |
if (npgType == NpgsqlDbType.Text) |
202 |
{ |
203 |
//npgType = NpgsqlDbType.Varchar; |
204 |
} |
205 |
|
206 |
p.NpgsqlDbType = npgType; |
207 |
} |
208 |
else if (param.DbType != null) |
209 |
{ |
210 |
var npgType = dbTypeConvert.GetNpgsqlDbType(param.DbType.Value).Value; |
211 |
|
212 |
if (npgType == NpgsqlDbType.Text) |
213 |
{ |
214 |
//npgType = NpgsqlDbType.Varchar; |
215 |
} |
216 |
|
217 |
p.NpgsqlDbType = npgType; |
218 |
} |
219 |
else |
220 |
{ |
221 |
throw new Exception($"{param.Name} npgSQL type을 알 수 없습니다."); |
222 |
} |
223 |
|
224 |
return p; |
225 |
} |
226 |
|
227 |
private static string Clean(string name) |
228 |
{ |
229 |
if (!string.IsNullOrEmpty(name)) |
230 |
{ |
231 |
switch (name[0]) |
232 |
{ |
233 |
case '@': |
234 |
case ':': |
235 |
case '?': |
236 |
return name.Substring(1); |
237 |
} |
238 |
} |
239 |
return name; |
240 |
} |
241 |
internal void Add(string name,object value = null,NpgsqlTypes.NpgsqlDbType? npgsqlDbType = null, DbType? dbType = null, ParameterDirection? direction = null, int? size = null) |
242 |
{ |
243 |
if(DBMS == DBMSType.MSSQL) |
244 |
{ |
245 |
switch (direction) |
246 |
{ |
247 |
case ParameterDirection.Output: |
248 |
case ParameterDirection.InputOutput: |
249 |
if(size == null) |
250 |
{ |
251 |
size = 50; |
252 |
//throw new Exception($"{name}는 MSSQL인 경우 direction이 {direction.ToString()}이므로 size를 지정하여야 합니다."); |
253 |
} |
254 |
break; |
255 |
default: |
256 |
break; |
257 |
} |
258 |
} |
259 |
|
260 |
_parameters[Clean(name)] = new ParamInfo |
261 |
{ |
262 |
Name = name, |
263 |
Value = value, |
264 |
ParameterDirection = direction ?? ParameterDirection.Input, |
265 |
DbType = dbType, |
266 |
NpgsqlType = npgsqlDbType, |
267 |
Size = size |
268 |
}; |
269 |
} |
270 |
|
271 |
internal T Get<T>(string name) |
272 |
{ |
273 |
var val = _parameters[name].AttachedParam.Value; |
274 |
if (val == DBNull.Value) |
275 |
{ |
276 |
if (default(T) != null) |
277 |
throw new ApplicationException("Attempting to cast a DBNull to a non nullable type!"); |
278 |
return default(T); |
279 |
} |
280 |
|
281 |
return (T)val; |
282 |
} |
283 |
|
284 |
|
285 |
public NpgsqlParameter Get(string name) |
286 |
{ |
287 |
return ParamInfoToNpgParam(_parameters[name]); |
288 |
} |
289 |
|
290 |
private class ParamInfo |
291 |
{ |
292 |
public string Name { get; set; } |
293 |
public object Value { get; set; } |
294 |
public ParameterDirection ParameterDirection { get; set; } |
295 |
public DbType? DbType { get; set; } |
296 |
|
297 |
public NpgsqlTypes.NpgsqlDbType? NpgsqlType { get; set; } |
298 |
public int? Size { get; set; } |
299 |
public IDbDataParameter AttachedParam { get; set; } |
300 |
internal Action<object, DynamicParameters> OutputCallback { get; set; } |
301 |
internal object OutputTarget { get; set; } |
302 |
internal bool CameFromTemplate { get; set; } |
303 |
|
304 |
public byte? Precision { get; set; } |
305 |
public byte? Scale { get; set; } |
306 |
} |
307 |
} |
308 |
} |