mockdb使用心得

这两天在写累积量模块的单元测试用例,由于AcmCalc.cpp里面涉及大量数据库操作,三千多行代码,为了增强对该文件的防护,提升覆盖率,就想着通过mockdb实现一下,但事与愿违,一直在碰壁,好好的数据放到mockdb怎么就捞不出来呢,于是花了两天时间研究了一下mockdbmockdb的实现。

其实很简单,从代码里看来,无外乎几个STL标准模板

typedef vector<TMockDBField> MockFields;

typedef list<MockFields> MockRecords;

typedef map<string,MockRecords> MockTables;

typedef map<string,MockTables *> MockDatas;

从上面几个容器,就构建出了一个简易数据库。第一个容器,就是用来存放表里的每个字段,TMockDBField类里存放的数据有如下几个:

 int data_type; 

 long long lvalue; 

 string svalue; 

 char name[128]; 

 bool bNull;

将一个表里的所有字段存放在第一个vector里面之后,就可以生成一个表,记录了同一张表里的多条记录。通过第二个list存放,这时就需要有一个表了,于是第三个map就出现了第三个map里的前面string就是表名,后面的list就是这个表里的所有记录。对于第四个map,这里应该是为了创建多个数据库准备的。大概了解mockdb数据存放方式之后,就可以进行插入更新删除操作了。

可惜的是,这些DML操作也实现得太简单了,具体情况如下:

本来我是通过:

static const char * const UPDATE_SUBS_ACM_DAILY_SQL="UPDATE SUBS_ACM_DAILY SET VALUE = VALUE + :VALUE \n"

" WHERE \n"

" SUBS_ID = :SUBS_ID AND RESOURCE_ID = :RESOURCE_ID AND DATE_STAMP = :DATE_STAMP";

语句去实现更新的,如果有记录,则更新。如果没有,则实现如下语句:

static const char * const INSERT_SUBS_ACM_DAILY_SQL="INSERT INTO SUBS_ACM_DAILY (SUBS_ID,RESOURCE_ID,VALUE,DATE_STAMP) \n"

" VALUES \n"

" (:SUBS_ID,:RESOURCE_ID,:VALUE,:DATE_STAMP)";

这没问题。当我第一个用例去实现订户日累积的时候,能正常插入数据。

但是当我第二个用例去对商品每天累积时,由于是同一张表SUBS_ACM_DAILY我先去实现update操作,mockdb居然就把我第一个用例的数据给更新掉了,更新掉了。。。。SUBS_ACM_DAILY表字段如下:

*SUBS_ID

*RESOURCE_ID

*DATE_STAMP 

VALUE

我在想,它是通过什么去更新的。发现它自己能创建出索引,所有update语句where后面所带的字段,它都自动认为是索引。那也是很好的,既然能自动为SUBS_ID、RESOURCE_ID、DATE_STAMP加上索引,那就进行匹配吧。结果匹配得也是神奇,三个字段,只要有一个字段匹配上,就算是找到记录了,前两个字段都没有匹配,只有第三个DATE_STAMP相同,于是乎就update了。如下函数就是去找数据库匹配的记录:

bool TMockDBQuery::FindSatisfiedRecord()

{

    bool bSatisfied = false;

    for (MockFields::iterator itF = m_fields.begin(); itF !=m_fields.end(); itF++)//检查绑定的参数

    {

        TMockDBField tInputField = *itF;//待检查的参数


        if(!IsIndex(tInputField.AsName()))//这里看上去是在匹配索引

        {

            continue;

        }


        MockFields vFields = *itCur;

        for (MockFields::iterator it = vFields.begin(); it != vFields.end(); it++)//对比fields

        {

            TMockDBField tSaveField = *it;//record中的参数

            if (tInputField == tSaveField)

            {

                bSatisfied = true;//这里只有一个索引值匹配时,就认为找到返回了,泪奔。。。

                break;

            }

            else

            {

                bSatisfied = false;

            }               

        }

    }

    return bSatisfied;

}

接下来的更新也是很神奇的。update语句里的SET VALUE = VALUE + :VALUE根本就不会去执行,只是将新记录的value去对原记录的覆盖。

    case OPER_UPDATE:

    if( it != pTables->end())

    {

        m_pRecords=&(it->second);

        itCur=m_pRecords->begin();

        while(itCur!=m_pRecords->end())

        {

            //查找满足条件的记录

            bool bSatisfied =FindSatisfiedRecord();

            if (bSatisfied)

            {

                //更新

                for (MockFields::iterator itF =m_fields.begin(); itF !=m_fields.end(); itF++)//检查绑定的参数

                {

                    TMockDBField tBindField = *itF;//绑定的参数                   


                    MockFields vFields =*itCur;

                    for (MockFields::iteratoriter = itCur->begin(); iter != itCur->end(); iter++)//对比fields

                    {

                        TMockDBFieldtRecordField = *iter;//record中的参数

                        if(!iter->IsName(tBindField.AsName()))//名字不同,跳过更新

                            continue;


                       if(!IsIndex(iter->AsName())) //如果不是索引字段,更新

                        {

                            cout <<"before: " <ToString()<

                            *iter = tBindField;

                            cout <<"after : " <ToString()<

                        }

                    }

                }

                iRowsAffected++;

                cout<<"TDBQuery::Execute() for update->OK!" << endl;

            }

            itCur++;

        }

    }

