J'obtiens un comportement inattendu avec un QueryOver JoinAlias dans NHibernate.
Mon entité ressemble essentiellement à ceci :
public class Field
{
public virtual long Id { get; protected set; }
public virtual Field Parent { get; protected set; }
public virtual FieldType Type { get; protected set; }
public virtual string Value { get; protected set; }
...(Ctors etc
}
Ma cartographie est telle :
public class FieldMap : ClassMap<Field>
{
public FieldMap()
{
Id(x => x.Id)
.GeneratedBy.Native();
References(x => x.Type)
.Column("FieldTypeId")
.LazyLoad()
.Cascade.All()
;
Map(x => x.Value);
References(x => x.Parent)
.Column("ParentFieldId")
.Nullable()
.LazyLoad()
.Cascade.All()
;
}
Ma question :
Field fieldAlias = null;
string typeAlias = null;
Field parentFieldAlias = null;
var query = getSession().QueryOver<Field>(() => fieldAlias)
.JoinAlias(() => fieldAlias.Type, () => typeAlias)
.Where(() => typeAlias.Name == type)
.JoinAlias(() => fieldAlias.Parent, () => parentFieldAlias)
.Where(() => (parentFieldAlias.Value == parentValue) || (parentFieldAlias == null))
;
En ce qui me concerne, cela devrait me donner un SQL comme celui-ci :
... WHERE (a.ParentFieldId == NULL) OR (a.ParentFieldId = c.FieldId AND c.Value = parentValue)
Mais j'obtiens une exception de référence nulle. (Je suppose que lorsque l'alias est résolu et que le Parent est nul).
Les détails de l'exception sont :
System.NullReferenceException occurred
HResult=-2147467261
Message=Object reference not set to an instance of an object.
Source=NHibernate
StackTrace:
at NHibernate.Criterion.ConstantProjection..ctor(Object value)
at NHibernate.Criterion.Projections.Constant(Object obj)
at NHibernate.Impl.ExpressionProcessor.FindMemberProjection(Expression expression)
at NHibernate.Impl.ExpressionProcessor.ProcessSimpleExpression(Expression left, Expression right, ExpressionType nodeType)
at NHibernate.Impl.ExpressionProcessor.ProcessSimpleExpression(BinaryExpression be)
at NHibernate.Impl.ExpressionProcessor.ProcessBinaryExpression(BinaryExpression expression)
at NHibernate.Impl.ExpressionProcessor.ProcessExpression(Expression expression)
at NHibernate.Impl.ExpressionProcessor.ProcessOrExpression(BinaryExpression expression)
at NHibernate.Impl.ExpressionProcessor.ProcessBinaryExpression(BinaryExpression expression)
at NHibernate.Impl.ExpressionProcessor.ProcessExpression(Expression expression)
at NHibernate.Impl.ExpressionProcessor.ProcessLambdaExpression(LambdaExpression expression)
at NHibernate.Impl.ExpressionProcessor.ProcessExpression(Expression`1 expression)
at NHibernate.Criterion.QueryOver`2.Add(Expression`1 expression)
at NHibernate.Criterion.QueryOver`2.Where(Expression`1 expression)
at NHibernate.Criterion.QueryOver`2.NHibernate.IQueryOver<TRoot,TSubType>.Where(Expression`1 expression)
at Ismoos.Director.FieldOptionsQuery.Execute(Service service, String type, String parentValue) in D:\Work\Ismoos\Ismoos\Director\Ismoos.Director\FieldOptionsQuery.cs:line 31
InnerException:
J'ai essayé plusieurs méthodes différentes, notamment :
.JoinAlias(() => fieldAlias.Parent, () => parentFieldAlias)
.Where(Restrictions.Or(Restrictions.On(() => fieldAlias.Parent).IsNotNull,
Restrictions.On(() => parentFieldAlias.Value).IsLike(parentValue))))
mais aucune d'entre elles ne fonctionne.
J'ai trouvé une solution de contournement, en laissant de côté la restriction sur la valeur du champ parent dans la requête, et en effectuant une requête LINQ après le retour de QueryOver, comme suit :
Field fieldAlias = null;
string typeAlias = null;
Field parentFieldAlias = null;
var query = getSession().QueryOver<Field>(() => fieldAlias)
.JoinAlias(() => fieldAlias.Type, () => typeAlias)
.Where(() => typeAlias.Name == type)
;
var list = query
.List<Field>()
;
return list
.Where(x => (x.Parent == null) || (x.Parent.Value == parentValue))
.ToList();
mais ce n'est pas aussi optimal que de le faire dans le QueryOver.
Des suggestions ?