forked from nhibernate/nhibernate-core
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathRandomHqlGenerator.cs
109 lines (101 loc) · 3.71 KB
/
RandomHqlGenerator.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
using System;
using System.Collections.ObjectModel;
using System.Linq.Expressions;
using System.Reflection;
using NHibernate.Engine;
using NHibernate.Hql.Ast;
using NHibernate.Linq.Visitors;
using NHibernate.Util;
using Environment = NHibernate.Cfg.Environment;
namespace NHibernate.Linq.Functions
{
public class RandomHqlGenerator : BaseHqlGeneratorForMethod, IAllowPreEvaluationHqlGenerator
{
private readonly MethodInfo _nextDouble = ReflectHelper.GetMethod<Random>(r => r.NextDouble());
private const string _randomFunctionName = "random";
private const string _floorFunctionName = "floor";
public RandomHqlGenerator()
{
SupportedMethods = new[]
{
_nextDouble,
ReflectHelper.GetMethod<Random>(r => r.Next()),
ReflectHelper.GetMethod<Random>(r => r.Next(2)),
ReflectHelper.GetMethod<Random>(r => r.Next(-1, 1))
};
}
public override HqlTreeNode BuildHql(
MethodInfo method,
Expression targetObject,
ReadOnlyCollection<Expression> arguments,
HqlTreeBuilder treeBuilder,
IHqlExpressionVisitor visitor)
{
if (method == _nextDouble)
return treeBuilder.MethodCall(_randomFunctionName);
switch (arguments.Count)
{
case 0:
return treeBuilder.Cast(
treeBuilder.MethodCall(
_floorFunctionName,
treeBuilder.Multiply(
treeBuilder.MethodCall(_randomFunctionName),
treeBuilder.Constant(int.MaxValue))),
typeof(int));
case 1:
return treeBuilder.Cast(
treeBuilder.MethodCall(
_floorFunctionName,
treeBuilder.Multiply(
treeBuilder.MethodCall(_randomFunctionName),
visitor.Visit(arguments[0]).AsExpression())),
typeof(int));
case 2:
var minValue = visitor.Visit(arguments[0]).AsExpression();
var maxValue = visitor.Visit(arguments[1]).AsExpression();
return treeBuilder.Cast(
treeBuilder.Add(
treeBuilder.MethodCall(
_floorFunctionName,
treeBuilder.Multiply(
treeBuilder.MethodCall(_randomFunctionName),
treeBuilder.Subtract(maxValue, minValue))),
minValue),
typeof(int));
default:
throw new NotSupportedException();
}
}
/// <inheritdoc />
public bool AllowPreEvaluation(MemberInfo member, ISessionFactoryImplementor factory)
{
if (factory.Dialect.Functions.ContainsKey(_randomFunctionName) &&
(member == _nextDouble || factory.Dialect.Functions.ContainsKey(_floorFunctionName)))
return false;
if (factory.Settings.LinqToHqlFallbackOnPreEvaluation)
return true;
var functionName = factory.Dialect.Functions.ContainsKey(_randomFunctionName)
? _floorFunctionName
: _randomFunctionName;
throw new QueryException(
$"Cannot translate {member.DeclaringType.Name}.{member.Name}: {functionName} is " +
$"not supported by {factory.Dialect}. Either enable the fallback on pre-evaluation " +
$"({Environment.LinqToHqlFallbackOnPreEvaluation}) or evaluate {member.Name} " +
"outside of the query.");
}
/// <inheritdoc />
public bool IgnoreInstance(MemberInfo member)
{
// The translation ignores the Random instance, so long if it was specifically seeded: the user should
// pass the random value as a local variable in the Linq query in such case.
// Returning false here would cause the method, when appearing in a select clause, to be post-evaluated.
// Contrary to pre-evaluation, the post-evaluation is done for each row so it at least would avoid having
// the same random value for each result.
// But that would still be not executed in database which would be unexpected, in my opinion.
// It would even cause failures if the random instance used for querying is shared among threads or is
// too similarly seeded between queries.
return true;
}
}
}