我们 RCV 在这里,经常跑concurrent request RTP: Receiving Transaction Processor, 它主要是用来处理 RCV_TRANSACTIONS_INTERFACE 内的数据.

这个 concurrent program 里面包括很多文件, 比較重要的有:

ident RVCTP
RVCTP:
$Header: rvctp.oc 120.0.12000000.1 2007/01/16 23:53:49 appldev ship $
$Header: rvtbm.lpc 120.13.12010000.10 2013/03/15 02:51:16 zhlee ship $
$Header: rvtpt.lpc 120.28.12010000.33 2013/07/15 07:12:37 smai ship $
$Header: rvtvq.lpc 120.8.12010000.17 2013/04/03 11:40:30 wayin ship $
$Header: rvtuq.lpc 120.9.12000000.7 2009/07/07 09:47:49 YUZZHANG01 ship $
$Header: pouer.lpc 120.2 2006/06/14 07:57:02 arudas noship $
$Header: rvtlo.lpc 120.1.12010000.7 2013/01/05 16:02:14 gke ship $
$Header: rvtoc.lpc 120.1.12010000.5 2013/03/28 22:38:31 vthevark ship $
$Header: rvtpa.lc 120.6.12010000.3 2009/07/07 08:58:22 ksivasa ship $
$Header: rvtpd.lc 120.1.12010000.3 2009/07/07 09:03:17 ksivasa ship $
$Header: rvtsh.lpc 120.10.12010000.3 2009/03/06 22:09:40 vthevark ship $
$Header: rvtth.lpc 120.29.12010000.9 2012/05/30 15:50:03 gke ship $
$Header: rvtvt.lpc 120.9.12010000.10 2013/03/28 18:19:11 vthevark ship $
$Header: rvsco.lpc 120.4.12010000.12 2012/09/27 23:10:24 vthevark ship $
$Header: rvsrv.lpc 120.4.12010000.16 2013/02/04 07:34:37 zhizheng ship $
$Header: rvsit.lpc 120.5.12010000.6 2013/02/04 07:26:30 zhizheng ship $
$Header: rvsdl.lpc 120.5.12010000.17 2011/11/28 05:19:19 xiameng ship $
$Header: rvssh.lpc 120.2.12010000.3 2009/07/07 09:56:26 ksivasa ship $
$Header: rvsut.lpc 120.3.12010000.14 2012/05/28 04:53:57 gke ship $
$Header: rvtcl.lpc 120.6.12010000.3 2009/07/07 09:15:25 ksivasa ship $
$Header: rvtii.lpc 120.19.12010000.27 2013/08/14 09:58:23 gke ship $
$Header: rvtls.lpc 120.7.12010000.14 2012/01/30 12:31:22 ksivasa ship $
$Header: rvtoo.lpc 120.3.12010000.3 2009/07/07 09:08:40 ksivasa ship $
$Header: rvsrq.lpc 120.5.12010000.3 2009/07/07 09:53:56 ksivasa ship $
$Header: rvspo.lpc 120.3.12010000.3 2009/07/07 09:53:12 ksivasa ship $

这些文件是在 RTP log 里面常常见到的. 这里我们主要看看, 在这些文件里是用如何的机制写 log 的.

首先, 要在要写日志的函数里面声明一个变量:

text     strbuf[BUFLEN+1];

然后用以下的语句:

    strcpy((char *)strbuf,"");
sprintf((char *)strbuf,"RVTUQ:370 req_line_id %ld\n", (long)req_line_id);
Debug(strbuf,TRUE,TRUE);

第一句把 strbuf 清空, 然后把后面的字符串和整型数写入 strbuf, 第三句 Debug 把 strbuf 写入日志. Debug 后面有两个參数:

#define Debug(message,check_sql_error,no_rows_is_error)

第二个參数 check_sql_error 表示 假设希望检查 SQL error, 就设为 TRUE, 否则设为 FALSE. no_rows_is_error 表示 是不是把 NO_DATA_FOUND 当做一个 error. 假设是, 设为TRUE.

Concurrent Request 里面的文件都是 .c 的代码, 假设里面要写SQL, 就要写一句:

    EXEC SQL

然后以下写上 SQL. 这句 SQL 有可能会出错, 这时须要一个方法把错误消息写到 log 里面去. 比方:

       EXEC SQL
select transaction_id,
unit_of_measure
into :parent_trx_id,
:rec_uom
from rcv_transactions
where transaction_type in ('RECEIVE','MATCH')
start with transaction_id = :p_trx_id
connect by transaction_id = prior parent_transaction_id;

