Files
harbour-core/harbour/contrib/tip/encmthd.c
Viktor Szakats 799edeb1e5 2007-10-01 23:50 UTC+0100 Viktor Szakats (harbour.01 syenar.hu)
* contrib/tip/Makefile
   * contrib/tip/makefile.bc
   + contrib/tip/makefile.vc
   + contrib/tip/make_vc.bat
   * contrib/tip/tip.ch
   * contrib/tip/utils.c
   * contrib/tip/base64x.c
   * contrib/tip/encmthd.c
   - contrib/tip/hbhex2n.c
   * contrib/tip/cgi.prg
   * contrib/tip/client.prg
   * contrib/tip/ftpcln.prg
   * contrib/tip/httpcln.prg
   * contrib/tip/mail.prg
   * contrib/tip/thtml.prg
   * contrib/tip/url.prg
     + Code converted to not require xhb lib.
     % Now compiled with -kM switch to avoid macro expansion 
       of strings. This is important here, as "&" char is used 
       quite often in the code, so expect a somewhat better 
       performance.
     - Removed hex number conversion functions, as compatible 
       versions are now available in Harbour with am hb_ prefix.
     + C code update to use Harbour API calls only.
     + Added VC build script.
     ! Fixed two C warnings.
     ! Standardized SVN header in one file.
     ! Set std SVN keywords and eol-style.
     ; Please test it.

   * include/hbextern.ch
   * source/rtl/hbinet.c
   * contrib/xhb/xhb.ch
   * contrib/xhb/xhbfunc.c
     * Renamed hb_IsInetSocket() to hb_inetIsSocket() to 
       keep all inet related functions in one namespace.
2007-10-01 22:17:21 +00:00

569 lines
14 KiB
C

