2004-02-20 07:19:18 +00:00
|
|
|
/***************************************************************************
|
2004-11-29 12:10:09 +00:00
|
|
|
* _ _ ____ _
|
|
|
|
* Project ___| | | | _ \| |
|
|
|
|
* / __| | | | |_) | |
|
|
|
|
* | (__| |_| | _ <| |___
|
2004-02-20 07:19:18 +00:00
|
|
|
* \___|\___/|_| \_\_____|
|
|
|
|
*
|
2012-04-10 17:32:06 +02:00
|
|
|
* Copyright (C) 1998 - 2012, Daniel Stenberg, <daniel@haxx.se>, et al.
|
2004-02-20 07:19:18 +00:00
|
|
|
*
|
|
|
|
* This software is licensed as described in the file COPYING, which
|
|
|
|
* you should have received as part of this distribution. The terms
|
|
|
|
* are also available at http://curl.haxx.se/docs/copyright.html.
|
2004-11-29 12:10:09 +00:00
|
|
|
*
|
2004-02-20 07:19:18 +00:00
|
|
|
* You may opt to use, copy, modify, merge, publish, distribute and/or sell
|
|
|
|
* copies of the Software, and permit persons to whom the Software is
|
|
|
|
* furnished to do so, under the terms of the COPYING file.
|
|
|
|
*
|
|
|
|
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
|
|
|
|
* KIND, either express or implied.
|
|
|
|
*
|
|
|
|
***************************************************************************/
|
2012-04-10 17:32:06 +02:00
|
|
|
#include "server_setup.h"
|
2004-11-29 21:44:23 +00:00
|
|
|
|
2004-02-20 07:19:18 +00:00
|
|
|
#include "getpart.h"
|
2002-02-07 09:39:15 +00:00
|
|
|
|
2010-02-26 16:42:33 +00:00
|
|
|
#define ENABLE_CURLX_PRINTF
|
|
|
|
/* make the curlx header define all printf() functions to use the curlx_*
|
|
|
|
versions instead */
|
|
|
|
#include "curlx.h" /* from the private lib dir */
|
2004-11-29 12:10:09 +00:00
|
|
|
|
2011-04-05 22:30:09 -07:00
|
|
|
/* just to please curl_base64.h we create a fake struct */
|
2007-01-04 23:04:50 +00:00
|
|
|
struct SessionHandle {
|
|
|
|
int fake;
|
|
|
|
};
|
|
|
|
|
2008-08-17 00:25:38 +00:00
|
|
|
#include "curl_base64.h"
|
2009-04-21 11:46:16 +00:00
|
|
|
#include "curl_memory.h"
|
2004-11-29 12:10:09 +00:00
|
|
|
|
|
|
|
/* include memdebug.h last */
|
|
|
|
#include "memdebug.h"
|
|
|
|
|
2010-02-14 13:14:17 +00:00
|
|
|
#define EAT_SPACE(p) while(*(p) && ISSPACE(*(p))) (p)++
|
2002-02-07 09:39:15 +00:00
|
|
|
|
2010-02-14 13:14:17 +00:00
|
|
|
#define EAT_WORD(p) while(*(p) && !ISSPACE(*(p)) && ('>' != *(p))) (p)++
|
|
|
|
|
|
|
|
#ifdef DEBUG_GETPART
|
2002-02-22 10:40:05 +00:00
|
|
|
#define show(x) printf x
|
|
|
|
#else
|
2011-09-03 16:06:10 +02:00
|
|
|
#define show(x) Curl_nop_stmt
|
2002-02-22 10:40:05 +00:00
|
|
|
#endif
|
|
|
|
|
2008-10-23 14:07:28 +00:00
|
|
|
#if defined(_MSC_VER) && defined(_DLL)
|
|
|
|
# pragma warning(disable:4232) /* MSVC extension, dllimport identity */
|
|
|
|
#endif
|
|
|
|
|
2004-11-29 12:10:09 +00:00
|
|
|
curl_malloc_callback Curl_cmalloc = (curl_malloc_callback)malloc;
|
|
|
|
curl_free_callback Curl_cfree = (curl_free_callback)free;
|
|
|
|
curl_realloc_callback Curl_crealloc = (curl_realloc_callback)realloc;
|
|
|
|
curl_strdup_callback Curl_cstrdup = (curl_strdup_callback)strdup;
|
|
|
|
curl_calloc_callback Curl_ccalloc = (curl_calloc_callback)calloc;
|
|
|
|
|
2008-10-23 14:07:28 +00:00
|
|
|
#if defined(_MSC_VER) && defined(_DLL)
|
|
|
|
# pragma warning(default:4232) /* MSVC extension, dllimport identity */
|
|
|
|
#endif
|
|
|
|
|
2010-02-14 13:14:17 +00:00
|
|
|
/*
|
|
|
|
* readline()
|
|
|
|
*
|
|
|
|
* Reads a complete line from a file into a dynamically allocated buffer.
|
|
|
|
*
|
|
|
|
* Calling function may call this multiple times with same 'buffer'
|
|
|
|
* and 'bufsize' pointers to avoid multiple buffer allocations. Buffer
|
|
|
|
* will be reallocated and 'bufsize' increased until whole line fits in
|
|
|
|
* buffer before returning it.
|
|
|
|
*
|
|
|
|
* Calling function is responsible to free allocated buffer.
|
|
|
|
*
|
|
|
|
* This function may return:
|
|
|
|
* GPE_OUT_OF_MEMORY
|
|
|
|
* GPE_END_OF_FILE
|
|
|
|
* GPE_OK
|
|
|
|
*/
|
|
|
|
|
|
|
|
static int readline(char **buffer, size_t *bufsize, FILE *stream)
|
|
|
|
{
|
|
|
|
size_t offset = 0;
|
|
|
|
size_t length;
|
|
|
|
char *newptr;
|
|
|
|
|
|
|
|
if(!*buffer) {
|
|
|
|
*buffer = malloc(128);
|
|
|
|
if(!*buffer)
|
|
|
|
return GPE_OUT_OF_MEMORY;
|
|
|
|
*bufsize = 128;
|
|
|
|
}
|
|
|
|
|
|
|
|
for(;;) {
|
2010-02-26 16:42:33 +00:00
|
|
|
int bytestoread = curlx_uztosi(*bufsize - offset);
|
|
|
|
|
|
|
|
if(!fgets(*buffer + offset, bytestoread, stream))
|
2010-02-14 13:14:17 +00:00
|
|
|
return (offset != 0) ? GPE_OK : GPE_END_OF_FILE ;
|
|
|
|
|
|
|
|
length = offset + strlen(*buffer + offset);
|
|
|
|
if(*(*buffer + length - 1) == '\n')
|
|
|
|
break;
|
|
|
|
offset = length;
|
|
|
|
if(length < *bufsize - 1)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
newptr = realloc(*buffer, *bufsize * 2);
|
|
|
|
if(!newptr)
|
|
|
|
return GPE_OUT_OF_MEMORY;
|
|
|
|
*buffer = newptr;
|
|
|
|
*bufsize *= 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
return GPE_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* appenddata()
|
|
|
|
*
|
|
|
|
* This appends data from a given source buffer to the end of the used part of
|
|
|
|
* a destination buffer. Arguments relative to the destination buffer are, the
|
|
|
|
* address of a pointer to the destination buffer 'dst_buf', the length of data
|
|
|
|
* in destination buffer excluding potential null string termination 'dst_len',
|
|
|
|
* the allocated size of destination buffer 'dst_alloc'. All three destination
|
|
|
|
* buffer arguments may be modified by this function. Arguments relative to the
|
|
|
|
* source buffer are, a pointer to the source buffer 'src_buf' and indication
|
|
|
|
* whether the source buffer is base64 encoded or not 'src_b64'.
|
|
|
|
*
|
|
|
|
* If the source buffer is indicated to be base64 encoded, this appends the
|
|
|
|
* decoded data, binary or whatever, to the destination. The source buffer
|
|
|
|
* may not hold binary data, only a null terminated string is valid content.
|
|
|
|
*
|
|
|
|
* Destination buffer will be enlarged and relocated as needed.
|
|
|
|
*
|
|
|
|
* Calling function is responsible to provide preallocated destination
|
|
|
|
* buffer and also to deallocate it when no longer needed.
|
|
|
|
*
|
|
|
|
* This function may return:
|
|
|
|
* GPE_OUT_OF_MEMORY
|
|
|
|
* GPE_OK
|
|
|
|
*/
|
|
|
|
|
|
|
|
static int appenddata(char **dst_buf, /* dest buffer */
|
|
|
|
size_t *dst_len, /* dest buffer data length */
|
|
|
|
size_t *dst_alloc, /* dest buffer allocated size */
|
|
|
|
char *src_buf, /* source buffer */
|
|
|
|
int src_b64) /* != 0 if source is base64 encoded */
|
2002-02-19 01:04:46 +00:00
|
|
|
{
|
2010-02-14 13:14:17 +00:00
|
|
|
size_t need_alloc, src_len;
|
2008-10-02 14:44:18 +00:00
|
|
|
union {
|
2010-02-14 13:14:17 +00:00
|
|
|
unsigned char *as_uchar;
|
|
|
|
char *as_char;
|
2008-10-02 14:44:18 +00:00
|
|
|
} buf64;
|
|
|
|
|
2010-02-14 13:14:17 +00:00
|
|
|
src_len = strlen(src_buf);
|
|
|
|
if(!src_len)
|
|
|
|
return GPE_OK;
|
2008-10-02 14:44:18 +00:00
|
|
|
|
|
|
|
buf64.as_char = NULL;
|
2004-11-29 12:10:09 +00:00
|
|
|
|
2010-02-14 13:14:17 +00:00
|
|
|
if(src_b64) {
|
|
|
|
/* base64 decode the given buffer */
|
2011-08-24 08:07:36 +02:00
|
|
|
int error = (int) Curl_base64_decode(src_buf, &buf64.as_uchar, &src_len);
|
|
|
|
if(error)
|
|
|
|
return GPE_OUT_OF_MEMORY;
|
2010-02-14 13:14:17 +00:00
|
|
|
src_buf = buf64.as_char;
|
|
|
|
if(!src_len || !src_buf) {
|
|
|
|
/*
|
|
|
|
** currently there is no way to tell apart an OOM condition in
|
|
|
|
** Curl_base64_decode() from zero length decoded data. For now,
|
|
|
|
** let's just assume it is an OOM condition, currently we have
|
|
|
|
** no input for this function that decodes to zero length data.
|
|
|
|
*/
|
|
|
|
if(buf64.as_char)
|
|
|
|
free(buf64.as_char);
|
|
|
|
return GPE_OUT_OF_MEMORY;
|
|
|
|
}
|
2004-11-29 12:10:09 +00:00
|
|
|
}
|
2002-02-19 01:04:46 +00:00
|
|
|
|
2010-02-14 13:14:17 +00:00
|
|
|
need_alloc = src_len + *dst_len + 1;
|
2003-09-03 15:37:30 +00:00
|
|
|
|
2010-02-14 13:14:17 +00:00
|
|
|
/* enlarge destination buffer if required */
|
|
|
|
if(need_alloc > *dst_alloc) {
|
|
|
|
size_t newsize = need_alloc * 2;
|
|
|
|
char *newptr = realloc(*dst_buf, newsize);
|
|
|
|
if(!newptr) {
|
2008-10-02 14:44:18 +00:00
|
|
|
if(buf64.as_char)
|
|
|
|
free(buf64.as_char);
|
2010-02-14 13:14:17 +00:00
|
|
|
return GPE_OUT_OF_MEMORY;
|
2005-02-24 18:54:23 +00:00
|
|
|
}
|
2010-02-14 13:14:17 +00:00
|
|
|
*dst_alloc = newsize;
|
|
|
|
*dst_buf = newptr;
|
2002-02-19 01:04:46 +00:00
|
|
|
}
|
2010-02-14 13:14:17 +00:00
|
|
|
|
2004-11-29 12:10:09 +00:00
|
|
|
/* memcpy to support binary blobs */
|
2010-02-14 13:14:17 +00:00
|
|
|
memcpy(*dst_buf + *dst_len, src_buf, src_len);
|
|
|
|
*dst_len += src_len;
|
|
|
|
*(*dst_buf + *dst_len) = '\0';
|
2002-02-19 01:04:46 +00:00
|
|
|
|
2008-10-02 14:44:18 +00:00
|
|
|
if(buf64.as_char)
|
|
|
|
free(buf64.as_char);
|
2005-02-22 12:10:30 +00:00
|
|
|
|
2010-02-14 13:14:17 +00:00
|
|
|
return GPE_OK;
|
2002-02-19 01:04:46 +00:00
|
|
|
}
|
|
|
|
|
2010-02-14 13:14:17 +00:00
|
|
|
/*
|
|
|
|
* getpart()
|
|
|
|
*
|
|
|
|
* This returns whole contents of specified XML-like section and subsection
|
|
|
|
* from the given file. This is mostly used to retrieve a specific part from
|
|
|
|
* a test definition file for consumption by test suite servers.
|
|
|
|
*
|
|
|
|
* Data is returned in a dynamically allocated buffer, a pointer to this data
|
|
|
|
* and the size of the data is stored at the addresses that caller specifies.
|
|
|
|
*
|
|
|
|
* If the returned data is a string the returned size will be the length of
|
|
|
|
* the string excluding null termination. Otherwise it will just be the size
|
|
|
|
* of the returned binary data.
|
|
|
|
*
|
|
|
|
* Calling function is responsible to free returned buffer.
|
|
|
|
*
|
|
|
|
* This function may return:
|
|
|
|
* GPE_NO_BUFFER_SPACE
|
|
|
|
* GPE_OUT_OF_MEMORY
|
|
|
|
* GPE_OK
|
|
|
|
*/
|
|
|
|
|
|
|
|
int getpart(char **outbuf, size_t *outlen,
|
|
|
|
const char *main, const char *sub, FILE *stream)
|
2002-02-07 09:39:15 +00:00
|
|
|
{
|
2010-02-14 13:14:17 +00:00
|
|
|
# define MAX_TAG_LEN 79
|
|
|
|
char couter[MAX_TAG_LEN+1]; /* current outermost section */
|
|
|
|
char cmain[MAX_TAG_LEN+1]; /* current main section */
|
|
|
|
char csub[MAX_TAG_LEN+1]; /* current sub section */
|
|
|
|
char ptag[MAX_TAG_LEN+1]; /* potential tag */
|
|
|
|
char patt[MAX_TAG_LEN+1]; /* potential attributes */
|
|
|
|
char *buffer = NULL;
|
2002-02-07 09:39:15 +00:00
|
|
|
char *ptr;
|
|
|
|
char *end;
|
2010-02-15 16:18:52 +00:00
|
|
|
union {
|
|
|
|
ssize_t sig;
|
|
|
|
size_t uns;
|
|
|
|
} len;
|
2010-02-14 13:14:17 +00:00
|
|
|
size_t bufsize = 0;
|
|
|
|
size_t outalloc = 256;
|
|
|
|
int in_wanted_part = 0;
|
|
|
|
int base64 = 0;
|
|
|
|
int error;
|
2002-02-07 09:39:15 +00:00
|
|
|
|
|
|
|
enum {
|
2010-02-14 13:14:17 +00:00
|
|
|
STATE_OUTSIDE = 0,
|
|
|
|
STATE_OUTER = 1,
|
|
|
|
STATE_INMAIN = 2,
|
|
|
|
STATE_INSUB = 3,
|
|
|
|
STATE_ILLEGAL = 4
|
2002-02-07 09:39:15 +00:00
|
|
|
} state = STATE_OUTSIDE;
|
|
|
|
|
2010-02-14 13:14:17 +00:00
|
|
|
*outlen = 0;
|
|
|
|
*outbuf = malloc(outalloc);
|
|
|
|
if(!*outbuf)
|
|
|
|
return GPE_OUT_OF_MEMORY;
|
|
|
|
*(*outbuf) = '\0';
|
2003-09-03 15:37:30 +00:00
|
|
|
|
2010-02-14 13:14:17 +00:00
|
|
|
couter[0] = cmain[0] = csub[0] = ptag[0] = patt[0] = '\0';
|
2004-11-29 12:10:09 +00:00
|
|
|
|
2010-02-14 13:14:17 +00:00
|
|
|
while((error = readline(&buffer, &bufsize, stream)) == GPE_OK) {
|
2002-02-07 09:39:15 +00:00
|
|
|
|
|
|
|
ptr = buffer;
|
|
|
|
EAT_SPACE(ptr);
|
|
|
|
|
|
|
|
if('<' != *ptr) {
|
2010-02-14 13:14:17 +00:00
|
|
|
if(in_wanted_part) {
|
2002-02-22 10:40:05 +00:00
|
|
|
show(("=> %s", buffer));
|
2010-02-14 13:14:17 +00:00
|
|
|
error = appenddata(outbuf, outlen, &outalloc, buffer, base64);
|
|
|
|
if(error)
|
|
|
|
break;
|
2002-02-07 09:39:15 +00:00
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
ptr++;
|
|
|
|
|
|
|
|
if('/' == *ptr) {
|
2010-02-14 13:14:17 +00:00
|
|
|
/*
|
|
|
|
** closing section tag
|
|
|
|
*/
|
2002-02-07 09:39:15 +00:00
|
|
|
|
2010-02-14 13:14:17 +00:00
|
|
|
ptr++;
|
2002-02-07 09:39:15 +00:00
|
|
|
end = ptr;
|
|
|
|
EAT_WORD(end);
|
2010-02-15 16:18:52 +00:00
|
|
|
if((len.sig = end - ptr) > MAX_TAG_LEN) {
|
2010-02-14 13:14:17 +00:00
|
|
|
error = GPE_NO_BUFFER_SPACE;
|
|
|
|
break;
|
2002-02-07 09:39:15 +00:00
|
|
|
}
|
2010-02-15 16:18:52 +00:00
|
|
|
memcpy(ptag, ptr, len.uns);
|
|
|
|
ptag[len.uns] = '\0';
|
2010-02-14 13:14:17 +00:00
|
|
|
|
|
|
|
if((STATE_INSUB == state) && !strcmp(csub, ptag)) {
|
|
|
|
/* end of current sub section */
|
|
|
|
state = STATE_INMAIN;
|
|
|
|
csub[0] = '\0';
|
|
|
|
if(in_wanted_part) {
|
|
|
|
/* end of wanted part */
|
|
|
|
in_wanted_part = 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if((STATE_INMAIN == state) && !strcmp(cmain, ptag)) {
|
|
|
|
/* end of current main section */
|
|
|
|
state = STATE_OUTER;
|
|
|
|
cmain[0] = '\0';
|
|
|
|
if(in_wanted_part) {
|
|
|
|
/* end of wanted part */
|
|
|
|
in_wanted_part = 0;
|
|
|
|
break;
|
|
|
|
}
|
2002-02-07 09:39:15 +00:00
|
|
|
}
|
2010-02-14 13:14:17 +00:00
|
|
|
else if((STATE_OUTER == state) && !strcmp(couter, ptag)) {
|
|
|
|
/* end of outermost file section */
|
|
|
|
state = STATE_OUTSIDE;
|
|
|
|
couter[0] = '\0';
|
|
|
|
if(in_wanted_part) {
|
|
|
|
/* end of wanted part */
|
|
|
|
in_wanted_part = 0;
|
|
|
|
break;
|
|
|
|
}
|
2007-01-23 02:25:56 +00:00
|
|
|
}
|
2010-02-14 13:14:17 +00:00
|
|
|
|
2002-02-07 09:39:15 +00:00
|
|
|
}
|
2010-02-14 13:14:17 +00:00
|
|
|
else if(!in_wanted_part) {
|
|
|
|
/*
|
|
|
|
** opening section tag
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* get potential tag */
|
2002-02-07 09:39:15 +00:00
|
|
|
end = ptr;
|
|
|
|
EAT_WORD(end);
|
2010-02-15 16:18:52 +00:00
|
|
|
if((len.sig = end - ptr) > MAX_TAG_LEN) {
|
2010-02-14 13:14:17 +00:00
|
|
|
error = GPE_NO_BUFFER_SPACE;
|
2002-02-07 09:39:15 +00:00
|
|
|
break;
|
2010-02-14 13:14:17 +00:00
|
|
|
}
|
2010-02-15 16:18:52 +00:00
|
|
|
memcpy(ptag, ptr, len.uns);
|
|
|
|
ptag[len.uns] = '\0';
|
2010-02-14 13:14:17 +00:00
|
|
|
|
|
|
|
/* ignore comments, doctypes and xml declarations */
|
|
|
|
if(('!' == ptag[0]) || ('?' == ptag[0])) {
|
|
|
|
show(("* ignoring (%s)", buffer));
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* get all potential attributes */
|
|
|
|
ptr = end;
|
|
|
|
EAT_SPACE(ptr);
|
|
|
|
end = ptr;
|
|
|
|
while(*end && ('>' != *end))
|
|
|
|
end++;
|
2010-02-15 16:18:52 +00:00
|
|
|
if((len.sig = end - ptr) > MAX_TAG_LEN) {
|
2010-02-14 13:14:17 +00:00
|
|
|
error = GPE_NO_BUFFER_SPACE;
|
2003-01-23 12:00:15 +00:00
|
|
|
break;
|
2002-02-07 09:39:15 +00:00
|
|
|
}
|
2010-02-15 16:18:52 +00:00
|
|
|
memcpy(patt, ptr, len.uns);
|
|
|
|
patt[len.uns] = '\0';
|
2004-11-29 12:10:09 +00:00
|
|
|
|
2010-02-14 13:14:17 +00:00
|
|
|
if(STATE_OUTSIDE == state) {
|
|
|
|
/* outermost element (<testcase>) */
|
|
|
|
strcpy(couter, ptag);
|
|
|
|
state = STATE_OUTER;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
else if(STATE_OUTER == state) {
|
|
|
|
/* start of a main section */
|
|
|
|
strcpy(cmain, ptag);
|
|
|
|
state = STATE_INMAIN;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
else if(STATE_INMAIN == state) {
|
|
|
|
/* start of a sub section */
|
|
|
|
strcpy(csub, ptag);
|
|
|
|
state = STATE_INSUB;
|
|
|
|
if(!strcmp(cmain, main) && !strcmp(csub, sub)) {
|
|
|
|
/* start of wanted part */
|
|
|
|
in_wanted_part = 1;
|
|
|
|
if(strstr(patt, "base64="))
|
|
|
|
/* bit rough test, but "mostly" functional, */
|
|
|
|
/* treat wanted part data as base64 encoded */
|
|
|
|
base64 = 1;
|
2004-11-29 12:10:09 +00:00
|
|
|
}
|
2010-02-14 13:14:17 +00:00
|
|
|
continue;
|
2004-11-29 12:10:09 +00:00
|
|
|
}
|
2002-02-07 09:39:15 +00:00
|
|
|
|
|
|
|
}
|
2010-02-14 13:14:17 +00:00
|
|
|
|
|
|
|
if(in_wanted_part) {
|
|
|
|
show(("=> %s", buffer));
|
|
|
|
error = appenddata(outbuf, outlen, &outalloc, buffer, base64);
|
|
|
|
if(error)
|
|
|
|
break;
|
2007-01-23 02:25:56 +00:00
|
|
|
}
|
2010-02-14 13:14:17 +00:00
|
|
|
|
|
|
|
} /* while */
|
|
|
|
|
|
|
|
if(buffer)
|
|
|
|
free(buffer);
|
|
|
|
|
|
|
|
if(error != GPE_OK) {
|
|
|
|
if(error == GPE_END_OF_FILE)
|
|
|
|
error = GPE_OK;
|
2002-02-07 09:39:15 +00:00
|
|
|
else {
|
2010-02-14 13:14:17 +00:00
|
|
|
if(*outbuf)
|
|
|
|
free(*outbuf);
|
|
|
|
*outbuf = NULL;
|
|
|
|
*outlen = 0;
|
2002-02-07 09:39:15 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-02-14 13:14:17 +00:00
|
|
|
return error;
|
2002-02-07 09:39:15 +00:00
|
|
|
}
|
|
|
|
|