Files
epics-base/src/ioc/db/test/dbChannelTest.c
2012-04-27 13:21:40 -04:00

217 lines
6.0 KiB
C

/*************************************************************************\
* Copyright (c) 2010 UChicago Argonne LLC, as Operator of Argonne
* National Laboratory.
* EPICS BASE is distributed subject to a Software License Agreement found
* in file LICENSE that is included with this distribution.
\*************************************************************************/
#include "dbChannel.h"
#include "dbStaticLib.h"
#include "dbAccessDefs.h"
#include "epicsUnitTest.h"
#include "testMain.h"
/* Expected call bit definitions */
#define e_start 0x00000001
#define e_abort 0x00000002
#define e_end 0x00000004
#define e_null 0x00000008
#define e_boolean 0x00000010
#define e_integer 0x00000020
#define e_double 0x00000040
#define e_string 0x00000080
#define e_start_map 0x00000100
#define e_map_key 0x00000200
#define e_end_map 0x00000400
#define e_start_array 0x00000800
#define e_end_array 0x00001000
#define e_open 0x00002000
#define e_report 0x00004000
#define e_close 0x00008000
#define r_any (e_start | e_abort | e_end | \
e_null | e_boolean | e_integer | e_double | e_string | \
e_start_map | e_map_key | e_end_map | e_start_array | e_end_array)
#define r_scalar (e_start | e_abort | e_end | \
e_null | e_boolean | e_integer | e_double | e_string)
unsigned int e, r;
#define p_ret(x) return r & x ? parse_continue : parse_stop
parse_result p_start(chFilter *filter)
{
testOk(e & e_start, "parse_start called");
p_ret(e_start);
}
void p_abort(chFilter *filter)
{
testOk(e & e_abort, "parse_abort called");
}
parse_result p_end(chFilter *filter)
{
testOk(e & e_end, "parse_end called");
p_ret(e_end);
}
parse_result p_null(chFilter *filter)
{
testOk(e & e_null, "parse_null called");
p_ret(e_null);
}
parse_result p_boolean(chFilter *filter, int boolVal)
{
testOk(e & e_boolean, "parse_boolean called, val = %d", boolVal);
p_ret(e_boolean);
}
parse_result p_integer(chFilter *filter, long integerVal)
{
testOk(e & e_integer, "parse_integer called, val = %ld", integerVal);
p_ret(e_integer);
}
parse_result p_double(chFilter *filter, double doubleVal)
{
testOk(e & e_double, "parse_double called, val = %g", doubleVal);
p_ret(e_double);
}
parse_result p_string(chFilter *filter, const char *stringVal, size_t stringLen)
{
testOk(e & e_string, "parse_string called, val = '%.*s'", stringLen,
stringVal);
p_ret(e_string);
}
parse_result p_start_map(chFilter *filter)
{
testOk(e & e_start_map, "parse_start_map called");
p_ret(e_start_map);
}
parse_result p_map_key(chFilter *filter, const char *key, size_t stringLen)
{
testOk(e & e_map_key, "parse_map_key called, key = '%.*s'", stringLen, key);
p_ret(e_map_key);
}
parse_result p_end_map(chFilter *filter)
{
testOk(e & e_end_map, "parse_end_map called");
p_ret(e_end_map);
}
parse_result p_start_array(chFilter *filter)
{
testOk(e & e_start_array, "parse_start_array called");
p_ret(e_start_array);
}
parse_result p_end_array(chFilter *filter)
{
testOk(e & e_end_array, "parse_end_array called");
p_ret(e_end_array);
}
long c_open(chFilter *filter)
{
testOk(e & e_open, "channel_open called");
return 0;
}
void c_report(chFilter *filter, int level)
{
testOk(e & e_report, "channel_report called, level = %d", level);
}
void c_close(chFilter *filter)
{
testOk(e & e_close, "channel_close called");
}
chFilterIf testIf =
{ p_start, p_abort, p_end, p_null, p_boolean, p_integer, p_double,
p_string, p_start_map, p_map_key, p_end_map, p_start_array, p_end_array,
c_open, c_report, c_close };
MAIN(dbChannelTest)
{
dbChannel *pch;
testPlan(64);
testOk1(!dbReadDatabase(&pdbbase, "dbChannelTest.dbx", ".:..", NULL));
testOk(!!pdbbase, "pdbbase was set");
r = e = 0;
/* Regular record and field names */
testOk1(!dbChannelTest("x"));
testOk1(!dbChannelTest("x.NAME"));
/* Long string field modifier */
testOk1(!dbChannelTest("x.NAME$"));
testOk1(!!(pch = dbChannelCreate("x.NAME$")));
testOk1(pch->addr.no_elements > 1);
testOk1(!dbChannelDelete(pch));
/* JSON field modifier validation */
testOk1(!dbChannelTest("x.{\"json\":true}"));
/* Ensure bad PVs get rejected */
testOk(dbChannelTest("y"), "Test nonexistent record");
testOk(dbChannelCreate("y") == NULL, "Create nonexistent record");
testOk1(dbChannelTest("x.{not-json}"));
testOk1(!dbChannelCreate("x.{\"none\":null}"));
dbRegisterFilter("any", &testIf);
/* Parser event rejection by filter */
e = e_start;
testOk1(!dbChannelCreate("x.{\"any\":null}"));
r = e_start;
e = e_start | e_null | e_abort;
testOk1(!dbChannelCreate("x.{\"any\":null}"));
r = e_start | e_null;
e = e_start | e_null | e_end;
testOk1(!dbChannelCreate("x.{\"any\":null}"));
/* Successful parsing... */
r = r_any;
e = e_start | e_null | e_end;
testOk1(!!(pch = dbChannelCreate("x.{\"any\":null}")));
e = e_close;
testOk1(!dbChannelDelete(pch));
dbRegisterFilter("scalar", &testIf);
e = e_start | e_null | e_end;
testOk1(!!(pch = dbChannelCreate("x.{\"scalar\":null}")));
e = e_report;
dbChannelReport(pch, 0);
e = e_close;
testOk1(!dbChannelDelete(pch));
e = e_start | e_start_array | e_boolean | e_integer | e_end_array
| e_end;
testOk1(!!(pch = dbChannelCreate("x.{\"any\":[true,1]}")));
e = e_close;
testOk1(!dbChannelDelete(pch));
e = e_start | e_start_map | e_map_key | e_double | e_string | e_end_map
| e_end;
testOk1(!!(pch = dbChannelCreate("x.{\"any\":{\"a\":2.7183,\"b\":\"c\"}}")));
e = e_close;
testOk1(!dbChannelDelete(pch));
/* More event rejection */
r = r_scalar;
e = e_start | e_start_array | e_abort;
testOk1(!dbChannelCreate("x.{\"scalar\":[null]}"));
e = e_start | e_start_map | e_abort;
testOk1(!dbChannelCreate("x.{\"scalar\":{}}"));
dbFreeBase(pdbbase);
return testDone();
}