Skip to content
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

PlainText Serializer incorrectly unmarshalls certain nested clusters/arrays #5

Closed
IlyaVino opened this issue Jan 11, 2022 · 11 comments
Closed

Comments

@IlyaVino
Copy link

I believe I have found a fix and will submit an incomplete pull request shortly, but an additional bug surfaces that prevents all the unit tests I added from passing:

If the datatype contains a nested numeric array, then the numeric values are not preserved. As far as I can tell, this second bug occurs inside OpenVariant.lvlib:Variant_KVP to Strict Variant.vi. I've submitted a separate issue inside that repository.

@francois-normandin
Copy link
Member

Can you provide a test vector for this issue?
What is an example of an input that does not deserialize correctly? And what do you think is the expected output of this deserialization?

VI or screenshots would help. Thanks.

@IlyaVino
Copy link
Author

Here with a modified set of tests demonstrating the issue.

Test Serializer.Text.zip

@IlyaVino
Copy link
Author

Some more context as requested in the pull request:

I'm interested in serializing/deserializing an array of clusters that contain a name string and numeric array. These are phase cycling schemes for a spectrometer. I was using the open source configuration manager to save/read these to/from a text file when I ran into the issue.

If I use the cluster array described above I get the serialized string: [{"A",[1,2,3]},{"B",[4,5,6]}], which throws an invalid type error upon deserializing. This error is thrown by Serializer.PlainText.lvclass:PlainTextToVariant.vi. My understanding is that this occurs because PlainTextToVariant.vi encounters '{' and switches to parsing the remaining string as a cluster, whereas it should be parsed as an array and then recursively call PlainTextToVariant.vi to parse the array's contents.

The proposed fix is to remove the shift register switch for '{' and '[' if already parsing the string as a cluster or array. That way, the contents are correctly passed to the recursive call of PlainTextToVariant.vi

I've added unit tests to Test Serializer.Text in the zip file in the preceding comment. These unit tests include a variant cast test and tests to see if the values are preserved for the following cases:

  1. Array of clusters [{"A",[1,2,3]},{"B",[4,5,6]}]
  2. Array of clusters no array [{"A",{"B",1}},{"C",{"D",2}}]
  3. Cluster with array {"A",[1,2,3]}
  4. Cluster with cluster array {"A",[{"B",1},{"C",2}]}

Before the proposed fix, all cases except number 2 failed. After the fix, all cases pass for cast but fail (except number 2) for preserving numeric array values.

The failure for preserving numeric array values appears to be a separate bug in LabVIEW-Open-Source/DataManipulation#5. I am unsure on how to fix this bug. I'll add a test vector when I get a chance tomorrow as a comment to that issue.

Finally, the most recent commit in the pull request contains the proposed fix and the added unit tests described above. Note that, as described above, not all tests pass. Therefore, I am calling this an incomplete fix.

@francois-normandin
Copy link
Member

Thanks for the test vectors. I don't have much time right now to look into this problem with the PlainText Serializer, but the test vectors are useful.

Note: Comparing Array of Floats through the text serializer will always fail... you will need to compare individual elements with the "Almost Equal" method... unless serializing the byte array representation, there is simply no way to preserve the full 64-bit resolution.

@IlyaVino
Copy link
Author

No worries, the object oriented framework makes it easy to find a workaround by using a different serializer.

Good to know about array of floats. The failing tests still fail with any numeric type except I64. I've attached a modified I16 version of the tests. Note that now "cluster with cluster array - nested string" also fails...

Test Serializer.Text.zip

@francois-normandin
Copy link
Member

I might have a fix @IlyaVino.
Question for you... Is your system configured to use "dot" or "comma" for decimals?
The unit tests were failing on my Canadian computer... where we use comma for decimal point. When I turned the flag "Use System Decimal Point" to false across the board, the test suite passes!

Found two related issues: #7 and #8

@francois-normandin
Copy link
Member

I have a fix for them all, which should effectively solve the current reported bug.

image

francois-normandin added a commit that referenced this issue Jan 15, 2022
…work for non-matching datatypes

[Fix #7] Serializing decimal points to enforce dot notation even for non-US systems (similar to JSON convention)
[Fix #8] Numerics of decimal types serialize with at least one decimal point after dot
@francois-normandin
Copy link
Member

version 1.2.0 is fixing this bug

@IlyaVino
Copy link
Author

Hi,

Thanks for getting back to this so quickly!

I tried your update, but unfortunately it still fails for arrays of clusters that contain arrays in them, which is the case I'm most interested in (test file re-linked from previous comment):

array of clusters test fail
array of clusters block diagram

I also ran the test that shipped with your update, and I get all passing, which I hope rules out possible differences between our computers/configuration settings.

array of clusters test all pass

I didn't have time yet to fully play around with your new update to narrow down the issue, but I'll take a more careful look tomorrow morning.

Thanks again for all your help!

@IlyaVino
Copy link
Author

I made some headway on the issue. It looks like the changes I made to PlainTextToVariant.vi were not completely merged in the most recent version, probably because they did not pass all tests/required intervention in the DataManipulation library for a full fix.

If I recreate those changes and (** important **) keep the type input to unmarshall.vi unwired, everything works! I'm not sure why keeping the type input wired fails.

array of clusters block diagram unwired type

My tests:
update1 array of clusters ilya test all pass

Your tests:
update1 array of clusters f-n test all pass

In addition, I took a second more careful look at that vi (PlainTextToVariant (standard).vi in v1.2.0) and discovered an additional but closely related bug for deeply nested clusters and arrays. This bug represents a fairly rare case but has a simple fix. I'll open a new issue for it for documentation.

@IlyaVino
Copy link
Author

Also uploaded in issue #10, here is the version of PlainTextToVariant.vi that contains the changes to pass the tests above:

PlainTextToVariant (Standard).zip

francois-normandin added a commit that referenced this issue Jan 20, 2022
…d clusters/arrays

[Fix #10] PlainText Serializer now correctly unmarshalls deeply nested clusters/arrays that contain multiple ] or } characters
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

2 participants