mirror of
https://github.com/bjc/prosody.git
synced 2025-04-03 21:27:38 +03:00
util.encodings: Support for ICU for IDNA operations.
This commit is contained in:
parent
d1d5bb6d1c
commit
93b0dcf56a
3 changed files with 168 additions and 6 deletions
26
configure
vendored
26
configure
vendored
|
@ -11,8 +11,10 @@ LUA_BINDIR="/usr/bin"
|
|||
LUA_INCDIR="/usr/include"
|
||||
LUA_LIBDIR="/usr/lib"
|
||||
IDN_LIB=idn
|
||||
ICU_FLAGS="-licui18n -licudata -licuuc"
|
||||
OPENSSL_LIB=crypto
|
||||
CC=gcc
|
||||
CXX=g++
|
||||
LD=gcc
|
||||
|
||||
CFLAGS="-fPIC -Wall"
|
||||
|
@ -43,6 +45,9 @@ Configure Prosody prior to building.
|
|||
Default is \$LUA_DIR/lib
|
||||
--with-idn=LIB The name of the IDN library to link with.
|
||||
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.
|
||||
Default is $OPENSSL_LIB
|
||||
--cflags=FLAGS Flags to pass to the compiler
|
||||
|
@ -96,7 +101,6 @@ do
|
|||
LUA_INCDIR_SET=yes
|
||||
LUA_LIBDIR=/usr/local/lib
|
||||
LUA_LIBDIR_SET=yes
|
||||
CFLAGS="-Wall"
|
||||
LDFLAGS="-bundle -undefined dynamic_lookup"
|
||||
fi
|
||||
if [ "$OSTYPE" = "linux" ]
|
||||
|
@ -143,6 +147,9 @@ do
|
|||
--with-idn=*)
|
||||
IDN_LIB="$value"
|
||||
;;
|
||||
--idn-library=*)
|
||||
IDN_LIBRARY="$value"
|
||||
;;
|
||||
--with-ssl=*)
|
||||
OPENSSL_LIB="$value"
|
||||
;;
|
||||
|
@ -261,6 +268,21 @@ then
|
|||
LUA_BINDIR="$LUA_DIR/bin"
|
||||
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... "
|
||||
lua_h="$LUA_INCDIR/lua.h"
|
||||
if [ -e "$lua_h" ]
|
||||
|
@ -311,10 +333,12 @@ LUA_LIBDIR=$LUA_LIBDIR
|
|||
LUA_BINDIR=$LUA_BINDIR
|
||||
REQUIRE_CONFIG=$REQUIRE_CONFIG
|
||||
IDN_LIB=$IDN_LIB
|
||||
IDNA_LIBS=$IDNA_LIBS
|
||||
OPENSSL_LIB=$OPENSSL_LIB
|
||||
CFLAGS=$CFLAGS
|
||||
LDFLAGS=$LDFLAGS
|
||||
CC=$CC
|
||||
CXX=$CXX
|
||||
LD=$LD
|
||||
|
||||
EOF
|
||||
|
|
|
@ -7,10 +7,18 @@ LUA_LIB?=lua$(LUA_SUFFIX)
|
|||
IDN_LIB?=idn
|
||||
OPENSSL_LIB?=crypto
|
||||
CC?=gcc
|
||||
CXX?=g++
|
||||
LD?=gcc
|
||||
|
||||
.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:
|
||||
$(CC) $(CFLAGS) -I$(LUA_INCDIR) -c -o $@ $<
|
||||
|
||||
|
|
|
@ -15,14 +15,16 @@
|
|||
// Newer MSVC compilers deprecate strcpy as unsafe, but we use it in a safe way
|
||||
#define _CRT_SECURE_NO_DEPRECATE
|
||||
|
||||
extern "C" {
|
||||
#include <string.h>
|
||||
#include <stdlib.h>
|
||||
|
||||
#include "lua.h"
|
||||
#include "lauxlib.h"
|
||||
|
||||
}
|
||||
/***************** BASE64 *****************/
|
||||
|
||||
#define LUALIB_API extern "C"
|
||||
|
||||
static const char code[]=
|
||||
"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
|
||||
|
||||
|
@ -118,7 +120,8 @@ static const luaL_Reg Reg_base64[] =
|
|||
};
|
||||
|
||||
/***************** STRINGPREP *****************/
|
||||
|
||||
#ifndef USE_STRINGPREP_ICU
|
||||
/****************** libidn ********************/
|
||||
#include <stringprep.h>
|
||||
|
||||
static int stringprep_prep(lua_State *L, const Stringprep_profile *profile)
|
||||
|
@ -164,11 +167,83 @@ static const luaL_Reg Reg_stringprep[] =
|
|||
{ 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 <idn-free.h>
|
||||
|
||||
static int Lidna_to_ascii(lua_State *L) /** idna.to_ascii(s) */
|
||||
{
|
||||
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
|
||||
}
|
||||
}
|
||||
#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[] =
|
||||
{
|
||||
|
@ -219,6 +346,9 @@ static const luaL_Reg Reg[] =
|
|||
|
||||
LUALIB_API int luaopen_util_encodings(lua_State *L)
|
||||
{
|
||||
#ifdef USE_STRINGPREP_ICU
|
||||
init_icu();
|
||||
#endif
|
||||
luaL_register(L, "encodings", Reg);
|
||||
|
||||
lua_pushliteral(L, "base64");
|
Loading…
Add table
Add a link
Reference in a new issue