为了捕获这句SQL 可能发生的错误, 我们在这句后面写一句:

pouersql("RVTUQ", "134", FALSE )

pouersql() 这个函数是在 pouer.lpc 文件 (po user error) 中定义的, 假设上面的SQL 发生了错误, 那么就会把 error message 写入 message stack. 而且返回 true.

这里面有三个參数, routine, location, no_rows_is_err.

routine 表示SQL 错误发生的文件名称, 上面的样例中是 RVTUQ.

location 表示SQL 错误发生的位置, 上面的样例中是 134. 这样我们看到日志里面的这个位置就非常easy定位到这句SQL.

no_rows_is_err 表示 是不是把 NO_DATA_FOUND 当做 error 来处理. 上面的样例中我们没有当做 error.

在 pouersql 这个函数的内部, 会去sqlca 里面提取 sqlcode. sqlca 全称是: sql communications area, 是一个用来记录SQL 运行信息的地方. 每次运行一个SQL 都会返回 sqlcode. 假设sqlcode = 0, 那么表示没有错误. 假设 sqlcode > 0, 表示有exception, 假设 sqlcode < 0, 表示发生 error, 要rollback. 我们平时接触的error 都是小于0 的, 可是 NO_DATA_FOUND
是 大于0 的, 代码 1403. pousersql() 函数会去 sqlca 里面提取 sqlcode, 然后推断是否 sqlcode < 0 还是 sqlcode > 0, 假设推断小于0, 那么才把错误消息写入 message stack. 假设大于0, 而且參数里面 no_rows_is_err 设为 TRUE, 那么也会写如message stack.

每次运行一次 SQL 就会更新一次 sqlca, 所以这个结构仅仅记录近期一次运行SQL 的信息 . 错误消息是记录在sqlca.sqlerrm.sqlerrmc 里面的. 所以会把这个里面的字符串写入message stack. 可是sqlerrmc 最多仅仅能记录70 个字符, 所以假设错误消息非常长的话, 就会被截断.

上面的SQL 错误发生的话, pouersql () 返回值为 TRUE. 那么在调用这个函数的地方, 比方 rvtpt.lpc 文件里, 就会捕捉到这个返回值, 推断返回值之后确定发生error 了, 就会调用:

pouertrace( "RVTUQ", "000", "rvtuqdebitmemo()" )

pouertrace () 这个函数也是在pouer.lpc 文件里定义的, 这个函数的返回值总是 false. 它的主要作用也是把错误消息写入 message stack. 三个參数分别为:

file: 表示错误发生的文件

location: 表示错误的位置

subroutine: 表示错误发生的函数.

上面我们提到, sqlerrmc 仅仅能保存70 个字符. 假设错误消息多余70 个字符, 想要得到全部的错误消息的话, 就须要用 sqlglm() 函数. 在使用这个函数之前, 我们要确保 sqlcode < 0, 否则取出来的错误消息是前面的SQL 的消息.

if (pouersql("RVTUQ","019",TRUE))
{
char msg[510];
size_t buf_len, msg_len;
buf_len = sizeof (msg);
sqlglm(msg, &buf_len, &msg_len); strcpy((char *)strbuf,"");
sprintf((char *)strbuf,"RVTUQ:001 YU SQL ERROR %s \n", (text *) msg);
Debug(strbuf,TRUE,TRUE);
return( FALSE );
}

sqlglm() 函数把错误消息写入 msg, 然后仅仅要把 msg 用 Debug() 函数写入日志就能够了. sqlglm() 函数对字数也有限制, 最大为512 个字符. 相比 sqlerrmc 的70 个字符多非常多了.

版权声明:本文博客原创文章,博客,未经同意,不得转载。

