source: branches/1.5x/libtransmission/json.c @ 7908

Last change on this file since 7908 was 7908, checked in by charles, 13 years ago

(1.5x libT) #1845: if settings.json is empty, T crashes with "Assertion: tr_bencIsDict( source ) failed"

  • Property svn:keywords set to Date Rev Author Id
File size: 4.4 KB
Line 
1/*
2 * This file Copyright (C) 2008-2009 Charles Kerr <charles@transmissionbt.com>
3 *
4 * This file is licensed by the GPL version 2.  Works owned by the
5 * Transmission project are granted a special exemption to clause 2(b)
6 * so that the bulk of its code can remain under the MIT license.
7 * This exemption does not extend to derived works not owned by
8 * the Transmission project.
9 *
10 * $Id: json.c 7908 2009-02-18 16:47:28Z charles $
11 */
12
13#include <assert.h>
14#include <ctype.h>
15#include <errno.h>
16#include <string.h>
17#include <stdio.h> /* printf */
18
19#include <event.h> /* evbuffer */
20
21#include "JSON_parser.h"
22
23#include "transmission.h"
24#include "bencode.h"
25#include "json.h"
26#include "ptrarray.h"
27#include "utils.h"
28
29struct json_benc_data
30{
31    tr_bool        hasContent;
32    tr_benc      * top;
33    tr_ptrArray    stack;
34    char         * key;
35};
36
37static tr_benc*
38getNode( struct json_benc_data * data )
39{
40    tr_benc * parent;
41    tr_benc * node = NULL;
42
43    if( tr_ptrArrayEmpty( &data->stack ) )
44        parent = NULL;
45    else
46        parent = tr_ptrArrayBack( &data->stack );
47
48    if( !parent )
49        node = data->top;
50    else if( tr_bencIsList( parent ) )
51        node = tr_bencListAdd( parent );
52    else if( tr_bencIsDict( parent ) && data->key )
53    {
54        node = tr_bencDictAdd( parent, data->key );
55        tr_free( data->key );
56        data->key = NULL;
57    }
58
59    return node;
60}
61
62static int
63callback( void *             vdata,
64          int                type,
65          const JSON_value * value )
66{
67    struct json_benc_data * data = vdata;
68    tr_benc *               node;
69
70    switch( type )
71    {
72        case JSON_T_ARRAY_BEGIN:
73            data->hasContent = TRUE;
74            node = getNode( data );
75            tr_bencInitList( node, 0 );
76            tr_ptrArrayAppend( &data->stack, node );
77            break;
78
79        case JSON_T_ARRAY_END:
80            tr_ptrArrayPop( &data->stack );
81            break;
82
83        case JSON_T_OBJECT_BEGIN:
84            data->hasContent = TRUE;
85            node = getNode( data );
86            tr_bencInitDict( node, 0 );
87            tr_ptrArrayAppend( &data->stack, node );
88            break;
89
90        case JSON_T_OBJECT_END:
91            tr_ptrArrayPop( &data->stack );
92            break;
93
94        case JSON_T_FLOAT:
95        {
96            char buf[128];
97            tr_snprintf( buf, sizeof( buf ), "%f",
98                         (double)value->vu.float_value );
99            tr_bencInitStr( getNode( data ), buf, -1 );
100            data->hasContent = TRUE;
101            break;
102        }
103
104        case JSON_T_NULL:
105            data->hasContent = TRUE;
106            tr_bencInitStr( getNode( data ), "", 0 );
107            break;
108
109        case JSON_T_INTEGER:
110            data->hasContent = TRUE;
111            tr_bencInitInt( getNode( data ), value->vu.integer_value );
112            break;
113
114        case JSON_T_TRUE:
115            data->hasContent = TRUE;
116            tr_bencInitInt( getNode( data ), 1 );
117            break;
118
119        case JSON_T_FALSE:
120            data->hasContent = TRUE;
121            tr_bencInitInt( getNode( data ), 0 );
122            break;
123
124        case JSON_T_STRING:
125            data->hasContent = TRUE;
126            tr_bencInitStr( getNode( data ),
127                            value->vu.str.value,
128                            value->vu.str.length );
129            break;
130
131        case JSON_T_KEY:
132            data->hasContent = TRUE;
133            assert( !data->key );
134            data->key = tr_strdup( value->vu.str.value );
135            break;
136    }
137
138    return 1;
139}
140
141int
142tr_jsonParse( const void     * vbuf,
143              size_t           len,
144              tr_benc        * setme_benc,
145              const uint8_t ** setme_end )
146{
147    int                         err = 0;
148    const unsigned char       * buf = vbuf;
149    const void                * bufend = buf + len;
150    struct JSON_config_struct   config;
151    struct JSON_parser_struct * checker;
152    struct json_benc_data       data;
153
154    init_JSON_config( &config );
155    config.callback = callback;
156    config.callback_ctx = &data;
157    config.depth = -1;
158
159    data.hasContent = FALSE;
160    data.key = NULL;
161    data.top = setme_benc;
162    data.stack = TR_PTR_ARRAY_INIT;
163
164    checker = new_JSON_parser( &config );
165    while( ( buf != bufend ) && JSON_parser_char( checker, *buf ) )
166        ++buf;
167
168    if( buf != bufend )
169        err = EILSEQ;
170
171    if( !data.hasContent )
172        err = EINVAL;
173
174    if( setme_end )
175        *setme_end = (const uint8_t*) buf;
176
177    delete_JSON_parser( checker );
178    tr_ptrArrayDestruct( &data.stack, NULL );
179    return err;
180}
181
Note: See TracBrowser for help on using the repository browser.