21625f40b5
I tried to get the latest versions from dockapps.org, but I haven't tested any of them. More dockapps will be added as time permits.
270 lines
5.3 KiB
C
270 lines
5.3 KiB
C
/* $Id: charutil.c,v 1.19 2004/04/28 00:19:03 bluehal Exp $ */
|
|
#ifdef HAVE_CONFIG_H
|
|
#include <config.h>
|
|
#endif
|
|
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <ctype.h>
|
|
#include <sys/types.h>
|
|
#include <regex.h>
|
|
#ifdef USE_DMALLOC
|
|
#include <dmalloc.h>
|
|
#endif
|
|
#include "charutil.h"
|
|
|
|
static __inline__ void LeftTrim(char *psValue)
|
|
{
|
|
char *psTmp = psValue;
|
|
|
|
while (*psTmp == ' ' || *psTmp == '\t')
|
|
psTmp++;
|
|
|
|
/* can't use strcpy here, as the strings must not
|
|
overlap, at least according to spec. */
|
|
if (psTmp > psValue) {
|
|
while (*psTmp != '\0') {
|
|
*(psValue++) = *(psTmp++);
|
|
}
|
|
*(psValue) = '\0';
|
|
}
|
|
}
|
|
|
|
static __inline__ void RightTrim(char *psValue)
|
|
{
|
|
long lLength = strlen(psValue) - 1;
|
|
|
|
while ((psValue[lLength] == ' ' || psValue[lLength] == '\t')
|
|
&& *psValue) {
|
|
lLength--;
|
|
}
|
|
|
|
psValue[++lLength] = '\000';
|
|
}
|
|
|
|
void FullTrim(char *psValue)
|
|
{
|
|
RightTrim(psValue);
|
|
LeftTrim(psValue);
|
|
}
|
|
|
|
void Bin2Hex(unsigned char *src, int length, char *dst)
|
|
{
|
|
static char hex_tbl[] = "0123456789abcdef";
|
|
|
|
int i = 0;
|
|
char *ptr = dst;
|
|
|
|
if (src && ptr) {
|
|
for (i = 0; i < length; i++) {
|
|
*ptr++ = hex_tbl[*src >> 4];
|
|
*ptr++ = hex_tbl[*src++ & 0xf];
|
|
}
|
|
*ptr = 0;
|
|
}
|
|
}
|
|
|
|
|
|
#define PAD '='
|
|
char ALPHABET[65] =
|
|
"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/\000";
|
|
|
|
/* find char in in alphabet, return offset, -1 otherwise */
|
|
int find_char(char c)
|
|
{
|
|
char *a = ALPHABET;
|
|
|
|
while (*a)
|
|
if (*(a++) == c)
|
|
return a - ALPHABET - 1;
|
|
|
|
return -1;
|
|
}
|
|
|
|
void Encode_Base64(char *src, char *dst)
|
|
{
|
|
int g = 0;
|
|
int c = 0;
|
|
|
|
if (!src || !dst)
|
|
return;
|
|
|
|
while (*src) {
|
|
g = (g << 8) | *src++;
|
|
if (c == 2) {
|
|
*dst++ = ALPHABET[0x3f & (g >> 18)];
|
|
*dst++ = ALPHABET[0x3f & (g >> 12)];
|
|
*dst++ = ALPHABET[0x3f & (g >> 6)];
|
|
*dst++ = ALPHABET[0x3f & g];
|
|
}
|
|
c = (c + 1) % 3;
|
|
}
|
|
|
|
if (c) {
|
|
if (c == 1) {
|
|
*dst++ = ALPHABET[0x3f & (g >> 2)];
|
|
*dst++ = ALPHABET[0x3f & (g << 4)];
|
|
*dst++ = PAD;
|
|
*dst++ = PAD;
|
|
} else {
|
|
*dst++ = ALPHABET[0x3f & (g >> 10)];
|
|
*dst++ = ALPHABET[0x3f & (g >> 4)];
|
|
*dst++ = ALPHABET[0x3f & (g << 2)];
|
|
*dst++ = PAD;
|
|
}
|
|
}
|
|
*dst = 0;
|
|
}
|
|
|
|
|
|
void Decode_Base64(char *src, char *dst)
|
|
{
|
|
int g = 0;
|
|
int c = 0;
|
|
int n = 0;
|
|
|
|
if (!src || !dst)
|
|
return;
|
|
|
|
while (*src) {
|
|
n = find_char(*src++);
|
|
if (n < 0)
|
|
continue;
|
|
|
|
g <<= 6;
|
|
g |= n;
|
|
if (c == 3) {
|
|
*dst++ = g >> 16;
|
|
*dst++ = g >> 8;
|
|
*dst++ = g;
|
|
g = 0;
|
|
}
|
|
c = (c + 1) % 4;
|
|
}
|
|
if (c) {
|
|
if (c == 1) {
|
|
/* shouldn't happen, but do something anyway */
|
|
*dst++ = g << 2;
|
|
} else if (c == 2) {
|
|
*dst++ = g >> 4;
|
|
} else {
|
|
*dst++ = g >> 10;
|
|
*dst++ = g >> 2;
|
|
}
|
|
}
|
|
*dst = 0;
|
|
}
|
|
|
|
/* helper function for the configuration line parser */
|
|
void
|
|
copy_substring(char *destination,
|
|
int startidx, int endidx, const char *source)
|
|
{
|
|
if (startidx > -1) {
|
|
strncpy(destination, source + startidx, endidx - startidx);
|
|
destination[endidx - startidx] = '\0';
|
|
}
|
|
}
|
|
|
|
/* common to Pop3 and Imap4 authentication list grabber. */
|
|
void grab_authList(const char *source, char *destination)
|
|
{
|
|
int i;
|
|
/* regex isn't all that helpful for lists of things. */
|
|
/* but does leave the end of the matched region in regs.end[0] */
|
|
/* what remains is a list of legal authentication schemes. */
|
|
if (isalnum(source[0])) {
|
|
/* copy, while turning caps into lower case */
|
|
for (i = 0; i < 99 && source[i] != '\0'; i++) {
|
|
destination[i] = tolower(source[i]);
|
|
}
|
|
destination[i] = '\0';
|
|
} else {
|
|
destination[0] = '\0';
|
|
}
|
|
}
|
|
|
|
#ifdef USE_GNU_REGEX
|
|
int compile_and_match_regex(const char *regex, const char *str, /*@out@ */
|
|
struct re_registers *regs)
|
|
{
|
|
|
|
const char *errstr;
|
|
int matchedchars;
|
|
struct re_pattern_buffer rpbuf;
|
|
|
|
/* compile the regex pattern */
|
|
memset(&rpbuf, 0, sizeof(struct re_pattern_buffer));
|
|
/* posix egrep interprets intervals (eg. {1,32}) nicely */
|
|
re_syntax_options = RE_SYNTAX_POSIX_EGREP;
|
|
errstr = re_compile_pattern(regex, strlen(regex), &rpbuf);
|
|
if (errstr != NULL) {
|
|
fprintf(stderr, "error in compiling regular expression: %s\n",
|
|
errstr);
|
|
return -1;
|
|
}
|
|
|
|
/* match the regex */
|
|
regs->num_regs = REGS_UNALLOCATED;
|
|
matchedchars = re_match(&rpbuf, str, strlen(str), 0, regs);
|
|
/* this can fail (return -1 or 0) without being an error,
|
|
if we're trying to apply a regex just to see if it
|
|
matched. */
|
|
|
|
#ifdef undef
|
|
printf("--\n");
|
|
for (i = 1; i < 6; i++) {
|
|
printf("%d %d, (%.*s)\n", regs.start[i], regs.end[i],
|
|
(regs.end[i] - regs.start[i]),
|
|
(regs.start[i] >= 0) ? &str[regs.start[i]] : "");
|
|
}
|
|
#endif
|
|
|
|
regfree(&rpbuf); // added 3 jul 02, appeasing valgrind
|
|
return matchedchars;
|
|
}
|
|
#endif
|
|
|
|
/* like perl chomp(); useful for dealing with input from popen */
|
|
void chomp(char *s)
|
|
{
|
|
int l = strlen(s) - 1;
|
|
if (l >= 0 && s[l] == '\n')
|
|
s[l] = '\0';
|
|
}
|
|
|
|
char *strdup_ordie(const char *c)
|
|
{
|
|
char *ret = strdup(c);
|
|
if (ret == NULL) {
|
|
fprintf(stderr, "ran out of memory\n");
|
|
exit(EXIT_FAILURE);
|
|
}
|
|
return (ret);
|
|
}
|
|
|
|
void StripComment(char *buf)
|
|
{
|
|
char *p;
|
|
|
|
/* Strip comments at beginning of line, or after whitespace.
|
|
a kludgy way of avoiding problems with #'s in passwords. */
|
|
if (buf[0] == '#')
|
|
buf[0] = '\0';
|
|
for (p = (char *) buf + 1; *p && !(*p == '#' && isspace(*(p - 1)));
|
|
p++);
|
|
if (*p) {
|
|
*p = '\0';
|
|
}
|
|
}
|
|
|
|
|
|
/* vim:set ts=4: */
|
|
/*
|
|
* Local Variables:
|
|
* tab-width: 4
|
|
* c-indent-level: 4
|
|
* c-basic-offset: 4
|
|
* End:
|
|
*/
|