675fcde74c
Doing setSourceType() unconditionally unintentionally also removed a configured syncFormat and forceSyncFormat setting for an existing configuration. Only setBackend() (as with this patch) is necessary and valid in all cases (never run a test with the wrong backend!).
604 lines
22 KiB
C++
604 lines
22 KiB
C++
/*
|
|
* Copyright (C) 2005-2009 Patrick Ohly <patrick.ohly@gmx.de>
|
|
* Copyright (C) 2009 Intel Corporation
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) version 3.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
|
|
* 02110-1301 USA
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include <ClientTest.h>
|
|
|
|
#include <exception>
|
|
#include <fstream>
|
|
#include <fcntl.h>
|
|
#include <unistd.h>
|
|
#include <errno.h>
|
|
#include <signal.h>
|
|
#ifdef HAVE_VALGRIND_VALGRIND_H
|
|
# include <valgrind/valgrind.h>
|
|
#endif
|
|
#ifdef HAVE_EXECINFO_H
|
|
# include <execinfo.h>
|
|
#endif
|
|
|
|
#include "CmdlineSyncClient.h"
|
|
#include "EvolutionSyncSource.h"
|
|
#include <syncevo/util.h>
|
|
#include <syncevo/VolatileConfigNode.h>
|
|
|
|
#include <syncevo/declarations.h>
|
|
|
|
#ifdef ENABLE_BUTEO_TESTS
|
|
#include "client-test-buteo.h"
|
|
#endif
|
|
|
|
SE_BEGIN_CXX
|
|
|
|
/*
|
|
* always provide this test class, even if not used:
|
|
* that way the test scripts can unconditionally
|
|
* invoke "client-test SyncEvolution"
|
|
*/
|
|
CPPUNIT_REGISTRY_ADD_TO_DEFAULT("SyncEvolution");
|
|
|
|
class EvolutionLocalTests : public LocalTests {
|
|
public:
|
|
EvolutionLocalTests(const std::string &name, ClientTest &cl, int sourceParam, ClientTest::Config &co) :
|
|
LocalTests(name, cl, sourceParam, co)
|
|
{}
|
|
|
|
virtual void addTests() {
|
|
LocalTests::addTests();
|
|
|
|
#ifdef ENABLE_MAEMO
|
|
if (config.createSourceA &&
|
|
config.createSourceB &&
|
|
config.templateItem &&
|
|
strstr(config.templateItem, "BEGIN:VCARD") &&
|
|
config.uniqueProperties) {
|
|
ADD_TEST(EvolutionLocalTests, testOssoDelete);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
private:
|
|
|
|
// insert am item,
|
|
// overwrite it with an additional X-OSSO-CONTACT-STATE:DELETED as Maemoe address book does,
|
|
// iterate again and check that our own code deleted the item
|
|
void testOssoDelete() {
|
|
// get into clean state with one template item added
|
|
deleteAll(createSourceA);
|
|
insert(createSourceA, config.templateItem, config.itemType);
|
|
|
|
// add X-OSSO-CONTACT-STATE:DELETED
|
|
string item = config.templateItem;
|
|
const char *comma = strchr(config.uniqueProperties, ':');
|
|
size_t offset = item.find(config.uniqueProperties, 0,
|
|
comma ? comma - config.uniqueProperties : strlen(config.uniqueProperties));
|
|
CPPUNIT_ASSERT(offset != item.npos);
|
|
item.insert(offset, "X-OSSO-CONTACT-STATE:DELETED\n");
|
|
update(createSourceA, item.c_str(), false);
|
|
|
|
// opening and preparing the source should delete the item
|
|
std::auto_ptr<TestingSyncSource> source;
|
|
SOURCE_ASSERT_NO_FAILURE(source.get(), source.reset(createSourceA()));
|
|
SOURCE_ASSERT_NO_FAILURE(source.get(), source->open());
|
|
SOURCE_ASSERT_NO_FAILURE(source.get(), source->beginSync("", "") );
|
|
CPPUNIT_ASSERT_EQUAL(0, countItemsOfType(source.get(), SyncSourceChanges::ANY));
|
|
CPPUNIT_ASSERT_EQUAL(0, countItemsOfType(source.get(), SyncSourceChanges::NEW));
|
|
CPPUNIT_ASSERT_EQUAL(0, countItemsOfType(source.get(), SyncSourceChanges::UPDATED));
|
|
CPPUNIT_ASSERT_EQUAL(1, countItemsOfType(source.get(), SyncSourceChanges::DELETED));
|
|
}
|
|
};
|
|
|
|
/**
|
|
* This is a workaround for libecal/libebook in Evolution >= 2.30.
|
|
* The storage daemons shut down after only 10 seconds of no client
|
|
* being attached. Due to limitations in libecal/libebook this is not
|
|
* detected when only using the synchronous API ("destroyed" signal
|
|
* not delivered, see e-cal.c), which then leads to D-Bus errors.
|
|
*
|
|
* The workaround consists of keeping one open SyncEvolution backend
|
|
* around for each of ical20 and vcard21/30, if they ever were used
|
|
* during testing.
|
|
*/
|
|
static map<string, boost::shared_ptr<TestingSyncSource> > lockEvolution;
|
|
static void CleanupSources()
|
|
{
|
|
lockEvolution.clear();
|
|
}
|
|
|
|
/**
|
|
* This code uses the ClientTest and and information provided by
|
|
* the backends in their RegisterSyncSourceTest instances to test
|
|
* real synchronization with a server.
|
|
*
|
|
* Configuration is done by environment variables which indicate which
|
|
* part below the root node "client-test" of the the configuration tree to use;
|
|
* beyond that everything needed for synchronization is read from the
|
|
* configuration tree.
|
|
*
|
|
* - CLIENT_TEST_SERVER = maps to name of root node in configuration tree
|
|
* - CLIENT_TEST_EVOLUTION_PREFIX = a common "evolutionsource" prefix for *all*
|
|
* sources; the source name followed by "_[12]"
|
|
* is appended to get unique names
|
|
* - CLIENT_TEST_EVOLUTION_USER = sets the "evolutionuser" property of all sources
|
|
* - CLIENT_TEST_EVOLUTION_PASSWORD = sets the "evolutionpassword" property of all sources
|
|
* - CLIENT_TEST_SOURCES = comma separated list of active sources,
|
|
* names as selected in their RegisterSyncSourceTest
|
|
* instances
|
|
* - CLIENT_TEST_DELAY = number of seconds to sleep between syncs, required
|
|
* by some servers
|
|
* - CLIENT_TEST_LOG = logfile name of a server, can be empty:
|
|
* if given, then the content of that file will be
|
|
* copied and stored together with the client log
|
|
* (only works on Unix)
|
|
* - CLIENT_TEST_NUM_ITEMS = numbers of contacts/events/... to use during
|
|
* local and sync tests which create artificial
|
|
* items
|
|
*
|
|
* The CLIENT_TEST_SERVER also has another meaning: it is used as hint
|
|
* by the synccompare.pl script and causes it to automatically ignore
|
|
* known, acceptable data modifications caused by sending an item to
|
|
* a server and back again. Currently the script recognizes "funambol",
|
|
* "scheduleworld", "synthesis" and "egroupware" as special server
|
|
* names.
|
|
*/
|
|
class TestEvolution : public ClientTest {
|
|
public:
|
|
/**
|
|
* can be instantiated as client A with id == "1" and client B with id == "2"
|
|
*/
|
|
TestEvolution(const string &id) :
|
|
ClientTest(getenv("CLIENT_TEST_DELAY") ? atoi(getenv("CLIENT_TEST_DELAY")) : 0,
|
|
getenv("CLIENT_TEST_LOG") ? getenv("CLIENT_TEST_LOG") : ""),
|
|
m_clientID(id),
|
|
m_configs(EvolutionSyncSource::getTestRegistry())
|
|
{
|
|
const char *server = getenv("CLIENT_TEST_SERVER");
|
|
|
|
if (id == "1") {
|
|
m_clientB.reset(new TestEvolution("2"));
|
|
}
|
|
|
|
/* check server */
|
|
if (!server) {
|
|
server = "funambol";
|
|
setenv("CLIENT_TEST_SERVER", "funambol", 1);
|
|
}
|
|
|
|
/* override Evolution database names? */
|
|
const char *evoprefix = getenv("CLIENT_TEST_EVOLUTION_PREFIX");
|
|
m_evoPrefix = evoprefix ? evoprefix : "SyncEvolution_Test_";
|
|
const char *evouser = getenv("CLIENT_TEST_EVOLUTION_USER");
|
|
if (evouser) {
|
|
m_evoUser = evouser;
|
|
}
|
|
const char *evopasswd = getenv("CLIENT_TEST_EVOLUTION_PASSWORD");
|
|
if (evopasswd) {
|
|
m_evoPassword = evopasswd;
|
|
}
|
|
|
|
/* check sources */
|
|
const char *sourcelist = getenv("CLIENT_TEST_SOURCES");
|
|
set<string> sources;
|
|
if (sourcelist) {
|
|
boost::split(sources, sourcelist, boost::is_any_of(","));
|
|
} else {
|
|
BOOST_FOREACH(const RegisterSyncSourceTest *test, m_configs) {
|
|
sources.insert(test->m_configName);
|
|
}
|
|
}
|
|
|
|
BOOST_FOREACH(const RegisterSyncSourceTest *test, m_configs) {
|
|
if (sources.find(test->m_configName) != sources.end()) {
|
|
m_syncSource2Config.push_back(test->m_configName);
|
|
}
|
|
}
|
|
|
|
/* Local Test SyncSource : remove all virtual datastores, inserting the
|
|
* sub datastores*/
|
|
ClientTest::Config conf;
|
|
BOOST_FOREACH (string source, sources) {
|
|
getSourceConfig (source, conf);
|
|
if (conf.subConfigs) {
|
|
vector<string> subs;
|
|
boost::split (subs, conf.subConfigs, boost::is_any_of(","));
|
|
BOOST_FOREACH (string sub, subs) {
|
|
pushLocalSource2Config(sub);
|
|
}
|
|
} else {
|
|
pushLocalSource2Config(source);
|
|
}
|
|
}
|
|
// get configuration and set obligatory fields
|
|
LoggerBase::instance().setLevel(Logger::DEBUG);
|
|
std::string root = std::string("evolution/") + server + "_" + id;
|
|
boost::shared_ptr<SyncConfig> config(new SyncConfig(string(server) + "_" + id));
|
|
boost::shared_ptr<SyncConfig> from = boost::shared_ptr<SyncConfig> ();
|
|
|
|
if (!config->exists()) {
|
|
// no configuration yet, create in different contexts because
|
|
// device ID is different
|
|
config.reset(new SyncConfig(string(server) + "_" + id + "@client-test-" + id));
|
|
config->setDefaults();
|
|
from = SyncConfig::createPeerTemplate(server);
|
|
if(from) {
|
|
set<string> filter;
|
|
config->copy(*from, &filter);
|
|
}
|
|
config->setDevID(id == "1" ? "sc-api-nat" : "sc-pim-ppc");
|
|
}
|
|
BOOST_FOREACH(const RegisterSyncSourceTest *test, m_configs) {
|
|
ClientTest::Config testconfig;
|
|
getSourceConfig(test, testconfig);
|
|
CPPUNIT_ASSERT(testconfig.type);
|
|
|
|
boost::shared_ptr<SyncSourceConfig> sc = config->getSyncSourceConfig(testconfig.sourceName);
|
|
if (!sc || !sc->exists()) {
|
|
// no configuration yet
|
|
config->setSourceDefaults(testconfig.sourceName);
|
|
sc = config->getSyncSourceConfig(testconfig.sourceName);
|
|
CPPUNIT_ASSERT(sc);
|
|
sc->setURI(testconfig.uri);
|
|
if(from && testconfig.sourceNameServerTemplate){
|
|
boost::shared_ptr<SyncSourceConfig> scServerTemplate = from->getSyncSourceConfig(testconfig.sourceNameServerTemplate);
|
|
sc->setURI(scServerTemplate->getURI());
|
|
}
|
|
}
|
|
|
|
// always set these properties: they might have changed since the last run
|
|
string database = getDatabaseName(test->m_configName);
|
|
sc->setDatabaseID(database);
|
|
sc->setUser(m_evoUser);
|
|
sc->setPassword(m_evoPassword);
|
|
sc->setBackend(SourceType(testconfig.type).m_backend);
|
|
}
|
|
config->flush();
|
|
}
|
|
|
|
virtual LocalTests *createLocalTests(const std::string &name, int sourceParam, ClientTest::Config &co) {
|
|
return new EvolutionLocalTests(name, *this, sourceParam, co);
|
|
}
|
|
|
|
virtual int getNumLocalSources() {
|
|
return m_localSource2Config.size();
|
|
}
|
|
|
|
virtual int getNumSyncSources() {
|
|
return m_syncSource2Config.size();
|
|
}
|
|
|
|
virtual void getLocalSourceConfig(int source, Config &config) {
|
|
getSourceConfig(m_configs[m_localSource2Config[source]], config);
|
|
}
|
|
|
|
virtual void getSyncSourceConfig(int source, Config &config) {
|
|
getSourceConfig(m_configs[m_syncSource2Config[source]], config);
|
|
}
|
|
|
|
virtual int getLocalSourcePosition(const string &configName) {
|
|
for (size_t i=0; i< m_localSource2Config.size(); i++) {
|
|
if(m_localSource2Config[i] == configName) {
|
|
return i;
|
|
break;
|
|
}
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
virtual void getSourceConfig (const string &configName, Config &config) {
|
|
return getSourceConfig (m_configs[configName], config);
|
|
}
|
|
|
|
static void getSourceConfig(const RegisterSyncSourceTest *test, Config &config) {
|
|
memset(&config, 0, sizeof(config));
|
|
ClientTest::getTestData(test->m_testCaseName.c_str(), config);
|
|
config.createSourceA = createSource;
|
|
config.createSourceB = createSource;
|
|
config.sourceName = test->m_configName.c_str();
|
|
|
|
test->updateConfig(config);
|
|
}
|
|
|
|
virtual ClientTest *getClientB() {
|
|
return m_clientB.get();
|
|
}
|
|
|
|
virtual bool isB64Enabled() {
|
|
return false;
|
|
}
|
|
|
|
#ifdef ENABLE_BUTEO_TESTS
|
|
virtual void setup() {
|
|
QtContactsSwitcher::prepare(*this);
|
|
}
|
|
#endif
|
|
|
|
virtual SyncMLStatus doSync(const int *sources,
|
|
const std::string &logbase,
|
|
const SyncOptions &options)
|
|
{
|
|
// check whether using buteo to do sync
|
|
const char *buteo = getenv("CLIENT_TEST_BUTEO");
|
|
bool useButeo = false;
|
|
if (buteo &&
|
|
(boost::equals(buteo, "1") || boost::iequals(buteo, "t"))) {
|
|
useButeo = true;
|
|
}
|
|
|
|
string server = getenv("CLIENT_TEST_SERVER") ? getenv("CLIENT_TEST_SERVER") : "funambol";
|
|
server += "_";
|
|
server += m_clientID;
|
|
|
|
|
|
if (useButeo) {
|
|
#ifdef ENABLE_BUTEO_TESTS
|
|
ButeoTest buteo(*this, server, logbase, options);
|
|
buteo.prepareSources(sources, m_syncSource2Config);
|
|
SyncReport report;
|
|
SyncMLStatus status = buteo.doSync(&report);
|
|
options.m_checkReport.check(status, report);
|
|
return status;
|
|
#else
|
|
throw runtime_error("This client-test was built without enabling buteo testing.");
|
|
#endif
|
|
}
|
|
class ClientTest : public CmdlineSyncClient {
|
|
public:
|
|
ClientTest(const string &server,
|
|
const string &logbase,
|
|
const SyncOptions &options) :
|
|
CmdlineSyncClient(server, false, true),
|
|
m_logbase(logbase),
|
|
m_options(options),
|
|
m_started(false)
|
|
{}
|
|
|
|
protected:
|
|
virtual void prepare() {
|
|
setLogDir(m_logbase, true);
|
|
setMaxLogDirs(0, true);
|
|
setMaxObjSize(m_options.m_maxObjSize, true);
|
|
setMaxMsgSize(m_options.m_maxMsgSize, true);
|
|
setWBXML(m_options.m_isWBXML, true);
|
|
setRetryDuration(m_options.m_retryDuration, true);
|
|
setRetryInterval(m_options.m_retryInterval, true);
|
|
if (m_options.m_syncMode == SYNC_TWO_WAY &&
|
|
m_options.m_checkReport.syncMode == SYNC_NONE) {
|
|
// For this test, any kind of final sync mode is
|
|
// acceptable. Disable slow sync prevention
|
|
// temporarily. The check for the requested sync
|
|
// mode is perhaps too conservative, but in
|
|
// practice the only test where slow sync
|
|
// prevention caused a test failure was
|
|
// Client::Sync::vcard30::testTwoWaySync after
|
|
// some other failed test, so let's be conservative...
|
|
setPreventSlowSync(false);
|
|
}
|
|
SyncContext::prepare();
|
|
if (m_options.m_prepareCallback &&
|
|
m_options.m_prepareCallback(*this, m_options)) {
|
|
m_options.m_isAborted = true;
|
|
}
|
|
}
|
|
|
|
virtual void displaySyncProgress(sysync::TProgressEventEnum type,
|
|
int32_t extra1, int32_t extra2, int32_t extra3)
|
|
{
|
|
if (!m_started) {
|
|
m_started = true;
|
|
if (m_options.m_startCallback(*this, m_options)) {
|
|
m_options.m_isAborted = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
virtual bool checkForAbort() { return m_options.m_isAborted; }
|
|
virtual bool checkForSuspend() {return m_options.m_isSuspended;}
|
|
|
|
virtual boost::shared_ptr<TransportAgent> createTransportAgent()
|
|
{
|
|
boost::shared_ptr<TransportAgent>wrapper = m_options.m_transport;
|
|
boost::shared_ptr<TransportAgent>agent =SyncContext::createTransportAgent();
|
|
if (!wrapper.get())
|
|
return agent;
|
|
dynamic_cast<TransportWrapper*>(wrapper.get())->setAgent(agent);
|
|
dynamic_cast<TransportWrapper*>(wrapper.get())->setSyncOptions(&m_options);
|
|
return wrapper;
|
|
}
|
|
|
|
private:
|
|
const string m_logbase;
|
|
SyncOptions m_options;
|
|
bool m_started;
|
|
} client(server, logbase, options);
|
|
|
|
// configure active sources with the desired sync mode,
|
|
// disable the rest
|
|
FilterConfigNode::ConfigFilter filter;
|
|
filter["sync"] = "none";
|
|
client.setConfigFilter(false, "", filter);
|
|
filter["sync"] =
|
|
PrettyPrintSyncMode(options.m_syncMode);
|
|
for(int i = 0; sources[i] >= 0; i++) {
|
|
std::string &name = m_syncSource2Config[sources[i]];
|
|
client.setConfigFilter(false, name, filter);
|
|
checkEvolutionSource(name);
|
|
}
|
|
|
|
SyncReport report;
|
|
SyncMLStatus status = client.sync(&report);
|
|
options.m_checkReport.check(status, report);
|
|
return status;
|
|
}
|
|
|
|
private:
|
|
string m_clientID;
|
|
std::auto_ptr<TestEvolution> m_clientB;
|
|
const TestRegistry &m_configs;
|
|
|
|
/** prefix, username, password to be used for local databases */
|
|
string m_evoPrefix, m_evoUser, m_evoPassword;
|
|
|
|
/**
|
|
* The ClientTest framework identifies active configs with an integer.
|
|
* This is the mapping to the corresponding config name, created when
|
|
* constructing this instance.
|
|
*/
|
|
vector<string> m_localSource2Config;
|
|
vector<string> m_syncSource2Config;
|
|
|
|
/** returns the name of the Evolution database */
|
|
string getDatabaseName(const string &configName) {
|
|
if (configName == "calendar+todo") {
|
|
return "ical20,itodo20";
|
|
} else if (configName == "file_calendar+todo") {
|
|
return "file_ical20,file_itodo20";
|
|
}
|
|
return m_evoPrefix + configName + "_" + m_clientID;
|
|
}
|
|
|
|
/** called by test frame work */
|
|
static TestingSyncSource *createSource(ClientTest &client, int source, bool isSourceA) {
|
|
TestEvolution &evClient((TestEvolution &)client);
|
|
string name = evClient.m_localSource2Config[source];
|
|
|
|
// implement Evolution shutdown workaround (see lockEvolution above)
|
|
evClient.checkEvolutionSource(name);
|
|
|
|
return evClient.createSource(name, isSourceA);
|
|
}
|
|
|
|
/** called internally in this class */
|
|
TestingSyncSource *createSource(const string &name, bool isSourceA) {
|
|
string database = getDatabaseName(name);
|
|
boost::shared_ptr<SyncConfig> context(new SyncConfig("source-config@client-test"));
|
|
SyncSourceNodes nodes = context->getSyncSourceNodes(name,
|
|
string("_") + m_clientID +
|
|
"_" + (isSourceA ? "A" : "B"));
|
|
|
|
// always set this property: the name might have changes since last test run
|
|
nodes.getProperties()->setProperty("evolutionsource", database.c_str());
|
|
nodes.getProperties()->setProperty("evolutionuser", m_evoUser.c_str());
|
|
nodes.getProperties()->setProperty("evolutionpassword", m_evoPassword.c_str());
|
|
|
|
SyncSourceParams params(name,
|
|
nodes,
|
|
context);
|
|
const RegisterSyncSourceTest *test = m_configs[name];
|
|
ClientTestConfig testConfig;
|
|
getSourceConfig(test, testConfig);
|
|
|
|
PersistentSyncSourceConfig sourceConfig(params.m_name, params.m_nodes);
|
|
sourceConfig.setSourceType(SourceType(testConfig.type));
|
|
|
|
// downcasting here: anyone who registers his sources for testing
|
|
// must ensure that they are indeed TestingSyncSource instances
|
|
SyncSource *ss = SyncSource::createSource(params);
|
|
return static_cast<TestingSyncSource *>(ss);
|
|
}
|
|
|
|
// push source into localsource2config if it doesn't exist in the vector
|
|
void pushLocalSource2Config(const string &source) {
|
|
bool finded = false;
|
|
BOOST_FOREACH(string element, m_localSource2Config) {
|
|
if (boost::iequals(element, source)) {
|
|
finded = true;
|
|
break;
|
|
}
|
|
}
|
|
if (!finded) {
|
|
m_localSource2Config.push_back (source);
|
|
}
|
|
}
|
|
|
|
void checkEvolutionSource(std::string &name)
|
|
{
|
|
string basename;
|
|
|
|
// hard-coded names as used by src/backends/evolution;
|
|
// if some other backend reuses them, it gets the
|
|
// same treatment, which shouldn't cause any harm
|
|
if (name == "vcard21" ||
|
|
name == "vcard30") {
|
|
basename = "ebook";
|
|
} else if (name == "ical20" ||
|
|
name == "text") {
|
|
basename = "ecal";
|
|
}
|
|
|
|
if (!basename.empty() &&
|
|
lockEvolution.find(basename) == lockEvolution.end()) {
|
|
lockEvolution[basename].reset(createSource(name, true));
|
|
lockEvolution[basename]->open();
|
|
ClientTest::registerCleanup(CleanupSources);
|
|
}
|
|
}
|
|
};
|
|
|
|
static void handler(int sig)
|
|
{
|
|
void *buffer[100];
|
|
int size;
|
|
|
|
fprintf(stderr, "\ncaught signal %d\n", sig);
|
|
fflush(stderr);
|
|
#ifdef HAVE_EXECINFO_H
|
|
size = backtrace(buffer, sizeof(buffer)/sizeof(buffer[0]));
|
|
backtrace_symbols_fd(buffer, size, 2);
|
|
#endif
|
|
#ifdef HAVE_VALGRIND_VALGRIND_H
|
|
VALGRIND_PRINTF_BACKTRACE("\ncaught signal %d\n", sig);
|
|
#endif
|
|
/* system("objdump -l -C -d client-test >&2"); */
|
|
struct sigaction act;
|
|
memset(&act, 0, sizeof(act));
|
|
act.sa_handler = SIG_DFL;
|
|
sigaction(SIGABRT, &act, NULL);
|
|
abort();
|
|
}
|
|
|
|
static class RegisterTestEvolution {
|
|
public:
|
|
RegisterTestEvolution() :
|
|
testClient("1") {
|
|
struct sigaction act;
|
|
|
|
memset(&act, 0, sizeof(act));
|
|
act.sa_handler = handler;
|
|
sigaction(SIGABRT, &act, NULL);
|
|
sigaction(SIGSEGV, &act, NULL);
|
|
sigaction(SIGILL, &act, NULL);
|
|
|
|
#if defined(HAVE_GLIB)
|
|
// this is required when using glib directly or indirectly
|
|
g_type_init();
|
|
g_thread_init(NULL);
|
|
g_set_prgname("client-test");
|
|
#endif
|
|
EDSAbiWrapperInit();
|
|
testClient.registerTests();
|
|
}
|
|
|
|
private:
|
|
TestEvolution testClient;
|
|
|
|
} testEvolution;
|
|
|
|
SE_END_CXX
|