-
-
Notifications
You must be signed in to change notification settings - Fork 19
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
[DotNet] Fix generation of classes that accept optional lists as constructor arguments #249
[DotNet] Fix generation of classes that accept optional lists as constructor arguments #249
Conversation
…tructor arguments The template was generating code that was intolerant of arguments that were null when it expected a List<T>. Added a test case for the regression.
@@ -45,7 +45,7 @@ string id | |||
RequireNonNull<string>(description, "Description", "Background.Description cannot be null"); | |||
this.Description = description; | |||
RequireNonNull<List<Step>>(steps, "Steps", "Background.Steps cannot be null"); | |||
this.Steps = new List<Step>(steps); | |||
this.Steps = steps == null ? new List<Step>() : new List<Step>(steps); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
How can steps
be null
? It is guarded by RequireNonNull
on the line above.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hmm. Hadn't been dealing with Background yet, so hadn't seen that. At worst, it's a bit redundant.
But look at TestStep (before my proposed change):
public TestStep(
string hookId,
string id,
string pickleStepId,
List<string> stepDefinitionIds,
List<StepMatchArgumentsList> stepMatchArgumentsLists
)
{
this.HookId = hookId;
RequireNonNull<string>(id, "Id", "TestStep.Id cannot be null");
this.Id = id;
this.PickleStepId = pickleStepId;
this.StepDefinitionIds = new List<string>(stepDefinitionIds);
this.StepMatchArgumentsLists = new List<StepMatchArgumentsList>(stepMatchArgumentsLists);
}
The schema here indicates that the list of StepDefinitionIds and the list of StepMatchArgumentsLists are optional.
If this code gets invoked with null for either, the code throws an exception as the List constructor, when given an argument, the argument must not be null.
I added a test in BasicMessageSerializationTests.cs to demonstrate the behavior.
Would you prefer that I make the template a bit more sophisticated to use the new pattern only when the property is optional?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ah. Yes. More sophisticated would be preferred. You can probably copy this from Java.
Modified the codegen so that constructor argument arrays do not use the conditional check for null (as the NotNull check on the previous line would have done that already).
Modified the generation so that for arguments that are required, the check for null is not done when invoking the new List. |
Can I get some assistance on how to control indenting of the generated lines of code? It's not intuitive to me why some lines are generated at different indent levels. It must have something to do with being embedded in <%- if -%> conditions, but I'm not proficient on this templating language. |
You can find an overview of the ERB syntax here. https://www.puppet.com/docs/puppet/5.5/lang_template_erb It is not the canonical documentation, but much better explained. But it looks like |
this.Id = id; | ||
this.PickleStepId = pickleStepId; | ||
this.StepDefinitionIds = new List<string>(stepDefinitionIds); | ||
this.StepMatchArgumentsLists = new List<StepMatchArgumentsList>(stepMatchArgumentsLists); | ||
this.StepDefinitionIds = stepDefinitionIds == null ? new List<string>() : new List<string>(stepDefinitionIds); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think this should be:
this.StepDefinitionIds = stepDefinitionIds == null ? new List<string>() : new List<string>(stepDefinitionIds); | |
this.StepDefinitionIds = stepDefinitionIds == null ? null : new List<string>(stepDefinitionIds); |
The field is not required, so it can be null. When serializing, if configured correctly, this will result in the omission of the field. And so it would be strange if non-required value that wasn't present suddenly became an empty list after serialisation.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
In principle, I agree with you, purely optional elements should be null and not emitted when serialized.
I'm curious then about these lines in the Hooks example in the CCK:
{"testCase":{"id":"33","pickleId":"20","testSteps":[{"hookId":"0","id":"29"},{"hookId":"1","id":"30"},{"id":"31","pickleStepId":"19","stepDefinitionIds":["2"],"stepMatchArgumentsLists":[{"stepMatchArguments":[]}]},{"hookId":"4","id":"32"}]}}
{"testCase":{"id":"38","pickleId":"22","testSteps":[{"hookId":"0","id":"34"},{"hookId":"1","id":"35"},{"id":"36","pickleStepId":"21","stepDefinitionIds":["3"],"stepMatchArgumentsLists":[{"stepMatchArguments":[]}]},{"hookId":"4","id":"37"}]}}
{"testCase":{"id":"43","pickleId":"24","testSteps":[{"hookId":"0","id":"39"},{"hookId":"1","id":"40"},{"id":"41","pickleStepId":"23","stepDefinitionIds":[],"stepMatchArgumentsLists":[]},{"hookId":"4","id":"42"}]}}
In the first two lines, the third step of each test case is matched with a StepDefinition; each of those stepDefinitions accept no parameters. So why are the stepMatchArgumentsLists and stepMatchArguments properties being serialized as empty lists? Why not omit them altogether? (They are optional according to the schema).
Likewise in the third example (the test case does not bind to any step definition (aka, pending). Why is the StepDefinitionIds array getting serialized when it is empty?
I will try out your proposal and see how well it works with Reqnroll.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
A TestStep
is either a PickleTestStep
or a HookTestStep
.
- When a
PickleTestStep
then theid
,pickleStepId
,stepDefinitionIds
andstepMatchArgumentsLists
are required. - When a
HookTestStep
then theid
andhookId
are required.
So in the union of these, only the id
is required.
There are now better ways to express this in a json schemas but either we didn't have those at the time, or they didn't quite work.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Oof. That makes a general solution almost impossible.
If the template were to deserialize missing optional lists as null then they won't round-trip correctly. (Nor would it round-trip correctly with the code as it exists or with what I had proposed)
I'll consider inserting a patch up phase in the Serializer helper class.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
How does it make round tripping correctly impossible? Not including null values when serializing and not replacing absent/null values with empty list during deserialisation is a successful round trip.
Or do you mean with the current implementation?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Well, OK, not impossible.
Under good conditions, the round-trip would be successful.
Under the principle of being forgiving of bad input and strict on output; if the input json string (for a Pickle TestStep) were to omit the stepDefinitionIds instead of representing it as an empty array, then this code would set the StepDefinitionIds property to null, and render it back out during serialization as null (rather than the required empty array).
Are those edge cases something that we should be guarding against?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There might be a use-case for it, but it would be somewhat out of scope for this PR.
In this example, an error correction to an empty list would be context sensitive. The correct solution would depend on whether we are dealing with a PickleTestStep
or a HookTestStep
.
Given that sort of complexity, probably best discussed separately, if at all.
Oh btw. You should have the rights to make branches directly in this repository. You don't need to fork, that should avoid some of the work that comes with keeping the fork up to date. |
codegen/templates/dotnet.dotnet.erb
Outdated
@@ -73,10 +73,14 @@ public sealed class <%= class_name(key) %> | |||
required = (schema['required'] || []).index(property_name) | |||
-%> | |||
<% if required -%> |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
<% if required -%> | |
<%- if required -%> |
codegen/templates/dotnet.dotnet.erb
Outdated
@@ -73,10 +73,14 @@ public sealed class <%= class_name(key) %> | |||
required = (schema['required'] || []).index(property_name) | |||
-%> | |||
<% if required -%> | |||
RequireNonNull<<%= type_for(class_name(key), property_name, property) -%>>(<%= property_name %>, "<%= capitalize(property_name) %>", "<%= class_name(key) %>.<%= capitalize(property_name) %> cannot be null"); | |||
RequireNonNull<<%= type_for(class_name(key), property_name, property) -%>>(<%= property_name %>, "<%= capitalize(property_name) %>", "<%= class_name(key) %>.<%= capitalize(property_name) %> cannot be null"); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
RequireNonNull<<%= type_for(class_name(key), property_name, property) -%>>(<%= property_name %>, "<%= capitalize(property_name) %>", "<%= class_name(key) %>.<%= capitalize(property_name) %> cannot be null"); | |
RequireNonNull<<%= type_for(class_name(key), property_name, property) -%>>(<%= property_name %>, "<%= capitalize(property_name) %>", "<%= class_name(key) %>.<%= capitalize(property_name) %> cannot be null"); |
I think I marked the likely culprit of the indentation problems. Can't test this. I haven't got my regular development machine available right now. |
Thanks for the pointers and suggestions. I'll work on them tomorrow. |
@clrudolphi if you need any help with erb templating we can pair on it some time unless you've cracked it here. I can also explain the diff semantics on tags better if needed? |
Cleaned up formatting produced by the dotnet template. Adopted @mpkorstanje suggestion on how to handle null values within constructors for optional properties that are Lists.
@mpkorstanje accepted your suggestion of how to handle the nulls and fixed up the indent formatting. |
@@ -6,6 +6,7 @@ The format is based on [Keep a Changelog](http://keepachangelog.com/) | |||
and this project adheres to [Semantic Versioning](http://semver.org/). | |||
|
|||
## [Unreleased] | |||
- [Dotnet] Fixed code generation for types that accept List<T> as parameters. Constructors were not properly handling null input. [clrudolphi] |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Spotted this after merging. Changes should be listed as one of Added
, Changed
, ect. I'll fix it on main.
🤔 What's changed?
Modified the Dotnet template.
The template was generating code that was intolerant of arguments that were null when it expected a List.
Added a test case for the regression.
🏷️ What kind of change is this?
📋 Checklist:
This text was originally generated from a template, then edited by hand. You can modify the template here.