util.encodings: Support for ICU for IDNA operations.

This commit is contained in:
Tobias Markmann 2010-12-14 15:03:37 +01:00
parent d1d5bb6d1c
commit 93b0dcf56a
3 changed files with 168 additions and 6 deletions

26
configure vendored
View file

@ -11,8 +11,10 @@ LUA_BINDIR="/usr/bin"
LUA_INCDIR="/usr/include" LUA_INCDIR="/usr/include"
LUA_LIBDIR="/usr/lib" LUA_LIBDIR="/usr/lib"
IDN_LIB=idn IDN_LIB=idn
ICU_FLAGS="-licui18n -licudata -licuuc"
OPENSSL_LIB=crypto OPENSSL_LIB=crypto
CC=gcc CC=gcc
CXX=g++
LD=gcc LD=gcc
CFLAGS="-fPIC -Wall" CFLAGS="-fPIC -Wall"
@ -43,6 +45,9 @@ Configure Prosody prior to building.
Default is \$LUA_DIR/lib Default is \$LUA_DIR/lib
--with-idn=LIB The name of the IDN library to link with. --with-idn=LIB The name of the IDN library to link with.
Default is $IDN_LIB Default is $IDN_LIB
--idn-library=(idn|icu) Select library to use for IDNA functionality.
idn: use GNU libidn (default)
icu: use ICU from IBM
--with-ssl=LIB The name of the SSL to link with. --with-ssl=LIB The name of the SSL to link with.
Default is $OPENSSL_LIB Default is $OPENSSL_LIB
--cflags=FLAGS Flags to pass to the compiler --cflags=FLAGS Flags to pass to the compiler
@ -96,7 +101,6 @@ do
LUA_INCDIR_SET=yes LUA_INCDIR_SET=yes
LUA_LIBDIR=/usr/local/lib LUA_LIBDIR=/usr/local/lib
LUA_LIBDIR_SET=yes LUA_LIBDIR_SET=yes
CFLAGS="-Wall"
LDFLAGS="-bundle -undefined dynamic_lookup" LDFLAGS="-bundle -undefined dynamic_lookup"
fi fi
if [ "$OSTYPE" = "linux" ] if [ "$OSTYPE" = "linux" ]
@ -143,6 +147,9 @@ do
--with-idn=*) --with-idn=*)
IDN_LIB="$value" IDN_LIB="$value"
;; ;;
--idn-library=*)
IDN_LIBRARY="$value"
;;
--with-ssl=*) --with-ssl=*)
OPENSSL_LIB="$value" OPENSSL_LIB="$value"
;; ;;
@ -261,6 +268,21 @@ then
LUA_BINDIR="$LUA_DIR/bin" LUA_BINDIR="$LUA_DIR/bin"
fi fi
if [ "$IDN_LIBRARY" = "icu" ]
then
IDNA_LIBS="$ICU_FLAGS"
CFLAGS="$CFLAGS -DUSE_STRINGPREP_ICU"
else
if [ "$IDN_LIBRARY" = "idn" ]
then
IDNA_LIBS="-l$IDN_LIB"
else
echo "Must use idn or icu as argument for --idn-library!"
exit 1
fi
fi
LDFLAGS="$LDFLAGS -llua"
echo -n "Checking Lua includes... " echo -n "Checking Lua includes... "
lua_h="$LUA_INCDIR/lua.h" lua_h="$LUA_INCDIR/lua.h"
if [ -e "$lua_h" ] if [ -e "$lua_h" ]
@ -311,10 +333,12 @@ LUA_LIBDIR=$LUA_LIBDIR
LUA_BINDIR=$LUA_BINDIR LUA_BINDIR=$LUA_BINDIR
REQUIRE_CONFIG=$REQUIRE_CONFIG REQUIRE_CONFIG=$REQUIRE_CONFIG
IDN_LIB=$IDN_LIB IDN_LIB=$IDN_LIB
IDNA_LIBS=$IDNA_LIBS
OPENSSL_LIB=$OPENSSL_LIB OPENSSL_LIB=$OPENSSL_LIB
CFLAGS=$CFLAGS CFLAGS=$CFLAGS
LDFLAGS=$LDFLAGS LDFLAGS=$LDFLAGS
CC=$CC CC=$CC
CXX=$CXX
LD=$LD LD=$LD
EOF EOF

