- Home
- LLBLGen Pro
- Bugs & Issues
Bug: in Linq-To-LLBLGenPro fails to generate valid SQL query from LLBLGen
Joined: 28-Mar-2012
Hi,
I'm using the LLBLGen Runtime Version 5.9.2 using the adapter template.
I have noticed a very troubling bug in the Linq-To-LLBLGenPro functionality. I'm using VB.NET to formulate a Linq query, and depending on the alias chosen for the tables it generates a different query, one of them incorrect.
Here is query variant 1. This one works fine:
    Dim e = (From TLD In dbLinq.TLD
             Join TLA In dbLinq.TLA
               On TLA.TLA_Cod Equals TLD.TLA_Cod
             Where TLD.TArb_Cod = elem.TArb_Cod _
              AndAlso TLD.TLK_Peri = elem.TLK_Peri _
              AndAlso TLD.TLK_Typ = elem.TLK_Typ _
              AndAlso TLD.TLK_Nr = elem.TLK_Nr _
              AndAlso {"A", "B", "C", "D", "E"}.Contains(TLA.TTypLA_Cod) _
              AndAlso TLA.TLA_CalcYN
             Select TLD.TLD_Ges).Sum
Generated query:
    SELECT TOP(@p2) SUM([LPA_L1].[TLD_Ges]) AS [LPAV_] 
     FROM (SELECT [LPA_L2].[TLD_Ges] 
            FROM ([mydb].[dbo].[TLD] [LPA_L2] 
             INNER JOIN [mydb].[dbo].[TLA] [LPA_L3] ON [LPA_L2].[TLA_Cod] = [LPA_L3].[TLA_Cod]) 
            WHERE ( ( ( ( ( ( ( ( [LPA_L2].[TArb_Cod] = @p3) 
              AND ( [LPA_L2].[TLK_Peri] = @p4)) 
              AND ( [LPA_L2].[TLK_Typ] = @p5)) 
              AND ( [LPA_L2].[TLK_Nr] = @p6)) 
              AND ( [LPA_L3].[TTypLA_Cod] IN (@p7, @p8, @p9, @p10, @p11))) 
              AND ( [LPA_L3].[TLA_CalcYN] = @p12))))
     ) [LPA_L1]
Here is query variant 2. The only change is the lowercase letter for the table name alias in the linq query. This one fails:
    Dim o = (From tLD In dbLinq.TLD
             Join tLA In dbLinq.TLA
               On tLA.TLA_Cod Equals tLD.TLA_Cod
             Where tLD.TArb_Cod = elem.TArb_Cod _
              AndAlso tLD.TLK_Peri = elem.TLK_Peri _
              AndAlso tLD.TLK_Typ = elem.TLK_Typ _
              AndAlso tLD.TLK_Nr = elem.TLK_Nr _
              AndAlso {"A", "B", "C", "D", "E"}.Contains(tLA.TTypLA_Cod) _
              AndAlso tLA.TLA_CalcYN
             Select tLD.TLD_Ges).Sum
Generated query:
    SELECT TOP(@p2) SUM([LPA_L1].[TLD_Ges]) AS [LPAV_] 
     FROM (SELECT [LPLA_3].[TLD_Ges] 
            FROM ([mydb].[dbo].[TLD] [LPA_L2] 
             INNER JOIN [mydb].[dbo].[TLA] [LPA_L3] ON [LPA_L2].[TLA_Cod] = [LPA_L3].[TLA_Cod])
            WHERE ( ( ( ( ( ( ( ( [LPLA_3].[TArb_Cod] = @p3) 
              AND ( [LPLA_3].[TLK_Peri] = @p4)) 
              AND ( [LPLA_3].[TLK_Typ] = @p5)) 
              AND ( [LPLA_3].[TLK_Nr] = @p6)) 
              AND ( [LPLA_3].[TTypLA_Cod] IN (@p7, @p8, @p9, @p10, @p11))) 
              AND ( [LPLA_3].[TLA_CalcYN] = @p12))))
           ) [LPA_L1]