RTP 记录 log 该机制的更多相关文章

  1. 简单搭建syslog-ng server记录log

    ### 简单搭建syslog-ng server记录log 安装syslog-ng apt-get install syslog-ng 安装syslog-ng 配置syslog-ng vim /etc ...

  2. 基于Java+Selenium的WebUI自动化测试框架(三)------记录LOG

    在有了Position类和接口类之后,我们是不是立刻就要着手开始写实现类了呢?按照一般的顺序是这样.但是,我们这里先停一下.原因有二: 1)既然是写一个框架,我们希望总体的功能上是全面的.实现类中,我 ...

  3. C# 添加Log文件、记录Log

    其实在平时的开发过程中都是不怎么写log的,觉得在debug中能看得一清二楚.同事小姐姐前辈,一直就我不写log进行批判,但是我从来不改,哈哈.也算是遇到报应了,在最近一个工程里,本地调试一切正常,到 ...

  4. RTP/RTCP的时间同步机制

    转自:http://blog.csdn.net/leesphone/article/details/5571972 RTP支持传送不同codec的steaming,不同codec的clock rate ...

  5. Unity3D Log 收集机制

    最近做项目的时候发现,需要有一个完整的log机制.这样不仅方便调试而且方便观察. 一.需求 目前我认为一个完善的log机制应该是这样的. 一.双击定位 二.生命周期是全局的 三.输出包括consloe ...

  6. android log写入机制

    这几天和华为的leader面试了下.感觉不错.关键是小女.不容易.是技术面啊.我说的不容易不是面试不容易,是说在华为写代码的小女不容易.哥走南闯北这么多年,女人代码写的好真不多. 其实在任何时候,只要 ...

  7. Hadoop记录-Federation联邦机制

    在Hadoop2.0之前,HDFS的单NameNode设计带来诸多问题:  单点故障.内存受限,制约集群扩展性和缺乏隔离机制(不同业务使用同一个NameNode导致业务相互影响)等 为了解决这些问题, ...

  8. JAVA记录-redis缓存机制介绍(一)

    1.redis介绍 Redis 简介 Redis 是完全开源免费的,遵守BSD协议,是一个高性能的key-value数据库. Redis 与其他 key - value 缓存产品有以下三个特点: Re ...

  9. 在Go语言中记录log:seelog包

    前两周调bug调的吐血,虽然解决了但是还是挺浪费时间的.跟同事聊了聊,觉得我们现在项目中的日志记录太少了,导致出了问题不知道怎么下手,还得自己改代码记录日志,然后排查问题.这样如果将来还有bug的话还 ...

随机推荐

  1. android 4.0 中出错 java.lang.UnsupportedOperationException

    在android4.0中  画图的时候使用: canvas.clipPath(path, Region.Op.XOR); 报错 java.lang.UnsupportedOperationExcept ...

  2. WMI 获取硬件信息的封装函数与获取联想台式机的出厂编号方法

    原文:WMI 获取硬件信息的封装函数与获取联想台式机的出厂编号方法 今天玩了一把WMI,查询了一下电脑的硬件信息,感觉很多代码都是可以提取出来的,就自己把那些公共部分提出出来,以后如果要获取 某部分的 ...

  3. Unity3D方法来隐藏和显示对象

    Unity3D作 在使用unity3d开发游戏的过程中.我们经常会遇到须要隐藏或者显示的操作,针对这一点,以下做了一些总结. 一.设置Renderer状态 在游戏的开发中,全部可以被渲染的物体都包括有 ...

  4. Java线程学习笔记(一个)

    一个.正在创建的线程: 老掉牙的话题了.继承 java.lang.Thread父类或者实现Runnalbe接口.这里就提一句: class Thread implements Runnable Thr ...

  5. UVA 439 Knight Moves(BFS)

    Knight Moves option=com_onlinejudge&Itemid=8&category=11&page=show_problem&problem=3 ...

  6. bat(批处理文件)初步 第一篇 基本符号

    近期我使用的一款软件中须要大量的环境变量设置,而我又不想讲这些变量都加入到系统的环境变量中,一方面是由于有一些同名的库文件的版本号却不一样,都 写在系统环境中会相互干扰:还有一方面则是大部分的路径仅仅 ...

  7. 第十二章——SQLServer统计信息(4)——在过滤索引上的统计信息

    原文:第十二章--SQLServer统计信息(4)--在过滤索引上的统计信息 前言: 从2008开始,引入了一个增强非聚集索引的新功能--过滤索引(filter index),可以使用带有where条 ...

  8. c语言实现hashtable,相似C++的map和iOS的NSDictionary

    跟线性数组和链表不同.HashTable是高速查找的数据结构.本文中的HashTable使用链表处理数组. 该HashTable能够指定table的长度.提供了遍历的方法. 包含table的长度的选择 ...

  9. 设计与实现简单而经常使用的权限系统(四):无需维护level,递归构建树

    第三篇中.我们通过维护节点的深度level,通过迭代全部的节点,仅仅须要一次,就构造了树.  本篇.换一种方式. 优点是:不维护节点的深度level,添加和改动节点时,也不用维护.递归实现,代码比較清 ...

  10. web自定义炫酷字体

    电脑有已经安装好的字体,但是如果你有特殊需要而要选择其他字体,则需要以下几个步骤 1.寻找适合你的字体 有下面几个站点提供字体下载: www.theleagueofmoveabletype.com w ...