* contrib/hbct/ctnet.c
! fixed copy and past typo in previous commit
* src/rdd/hbsix/sxcompat.prg
! fixed sx_Compress() and sx_Decompress() used with arrays
; modifications below comes from Viktor fork:
* include/dbinfo.ch
* src/rdd/workarea.c
+ added support for new dbInfo() action: DBI_CODEPAGE
It returns Harbour CP used by current work area.
* src/common/expropt2.c
* change instruction order to protect against NULL references
if we ever add support for dummy lists
* src/common/hbver.c
! CLANG detection
* src/lang/l_fr.c
* src/lang/l_hu.c
* updated translations
* src/rtl/cdpapihb.c
* include/harbour.hbx
+ added function hb_cdpExists( <cCDP> ) -> <lExists>
it returns .T. if the passed codepage is supported
and linked, .F. otherwise
* src/rtl/hbrand.c
* include/harbour.hbx
+ added hb_randStr( <nLen> ) -> <cBytes> which returns a strong
random stream of bytes of requested length
* src/vm/hashfunc.c
* include/harbour.hbx
+ New hash function to easily check for the existence
of a key and at the same time retrieving its corresponding
value:
hb_HGetRef( <hHash>, <xKey>[, <@xValue> ] ) -> <lFound>
If the key is not found, <xValue> is also reset to NIL.
Useful to replace hb_HPos()/hb_HValueAt() call pairs and
associated temp variable, or to avoid multiple hash lookups
for the cost of one temp variable.
* src/rtl/copyfile.c
% use 64KB buffer
* src/rtl/tget.prg
! Get():Insert()/Get():OverStrike() GET freeze fix reworked
using better solution by Heinz.
* src/rtl/tgetlist.prg
* src/rtl/tmenuitm.prg
* src/rtl/tmenusys.prg
* src/rtl/tpopup.prg
! fixed to precisely verify against multiple expected class names
previously a custom class named f.e. 'LIST' was accepted in some
places for LISTBOX
! fixed TOFIX where CP/encoding dependent string was
used to determine whether a menu item is separator
throughout the lifetime of the menu.
* src/common/hbdate.c
* src/common/hbprintf.c
* src/common/hbtrace.c
* src/common/hbver.c
* src/compiler/compi18n.c
* src/compiler/genc.c
* src/compiler/gencc.c
* src/compiler/hbusage.c
* src/debug/dbgthsh.prg
* src/debug/dbgtmenu.prg
* src/debug/debugger.prg
* src/debug/tbrwtext.prg
* src/pp/hbpp.c
* src/rdd/dbfcdx/dbfcdx1.c
* src/rdd/dbfuncs.prg
* src/rdd/dbtotal.prg
* src/rdd/hbsix/sxcompat.prg
* src/rdd/hbsix/sxini.prg
* src/rdd/usrrdd/rdds/arrayrdd.prg
* src/rdd/usrrdd/rdds/hscdx.prg
* src/rtl/adir.prg
* src/rtl/arc4.c
* src/rtl/cdpdet.prg
* src/rtl/dbedit.prg
* src/rtl/dircmd.prg
* src/rtl/diskspac.c
* src/rtl/disksphb.c
* src/rtl/errint.c
* src/rtl/errintlo.c
* src/rtl/errsys.prg
* src/rtl/filesys.c
* src/rtl/fslink.c
* src/rtl/gtkbstat.c
* src/rtl/gtwin/gtwin.c
* src/rtl/gtwvt/gtwvt.c
* src/rtl/hbbffnc.c
* src/rtl/hbi18n1.c
* src/rtl/hbi18n2.prg
* src/rtl/hbini.prg
* src/rtl/hbjson.c
* src/rtl/listbox.prg
* src/rtl/menusys.prg
* src/rtl/mtran.c
* src/rtl/radiogrp.prg
* src/rtl/tgetlist.prg
* src/rtl/tmenusys.prg
* src/rtl/tobject.prg
* src/rtl/tpersist.prg
* src/rtl/tpopup.prg
* src/rtl/tpopuphb.prg
* src/rtl/treport.prg
* src/rtl/tscalar.prg
* src/rtl/ttopbar.prg
* src/rtl/tget.prg
* src/rtl/tlabel.prg
* src/rtl/tmenuitm.prg
* src/rtl/valtoexp.prg
* src/rtl/valtype.c
* src/vm/dlmalloc.c
* src/vm/fm.c
! check module handle returned by GetModuleHandle() in MS-Windows
builds
! use %u for unsigned printf formatting
* casting, formatting, casing and some modifications in names
* use #if 0 / #endif to mark commented blocks
* updated year to 2014
% use FOR EACH, SWITCH, DO CASE statements
% use :__enumFirst() / :__enumLast()
% eliminated some local variables and redundant declarations
% use hb_defaultValue()
% use $ operator instead of At() and hb_HHasKey()
% use hb_ADel() instead of ADel()+ASize()
* replace some functions with procedures
* pacified some MSC warnings
* use hb_StrFormat() for user messages
* use :: instead of Self:
* use ANSI date delimiters
705 lines
20 KiB
C
705 lines
20 KiB
C
/*
|
|
* Harbour Project source code:
|
|
* JavaScript Object Notation (JSON)
|
|
*
|
|
* Copyright 2010 Mindaugas Kavaliauskas <dbtopas / at / dbtopas.lt>
|
|
* www - http://harbour-project.org/
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2, or (at your option)
|
|
* any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this software; see the file COPYING.txt. If not, write to
|
|
* the Free Software Foundation, Inc., 59 Temple Place, Suite 330,
|
|
* Boston, MA 02111-1307 USA (or visit the web site http://www.gnu.org/).
|
|
*
|
|
* As a special exception, the Harbour Project gives permission for
|
|
* additional uses of the text contained in its release of Harbour.
|
|
*
|
|
* The exception is that, if you link the Harbour libraries with other
|
|
* files to produce an executable, this does not by itself cause the
|
|
* resulting executable to be covered by the GNU General Public License.
|
|
* Your use of that executable is in no way restricted on account of
|
|
* linking the Harbour library code into it.
|
|
*
|
|
* This exception does not however invalidate any other reasons why
|
|
* the executable file might be covered by the GNU General Public License.
|
|
*
|
|
* This exception applies only to the code released by the Harbour
|
|
* Project under the name Harbour. If you copy code from other
|
|
* Harbour Project or Free Software Foundation releases into a copy of
|
|
* Harbour, as the General Public License permits, the exception does
|
|
* not apply to the code that you add in this way. To avoid misleading
|
|
* anyone as to the status of such modified files, you must delete
|
|
* this exception notice from them.
|
|
*
|
|
* If you write modifications of your own for Harbour, it is your choice
|
|
* whether to permit this exception to apply to your modifications.
|
|
* If you do not wish that, delete this exception notice.
|
|
*
|
|
*/
|
|
|
|
#include "hbapi.h"
|
|
#include "hbapiitm.h"
|
|
#include "hbapistr.h"
|
|
#include "hbset.h"
|
|
|
|
/*
|
|
The application/json Media Type for JavaScript Object Notation (JSON)
|
|
http://www.ietf.org/rfc/rfc4627.txt
|
|
|
|
C level functions:
|
|
char * hb_jsonEncode( PHB_ITEM pValue, HB_SIZE * pnLen, HB_BOOL fHuman );
|
|
pValue - value to encode;
|
|
pnLen - if pnLen is not NULL, length of returned buffer is
|
|
stored to *pnLen;
|
|
fHuman - format to be human redable;
|
|
returns pointer to encoded JSON buffer. buffer must be fried
|
|
by the caller.
|
|
|
|
HB_SIZE hb_jsonDecode( const char * szSource, PHB_ITEM pValue );
|
|
szSource - JSON source;
|
|
pValue - item to store decoded value. Item value is
|
|
undetermined in case of error;
|
|
returns number of bytes decoded from the buffer. This allows
|
|
to use the remaining part of the buffer for some other
|
|
purposes. Returns 0 on error.
|
|
|
|
Harbour level functions:
|
|
hb_jsonDecode( cJSON, @xValue ) --> nLengthDecoded
|
|
hb_jsonEncode( xValue [, lHuman = .F. ] ) --> cJSON
|
|
|
|
Note:
|
|
- JSON encode functions are safe for recursive arrays and hashes.
|
|
Recursive part of array or hash will be stored as null. JSON
|
|
encoder still allows to use same structure in the leaves, in
|
|
this case content will be duplicate.
|
|
I.e.:
|
|
xI := { 1, NIL }
|
|
xI[ 2 ] := xI
|
|
? hb_jsonEncode( xI ) // [1,null]
|
|
but:
|
|
xI := { 1, .T. }
|
|
xI := { 2, xI, xI }
|
|
? hb_jsonEncode( xI ) // [2,[1,true],[1,true]]
|
|
*/
|
|
|
|
typedef struct
|
|
{
|
|
char * pBuffer;
|
|
char * pHead;
|
|
HB_SIZE nAlloc;
|
|
void ** pId;
|
|
HB_SIZE nAllocId;
|
|
HB_BOOL fHuman;
|
|
int iEolLen;
|
|
const char * szEol;
|
|
} HB_JSON_ENCODE_CTX, * PHB_JSON_ENCODE_CTX;
|
|
|
|
|
|
#define INDENT_SIZE 2
|
|
|
|
static void _hb_jsonCtxAdd( PHB_JSON_ENCODE_CTX pCtx, const char * szString, HB_SIZE nLen )
|
|
{
|
|
if( pCtx->pHead + nLen >= pCtx->pBuffer + pCtx->nAlloc )
|
|
{
|
|
HB_SIZE nSize = pCtx->pHead - pCtx->pBuffer;
|
|
|
|
pCtx->nAlloc += ( pCtx->nAlloc << 1 ) + nLen;
|
|
pCtx->pBuffer = ( char * ) hb_xrealloc( pCtx->pBuffer, pCtx->nAlloc );
|
|
pCtx->pHead = pCtx->pBuffer + nSize;
|
|
}
|
|
if( szString )
|
|
{
|
|
hb_xmemcpy( pCtx->pHead, szString, nLen );
|
|
pCtx->pHead += nLen;
|
|
}
|
|
}
|
|
|
|
static void _hb_jsonCtxAddIndent( PHB_JSON_ENCODE_CTX pCtx, HB_SIZE nCount )
|
|
{
|
|
if( nCount > 0 )
|
|
{
|
|
if( pCtx->pHead + nCount >= pCtx->pBuffer + pCtx->nAlloc )
|
|
{
|
|
HB_SIZE nSize = pCtx->pHead - pCtx->pBuffer;
|
|
|
|
pCtx->nAlloc += ( pCtx->nAlloc << 1 ) + nCount;
|
|
pCtx->pBuffer = ( char * ) hb_xrealloc( pCtx->pBuffer, pCtx->nAlloc );
|
|
pCtx->pHead = pCtx->pBuffer + nSize;
|
|
}
|
|
hb_xmemset( pCtx->pHead, ' ', nCount );
|
|
pCtx->pHead += nCount;
|
|
}
|
|
}
|
|
|
|
static void _hb_jsonEncode( PHB_ITEM pValue, PHB_JSON_ENCODE_CTX pCtx,
|
|
HB_SIZE nLevel, HB_BOOL fEOL )
|
|
{
|
|
/* Protection against recursive structures */
|
|
if( ( HB_IS_ARRAY( pValue ) || HB_IS_HASH( pValue ) ) && hb_itemSize( pValue ) > 0 )
|
|
{
|
|
void * id = HB_IS_HASH( pValue ) ? hb_hashId( pValue ) : hb_arrayId( pValue );
|
|
HB_SIZE nIndex;
|
|
|
|
for( nIndex = 0; nIndex < nLevel; nIndex++ )
|
|
{
|
|
if( pCtx->pId[ nIndex ] == id )
|
|
{
|
|
if( ! fEOL && pCtx->fHuman )
|
|
_hb_jsonCtxAddIndent( pCtx, nLevel * INDENT_SIZE );
|
|
_hb_jsonCtxAdd( pCtx, "null", 4 );
|
|
return;
|
|
}
|
|
}
|
|
if( nLevel >= pCtx->nAllocId )
|
|
{
|
|
pCtx->nAllocId += 8;
|
|
pCtx->pId = ( void ** ) hb_xrealloc( pCtx->pId, sizeof( void * ) * pCtx->nAllocId );
|
|
}
|
|
pCtx->pId[ nLevel ] = id;
|
|
}
|
|
|
|
if( fEOL )
|
|
{
|
|
--pCtx->pHead;
|
|
_hb_jsonCtxAdd( pCtx, pCtx->szEol, pCtx->iEolLen );
|
|
}
|
|
|
|
if( HB_IS_STRING( pValue ) )
|
|
{
|
|
const char * szString = hb_itemGetCPtr( pValue );
|
|
HB_SIZE nPos, nPos2, nLen = hb_itemGetCLen( pValue );
|
|
|
|
_hb_jsonCtxAdd( pCtx, "\"", 1 );
|
|
|
|
nPos = 0;
|
|
while( nPos < nLen )
|
|
{
|
|
nPos2 = nPos;
|
|
while( *( ( const unsigned char * ) szString + nPos2 ) >= ' ' &&
|
|
szString[ nPos2 ] != '\\' && szString[ nPos2 ] != '\"' )
|
|
nPos2++;
|
|
if( nPos2 > nPos )
|
|
{
|
|
_hb_jsonCtxAdd( pCtx, szString + nPos, nPos2 - nPos );
|
|
nPos = nPos2;
|
|
continue;
|
|
}
|
|
|
|
switch( szString[ nPos ] )
|
|
{
|
|
case '\\':
|
|
_hb_jsonCtxAdd( pCtx, "\\\\", 2 );
|
|
break;
|
|
case '\"':
|
|
_hb_jsonCtxAdd( pCtx, "\\\"", 2 );
|
|
break;
|
|
case '\b':
|
|
_hb_jsonCtxAdd( pCtx, "\\b", 2 );
|
|
break;
|
|
case '\f':
|
|
_hb_jsonCtxAdd( pCtx, "\\f", 2 );
|
|
break;
|
|
case '\n':
|
|
_hb_jsonCtxAdd( pCtx, "\\n", 2 );
|
|
break;
|
|
case '\r':
|
|
_hb_jsonCtxAdd( pCtx, "\\r", 2 );
|
|
break;
|
|
case '\t':
|
|
_hb_jsonCtxAdd( pCtx, "\\t", 2 );
|
|
break;
|
|
default:
|
|
{
|
|
char buf[ 8 ];
|
|
hb_snprintf( buf, sizeof( buf ), "\\u00%02X", ( unsigned char ) szString[ nPos ] );
|
|
_hb_jsonCtxAdd( pCtx, buf, 6 );
|
|
break;
|
|
}
|
|
}
|
|
nPos++;
|
|
}
|
|
_hb_jsonCtxAdd( pCtx, "\"", 1 );
|
|
}
|
|
else if( HB_IS_NUMINT( pValue ) )
|
|
{
|
|
char buf[ 32 ];
|
|
|
|
hb_snprintf( buf, sizeof( buf ), "%" PFHL "d", hb_itemGetNInt( pValue ) );
|
|
_hb_jsonCtxAdd( pCtx, buf, strlen( buf ) );
|
|
}
|
|
else if( HB_IS_NUMERIC( pValue ) )
|
|
{
|
|
char buf[ 64 ];
|
|
int iDec;
|
|
double dblValue = hb_itemGetNDDec( pValue, &iDec );
|
|
|
|
hb_snprintf( buf, sizeof( buf ), "%.*f", iDec, dblValue );
|
|
_hb_jsonCtxAdd( pCtx, buf, strlen( buf ) );
|
|
}
|
|
else if( HB_IS_NIL( pValue ) )
|
|
{
|
|
_hb_jsonCtxAdd( pCtx, "null", 4 );
|
|
}
|
|
else if( HB_IS_LOGICAL( pValue ) )
|
|
{
|
|
if( hb_itemGetL( pValue ) )
|
|
_hb_jsonCtxAdd( pCtx, "true", 4 );
|
|
else
|
|
_hb_jsonCtxAdd( pCtx, "false", 5 );
|
|
|
|
}
|
|
else if( HB_IS_DATE( pValue ) )
|
|
{
|
|
char szBuffer[ 10 ];
|
|
|
|
hb_itemGetDS( pValue, szBuffer + 1 );
|
|
szBuffer[ 0 ] = '\"';
|
|
szBuffer[ 9 ] = '\"';
|
|
_hb_jsonCtxAdd( pCtx, szBuffer, 10 );
|
|
}
|
|
else if( HB_IS_TIMESTAMP( pValue ) )
|
|
{
|
|
char szBuffer[ 19 ];
|
|
hb_itemGetTS( pValue, szBuffer + 1 );
|
|
szBuffer[ 0 ] = '\"';
|
|
szBuffer[ 18 ] = '\"';
|
|
_hb_jsonCtxAdd( pCtx, szBuffer, 19 );
|
|
}
|
|
else if( HB_IS_ARRAY( pValue ) )
|
|
{
|
|
HB_SIZE nLen = hb_itemSize( pValue );
|
|
|
|
if( nLen )
|
|
{
|
|
HB_SIZE nIndex;
|
|
|
|
if( pCtx->fHuman )
|
|
_hb_jsonCtxAddIndent( pCtx, nLevel * INDENT_SIZE );
|
|
|
|
_hb_jsonCtxAdd( pCtx, "[", 1 );
|
|
|
|
for( nIndex = 1; nIndex <= nLen; nIndex++ )
|
|
{
|
|
PHB_ITEM pItem = hb_arrayGetItemPtr( pValue, nIndex );
|
|
|
|
if( nIndex > 1 )
|
|
_hb_jsonCtxAdd( pCtx, ",", 1 );
|
|
|
|
if( pCtx->fHuman )
|
|
_hb_jsonCtxAdd( pCtx, pCtx->szEol, pCtx->iEolLen );
|
|
|
|
if( pCtx->fHuman &&
|
|
! ( ( HB_IS_ARRAY( pItem ) || HB_IS_HASH( pItem ) ) &&
|
|
hb_itemSize( pItem ) > 0 ) )
|
|
_hb_jsonCtxAddIndent( pCtx, ( nLevel + 1 ) * INDENT_SIZE );
|
|
|
|
_hb_jsonEncode( pItem, pCtx, nLevel + 1, HB_FALSE );
|
|
}
|
|
if( pCtx->fHuman )
|
|
{
|
|
_hb_jsonCtxAdd( pCtx, pCtx->szEol, pCtx->iEolLen );
|
|
_hb_jsonCtxAddIndent( pCtx, nLevel * INDENT_SIZE );
|
|
}
|
|
_hb_jsonCtxAdd( pCtx, "]", 1 );
|
|
}
|
|
else
|
|
_hb_jsonCtxAdd( pCtx, "[]", 2 );
|
|
}
|
|
else if( HB_IS_HASH( pValue ) )
|
|
{
|
|
HB_SIZE nLen = hb_hashLen( pValue );
|
|
|
|
if( nLen )
|
|
{
|
|
HB_SIZE nIndex;
|
|
|
|
if( pCtx->fHuman )
|
|
_hb_jsonCtxAddIndent( pCtx, nLevel * INDENT_SIZE );
|
|
|
|
_hb_jsonCtxAdd( pCtx, "{", 1 );
|
|
|
|
for( nIndex = 1; nIndex <= nLen; nIndex++ )
|
|
{
|
|
PHB_ITEM pKey = hb_hashGetKeyAt( pValue, nIndex );
|
|
|
|
if( HB_IS_STRING( pKey ) )
|
|
{
|
|
PHB_ITEM pItem = hb_hashGetValueAt( pValue, nIndex );
|
|
HB_BOOL fEOL = HB_FALSE;
|
|
|
|
if( nIndex > 1 )
|
|
_hb_jsonCtxAdd( pCtx, ",", 1 );
|
|
|
|
if( pCtx->fHuman )
|
|
{
|
|
_hb_jsonCtxAdd( pCtx, pCtx->szEol, pCtx->iEolLen );
|
|
_hb_jsonCtxAddIndent( pCtx, ( nLevel + 1 ) * INDENT_SIZE );
|
|
}
|
|
_hb_jsonEncode( pKey, pCtx, nLevel + 1, HB_FALSE );
|
|
|
|
if( pCtx->fHuman )
|
|
{
|
|
_hb_jsonCtxAdd( pCtx, ": ", 2 );
|
|
fEOL = ( HB_IS_ARRAY( pItem ) || HB_IS_HASH( pItem ) ) && hb_itemSize( pItem ) > 0;
|
|
}
|
|
else
|
|
_hb_jsonCtxAdd( pCtx, ":", 1 );
|
|
|
|
_hb_jsonEncode( pItem, pCtx, nLevel + 1, fEOL );
|
|
}
|
|
}
|
|
if( pCtx->fHuman )
|
|
{
|
|
_hb_jsonCtxAdd( pCtx, pCtx->szEol, pCtx->iEolLen );
|
|
_hb_jsonCtxAddIndent( pCtx, nLevel * INDENT_SIZE );
|
|
}
|
|
_hb_jsonCtxAdd( pCtx, "}", 1 );
|
|
}
|
|
else
|
|
_hb_jsonCtxAdd( pCtx, "{}", 2 );
|
|
}
|
|
else
|
|
{
|
|
/* All unsupported types are replacd by null */
|
|
_hb_jsonCtxAdd( pCtx, "null", 4 );
|
|
}
|
|
}
|
|
|
|
|
|
static const char * _skipws( const char * szSource )
|
|
{
|
|
while( *szSource == ' ' || *szSource == '\t' || *szSource == '\n' || *szSource == '\r' )
|
|
szSource++;
|
|
return szSource;
|
|
}
|
|
|
|
static const char * _hb_jsonDecode( const char * szSource, PHB_ITEM pValue )
|
|
{
|
|
if( *szSource == '\"' )
|
|
{
|
|
char * szDest, * szHead;
|
|
HB_SIZE nAlloc = 16;
|
|
|
|
szHead = szDest = ( char * ) hb_xgrab( nAlloc );
|
|
szSource++;
|
|
while( *szSource != '\"' )
|
|
{
|
|
if( szHead + 6 >= szDest + nAlloc )
|
|
{
|
|
HB_SIZE nLen = szHead - szDest;
|
|
nAlloc += nAlloc << 1;
|
|
szDest = ( char * ) hb_xrealloc( szDest, nAlloc );
|
|
szHead = szDest + nLen;
|
|
}
|
|
if( *szSource == '\\' )
|
|
{
|
|
szSource++;
|
|
switch( *szSource )
|
|
{
|
|
case '\"':
|
|
*szHead++ = '\"';
|
|
break;
|
|
case '\\':
|
|
*szHead++ = '\\';
|
|
break;
|
|
case '/':
|
|
*szHead++ = '/';
|
|
break;
|
|
case 'b':
|
|
*szHead++ = '\b';
|
|
break;
|
|
case 'f':
|
|
*szHead++ = '\f';
|
|
break;
|
|
case 'n':
|
|
*szHead++ = '\n';
|
|
break;
|
|
case 'r':
|
|
*szHead++ = '\r';
|
|
break;
|
|
case 't':
|
|
*szHead++ = '\t';
|
|
break;
|
|
case 'u':
|
|
{
|
|
HB_WCHAR wc = 0;
|
|
int i;
|
|
|
|
for( i = 0; i < 4; i++ )
|
|
{
|
|
char c = *++szSource;
|
|
wc <<= 4;
|
|
if( c >= '0' && c <= '9' )
|
|
wc += c - '0';
|
|
else if( c >= 'A' && c <= 'F' )
|
|
wc += c - 'A' + 10;
|
|
else if( c >= 'a' && c <= 'f' )
|
|
wc += c - 'a' + 10;
|
|
else
|
|
{
|
|
hb_xfree( szDest );
|
|
return NULL;
|
|
}
|
|
}
|
|
szHead += hb_cdpU16ToStr( hb_vmCDP(), HB_CDP_ENDIAN_NATIVE,
|
|
&wc, 1,
|
|
szHead, szDest + nAlloc - szHead );
|
|
break;
|
|
}
|
|
default:
|
|
hb_xfree( szDest );
|
|
return NULL;
|
|
}
|
|
szSource++;
|
|
}
|
|
else if( *( const unsigned char * ) szSource >= ' ' )
|
|
*szHead++ = *szSource++;
|
|
else
|
|
{
|
|
hb_xfree( szDest );
|
|
return NULL;
|
|
}
|
|
}
|
|
hb_itemPutCL( pValue, szDest, szHead - szDest );
|
|
hb_xfree( szDest );
|
|
return szSource + 1;
|
|
}
|
|
else if( *szSource == '-' || ( *szSource >= '0' && *szSource <= '9' ) )
|
|
{
|
|
/* NOTE: this function is much less strict to number format than
|
|
JSON syntax definition. This is allowed behaviour [Mindaugas] */
|
|
HB_MAXINT nValue = 0;
|
|
double dblValue = 0;
|
|
HB_BOOL fNeg, fDbl = HB_FALSE;
|
|
int iDec = 0;
|
|
|
|
fNeg = *szSource == '-';
|
|
if( fNeg )
|
|
szSource++;
|
|
|
|
while( *szSource >= '0' && *szSource <= '9' )
|
|
{
|
|
nValue = nValue * 10 + *szSource - '0';
|
|
szSource++;
|
|
}
|
|
if( *szSource == '.' )
|
|
{
|
|
double mult = 1;
|
|
|
|
dblValue = ( double ) nValue;
|
|
fDbl = HB_TRUE;
|
|
szSource++;
|
|
while( *szSource >= '0' && *szSource <= '9' )
|
|
{
|
|
mult /= 10;
|
|
dblValue += ( ( double ) ( *szSource - '0' ) ) * mult;
|
|
szSource++;
|
|
iDec++;
|
|
}
|
|
}
|
|
if( *szSource == 'e' || *szSource == 'E' )
|
|
{
|
|
HB_BOOL fNegExp;
|
|
int iExp = 0;
|
|
|
|
szSource++;
|
|
fNegExp = *szSource == '-';
|
|
if( fNegExp )
|
|
szSource++;
|
|
|
|
while( *szSource >= '0' && *szSource <= '9' )
|
|
{
|
|
iExp = iExp * 10 + *szSource - '0';
|
|
szSource++;
|
|
}
|
|
if( ! fDbl )
|
|
{
|
|
dblValue = ( double ) nValue;
|
|
fDbl = HB_TRUE;
|
|
}
|
|
if( fNegExp )
|
|
iDec += iExp;
|
|
dblValue = hb_numExpConv( dblValue, fNegExp ? iExp : -iExp );
|
|
}
|
|
|
|
if( fDbl )
|
|
hb_itemPutNDDec( pValue, hb_numRound( fNeg ? -dblValue : dblValue, iDec ), iDec );
|
|
else
|
|
hb_itemPutNInt( pValue, fNeg ? -nValue : nValue );
|
|
return szSource;
|
|
}
|
|
else if( ! strncmp( szSource, "null", 4 ) )
|
|
{
|
|
hb_itemClear( pValue );
|
|
return szSource + 4;
|
|
}
|
|
else if( ! strncmp( szSource, "true", 4 ) )
|
|
{
|
|
hb_itemPutL( pValue, HB_TRUE );
|
|
return szSource + 4;
|
|
}
|
|
else if( ! strncmp( szSource, "false", 5 ) )
|
|
{
|
|
hb_itemPutL( pValue, HB_FALSE );
|
|
return szSource + 5;
|
|
}
|
|
else if( *szSource == '[' )
|
|
{
|
|
hb_arrayNew( pValue, 0 );
|
|
szSource = _skipws( szSource + 1 );
|
|
if( *szSource != ']' )
|
|
{
|
|
PHB_ITEM pItem = hb_itemNew( NULL );
|
|
|
|
for( ;; )
|
|
{
|
|
szSource = _hb_jsonDecode( szSource, pItem );
|
|
if( ! szSource )
|
|
{
|
|
hb_itemRelease( pItem );
|
|
return NULL;
|
|
}
|
|
hb_arrayAddForward( pValue, pItem );
|
|
|
|
szSource = _skipws( szSource );
|
|
if( *szSource == ',' )
|
|
{
|
|
szSource = _skipws( szSource + 1 );
|
|
continue;
|
|
}
|
|
else if( *szSource == ']' )
|
|
break;
|
|
else
|
|
{
|
|
hb_itemRelease( pItem );
|
|
return NULL;
|
|
}
|
|
}
|
|
hb_itemRelease( pItem );
|
|
}
|
|
return szSource + 1;
|
|
}
|
|
else if( *szSource == '{' )
|
|
{
|
|
hb_hashNew( pValue );
|
|
szSource = _skipws( szSource + 1 );
|
|
if( *szSource != '}' )
|
|
{
|
|
PHB_ITEM pItemKey = hb_itemNew( NULL );
|
|
PHB_ITEM pItemValue = hb_itemNew( NULL );
|
|
|
|
for( ;; )
|
|
{
|
|
/* Do we need to check if key does not exist yet? */
|
|
if( ( szSource = _hb_jsonDecode( szSource, pItemKey ) ) == NULL ||
|
|
! HB_IS_STRING( pItemKey ) ||
|
|
* ( szSource = _skipws( szSource ) ) != ':' ||
|
|
( szSource = _hb_jsonDecode( _skipws( szSource + 1 ), pItemValue ) ) == NULL)
|
|
{
|
|
hb_itemRelease( pItemKey );
|
|
hb_itemRelease( pItemValue );
|
|
return NULL;
|
|
}
|
|
|
|
hb_hashAdd( pValue, pItemKey, pItemValue );
|
|
szSource = _skipws( szSource );
|
|
if( *szSource == ',' )
|
|
{
|
|
szSource = _skipws( szSource + 1 );
|
|
continue;
|
|
}
|
|
else if( *szSource == '}' )
|
|
break;
|
|
else
|
|
{
|
|
hb_itemRelease( pItemKey );
|
|
hb_itemRelease( pItemValue );
|
|
return NULL;
|
|
}
|
|
}
|
|
hb_itemRelease( pItemKey );
|
|
hb_itemRelease( pItemValue );
|
|
}
|
|
return szSource + 1;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
/* C level API functions */
|
|
|
|
char * hb_jsonEncode( PHB_ITEM pValue, HB_SIZE * pnLen, HB_BOOL fHuman )
|
|
{
|
|
PHB_JSON_ENCODE_CTX pCtx;
|
|
char * szRet;
|
|
HB_SIZE nLen;
|
|
|
|
pCtx = ( PHB_JSON_ENCODE_CTX ) hb_xgrab( sizeof( HB_JSON_ENCODE_CTX ) );
|
|
pCtx->nAlloc = 16;
|
|
pCtx->pHead = pCtx->pBuffer = ( char * ) hb_xgrab( pCtx->nAlloc );
|
|
pCtx->nAllocId = 8;
|
|
pCtx->pId = ( void ** ) hb_xgrab( sizeof( void * ) * pCtx->nAllocId );
|
|
pCtx->fHuman = fHuman;
|
|
pCtx->szEol = hb_setGetEOL();
|
|
if( ! pCtx->szEol || ! pCtx->szEol[ 0 ] )
|
|
pCtx->szEol = hb_conNewLine();
|
|
pCtx->iEolLen = ( int ) strlen( pCtx->szEol );
|
|
|
|
_hb_jsonEncode( pValue, pCtx, 0, HB_FALSE );
|
|
|
|
nLen = pCtx->pHead - pCtx->pBuffer;
|
|
szRet = ( char * ) hb_xrealloc( pCtx->pBuffer, nLen + 1 );
|
|
szRet[ nLen ] = '\0';
|
|
hb_xfree( pCtx->pId );
|
|
hb_xfree( pCtx );
|
|
if( pnLen )
|
|
*pnLen = nLen;
|
|
return szRet;
|
|
}
|
|
|
|
HB_SIZE hb_jsonDecode( const char * szSource, PHB_ITEM pValue )
|
|
{
|
|
PHB_ITEM pItem = pValue ? pValue : hb_itemNew( NULL );
|
|
const char * sz;
|
|
|
|
sz = szSource ? _hb_jsonDecode( _skipws( szSource ), pItem ) : NULL;
|
|
if( ! pValue )
|
|
hb_itemRelease( pItem );
|
|
if( sz )
|
|
return sz - szSource;
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* Harbour level API functions */
|
|
|
|
HB_FUNC( HB_JSONENCODE )
|
|
{
|
|
PHB_ITEM pItem = hb_param( 1, HB_IT_ANY );
|
|
|
|
if( pItem )
|
|
{
|
|
HB_SIZE nLen;
|
|
|
|
char * szRet = hb_jsonEncode( pItem, &nLen, hb_parl( 2 ) );
|
|
hb_retclen_buffer( szRet, nLen );
|
|
}
|
|
}
|
|
|
|
HB_FUNC( HB_JSONDECODE )
|
|
{
|
|
PHB_ITEM pItem = hb_itemNew( NULL );
|
|
|
|
hb_retns( ( HB_ISIZ ) hb_jsonDecode( hb_parc( 1 ), pItem ) );
|
|
hb_itemParamStoreForward( 2, pItem );
|
|
hb_itemRelease( pItem );
|
|
}
|