You can see that the alias for TLD is still LPA_L2 as before, but the SELECT and the WHERE clauses all reference an alias LPLA_3 that does not exist.
This throws a SD.LLBLGen.Pro.ORMSupportClasses.ORMQueryExecutionException with the following text:
An exception was caught during the execution of a retrieval query: The multi-part identifier "LPLA_3.TArb_Cod" could not be bound.
The multi-part identifier "LPLA_3.TLK_Peri" could not be bound.
The multi-part identifier "LPLA_3.TLK_Typ" could not be bound.
The multi-part identifier "LPLA_3.TLK_Nr" could not be bound.
The multi-part identifier "LPLA_3.TTypLA_Cod" could not be bound.
The multi-part identifier "LPLA_3.TTypLA_Cod" could not be bound.
The multi-part identifier "LPLA_3.TTypLA_Cod" could not be bound.
The multi-part identifier "LPLA_3.TTypLA_Cod" could not be bound.
The multi-part identifier "LPLA_3.TTypLA_Cod" could not be bound.
The multi-part identifier "LPLA_3.TLA_CalcYN" could not be bound.
The multi-part identifier "LPLA_3.TLD_Ges" could not be bound..                          
What's troubling is that the only difference is one lowercase letter in the alias.
For good measure, here another query variant that works fine:
    Dim y = (From det In dbLinq.TLD
             Join la In dbLinq.TLA
               On la.TLA_Cod Equals det.TLA_Cod
             Where det.TArb_Cod = elem.TArb_Cod _
              AndAlso det.TLK_Peri = elem.TLK_Peri _
              AndAlso det.TLK_Typ = elem.TLK_Typ _
              AndAlso det.TLK_Nr = elem.TLK_Nr _
              AndAlso {"A", "B", "C", "D", "E"}.Contains(la.TTypLA_Cod) _
              AndAlso la.TLA_CalcYN
             Select det.TLD_Ges).Sum
