blob: 672c1b8b34b2485e9e5e2fb25ddaf3cfed62e96b [file] [log] [blame]
/*
* rivetWWW.c - Rivet commands designed for use with the world wide web.
*/
/* Copyright 2002-2004 The Apache Software Foundation
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
/* $Id$ */
/* Rivet config */
#ifdef HAVE_CONFIG_H
#include <rivet_config.h>
#endif
#include <tcl.h>
#include <ctype.h>
/* Function prototypes are defined with EXTERN. Since we are in the same DLL,
* no need to keep this extern... */
#ifdef EXTERN
# undef EXTERN
# define EXTERN
#endif /* EXTERN */
#include "rivet.h"
#ifdef WIN32
#define snprintf _snprintf
#endif
/*
*-----------------------------------------------------------------------------
*
* Rivet_HexToDigit --
* Helper function to convert a hex character into the equivalent integer.
*
* Results:
* The integer, or -1 if an illegal hex character is encountered.
*
*-----------------------------------------------------------------------------
*/
static int
Rivet_HexToDigit(int c) {
if (c >= 'a' && c <= 'f') {
return (c - 'a' + 10);
}
if (c >= 'A' && c <= 'F') {
return (c - 'A' + 10);
}
if (c >= '0' && c <= '9') {
return (c - '0');
}
return (-1);
}
/*
*-----------------------------------------------------------------------------
*
* Rivet_UnescapeStringCmd --
* Implements the TCL unescape_string command:
* unescape_string string
*
* Results:
* Standard TCL results.
*
*-----------------------------------------------------------------------------
*/
TCL_CMD_HEADER( Rivet_UnescapeStringCmd )
{
char *origString, *newString, *origStringP, *newStringP;
int origLength;
int digit1, digit2;
if ( objc != 2 ) {
Tcl_WrongNumArgs( interp, 1, objv, "string" );
return TCL_ERROR;
}
origString = Tcl_GetStringFromObj( objv[1], &origLength );
newString = Tcl_Alloc( (unsigned)origLength + 1);
/* for all the characters in the source string */
for (origStringP = origString, newStringP = newString;
*origStringP != '\0';
origStringP++) {
char c = *origStringP;
char c2;
/* map plus to space */
if (c == '+') {
*newStringP++ = ' ';
continue;
}
/* if there's a percent sign, decode the two-character
* hex sequence that follows and copy it to the target
* string */
if (c == '%') {
digit1 = Rivet_HexToDigit(c = *++origStringP);
digit2 = Rivet_HexToDigit(c2 = *++origStringP);
if (digit1 == -1 || digit2 == -1) {
char buf[3];
snprintf (buf,3,"%c%c",c,c2);
Tcl_AppendResult( interp,
Tcl_GetStringFromObj( objv[0], NULL ),
": bad char in hex sequence %", buf, (char *)NULL );
return TCL_ERROR;
}
*newStringP++ = (digit1 * 16 + digit2);
continue;
}
/* it wasn't a plus or percent, just copy the char across */
*newStringP++ = c;
}
/* Don't forget to null-terminate the target string */
*newStringP = '\0';
Tcl_SetObjResult( interp, Tcl_NewStringObj( newString, -1 ) );
Tcl_Free(newString);
return TCL_OK;
}
/*
*-----------------------------------------------------------------------------
*
* Rivet_DigitToHex
* Helper function to convert a number 0 - 15 into the equivalent hex
* character.
*
* Results:
* The integer, or -1 if an illegal hex character is encountered.
*
*-----------------------------------------------------------------------------
*/
static int
Rivet_DigitToHex(int c) {
if (c < 10) {
return c + '0';
}
return c - 10 + 'a';
}
/*
*-----------------------------------------------------------------------------
*
* Rivet_EscapeStringCmd --
* Implements the TCL escape_string command:
* escape_string string
*
* Results:
* Standard TCL results.
*
*-----------------------------------------------------------------------------
*/
TCL_CMD_HEADER( Rivet_EscapeStringCmd )
{
char *origString, *newString, *origStringP, *newStringP;
int origLength;
if ( objc != 2 ) {
Tcl_WrongNumArgs( interp, 1, objv, "string" );
return TCL_ERROR;
}
origString = Tcl_GetStringFromObj( objv[1], &origLength );
/* If they sent us an empty string, we're done */
if (origLength == 0) return TCL_OK;
newString = (char *)Tcl_Alloc( (unsigned)origLength * 3 + 1 );
/* for all the characters in the source string */
for (origStringP = origString, newStringP = newString;
*origStringP != '\0';
origStringP++) {
char c = *origStringP;
if (isalnum ((int)c)) {
*newStringP++ = c;
} else if (c == ' ') {
*newStringP++ = '+';
} else {
*newStringP++ = '%';
*newStringP++ = Rivet_DigitToHex((c >> 4) & 0x0f);
*newStringP++ = Rivet_DigitToHex(c & 0x0f);
}
}
/* Don't forget to null-terminate the target string */
*newStringP = '\0';
Tcl_SetObjResult( interp, Tcl_NewStringObj( newString, -1 ) );
Tcl_Free(newString);
return TCL_OK;
}
/*
*-----------------------------------------------------------------------------
*
* Rivet_EscapeSgmlCharsCmd --
* Implements the TCL escape_sgml_chars command:
* escape_sgml_chars string
*
* Results:
* Standard TCL results.
*
*-----------------------------------------------------------------------------
*/
TCL_CMD_HEADER( Rivet_EscapeSgmlCharsCmd )
{
char *origString, *newString, *origStringP, *newStringP;
int origLength;
if( objc != 2 ) {
Tcl_WrongNumArgs( interp, 1, objv, "string" );
return TCL_ERROR;
}
origString = Tcl_GetStringFromObj( objv[1], &origLength );
/* If they sent us an empty string, we're done */
if (origLength == 0) return TCL_OK;
newString = (char *)Tcl_Alloc( (unsigned)origLength * 6 + 1 );
/* for all the characters in the source string */
for (origStringP = origString, newStringP = newString;
*origStringP != '\0';
origStringP++) {
char c = *origStringP;
switch(c) {
case '&':
*newStringP++ = '&';
*newStringP++ = 'a';
*newStringP++ = 'm';
*newStringP++ = 'p';
*newStringP++ = ';';
break;
case '<':
*newStringP++ = '&';
*newStringP++ = 'l';
*newStringP++ = 't';
*newStringP++ = ';';
break;
case '>':
*newStringP++ = '&';
*newStringP++ = 'g';
*newStringP++ = 't';
*newStringP++ = ';';
break;
case '\'':
*newStringP++ = '&';
*newStringP++ = '#';
*newStringP++ = '3';
*newStringP++ = '9';
*newStringP++ = ';';
break;
case '"':
*newStringP++ = '&';
*newStringP++ = 'q';
*newStringP++ = 'u';
*newStringP++ = 'o';
*newStringP++ = 't';
*newStringP++ = ';';
break;
default:
*newStringP++ = c;
break;
}
}
/* Don't forget to null-terminate the target string */
*newStringP = '\0';
Tcl_SetObjResult( interp, Tcl_NewStringObj( newString, -1 ) );
Tcl_Free(newString);
return TCL_OK;
}
/*
*-----------------------------------------------------------------------------
*
* Rivet_EscapeShellCommandCmd --
* Implements the TCL www_escape_shell_command command:
* www_escape_shell_command string
*
* Results:
* Standard TCL results.
*
*-----------------------------------------------------------------------------
*/
TCL_CMD_HEADER( Rivet_EscapeShellCommandCmd )
{
char *origString, *newString, *origStringP, *newStringP, *checkP;
int origLength;
if( objc != 2) {
Tcl_WrongNumArgs( interp, 1, objv, "string" );
return TCL_ERROR;
}
origString = Tcl_GetStringFromObj( objv[1], &origLength );
newString = Tcl_Alloc( (unsigned)origLength * 2 + 1 );
/* for all the characters in the source string */
for (origStringP = origString, newStringP = newString;
*origStringP != '\0';
origStringP++) {
char c = *origStringP;
/* if the character is a shell metacharacter, quote it */
for (checkP = "&;`'|*?-~<>^()[]{}$\\"; *checkP != '\0'; checkP++) {
if (c == *checkP) {
*newStringP++ = '\\';
break;
}
}
*newStringP++ = c;
}
/* Don't forget to null-terminate the target string */
*newStringP = '\0';
Tcl_SetObjResult( interp, Tcl_NewStringObj( newString, -1 ) );
Tcl_Free(newString);
return TCL_OK;
}
/*
*-----------------------------------------------------------------------------
* Rivet_InitWWW --
*
* Initialize the WWW functions.
*
* These functions have been examined and are believed to be safe for use
* in safe interpreters, as they process strings only.
*-----------------------------------------------------------------------------
*/
#if RIVET_NAMESPACE_EXPORT == 1
extern Tcl_Namespace* Rivet_GetNamespace( Tcl_Interp* interp);
#endif
int
Rivet_InitWWW( Tcl_Interp *interp)
{
RIVET_OBJ_CMD ("unescape_string",Rivet_UnescapeStringCmd,NULL);
RIVET_OBJ_CMD ("escape_string",Rivet_EscapeStringCmd,NULL);
RIVET_OBJ_CMD ("escape_sgml_chars",Rivet_EscapeSgmlCharsCmd,NULL);
RIVET_OBJ_CMD ("escape_shell_command",Rivet_EscapeShellCommandCmd,NULL);
#if RIVET_NAMESPACE_EXPORT == 1
{
Tcl_Namespace* rivet_ns = Rivet_GetNamespace(interp);
RIVET_EXPORT_CMD(interp,rivet_ns,"unescape_string");
RIVET_EXPORT_CMD(interp,rivet_ns,"escape_string");
RIVET_EXPORT_CMD(interp,rivet_ns,"escape_sgml_chars");
RIVET_EXPORT_CMD(interp,rivet_ns,"escape_shell_command");
}
#endif
return TCL_OK;
}