View file

@ -7,10 +7,18 @@ LUA_LIB?=lua$(LUA_SUFFIX)
IDN_LIB?=idn IDN_LIB?=idn
OPENSSL_LIB?=crypto OPENSSL_LIB?=crypto
CC?=gcc CC?=gcc
CXX?=g++
LD?=gcc LD?=gcc
.SUFFIXES: .c .o .so .SUFFIXES: .c .o .so
encodings.o:
$(CXX) $(CFLAGS) -I$(LUA_INCDIR) -c -o encodings.o encodings.cpp
encodings.so: encodings.o
MACOSX_DEPLOYMENT_TARGET="10.3"; export MACOSX_DEPLOYMENT_TARGET;
$(CXX) $(LDFLAGS) $(IDNA_LIBS) -o encodings.so encodings.o -lcrypto
.c.o: .c.o:
$(CC) $(CFLAGS) -I$(LUA_INCDIR) -c -o $@ $< $(CC) $(CFLAGS) -I$(LUA_INCDIR) -c -o $@ $<

View file

@ -15,14 +15,16 @@
// Newer MSVC compilers deprecate strcpy as unsafe, but we use it in a safe way // Newer MSVC compilers deprecate strcpy as unsafe, but we use it in a safe way
#define _CRT_SECURE_NO_DEPRECATE #define _CRT_SECURE_NO_DEPRECATE
extern "C" {
#include <string.h> #include <string.h>
#include <stdlib.h> #include <stdlib.h>
#include "lua.h" #include "lua.h"
#include "lauxlib.h" #include "lauxlib.h"
}
/***************** BASE64 *****************/ /***************** BASE64 *****************/
#define LUALIB_API extern "C"
static const char code[]= static const char code[]=
"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"; "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
@ -118,7 +120,8 @@ static const luaL_Reg Reg_base64[] =
}; };
/***************** STRINGPREP *****************/ /***************** STRINGPREP *****************/
#ifndef USE_STRINGPREP_ICU
/****************** libidn ********************/
#include <stringprep.h> #include <stringprep.h>
static int stringprep_prep(lua_State *L, const Stringprep_profile *profile) static int stringprep_prep(lua_State *L, const Stringprep_profile *profile)
@ -164,11 +167,83 @@ static const luaL_Reg Reg_stringprep[] =
{ NULL, NULL } { NULL, NULL }
}; };
/***************** IDNA *****************/ #else
#include <unicode/usprep.h>
//#include <unicode/stringpiece.h>
#include <unicode/unistr.h>
#include <unicode/utrace.h>
static int icu_stringprep_prep(lua_State *L, const UStringPrepProfile *profile)
{
size_t len;
const char *s;
UnicodeString ustr;
UErrorCode err = U_ZERO_ERROR;
UChar dest[1024];
char output[1024];
if(!lua_isstring(L, 1)) {
lua_pushnil(L);
return 1;
}
s = lua_tolstring(L, 1, &len);
if (len >= 1024) {
lua_pushnil(L);
return 1; // TODO return error message
}
ustr = UnicodeString::fromUTF8(s);
len = usprep_prepare(profile, ustr.getBuffer(), ustr.length(), dest, 1024, 0, NULL, &err);
if (U_FAILURE(err)) {
lua_pushnil(L);
return 1;
} else {
CheckedArrayByteSink output_sink(output, 1024);
UnicodeString dest_str(TRUE, dest, len);
dest_str.toUTF8(output_sink);
lua_pushstring(L, output);
return 1;
}
}
UStringPrepProfile *icu_nameprep;
UStringPrepProfile *icu_nodeprep;
UStringPrepProfile *icu_resourceprep;
UStringPrepProfile *icu_saslprep;
/* initialize global ICU stringprep profiles */
void init_icu()
{
UErrorCode err = U_ZERO_ERROR;
utrace_setLevel(UTRACE_VERBOSE);
icu_nameprep = usprep_openByType(USPREP_RFC3491_NAMEPREP, &err);
icu_nodeprep = usprep_openByType(USPREP_RFC3920_NODEPREP, &err);
icu_resourceprep = usprep_openByType(USPREP_RFC3920_RESOURCEPREP, &err);
icu_saslprep = usprep_openByType(USPREP_RFC4013_SASLPREP, &err);
if (U_FAILURE(err)) fprintf(stderr, "[c] util.encodings: error: %s\n", u_errorName((UErrorCode)err));
}
#define MAKE_PREP_FUNC(myFunc, prep) \
static int myFunc(lua_State *L) { return icu_stringprep_prep(L, prep); }
MAKE_PREP_FUNC(Lstringprep_nameprep, icu_nameprep) /** stringprep.nameprep(s) */
MAKE_PREP_FUNC(Lstringprep_nodeprep, icu_nodeprep) /** stringprep.nodeprep(s) */
MAKE_PREP_FUNC(Lstringprep_resourceprep, icu_resourceprep) /** stringprep.resourceprep(s) */
MAKE_PREP_FUNC(Lstringprep_saslprep, icu_saslprep) /** stringprep.saslprep(s) */
static const luaL_Reg Reg_stringprep[] =
{
{ "nameprep", Lstringprep_nameprep },
{ "nodeprep", Lstringprep_nodeprep },
{ "resourceprep", Lstringprep_resourceprep },
{ "saslprep", Lstringprep_saslprep },
{ NULL, NULL }
};
#endif
/***************** IDNA *****************/
#ifndef USE_STRINGPREP_ICU
/****************** libidn ********************/
#include <idna.h> #include <idna.h>
#include <idn-free.h> #include <idn-free.h>
static int Lidna_to_ascii(lua_State *L) /** idna.to_ascii(s) */ static int Lidna_to_ascii(lua_State *L) /** idna.to_ascii(s) */
{ {
size_t len; size_t len;
@ -202,6 +277,58 @@ static int Lidna_to_unicode(lua_State *L) /** idna.to_unicode(s) */
return 1; // TODO return error message return 1; // TODO return error message
} }
} }
#else
#include <unicode/ustdio.h>
#include <unicode/uidna.h>
/* IDNA2003 or IDNA2008 ? ? ? */
static int Lidna_to_ascii(lua_State *L) /** idna.to_ascii(s) */
{
size_t len;
int32_t out_len;
const char *s = luaL_checklstring(L, 1, &len);
UnicodeString ustr;
UErrorCode err = U_ZERO_ERROR;
UChar dest[1024];
char output[1024];
ustr = UnicodeString::fromUTF8(s);
out_len = uidna_IDNToASCII(ustr.getBuffer(), ustr.length(), dest, 1024, UIDNA_USE_STD3_RULES, NULL, &err);
if (U_FAILURE(err)) {
lua_pushnil(L);
return 1;
} else {
CheckedArrayByteSink output_sink(output, 1024);
UnicodeString dest_str(TRUE, dest, out_len);
dest_str.toUTF8(output_sink);
lua_pushstring(L, output);
return 1;
}
}
static int Lidna_to_unicode(lua_State *L) /** idna.to_unicode(s) */
{
size_t len;
int32_t out_len;
const char *s = luaL_checklstring(L, 1, &len);
UnicodeString ustr;
UErrorCode err = U_ZERO_ERROR;
UChar dest[1024];
char output[1024];
ustr = UnicodeString::fromUTF8(s);
out_len = uidna_IDNToUnicode(ustr.getBuffer(), ustr.length(), dest, 1024, UIDNA_USE_STD3_RULES, NULL, &err);
if (U_FAILURE(err)) {
lua_pushnil(L);
return 1;
} else {
CheckedArrayByteSink output_sink(output, 1024);
UnicodeString dest_str(TRUE, dest, out_len);
dest_str.toUTF8(output_sink);
lua_pushstring(L, output);
return 1;
}
}
#endif
static const luaL_Reg Reg_idna[] = static const luaL_Reg Reg_idna[] =
{ {
@ -219,6 +346,9 @@ static const luaL_Reg Reg[] =
LUALIB_API int luaopen_util_encodings(lua_State *L) LUALIB_API int luaopen_util_encodings(lua_State *L)
{ {
#ifdef USE_STRINGPREP_ICU
init_icu();
#endif
luaL_register(L, "encodings", Reg); luaL_register(L, "encodings", Reg);
lua_pushliteral(L, "base64"); lua_pushliteral(L, "base64");