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

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

(1.5x libT) #1832: Transmission should notify when JSON parser fails

  • Property svn:keywords set to Date Rev Author Id
File size: 4.7 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 7963 2009-02-27 00:57:44Z 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                         line = 1;
148    int                         column = 1;
149    int                         err = 0;
150    const unsigned char       * buf = vbuf;
151    const void                * bufend = buf + len;
152    struct JSON_config_struct   config;
153    struct JSON_parser_struct * checker;
154    struct json_benc_data       data;
155
156    init_JSON_config( &config );
157    config.callback = callback;
158    config.callback_ctx = &data;
159    config.depth = -1;
160
161    data.hasContent = FALSE;
162    data.key = NULL;
163    data.top = setme_benc;
164    data.stack = TR_PTR_ARRAY_INIT;
165
166    checker = new_JSON_parser( &config );
167    while( ( buf != bufend ) && JSON_parser_char( checker, *buf ) ) {
168        if( *buf != '\n' )
169            ++column;
170        else {
171            ++line;
172            column = 1;
173        }
174        ++buf;
175    }
176
177    if( buf != bufend ) {
178        tr_err( "JSON parser failed at line %d, column %d: \"%.16s\"", line, column, buf );
179        err = EILSEQ;
180    }
181
182    if( !data.hasContent )
183        err = EINVAL;
184
185    if( setme_end )
186        *setme_end = (const uint8_t*) buf;
187
188    delete_JSON_parser( checker );
189    tr_ptrArrayDestruct( &data.stack, NULL );
190    return err;
191}
192
Note: See TracBrowser for help on using the repository browser.