Could you please look into that.
Thanks,
andreas
| Filename | File size | Added on | Approval | 
|---|---|---|---|
| Queries.zip | 3,745 | 23-Nov-2022 17:15.37 | Approved | 
This doesn't make any sense. The casing shouldn't matter, as all elements are used properly (you're not mixing casing which happens to compile on vb.net because it's case insensitive). My suspicion is that the compiler makes a mistake somewhere (usually blaming a compiler is silly, but in vb.net it's not) where it emits the wrong expression tree.
Could you check if the expression trees of both are the same? I.e. by using tostring on the IQueryable?
Joined: 28-Mar-2012
Indeed, it does not make sense. FYI, removing the ().Sum() at the end does not change anything.
What do you mean by ToString on the IQueryable? Simply calling .ToString() instead of .Sum() yields nothing of interest (just the type name as a string).
acl wrote:
Indeed, it does not make sense. FYI, removing the ().Sum() at the end does not change anything.
What do you mean by ToString on the IQueryable? Simply calling .ToString() instead of .Sum() yields nothing of interest (just the type name as a string).
Remove the Sum() call and then in the debugger, put a break point on the statement after the Dim e=.... If you now hover over the 'e' variable it should show a popup (at least in vs) which offers you a debug string which represents the expression tree as string. Please give these for both.
I think it's vb.net's that's messing up. I see something's wrong here:
.Lambda #Lambda3<System.Func`3[Our.Project.Library.EntityClasses.TLDEntity,Our.Project.Library.EntityClasses.TLAEntity,VB$AnonymousType_0`2[Our.Project.Library.EntityClasses.TLDEntity,Our.Project.Library.EntityClasses.TLAEntity]]>(
    Our.Project.Library.EntityClasses.TLDEntity $tLD,
    Our.Project.Library.EntityClasses.TLAEntity $tLA) {
    .New VB$AnonymousType_0`2[Our.Project.Library.EntityClasses.TLDEntity,Our.Project.Library.EntityClasses.TLAEntity](
        $tLD,      <<<<<<<<<<<< lower case!
        $tLA)
}
.Lambda #Lambda4<System.Func`2[VB$AnonymousType_0`2[Our.Project.Library.EntityClasses.TLDEntity,Our.Project.Library.EntityClasses.TLAEntity],System.Boolean]>(VB$AnonymousType_0`2[Our.Project.Library.EntityClasses.TLDEntity,Our.Project.Library.EntityClasses.TLAEntity] $$VB$It)
{
    .Call Microsoft.VisualBasic.CompilerServices.Operators.CompareString(
        ($$VB$It.TLD).TArb_Cod,          <<<<<<<<< Upper case!
        (.Constant<Our.Project.Class+_Closure$__121-1>(Our.Project.Class+_Closure$__121-1).$VB$Local_elem).TArb_Cod,
        False) == 0 && ($$VB$It.TLD).TLK_Peri == (.Constant<Our.Project.Class+_Closure$__121-1>(Our.Project.Class+_Closure$__121-1).$VB$Local_elem).TLK_Peri
    && .Call Microsoft.VisualBasic.CompilerServices.Operators.CompareString(
        ($$VB$It.TLD).TLK_Typ,
        (.Constant<Our.Project.Class+_Closure$__121-1>(Our.Project.Class+_Closure$__121-1).$VB$Local_elem).TLK_Typ,
        False) == 0 && ($$VB$It.TLD).TLK_Nr == (.Constant<Our.Project.Class+_Closure$__121-1>(Our.Project.Class+_Closure$__121-1).$VB$Local_elem).TLK_Nr
    && .Call System.Linq.Enumerable.Contains(
        (System.Collections.Generic.IEnumerable`1[System.String]).NewArray System.String[] {
            .Call Our.Helper.Library.EnumsExtensionModule.StringValue((System.Enum).Constant<Our.Project.Library.TTypLAEnum>(ValA))
            ,
            .Call Our.Helper.Library.EnumsExtensionModule.StringValue((System.Enum).Constant<Our.Project.Library.TTypLAEnum>(ValB))
            ,
            .Call Our.Helper.Library.EnumsExtensionModule.StringValue((System.Enum).Constant<Our.Project.Library.TTypLAEnum>(ValC))
            ,
            .Call Our.Helper.Library.EnumsExtensionModule.StringValue((System.Enum).Constant<Our.Project.Library.TTypLAEnum>(ValD))
            ,
            .Call Our.Helper.Library.EnumsExtensionModule.StringValue((System.Enum).Constant<Our.Project.Library.TTypLAEnum>(ValE))
        },
        ($$VB$It.TLA).TTypLA_Cod) && ($$VB$It.TLA).TLA_CalcYN
}
.Lambda #Lambda5<System.Func`2[VB$AnonymousType_0`2[Our.Project.Library.EntityClasses.TLDEntity,Our.Project.Library.EntityClasses.TLAEntity],System.Decimal]>(VB$AnonymousType_0`2[Our.Project.Library.EntityClasses.TLDEntity,Our.Project.Library.EntityClasses.TLAEntity] $$VB$It)
{
    ($$VB$It.TLD).TLD_Ges
}
Creating an anonymous type by specifying tLD and tLA, creates 2 fields in the anonymous type with those names. However the lambda exposes it in the call with upper case characters. (see my markings). This creates a problem as they're different and therefore get a different alias assigned to them. 
I have honestly no idea why the VB.NET compiler makes this mistake. Perhaps because everything is case insensitive but for handlers of the lambda expression, case sensitivity is essential.
Joined: 28-Mar-2012
Wow, that's interesting. What is strange is that I have lots of queries with lowercase letters aliases, but the query above is the only one that actually throws an error.
So nothing both of us can do, right?
Seems that I need to file a bug with Microsoft, not sure that's worth the effort.
acl wrote:
Wow, that's interesting. What is strange is that I have lots of queries with lowercase letters aliases, but the query above is the only one that actually throws an error.
So nothing both of us can do, right?
Seems that I need to file a bug with Microsoft, not sure that's worth the effort.
We can't do much about this, no. The VB team has been abandoned so I doubt there's anyone fixing stuff at this point. But, perhaps it's something else, could you test this?
Dim y = (From deT In dbLinq.TLD
         Join lA In dbLinq.TLA
           On lA.TLA_Cod Equals deT.TLA_Cod
         Where deT.TArb_Cod = elem.TArb_Cod _
          AndAlso deT.TLK_Peri = elem.TLK_Peri _
          AndAlso deT.TLK_Typ = elem.TLK_Typ _
          AndAlso deT.TLK_Nr = elem.TLK_Nr _
          AndAlso {"A", "B", "C", "D", "E"}.Contains(lA.TTypLA_Cod) _
          AndAlso lA.TLA_CalcYN
         Select deT.TLD_Ges).Sum
As your 3rd example works fine, but that would create the same problem, so it might be the difference I marked above isn't the real problem. It's the one difference I could spot in the trees which are otherwise identical. Perhaps the VB.NET compiler creates an all caps reference as soon as there's 1 character in upper case.
Joined: 28-Mar-2012
I did what you asked. So your query variant works.
I also tried the following, which does not work (its the same as the other query that didn't work, but with uppercase and lowercase inverted):
Dim b = (From Tld In dbLinq.TLD
           Join Tla In dbLinq.TLA
             On Tla.TLA_Cod Equals Tld.TLA_Cod
           Where Tld.TArb_Cod = elem.TArb_Cod _
            AndAlso Tld.TLK_Peri = elem.TLK_Peri _
            AndAlso Tld.TLK_Typ = elem.TLK_Typ _
            AndAlso Tld.TLK_Nr = elem.TLK_Nr _
            AndAlso {"A", "B", "C", "D", "E"}.Contains(Tla.TTypLA_Cod) _
            AndAlso Tla.TLA_CalcYN
           Select Tld.TLD_Ges)
I believe that the problem occurs only when an alias is used, whose name is identical to the table name, but not using the same case as the table name. That's what happens in the two "bad" queries. It's not happening in the others.
I replaced the original attachment by a zip file with 4 queries and expressions. The two from my first post, the one from your post and the one from this post.
Very strange problem, we can leave it at that. 
Ok, so as long as the variable name is equal to the entity property name but with different casing it won't work, but if you pick another name that's not equal to the entity property name it will work.... weird! 
If you can live with it, we can leave it with this, as I have no idea what to do about it, sorry ...
acl wrote:
I can live with it. Thanks for taking your time to look into it.
I think it might even be related to compiler version or something. I use Rider 2022.1.2, which uses
C:\Program Files\Microsoft Visual Studio\2022\Professional\MSBuild\Current\Bin\Roslyn\vbc.exe
on .NET FX: 
I tried to reproduce it, but here things are done alright, see the expression tree, where the names are correct now, including the casing.
value(SD.LLBLGen.Pro.LinqSupportClasses.DataSource2`1[NW26.Adapter.EntityClasses.OrderEntity])
    .Join(value(SD.LLBLGen.Pro.LinqSupportClasses.DataSource2`1[NW26.Adapter.EntityClasses.CustomerEntity]), 
        oRDER => oRDER.CustomerId, 
        cUSTOMER => cUSTOMER.CustomerId, 
        (oRDER, cUSTOMER) => new VB$AnonymousType_7`2(oRDER = oRDER, cUSTOMER = cUSTOMER))
    .Where($VB$It => ((($VB$It.oRDER.EmployeeId == ConvertChecked(2)) 
        AndAlso Convert(Convert(new [] {"USA", "Germany"}).Contains($VB$It.cUSTOMER.Country))) ?? False))
    .Select($VB$It => $VB$It.oRDER.OrderId)
original query:
<Test()>
Public Sub WeirdNameCasingIssueTest()
    Using adapter as New DataAccessAdapter()
        Dim metaData as new LinqMetaData(adapter)
        Dim q = From oRDER in metaData.Order
                Join cUSTOMER in metaData.Customer
                on cUSTOMER.CustomerId equals oRDER.CustomerId
                Where oRDER.EmployeeId = 2 _
                    AndAlso { "USA", "Germany" }.Contains(cUSTOMER.Country) _
                Select oRDER.OrderId
        Dim r = q.ToList()
    End Using
End Sub
SELECT [LPA_L1].[OrderID] AS [OrderId]
FROM   ([Northwind].[dbo].[Orders] [LPA_L1]
        INNER JOIN [Northwind].[dbo].[Customers] [LPA_L2]
            ON [LPA_L1].[CustomerID] = [LPA_L2].[CustomerID])
WHERE  (((((([LPA_L1].[EmployeeID] = @p1)
        AND ([LPA_L2].[Country] IN (@p2, @p3))))))) 
Perhaps worth looking into, if you try to use the 2022 vb.net compiler. Also compiles and runs fine on .net core 3.1
 
						