Skip to content

Commit b6af5b6

Browse files
committed
Prevent the production of invalid JSON files due to NaN, Infinity, or -Infinity real values. If these value are encountered, they are serialized as strings (in quotes). See #395
1 parent 768f31a commit b6af5b6

File tree

2 files changed

+146
-12
lines changed

2 files changed

+146
-12
lines changed

src/json_string_utilities.F90

+39-12
Original file line numberDiff line numberDiff line change
@@ -147,12 +147,20 @@ end subroutine string_to_integer
147147
!
148148
! Convert a real value to a string.
149149
!
150+
!### Note
151+
! If the value is NaN, Infinity, or -Infinity, the string
152+
! will be returned in quotes. This is so it will be printed
153+
! in JSON as a string.
154+
!
150155
!### Modified
151-
! * Izaak Beekman : 02/24/2015 : added the compact option.
156+
! * Izaak Beekman : 02/24/2015 : added the compact option.
152157
! * Jacob Williams : 10/27/2015 : added the star option.
158+
! * Jacob Williams : 07/07/2019 : added ieee cases.
153159

154160
subroutine real_to_string(rval,real_fmt,compact_real,str)
155161

162+
use,intrinsic :: ieee_arithmetic
163+
156164
implicit none
157165

158166
real(RK),intent(in) :: rval !! real value.
@@ -161,20 +169,39 @@ subroutine real_to_string(rval,real_fmt,compact_real,str)
161169
!! displayed with fewer characters
162170
character(kind=CK,len=*),intent(out) :: str !! `rval` converted to a string.
163171

164-
integer(IK) :: istat
172+
integer(IK) :: istat !! write `iostat` flag
165173

166-
if (real_fmt==star) then
167-
write(str,fmt=*,iostat=istat) rval
168-
else
169-
write(str,fmt=real_fmt,iostat=istat) rval
170-
end if
174+
if (ieee_is_finite(rval) .and. .not. ieee_is_nan(rval)) then
175+
176+
! normal real numbers
177+
178+
if (real_fmt==star) then
179+
write(str,fmt=*,iostat=istat) rval
180+
else
181+
write(str,fmt=real_fmt,iostat=istat) rval
182+
end if
183+
184+
if (istat==0) then
185+
!in this case, the default string will be compacted,
186+
! so that the same value is displayed with fewer characters.
187+
if (compact_real) call compact_real_string(str)
188+
else
189+
str = repeat(star,len(str)) ! error
190+
end if
171191

172-
if (istat==0) then
173-
!in this case, the default string will be compacted,
174-
! so that the same value is displayed with fewer characters.
175-
if (compact_real) call compact_real_string(str)
176192
else
177-
str = repeat(star,len(str))
193+
! special case for NaN, Infinity, and -Infinity
194+
195+
! Let the compiler do the real to string conversion
196+
! like before, but put the result in quotes so it
197+
! gets printed as a string
198+
write(str,fmt=*,iostat=istat) rval
199+
if (istat==0) then
200+
str = quotation_mark//trim(adjustl(str))//quotation_mark
201+
else
202+
str = repeat(star,len(str)) ! error
203+
end if
204+
178205
end if
179206

180207
end subroutine real_to_string

src/tests/jf_test_42.F90

+107
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,107 @@
1+
!*****************************************************************************************
2+
!>
3+
! Module for the 42nd unit test
4+
5+
module jf_test_42_mod
6+
7+
use json_module, CK => json_CK, IK => json_IK, RK => json_RK, LK => json_LK
8+
use, intrinsic :: iso_fortran_env , only: error_unit, output_unit
9+
10+
implicit none
11+
12+
private
13+
public :: test_42
14+
15+
contains
16+
17+
subroutine test_42(error_cnt)
18+
19+
!! Test of NaN and Infinity
20+
21+
implicit none
22+
23+
type(json_file) :: json !! the JSON structure read from the file
24+
integer,intent(out) :: error_cnt !! error counter
25+
26+
character(kind=CK,len=*),parameter :: str = CK_'{"bad_reals": [1.0, "NaN", "+Infinity", "-Infinity", 4.0]}'
27+
28+
real(rk),dimension(:),allocatable :: bad_reals
29+
logical(lk) :: found
30+
31+
write(error_unit,'(A)') ''
32+
write(error_unit,'(A)') '================================='
33+
write(error_unit,'(A)') ' TEST 42'
34+
write(error_unit,'(A)') '================================='
35+
write(error_unit,'(A)') ''
36+
37+
error_cnt = 0
38+
39+
! parse the json string:
40+
write(error_unit,'(A)') ''
41+
write(error_unit,'(A)') 'parsing string... '
42+
call json%load_from_string(str)
43+
if (json%failed()) then
44+
call json%print_error_message(error_unit)
45+
error_cnt = error_cnt + 1
46+
end if
47+
write(error_unit,'(A)') ''
48+
write(error_unit,'(A)') 'printing...'
49+
call json%print_file(int(error_unit,IK))
50+
51+
write(error_unit,'(A)') ''
52+
write(error_unit,'(A)') 'get values as real...'
53+
call json%get('bad_reals',bad_reals,found)
54+
55+
if (json%failed()) then !if there was an error reading the file
56+
57+
call json%print_error_message(error_unit)
58+
error_cnt = error_cnt + 1
59+
60+
else
61+
62+
write(error_unit,'(A)') 'printing...'
63+
write(error_unit,*) bad_reals
64+
write(error_unit,'(A)') ''
65+
66+
end if
67+
68+
call json%destroy()
69+
70+
write(error_unit,'(A)') ''
71+
write(error_unit,'(A)') 'now add back as a real...'
72+
call json%add('bad_reals', bad_reals)
73+
call json%print_file(int(error_unit,IK))
74+
write(error_unit,'(A)') ''
75+
76+
if (json%failed()) then
77+
call json%print_error_message(error_unit)
78+
error_cnt = error_cnt + 1
79+
end if
80+
81+
if (error_cnt==0) then
82+
write(error_unit,'(A)') 'Success!'
83+
else
84+
write(error_unit,'(A)') 'Failed!'
85+
end if
86+
87+
end subroutine test_42
88+
89+
end module jf_test_42_mod
90+
!*****************************************************************************************
91+
92+
#ifndef INTEGRATED_TESTS
93+
!*****************************************************************************************
94+
program jf_test_42
95+
96+
!! 42nd unit test.
97+
98+
use jf_test_42_mod , only: test_42
99+
implicit none
100+
integer :: n_errors
101+
n_errors = 0
102+
call test_42(n_errors)
103+
if (n_errors /= 0) stop 1
104+
105+
end program jf_test_42
106+
!*****************************************************************************************
107+
#endif

0 commit comments

Comments
 (0)