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

Json produces invalid JSON output for integer keys #109

Open
LinqLover opened this issue Jan 8, 2024 · 7 comments
Open

Json produces invalid JSON output for integer keys #109

LinqLover opened this issue Jan 8, 2024 · 7 comments
Labels
bug [WHAT] Something isn't working as expected. Automated tests beneficial. :- tools and libraries [SCOPE] From #asTextToHtml over JSON parsing to Debugger and Browser.

Comments

@LinqLover
Copy link
Contributor

Dictionary new
	at: 1 put: 2;
	asJsonString --> '{1:2}'

What would be the expected behavior?

  • '{"1":2}'
  • raise an error: invalid key for json

What in this case?

Dictionary new
	at: 1 put: 2;
	at: '1' put: '2';
	asJsonString
  • '{"1":2}'
  • '{"1":"2"}'
  • raise an error: duplicate keys

My tendency would be the variants with a star ...

@LinqLover LinqLover added tools and libraries [SCOPE] From #asTextToHtml over JSON parsing to Debugger and Browser. bug [WHAT] Something isn't working as expected. Automated tests beneficial. :- labels Jan 8, 2024
@timrowledge
Copy link

Looking at https://www.json.org/json-en.html & https://jsonlint.com/ I think the expectation is that the key must be a string and the value must be the json-ised value, so

Dictionary new
	at: 1 put: 2;
	asJsonString 

should result in an error.

Consider the first EBNF diagram on https://www.json.org/json-en.html - it can be simplified for this case to
{ string : jsonised-value}.

Since we're exceptionally clever we could consider doing "make the key be a string of whatever we have" except that trying to reliably convert non-string keys might open up entire pallet-loads of worms. What if somebody is making dictionaries where the keys are complex objects?

Might also want to check on STON as well (https://wiki.squeak.org/squeak/6504). If we're attempting to produce sometihng that meets a standard we should probably actually meet the standard...

@codefrau
Copy link
Member

codefrau commented Apr 1, 2024

The second should raise an error. Our dictionaries do not preserve insertion order so we don't even know which one is the duplicate.

For the first case where it's unambiguous I'd find it more convenient to allow this and automatically convert the key to a string, just like Association>> jsonWriteOn: does (there is an argument for removing this method though, because unlike all other methods it does not by itself produce valid JSON)

@LinqLover
Copy link
Contributor Author

I'm personally rather with Tim on that. Let's not bring the joy of [object Object] to Squeak. :D I'm personally more a fan of EAFP than LBYL, but silent invalidation of data just makes debugging harder IMO. I even wonder whether JsonObject should reject any non-string keys during construction and manipulation (e.g., add:) in the first place. However, the latter idea might impact compatibility (but seriously, the name JsonObject conveys a pretty clear idea what types of keys might be allowed there.)

just like Association>> jsonWriteOn: does (there is an argument for removing this method though, because unlike all other methods it does not by itself produce valid JSON)

Oh yes indeed. I placed an #isThisEverCalled in this method in my image and if we do not identify any surprising usages, I support removing this method (or deprecating it at least).

@tonyg
Copy link
Member

tonyg commented May 22, 2024

IMO JsonObject was a mistake :-) but rejecting non-string keys there would make sense. For Dictionary, the best we can do is signal an error when trying to write a non-string key as JSON. But we should definitely do that rather than silently emit non-JSON. I do not like the idea of automagic conversion to strings: this seems guaranteed to produce unpleasant surprises downstream, and also might result in (technically legal JSON!!) monstrosities like {"1": 2, "1": 2} for the case where a Dictionary has a key of 1 and another of '1'.

To be specific wrt the test cases from the top post here: the first case should signal an error during asJsonString, and the second should signal an error when trying to render the key 1. In both cases this would be some kind of "invalid JSON dictionary key type" error.

@LinqLover
Copy link
Contributor Author

There are other kinds of invalid keys, such as String cr, that we should detect and treat equivantly.

@tonyg
Copy link
Member

tonyg commented Jun 8, 2024

There are other kinds of invalid keys, such as String cr, that we should detect and treat equivantly.

That's not an invalid key: it should serialize to

{ "\r": 123 }

for example. So long as it's a valid JSON string, it should be a valid JSON key.

@LinqLover
Copy link
Contributor Author

There are other kinds of invalid keys, such as String cr, that we should detect and treat equivantly.

That's not an invalid key: it should serialize to

{ "\r": 123 }

for example. So long as it's a valid JSON string, it should be a valid JSON key.

My bad. I tested it in JavaScript and Python but forgot to escape the backslash ...

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
bug [WHAT] Something isn't working as expected. Automated tests beneficial. :- tools and libraries [SCOPE] From #asTextToHtml over JSON parsing to Debugger and Browser.
Projects
None yet
Development

No branches or pull requests

4 participants