/*
* $Id$
*/
/*
* xHarbour Project source code:
* TIP Class oriented Internet protocol library
*
* Copyright 2003 Giancarlo Niccolai <gian@niccolai.ws>
*
* www - http://www.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. 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 "hbapicls.h"
#include "hbapierr.h"
#include "hbstack.h"
HB_FUNC( TIPENCODERBASE64_ENCODE )
{
char *cData = hb_parc(1);
char *cRet;
int nLen = hb_parclen(1);
int nPos = 0, nPosRet = 0;
int nPosBlock = 0, nLineCount = 0;
ULONG nFinalLen;
unsigned char cElem, cElem1;
BOOL bExcept;
if ( ! cData )
{
hb_errRT_BASE( EG_ARG, 3012, NULL,
"TIPENCODERBASE64_ENCODE", 1, hb_paramError(1) );
return;
}
if ( ! nLen )
{
hb_retc( "" );
return;
}
// read the status of bHttpExcept
if ( hb_pcount() > 1 )
{
// this makes this function static!!!!
bExcept = hb_parl( 2 );
}
else
{
hb_objSendMsg( hb_stackSelfItem(), "BHTTPEXCEPT", 0 );
bExcept = hb_parl( -1 );
}
// we know exactly the renturned length.
nFinalLen = (ULONG) ((nLen / 3 + 2) * 4);
// add line termination padding, CRLF each 76 output bytes
nFinalLen += (nFinalLen / 72 +1) * 2;
cRet = (char *) hb_xgrab( nFinalLen );
while ( nPos < nLen )
{
cElem = (unsigned char) cData[ nPos ];
// NOT using trailing 0 here as some buggy 3dparty func
// will create strings without trailing 0.
nPosBlock++;
switch( nPosBlock )
{
case 1:
cElem = cElem >> 2;
break;
case 2:
cElem1 = nPos < nLen -1 ? (unsigned char) cData[ nPos + 1] : 0;
cElem = ((cElem & 0x3) << 4) | (cElem1 >> 4);
nPos++;
break;
case 3:
cElem1 = nPos < nLen -1 ? (unsigned char) cData[ nPos + 1] : 0;
cElem = ((cElem & 0xF) << 2) | (cElem1 >> 6);
nPos++;
break;
case 4:
cElem = cElem & 0x3f;
nPos++;
nPosBlock = 0;
break;
}
if ( cElem < 26 )
{
cRet[nPosRet++] = cElem + 'A';
}
else if ( cElem < 52 )
{
cRet[nPosRet++] = ( cElem - 26 ) + 'a';
}
else if ( cElem < 62 )
{
cRet[nPosRet++] = ( cElem - 52 ) + '0';
}
else if ( cElem == 62 )
{
cRet[nPosRet++] = '+';
}
else
{
cRet[nPosRet++] = '/';
}
if ( ! bExcept )
{
nLineCount ++ ;
/* RFC says to add a CRLF each 76 chars, but is pretty unclear about
the fact of this 76 chars counting CRLF or not. Common
practice is to limit line size to 72 chars */
if ( nLineCount == 72 )
{
cRet[nPosRet++] = '\r';
cRet[nPosRet++] = '\n';
nLineCount = 0;
}
}
}
switch( nPos % 3 )
{
case 1:
cRet[ nPosRet++ ] = '=';
/* fallthrough */
case 2:
cRet[ nPosRet++ ] = '=';
/* fallthrough */
}
/* RFC is not explicit, but CRLF SHOULD be added at bottom
during encoding phase */
if ( ! bExcept )
{
cRet[nPosRet++] = '\r';
cRet[nPosRet++] = '\n';
}
/* this function also adds a zero */
hb_retclen_buffer( cRet, nPosRet );
}
HB_FUNC( TIPENCODERBASE64_DECODE )
{
char *cData = hb_parc(1);
unsigned char *cRet;
int nLen = hb_parclen(1);
int nPos = 0, nPosRet = 0, nPosBlock = 0;
unsigned char cElem;
if ( ! cData )
{
hb_errRT_BASE( EG_ARG, 3012, NULL,
"TIPENCODERBASE64_DECODE", 1, hb_paramError(1) );
return;
}
if ( ! nLen )
{
hb_retc( "" );
return;
}
// we know exactly the renturned length.
cRet = (unsigned char *) hb_xgrab( (nLen / 4 + 1) * 3 );
while ( nPos < nLen )
{
cElem = cData[ nPos ];
if ( cElem >= 'A' && cElem <= 'Z' )
{
cElem -= 'A';
}
else if ( cElem >= 'a' && cElem <= 'z' )
{
cElem = cElem - 'a' + 26;
}
else if ( cElem >= '0' && cElem <= '9' )
{
cElem = cElem - '0' + 52;
}
else if ( cElem == '+' )
{
cElem = 62;
}
else if ( cElem == '/' )
{
cElem = 63;
}
// end of stream?
else if ( cElem == '=' )
{
break;
}
// RFC 2045 specifies characters not in base64 must be ignored
else
{
nPos++;
continue;
}
switch( nPosBlock )
{
case 0:
cRet[nPosRet] = cElem << 2;
nPosBlock++;
break;
case 1:
// higer bits are zeros
cRet[nPosRet] |= cElem >> 4;
nPosRet++;
cRet[nPosRet] = cElem << 4;
nPosBlock++;
break;
case 2:
// higer bits are zeros
cRet[nPosRet] |= cElem >> 2;
nPosRet++;
cRet[nPosRet] = cElem << 6;
nPosBlock++;
break;
case 3:
cRet[nPosRet] |= cElem;
nPosRet++;
nPosBlock = 0;
break;
}
nPos++;
}
/* this function also adds a zero */
/* hopefully reduce the size of cRet */
cRet = (unsigned char *) hb_xrealloc( cRet, nPosRet + 1 );
hb_retclen_buffer( (char *)cRet, nPosRet );
}
HB_FUNC( TIPENCODERQP_ENCODE )
{
char *cData = hb_parc(1);
int nLen = hb_parclen(1);
char *cRet;
unsigned char cElem;
int nVal, iLineLen = 0;
int nPosRet = 0, nPos = 0;
if ( ! cData )
{
hb_errRT_BASE( EG_ARG, 3012, NULL,
"TIPENCODEQP_ENCODE", 1, hb_paramError(1) );
return;
}
if ( ! nLen )
{
hb_retc( "" );
return;
}
// Preallocating maximum possible length
cRet = (char *) hb_xgrab( nLen * 3 + ( nLen/72 ) *3 + 3 );
// last +3 is trailing \r\n\0
while ( nPos < nLen )
{
cElem = (unsigned char) cData[ nPos ];
// We chose not to encode spaces and tab here.
// cElem is signed and ranges from -126 to +127.
// negative values are automatically encoded
if ( (cElem >=33 && cElem <= 60) || cElem >= 62 ||
cElem == 9 || cElem == 32 )
{
cRet[nPosRet++] = (char) cElem;
iLineLen++;
}
else
{
cRet[nPosRet++] = '=';
nVal = cElem >> 4;
cRet[nPosRet++] = (char) (nVal < 10 ? '0' + nVal : 'A' + nVal - 10);
nVal = cElem & 0x0f;
cRet[nPosRet++] = (char) (nVal < 10 ? '0' + nVal : 'A' + nVal - 10);
iLineLen+=3;
}
nPos++;
if ( iLineLen >= 72 )
{
cRet[nPosRet++] = '=';
cRet[nPosRet++] = '\r';
cRet[nPosRet++] = '\n';
iLineLen = 0;
}
}
/* Securing last line trailing space, if needed */
cElem = (unsigned char) cRet[nPosRet - 1];
if ( cElem == 9 || cElem == 32 )
{
cRet[nPosRet++] = '=';
cRet[nPosRet++] = '\r';
cRet[nPosRet++] = '\n';
}
/* Adding canonical new line for RFC2045 blocks */
/* this function also adds a zero */
cRet = (char *) hb_xrealloc( cRet, nPosRet + 1 );
hb_retclen_buffer( cRet, nPosRet );
}
HB_FUNC( TIPENCODERQP_DECODE )
{
char *cData = hb_parc(1);
int nLen = hb_parclen(1);
char *cRet;
int nPos = 0, nPosRet = 0, nVal;
unsigned char cElem, cCipher;
if ( ! cData )
{
hb_errRT_BASE( EG_ARG, 3012, NULL,
"TIPENCODERQP_DECODE", 1, hb_paramError(1) );
return;
}
if ( ! nLen )
{
hb_retc( "" );
return;
}
// allocate maximum possible lenght.
cRet = (char *) hb_xgrab( nLen + 1 );
while ( nPos < nLen )
{
cElem = (unsigned char) cData[ nPos ];
if ( cElem == '=' )
{
if ( nPos < nLen - 2 )
{
cCipher = (unsigned char) cData[ ++nPos ];
//soft line break
if ( cCipher == '\r' )
{
nPos += 2;
continue;
}
else {
nVal = cCipher >= 'A' && cCipher <= 'F' ? cCipher - 'A' + 10 :
cCipher - '0';
nVal *= 16;
cCipher = (unsigned char) cData[ ++nPos ];
nVal += cCipher >= 'A' && cCipher <= 'F' ? cCipher - 'A' + 10 :
cCipher - '0';
cRet[ nPosRet++ ] = (char) nVal;
}
}
// else the encoding is malformed
else
{
if (nPosRet > 0 )
{
break;
}
}
}
else
{
cRet[ nPosRet++ ] = (char) cElem;
}
nPos ++;
}
/* this function also adds a zero */
/* hopefully reduce the size of cRet */
cRet = (char *) hb_xrealloc( cRet, nPosRet + 1 );
hb_retclen_buffer( cRet, nPosRet );
}
HB_FUNC( TIPENCODERURL_ENCODE )
{
char *cData = hb_parc(1);
int nLen = hb_parclen(1);
BOOL bComplete = hb_parl(2);
char *cRet;
int nPos = 0, nPosRet = 0, nVal;
char cElem;
if ( hb_pcount() < 2 )
{
bComplete = TRUE;
}
if ( ! cData )
{
hb_errRT_BASE( EG_ARG, 3012, NULL,
"TIPENCODERBASE64_ENCODE", 1, hb_paramError(1) );
return;
}
if ( ! nLen )
{
hb_retc( "" );
return;
}
// Giving maximum final length possible
cRet = (char *) hb_xgrab( nLen * 3 +1);
while ( nPos < nLen )
{
cElem = cData[ nPos ];
if ( cElem == ' ' )
{
cRet[ nPosRet ] = '+';
}
else if (
(cElem >= 'A' && cElem <= 'Z') ||
(cElem >= 'a' && cElem <= 'z') ||
(cElem >= '0' && cElem <= '9') ||
cElem == '.' || cElem == ',' || cElem == '&' ||
cElem == '/' || cElem == ';' || cElem =='_' )
{
cRet[ nPosRet ] = cElem;
}
else if ( ! bComplete && ( cElem == ':' || cElem == '?' || cElem == '=' ) )
{
cRet[ nPosRet ] = cElem;
}
else // encode!
{
cRet[ nPosRet++] = '%';
nVal = ((unsigned char) cElem) >> 4;
cRet[ nPosRet++] = nVal < 10 ? '0' + nVal : 'A' + nVal - 10;
nVal = ((unsigned char) cElem) & 0x0f;
cRet[ nPosRet ] = nVal < 10 ? '0' + nVal : 'A' + nVal - 10;
}
nPosRet++;
nPos++;
}
hb_retclen_buffer( cRet, nPosRet );
}
HB_FUNC( TIPENCODERURL_DECODE )
{
char *cData = hb_parc(1);
int nLen = hb_parclen(1);
char *cRet;
int nPos = 0, nPosRet = 0;
char cElem;
if ( ! cData )
{
hb_errRT_BASE( EG_ARG, 3012, NULL,
"TIPENCODERBASE64_DECODE", 1, hb_paramError(1) );
return;
}
if ( ! nLen )
{
hb_retc( "" );
return;
}
// maximum possible lenght
cRet = (char *) hb_xgrab( nLen );
while ( nPos < nLen )
{
cElem = cData[ nPos ];
if ( cElem == '+' )
{
cRet[ nPosRet ] = ' ';
}
else if ( cElem == '%' )
{
if ( nPos < nLen - 2 )
{
cElem = cData[ ++nPos ];
cRet[ nPosRet ] = cElem < 'A' ? cElem - '0' : cElem - 'A' + 10;
cRet[ nPosRet ] *= 16;
cElem = cData[ ++nPos ];
cRet[ nPosRet ] |= cElem < 'A' ? cElem - '0' : cElem - 'A' + 10;
}
else
{
if (nPosRet > 0 )
{
break;
}
}
}
else
{
cRet[ nPosRet ] = cElem;
}
nPos++;
nPosRet++;
}
/* this function also adds a zero */
/* hopefully reduce the size of cRet */
cRet = (char *) hb_xrealloc( cRet, nPosRet + 1 );
hb_retclen_buffer( cRet, nPosRet );
}