Renaming a Navigator on a nullable field clears the "Optional" flag for the underlying field

Posts   
 
    
TL66
User
Posts: 5
Joined: 04-Dec-2020
# Posted on: 21-Jan-2021 16:00:39   

LLBLGen Pro 5.7.3, using Database First:

I believe this is a both a reverse engineer and designer issue, not related to the target framework.

Scenario: (MSSQL server) With SQL tables configured as below

CREATE TABLE dbo.A (
  Pk1 INT not null primary key
);
CREATE TABLE dbo.C (
 Pk1 INT not null,
 Pk2 INT not null,
 [Data] nvarchar(100) null,
 CONSTRAINT [PK_C] PRIMARY KEY (Pk1,Pk2),
 CONSTRAINT [FK_C_A] FOREIGN KEY (Pk1) REFERENCES dbo.A (Pk1)
);

CREATE TABLE dbo.B (
  Pk1 INT not null,
  Pk2 INT not null,
 Link INT null,
 CONSTRAINT [PK_B] PRIMARY KEY (Pk1,Pk2),
 CONSTRAINT [FK_B_A] FOREIGN KEY (Pk1) REFERENCES dbo.A (Pk1),
 CONSTRAINT [FK_B_Link] FOREIGN KEY (Pk1,Link) REFERENCES dbo.C (Pk1,Pk2)
);

There are two issues around this.

  1. When reverse engineering Tables to Entities, the model created sets field B.Link as not optional.

  2. If edit B.Link in designer, and set the "Optional" flag (so correct underlying object model is created), then renaming the Navigator associate with B.Link causes the "Optional" flag for B.Link to be cleared.

Inspecting the Model Relational data (created by database first process) confirms that B.Link field has correctly has the "Is Nullable" value set to True.

If the "Optional" flag for B.Link is not manually corrected before generating code, this bug results in broken code where entities of type B cannot be loaded if any of the rows contain NULL entries for Link column.

Walaa avatar
Walaa
Support Team
Posts: 14987
Joined: 21-Aug-2005
# Posted on: 21-Jan-2021 23:48:01   

I can't reproduce the first issue. I created a new project based on Northwind database, and the Product Entity has IsOptional set to true for the SupplierId and the CategoryId fields. The same for the second issue, renaming the Navigators doesn't rest the IsOptional flag.

TL66
User
Posts: 5
Joined: 04-Dec-2020
# Posted on: 22-Jan-2021 00:07:40   

Walaa wrote:

I can't reproduce the first issue. I created a new project based on Northwind database, and the Product Entity has IsOptional set to true for the SupplierId and the CategoryId fields. The same for the second issue, renaming the Navigators doesn't rest the IsOptional flag.

If you create a test database - then run the above SQL script against that database, it will demonstrate the problem. It's also a much simpler test case compared to Northwind to focus on the problem. I tested using the script above before posting this.

I suspect that compound foreign key is a key element in the issue where one part is not nullable, and the other part is. In my example, the not nullable part is also part of PK for that table.

Attached is the LLBLGen project file created when doing the test. To demo item 1, simply delete the model data, then re-create re reverse engineering tables to entities. At the end of reverse a warning conveniently highlights the issue:

DZ0009: The mapping of sub-element 'Link' of entity 'B' for the target ... has following issue The IsNullable flag on the target field 'dbo'B' is set to 'True' while the IsOptional flag flag of the mapped field 'Link (FK)' is set to 'False'

Checking the Optional flag for B.Link removes the warning. Point is the reverse engineering got it wrong in the first place, then gets it wrong again when editing the Navigator using B.Link.

Attachments
Filename File size Added on Approval
Test.llblgenproj 16,383 22-Jan-2021 00:27.30 Approved
Otis avatar
Otis
LLBLGen Pro Team
Posts: 39769
Joined: 17-Aug-2003
# Posted on: 22-Jan-2021 09:51:18   

We'll look into it

Frans Bouma | Lead developer LLBLGen Pro
Otis avatar
Otis
LLBLGen Pro Team
Posts: 39769
Joined: 17-Aug-2003
# Posted on: 22-Jan-2021 10:32:17   

Reasoning about what's going on, I'll be honest, I can only conclude: for me the construction doesn't make much sense: if the PK side is optional (so the FK fields can be nullable), it's weird to make these part of the PK, but alas, it's a given there are databases out there which are, let's say, unconventional. wink

That said, we should support this scenario and currently don't. E.g. we also support compound FKs with FK fields which are used in multiple FKs (which this model appears to have too btw), it doesn't make sense theoretically but it happens....

Reverse engineering B alone things are ok, but reversing C after that indeed resets the field, it's done in the FK field assignment. It's a bug in our code (or rather: a limitation as the rule is too strict) : it checks if all FK fields are nullable and only then it markes the PK side as optional. As it doesn't mark it as optional, assigning the FK fields enforces all FK fields to be non-optional (as otherwise the PK side can be optional).

Tip

As a current workaround till we have fixed this: editing the relationship manually, setting it to Many to 0..1 instead of Many to 1 will allow you to set the Link field to optional and rename the navigator.

We'll look into fixing this restriction today.

Important!

I see you're using EF 6. I'm not sure if EF 6 supports this tho, but the validator won't object as it won't test on this

Frans Bouma | Lead developer LLBLGen Pro
Otis avatar
Otis
LLBLGen Pro Team
Posts: 39769
Joined: 17-Aug-2003
# Posted on: 22-Jan-2021 11:30:04   

The fix is now available: v5.6.6 hotfix and v5.7.3 hotfix builds.

Frans Bouma | Lead developer LLBLGen Pro