LCOV - code coverage report
Current view: top level - ogr/ogrsf_frmts/geojson/libjson - json_util.c (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 21 106 19.8 %
Date: 2025-01-18 12:42:00 Functions: 3 13 23.1 %

          Line data    Source code
       1             : /*
       2             :  * $Id: json_util.c,v 1.4 2006/01/30 23:07:57 mclark Exp $
       3             :  *
       4             :  * Copyright (c) 2004, 2005 Metaparadigm Pte. Ltd.
       5             :  * Michael Clark <michael@metaparadigm.com>
       6             :  *
       7             :  * This library is free software; you can redistribute it and/or modify
       8             :  * it under the terms of the MIT license. See COPYING for details.
       9             :  *
      10             :  */
      11             : 
      12             : #include "config.h"
      13             : #undef realloc
      14             : 
      15             : #include "strerror_override.h"
      16             : 
      17             : #include <ctype.h>
      18             : #include <limits.h>
      19             : #include <stdarg.h>
      20             : #include <stddef.h>
      21             : #include <stdio.h>
      22             : #include <stdlib.h>
      23             : #include <string.h>
      24             : 
      25             : #ifdef HAVE_SYS_TYPES_H
      26             : #include <sys/types.h>
      27             : #endif /* HAVE_SYS_TYPES_H */
      28             : 
      29             : #ifdef HAVE_SYS_STAT_H
      30             : #include <sys/stat.h>
      31             : #endif /* HAVE_SYS_STAT_H */
      32             : 
      33             : #ifdef HAVE_FCNTL_H
      34             : #include <fcntl.h>
      35             : #endif /* HAVE_FCNTL_H */
      36             : 
      37             : #ifdef HAVE_UNISTD_H
      38             : #include <unistd.h>
      39             : #endif /* HAVE_UNISTD_H */
      40             : 
      41             : #ifdef _WIN32
      42             : #define WIN32_LEAN_AND_MEAN
      43             : #include <io.h>
      44             : #include <windows.h>
      45             : #endif /* defined(_WIN32) */
      46             : 
      47             : #if !defined(HAVE_OPEN) && defined(_WIN32)
      48             : #define open _open
      49             : #endif
      50             : 
      51             : #include "snprintf_compat.h"
      52             : 
      53             : #include "debug.h"
      54             : #include "json_inttypes.h"
      55             : #include "json_object.h"
      56             : #include "json_object_private.h"
      57             : #include "json_tokener.h"
      58             : #include "json_util.h"
      59             : #include "printbuf.h"
      60             : 
      61             : #include "cpl_string.h"
      62             : 
      63             : static int _json_object_to_fd(int fd, struct json_object *obj, int flags, const char *filename);
      64             : 
      65             : static char _last_err[256] = "";
      66             : 
      67           0 : const char *json_util_get_last_err()
      68             : {
      69           0 :   if (_last_err[0] == '\0')
      70           0 :     return NULL;
      71           0 :   return _last_err;
      72             : }
      73             : 
      74           1 : void _json_c_set_last_err(const char *err_fmt, ...)
      75             : {
      76             :   va_list ap;
      77           1 :   va_start(ap, err_fmt);
      78             :   // Ignore (attempted) overruns from snprintf
      79           1 :   (void)vsnprintf(_last_err, sizeof(_last_err), err_fmt, ap);
      80           1 :   va_end(ap);
      81           1 : }
      82             : 
      83           0 : struct json_object *json_object_from_fd(int fd)
      84             : {
      85           0 :   return json_object_from_fd_ex(fd, -1);
      86             : }
      87           0 : struct json_object *json_object_from_fd_ex(int fd, int in_depth)
      88             : {
      89             :   struct printbuf *pb;
      90             :   struct json_object *obj;
      91             :   char buf[JSON_FILE_BUF_SIZE];
      92             :   int ret;
      93           0 :   int depth = JSON_TOKENER_DEFAULT_DEPTH;
      94             :   json_tokener *tok;
      95             : 
      96           0 :   if (!(pb = printbuf_new()))
      97             :   {
      98           0 :     _json_c_set_last_err("json_object_from_file: printbuf_new failed\n");
      99           0 :     return NULL;
     100             :   }
     101             : 
     102           0 :   if (in_depth != -1)
     103           0 :     depth = in_depth;
     104           0 :   tok = json_tokener_new_ex(depth);
     105           0 :   if (!tok)
     106             :   {
     107           0 :     _json_c_set_last_err(
     108             :         "json_object_from_fd: unable to allocate json_tokener(depth=%d): %s\n", depth,
     109           0 :         strerror(errno));
     110           0 :     printbuf_free(pb);
     111           0 :     return NULL;
     112             :   }
     113             : 
     114           0 :   while ((ret = (int)read(fd, buf, JSON_FILE_BUF_SIZE)) > 0)
     115             :   {
     116           0 :     printbuf_memappend(pb, buf, ret);
     117             :   }
     118           0 :   if (ret < 0)
     119             :   {
     120           0 :     _json_c_set_last_err("json_object_from_fd: error reading fd %d: %s\n", fd,
     121           0 :                          strerror(errno));
     122           0 :     json_tokener_free(tok);
     123           0 :     printbuf_free(pb);
     124           0 :     return NULL;
     125             :   }
     126             : 
     127           0 :   obj = json_tokener_parse_ex(tok, pb->buf, printbuf_length(pb));
     128           0 :   if (obj == NULL)
     129           0 :     _json_c_set_last_err("json_tokener_parse_ex failed: %s\n",
     130             :                          json_tokener_error_desc(json_tokener_get_error(tok)));
     131             : 
     132           0 :   json_tokener_free(tok);
     133           0 :   printbuf_free(pb);
     134           0 :   return obj;
     135             : }
     136             : 
     137           0 : struct json_object *json_object_from_file(const char *filename)
     138             : {
     139             :   struct json_object *obj;
     140             :   int fd;
     141             : 
     142           0 :   if ((fd = open(filename, O_RDONLY)) < 0)
     143             :   {
     144           0 :     _json_c_set_last_err("json_object_from_file: error opening file %s: %s\n", filename,
     145           0 :                          strerror(errno));
     146           0 :     return NULL;
     147             :   }
     148           0 :   obj = json_object_from_fd(fd);
     149           0 :   close(fd);
     150           0 :   return obj;
     151             : }
     152             : 
     153             : /* extended "format and write to file" function */
     154             : 
     155           0 : int json_object_to_file_ext(const char *filename, struct json_object *obj, int flags)
     156             : {
     157             :   int fd, ret;
     158             :   int saved_errno;
     159             : 
     160           0 :   if (!obj)
     161             :   {
     162           0 :     _json_c_set_last_err("json_object_to_file: object is null\n");
     163           0 :     return -1;
     164             :   }
     165             : 
     166           0 :   if ((fd = open(filename, O_WRONLY | O_TRUNC | O_CREAT, 0644)) < 0)
     167             :   {
     168           0 :     _json_c_set_last_err("json_object_to_file: error opening file %s: %s\n", filename,
     169           0 :                          strerror(errno));
     170           0 :     return -1;
     171             :   }
     172           0 :   ret = _json_object_to_fd(fd, obj, flags, filename);
     173           0 :   saved_errno = errno;
     174           0 :   close(fd);
     175           0 :   errno = saved_errno;
     176           0 :   return ret;
     177             : }
     178             : 
     179           0 : int json_object_to_fd(int fd, struct json_object *obj, int flags)
     180             : {
     181           0 :   if (!obj)
     182             :   {
     183           0 :     _json_c_set_last_err("json_object_to_fd: object is null\n");
     184           0 :     return -1;
     185             :   }
     186             : 
     187           0 :   return _json_object_to_fd(fd, obj, flags, NULL);
     188             : }
     189           0 : static int _json_object_to_fd(int fd, struct json_object *obj, int flags, const char *filename)
     190             : {
     191             :   ssize_t ret;
     192             :   const char *json_str;
     193             :   unsigned int wpos, wsize;
     194             : 
     195           0 :   filename = filename ? filename : "(fd)";
     196             : 
     197           0 :   if (!(json_str = json_object_to_json_string_ext(obj, flags)))
     198             :   {
     199           0 :     return -1;
     200             :   }
     201             : 
     202             :   /* CAW: probably unnecessary, but the most 64bit safe */
     203           0 :   wsize = (unsigned int)(strlen(json_str) & UINT_MAX);
     204           0 :   wpos = 0;
     205           0 :   while (wpos < wsize)
     206             :   {
     207             :     /* coverity[overflow_sink] */
     208           0 :     if ((ret = write(fd, json_str + wpos, wsize - wpos)) < 0)
     209             :     {
     210           0 :       _json_c_set_last_err("json_object_to_file: error writing file %s: %s\n",
     211           0 :                            filename, strerror(errno));
     212           0 :       return -1;
     213             :     }
     214             : 
     215             :     /* because of the above check for ret < 0, we can safely cast and add */
     216           0 :     wpos += (unsigned int)ret;
     217             :   }
     218             : 
     219           0 :   return 0;
     220             : }
     221             : 
     222             : // backwards compatible "format and write to file" function
     223             : 
     224           0 : int json_object_to_file(const char *filename, struct json_object *obj)
     225             : {
     226           0 :   return json_object_to_file_ext(filename, obj, JSON_C_TO_STRING_PLAIN);
     227             : }
     228             : 
     229             : // Deprecated json_parse_double function.  See json_tokener_parse_double instead.
     230           0 : int json_parse_double(const char *buf, double *retval)
     231             : {
     232             :   char *end;
     233           0 :   *retval = CPLStrtod(buf, &end);
     234           0 :   return end == buf ? 1 : 0;
     235             : }
     236             : 
     237       13317 : int json_parse_int64(const char *buf, int64_t *retval)
     238             : {
     239       13317 :   char *end = NULL;
     240             :   int64_t val;
     241             : 
     242       13317 :   errno = 0;
     243       13317 :   val = strtoll(buf, &end, 10);
     244       13317 :   if (end != buf)
     245       13316 :     *retval = val;
     246       13317 :   return ((val == 0 && errno != 0) || (end == buf)) ? 1 : 0;
     247             : }
     248             : 
     249      281509 : int json_parse_uint64(const char *buf, uint64_t *retval)
     250             : {
     251      281509 :   char *end = NULL;
     252             :   uint64_t val;
     253             : 
     254      281509 :   errno = 0;
     255      281509 :   while (*buf == ' ')
     256           0 :     buf++;
     257      281509 :   if (*buf == '-')
     258           0 :     return 1; /* error: uint cannot be negative */
     259             : 
     260      281509 :   val = strtoull(buf, &end, 10);
     261      281509 :   if (end != buf)
     262      281509 :     *retval = val;
     263      281509 :   return ((val == 0 && errno != 0) || (end == buf)) ? 1 : 0;
     264             : }
     265             : 
     266             : #ifndef HAVE_REALLOC
     267             : void *rpl_realloc(void *p, size_t n)
     268             : {
     269             :   if (n == 0)
     270             :     n = 1;
     271             :   if (p == 0)
     272             :     return malloc(n);
     273             :   return realloc(p, n);
     274             : }
     275             : #endif
     276             : 
     277             : #define NELEM(a) (sizeof(a) / sizeof(a[0]))
     278             : /* clang-format off */
     279             : static const char *json_type_name[] = {
     280             :   /* If you change this, be sure to update the enum json_type definition too */
     281             :   "null",
     282             :   "boolean",
     283             :   "double",
     284             :   "int",
     285             :   "object",
     286             :   "array",
     287             :   "string",
     288             : };
     289             : /* clang-format on */
     290             : 
     291           0 : const char *json_type_to_name(enum json_type o_type)
     292             : {
     293           0 :   int o_type_int = (int)o_type;
     294           0 :   if (o_type_int < 0 || o_type_int >= (int)NELEM(json_type_name))
     295             :   {
     296           0 :     _json_c_set_last_err("json_type_to_name: type %d is out of range [0,%u]\n", o_type,
     297             :                          (unsigned)NELEM(json_type_name));
     298           0 :     return NULL;
     299             :   }
     300           0 :   return json_type_name[o_type];
     301             : }

Generated by: LCOV version 1.14