gdb打印的日志显示:

(gdb) n

before:TDBField[VALUE]:  Type   = 0; Values = ;  lvalue = 60;

243                                     *iter =tBindField;

(gdb) n

244                                     cout<<  "after : "<ToString()<

(gdb) n

after :TDBField[VALUE]:  Type   = 0; Values = ;  lvalue = 240;

236                                 TMockDBFieldtRecordField = *iter;//record中的参数

(gdb) n

这里之前的累积量就是60,本次累积量就是240,直接覆盖。。。


导致第二个测试用例去捞数据的时候,就没捞到

再次详细分析了一下为什么没捞到,是怎么匹配的关键字

select语句的索引创建时,是在SET_PARAM时将需要匹配的条件字段放在了vector m_fields中,这个存放规则也有问题,因为如果你有多个匹配字段,不小心将最后一个需要比较的字段放到里面,而此时表里正好有一条记录的这个字段能够匹配,程序却也认为这条记录就是你需要捞取的记录。这段代码如下:

    for (; itCur != m_pRecords->end();itCur++)//遍历表里每一条record

    {

        for (MockFields::iterator itF =m_fields.begin(); itF !=m_fields.end(); itF++)//检查绑定的参数,就是你select时where里的条件

        {

            TMockDBField tInputField = *itF;//待检查的参数

            if(!IsIndex(tInputField.AsName()))

            {

                continue;

            }

            MockFields vFields = *itCur;

            for (MockFields::iterator it =vFields.begin(); it != vFields.end(); it++)//对比fields

            {

                TMockDBField tSaveField =*it;//record中的参数

                if (tInputField == tSaveField)

                {                   

                    bFind = true;//当最后一个需要匹配的条件字段恰好跟表里当时那条记录对应字段相等时,这里就认为找到记录了

                    break;

                }

                else

                {                   

                    bFind = false;

                }               

            }

        }

        if (m_fields.size() == 0)

        {

            bFind = true;

        }

        if (bFind)

        {

            m_outputFields = *itCur;

            itCur++;

            return true;

        }

    }

这个地方我通过在设置:

    原始:

   m_pOdbcQuery->SetParameter("SUBS_ID",pRatableEvent->GetSubsId());

   m_pOdbcQuery->SetParameter("DATE_STAMP",iDate);

   m_pOdbcQuery->SetParameter("RESOURCE_ID", iResourceId);

    改为:

   m_pOdbcQuery->SetParameter("SUBS_ID",pRatableEvent->GetSubsId());

   m_pOdbcQuery->SetParameter("RESOURCE_ID", iResourceId);

   m_pOdbcQuery->SetParameter("DATE_STAMP",iDate);//因为这个字段跟第一个用例里的相同,所以最后匹配这个字段时相同,就认为是捞取出来了

最终结果跑出来确是正确的:

[       OK ] TAcmCalcTest.Give_duration1_AcmType7_When_rum60_Then_Acm60(7 ms)

[ RUN ]TAcmCalcTest.Give_duration181_AcmType9_When_rum60_Then_Acm240

[       OK ]TAcmCalcTest.Give_duration181_AcmType9_When_rum60_Then_Acm240 (9 ms)

[----------] 2

tests from TAcmCalcTest (16 ms total)

[----------]

Global test environment tear-down

[==========] 2

tests from 1 test case ran. (497 ms total)

[  PASSED ] 2 tests.

这里的成功,也多亏了前面update时没有加上原始值的bug,不然这里捞出来就是240+60=300了


还没想到能有什么好的方式去规避这个问题

想过每个用例跑完之后rollback,但是之前订户资料,资费等信息也在里面,担心rollback所有数据都没有,而且也没有提供这样的接口,因为一开始就没有commit动作

感觉mockdb只能是最简单的存放单条数据,而且不能写复杂的sql,包括where后面带and或者or,也不能在update里带计算。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,711评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,079评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,194评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,089评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,197评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,306评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,338评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,119评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,541评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,846评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,014评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,694评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,322评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,026评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,257评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,863评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,895评论 2 351

推荐阅读更多精彩内容