mirror of
https://github.com/bjc/prosody.git
synced 2025-04-04 13:47:41 +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_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
|
||||||
|
|
|
@ -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 $@ $<
|
||||||
|
|
||||||
|
|
|
@ -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");
|
Loading…
Add table
Add a link
Reference in a new issue