2015-07-14 20:32:22 +00:00
|
|
|
#include <exception>
|
|
|
|
#include <string>
|
|
|
|
|
|
|
|
#include "phonenumbers/phonenumberutil.h"
|
|
|
|
|
|
|
|
extern "C" {
|
|
|
|
#include "postgres.h"
|
|
|
|
#include "libpq/pqformat.h"
|
|
|
|
#include "fmgr.h"
|
|
|
|
}
|
|
|
|
|
2015-07-22 17:44:00 +00:00
|
|
|
#include "error_handling.h"
|
2015-07-17 20:22:30 +00:00
|
|
|
#include "short_phone_number.h"
|
|
|
|
|
2015-07-14 20:32:22 +00:00
|
|
|
using namespace i18n::phonenumbers;
|
|
|
|
|
2015-07-17 20:22:30 +00:00
|
|
|
static const PhoneNumberUtil* const phoneUtil = PhoneNumberUtil::GetInstance();
|
2015-07-17 18:46:26 +00:00
|
|
|
|
2015-07-16 19:56:23 +00:00
|
|
|
/*
|
|
|
|
* Utility functions
|
|
|
|
*/
|
|
|
|
|
2015-07-22 16:57:44 +00:00
|
|
|
static char* textToCString(const text* text) {
|
|
|
|
size_t len = VARSIZE(text) - VARHDRSZ;
|
|
|
|
char* str = (char*)palloc(len + 1);
|
|
|
|
memcpy(str, VARDATA(text), len);
|
|
|
|
str[len] = '\0';
|
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
2015-07-16 19:56:23 +00:00
|
|
|
//Internal function used by phone_number_in and parse_phone_number
|
2015-07-22 16:57:44 +00:00
|
|
|
//TODO: take a std::string to minimize copying?
|
2015-07-21 17:32:30 +00:00
|
|
|
ShortPhoneNumber* parsePhoneNumber(const char* number_str, const char* country) throw() {
|
2015-07-17 18:46:26 +00:00
|
|
|
try {
|
2015-07-21 17:32:30 +00:00
|
|
|
PhoneNumber number;
|
|
|
|
ShortPhoneNumber* short_number;
|
2015-07-17 18:46:26 +00:00
|
|
|
|
2015-07-21 17:32:30 +00:00
|
|
|
short_number = (ShortPhoneNumber*)palloc0(sizeof(ShortPhoneNumber));
|
|
|
|
if(short_number == nullptr) {
|
2015-07-17 18:46:26 +00:00
|
|
|
throw std::bad_alloc();
|
|
|
|
}
|
|
|
|
|
|
|
|
PhoneNumberUtil::ErrorType error;
|
2015-07-21 17:32:30 +00:00
|
|
|
error = phoneUtil->Parse(number_str, country, &number);
|
2015-07-17 18:46:26 +00:00
|
|
|
if(error == PhoneNumberUtil::NO_PARSING_ERROR) {
|
2015-07-21 17:36:40 +00:00
|
|
|
//Initialize short_number using placement new.
|
2015-07-21 17:32:30 +00:00
|
|
|
new(short_number) ShortPhoneNumber(number);
|
|
|
|
return short_number;
|
2015-07-17 18:46:26 +00:00
|
|
|
} else {
|
|
|
|
reportParseError(number_str, error);
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
//TODO: check number validity.
|
2015-07-21 21:41:44 +00:00
|
|
|
} catch(const std::bad_alloc& e) {
|
2015-07-17 18:46:26 +00:00
|
|
|
reportOutOfMemory();
|
2015-07-21 21:41:44 +00:00
|
|
|
//TODO: figure out why we need this.
|
|
|
|
} catch(const PhoneNumberTooLongException& e) {
|
|
|
|
reportGenericError(e);
|
|
|
|
} catch(const std::exception& e) {
|
2015-07-17 18:46:26 +00:00
|
|
|
reportGenericError(e);
|
2015-07-16 19:56:23 +00:00
|
|
|
}
|
|
|
|
|
2015-07-17 18:46:26 +00:00
|
|
|
return nullptr;
|
2015-07-16 19:56:23 +00:00
|
|
|
}
|
|
|
|
|
2015-07-22 17:44:00 +00:00
|
|
|
//TODO: check null args (PG_ARGISNULL) and make non-strict?
|
2015-07-16 19:56:23 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Extension functions
|
|
|
|
*/
|
2015-07-14 20:32:22 +00:00
|
|
|
|
|
|
|
extern "C" {
|
|
|
|
#ifdef PG_MODULE_MAGIC
|
|
|
|
PG_MODULE_MAGIC;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
PGDLLEXPORT PG_FUNCTION_INFO_V1(phone_number_in);
|
|
|
|
|
|
|
|
PGDLLEXPORT
|
|
|
|
Datum
|
|
|
|
phone_number_in(PG_FUNCTION_ARGS)
|
|
|
|
{
|
2015-07-16 19:56:23 +00:00
|
|
|
const char *number_str = PG_GETARG_CSTRING(0);
|
|
|
|
|
2015-07-21 17:32:30 +00:00
|
|
|
ShortPhoneNumber* number = parsePhoneNumber(number_str, "US");
|
2015-07-17 18:46:26 +00:00
|
|
|
if(number) {
|
|
|
|
PG_RETURN_POINTER(number);
|
|
|
|
} else {
|
|
|
|
PG_RETURN_NULL();
|
2015-07-16 19:56:23 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
PGDLLEXPORT PG_FUNCTION_INFO_V1(parse_phone_number);
|
|
|
|
|
|
|
|
PGDLLEXPORT
|
|
|
|
Datum
|
|
|
|
parse_phone_number(PG_FUNCTION_ARGS)
|
|
|
|
{
|
2015-07-22 16:57:44 +00:00
|
|
|
try {
|
|
|
|
const text* number_text = PG_GETARG_TEXT_P(0);
|
|
|
|
const text* country_text = PG_GETARG_TEXT_P(1);
|
|
|
|
|
|
|
|
char* number_str = textToCString(number_text);
|
|
|
|
char* country = textToCString(country_text);
|
|
|
|
|
|
|
|
ShortPhoneNumber* number = parsePhoneNumber(number_str, country);
|
|
|
|
//TODO: prevent leaks.
|
|
|
|
pfree(number_str);
|
|
|
|
pfree(country);
|
|
|
|
if(number) {
|
|
|
|
PG_RETURN_POINTER(number);
|
|
|
|
} else {
|
|
|
|
PG_RETURN_NULL();
|
|
|
|
}
|
|
|
|
} catch(std::bad_alloc e) {
|
|
|
|
reportOutOfMemory();
|
|
|
|
} catch(std::exception& e) {
|
|
|
|
reportGenericError(e);
|
2015-07-14 20:32:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
PGDLLEXPORT PG_FUNCTION_INFO_V1(phone_number_out);
|
|
|
|
|
|
|
|
PGDLLEXPORT
|
|
|
|
Datum
|
|
|
|
phone_number_out(PG_FUNCTION_ARGS)
|
|
|
|
{
|
|
|
|
try {
|
2015-07-21 17:32:30 +00:00
|
|
|
const ShortPhoneNumber* short_number = (ShortPhoneNumber*)PG_GETARG_POINTER(0);
|
|
|
|
PhoneNumber number = *short_number;
|
2015-07-16 18:25:44 +00:00
|
|
|
|
2015-07-21 17:32:30 +00:00
|
|
|
std::string formatted;
|
|
|
|
phoneUtil->Format(number, PhoneNumberUtil::INTERNATIONAL, &formatted);
|
2015-07-16 18:25:44 +00:00
|
|
|
|
|
|
|
//Copy the formatted number to a C-style string.
|
|
|
|
//We must use the PostgreSQL allocator, not new/malloc.
|
|
|
|
size_t len = formatted.length();
|
2015-07-21 17:32:30 +00:00
|
|
|
char* result = (char*)palloc(len + 1);
|
2015-07-16 18:25:44 +00:00
|
|
|
if(result == nullptr) {
|
2015-07-21 17:32:30 +00:00
|
|
|
throw std::bad_alloc();
|
2015-07-16 18:25:44 +00:00
|
|
|
}
|
|
|
|
memcpy(result, formatted.data(), len);
|
|
|
|
result[len] = '\0';
|
|
|
|
|
|
|
|
PG_RETURN_CSTRING(result);
|
2015-07-21 21:41:44 +00:00
|
|
|
} catch(const std::bad_alloc& e) {
|
2015-07-14 20:32:22 +00:00
|
|
|
reportOutOfMemory();
|
2015-07-21 21:41:44 +00:00
|
|
|
} catch (const std::exception& e) {
|
2015-07-14 20:32:22 +00:00
|
|
|
reportGenericError(e);
|
|
|
|
}
|
2015-07-16 19:56:23 +00:00
|
|
|
|
2015-07-16 18:25:44 +00:00
|
|
|
PG_RETURN_NULL();
|
2015-07-14 20:32:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
PGDLLEXPORT PG_FUNCTION_INFO_V1(phone_number_recv);
|
|
|
|
|
|
|
|
PGDLLEXPORT
|
|
|
|
Datum
|
2015-07-22 19:32:52 +00:00
|
|
|
phone_number_recv(PG_FUNCTION_ARGS) {
|
|
|
|
try {
|
|
|
|
StringInfo buf = (StringInfo)PG_GETARG_POINTER(0);
|
2015-07-22 19:41:05 +00:00
|
|
|
ShortPhoneNumber* number;
|
2015-07-22 19:32:52 +00:00
|
|
|
|
2015-07-22 19:41:05 +00:00
|
|
|
number = (ShortPhoneNumber*)palloc(sizeof(ShortPhoneNumber));
|
|
|
|
//TODO: make portable (fix endianness issues, etc.).
|
|
|
|
pq_copymsgbytes(buf, (char*)number, sizeof(ShortPhoneNumber));
|
|
|
|
PG_RETURN_POINTER(number);
|
|
|
|
} catch(const std::bad_alloc& e) {
|
|
|
|
reportOutOfMemory();
|
|
|
|
} catch (const std::exception& e) {
|
|
|
|
reportGenericError(e);
|
|
|
|
}
|
|
|
|
|
|
|
|
PG_RETURN_NULL();
|
2015-07-14 20:32:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
PGDLLEXPORT PG_FUNCTION_INFO_V1(phone_number_send);
|
|
|
|
|
|
|
|
PGDLLEXPORT
|
|
|
|
Datum
|
2015-07-22 19:41:05 +00:00
|
|
|
phone_number_send(PG_FUNCTION_ARGS) {
|
|
|
|
try {
|
|
|
|
const ShortPhoneNumber *number = (ShortPhoneNumber*)PG_GETARG_POINTER(0);
|
|
|
|
StringInfoData buf;
|
|
|
|
|
|
|
|
pq_begintypsend(&buf);
|
|
|
|
pq_sendbytes(&buf, (const char*)number, sizeof(ShortPhoneNumber));
|
|
|
|
PG_RETURN_BYTEA_P(pq_endtypsend(&buf));
|
|
|
|
} catch(const std::bad_alloc& e) {
|
|
|
|
reportOutOfMemory();
|
|
|
|
} catch (const std::exception& e) {
|
|
|
|
reportGenericError(e);
|
|
|
|
}
|
2015-07-14 20:32:22 +00:00
|
|
|
|
2015-07-22 19:41:05 +00:00
|
|
|
PG_RETURN_NULL();
|
2015-07-14 20:32:22 +00:00
|
|
|
}
|
|
|
|
}
|