手机自动化测试:Appium源码分析之跟踪代码分析九
手机自动化测试:Appium源码分析之跟踪代码分析九
poptest是国内唯一一家培养测试开发工程师的培训机构,以学员能胜任自动化测试,性能测试,测试工具开发等工作为目标。如果对课程感兴趣,请大家咨询qq:908821478。
// set up distributed logging before everything else
var npmlog = global._global_npmlog = require('npmlog');
// npmlog is used only for emitting, we use winston for output
//不利用npmlog输出信息,而是利用winston来输出,所以将npmlog的等级设置为silent
npmlog.level = "silent";
//日志管理系统
var winston = require('winston')
//文件操作模块
, fs = require('fs')
//获取系统信息的模块
, os = require('os')
//处理和转换文件路径
, path = require('path')
//工具模块
, util = require('util');
//日期处理模块
require('date-utils');
//log等级定义
var levels = {
debug: 1
, info: 2
, warn: 3
, error: 4
};
//各等级对应的字体颜色
var colors = {
info: 'cyan'
, debug: 'grey'
, warn: 'yellow'
, error: 'red'
};
//定义log等级对应关系的字典
var npmToWinstonLevels = {
silly: 'debug'
, verbose: 'debug'
, info: 'info'
, http: 'info'
, warn: 'warn'
, error: 'error'
};
var logger = null;
//时区
var timeZone = null;
//堆栈
var stackTrace = null;
// capture any logs emitted by other packages using our global distributed
// logger and pass them through winston
npmlog.on('log', function (logObj) {
//根据传入的参数,得到对应的log等级,如果参数未被定义过,也就是说在字典中未找到,那么就设置为info
var winstonLevel = npmToWinstonLevels[logObj.level] || 'info';
//获得消息体
var msg = logObj.message && logObj.prefix ?
(logObj.prefix + ": " + logObj.message) :
(logObj.prefix || logObj.message);
logger[winstonLevel](msg);
console.log("==========npmlog.on=================");
});
//将当前日期转化为YYYY-MM-DD HH24:MI:SS:LL形式
var timestamp = function () {
var date = new Date();
if (!timeZone) {
//date.getTimezoneOffset()时差,精确到分钟,乘以60000,精确到毫秒数
date = new Date(date.valueOf() + date.getTimezoneOffset() * 60000);
}
return date.toFormat("YYYY-MM-DD HH24:MI:SS:LL");
};
// Strip the color marking within messages.
// We need to patch the transports, because the stripColor functionality in
// Winston is wrongly implemented at the logger level, and we want to avoid
// having to create 2 loggers.
//解决winston在日志实现上bug,有时候会产生2个logger器
function applyStripColorPatch(transport) {
var _log = transport.log.bind(transport);
transport.log = function (level, msg, meta, callback) {
var code = /\u001b\[(\d+(;\d+)*)?m/g;
msg = ('' + msg).replace(code, '');
_log(level, msg, meta, callback);
};
}
//将log信息传输到控制台上
var _createConsoleTransport = function (args, logLvl) {
var transport = new (winston.transports.Console)({
name: "console"
, timestamp: args.logTimestamp ? timestamp : undefined
, colorize: !args.logNoColors
, handleExceptions: true
, exitOnError: false
, json: false
, level: logLvl
});
if (args.logNoColors) applyStripColorPatch(transport);
return transport;
};
//将log信息传输到文件中
var _createFileTransport = function (args, logLvl) {
var transport = new (winston.transports.File)({
name: "file"
, timestamp: timestamp
, filename: args.log
, maxFiles: 1
, handleExceptions: true
, exitOnError: false
, json: false
, level: logLvl
}
);
applyStripColorPatch(transport);
return transport;
};
//将log信息传输到webhook中,webhook就是将log信息传输到某一个指定url上,将log信息传输到web服务器上
var _createWebhookTransport = function (args, logLvl) {
var host = null,
port = null;
if (args.webhook.match(':')) {
var hostAndPort = args.webhook.split(':');
host = hostAndPort[0];
port = parseInt(hostAndPort[1], 10);
}
var transport = new (winston.transports.Webhook)({
name: "webhook"
, host: host || '127.0.0.1'
, port: port || 9003
, path: '/'
, handleExceptions: true
, exitOnError: false
, json: false
, level: logLvl
});
applyStripColorPatch(transport);
return transport;
};
//管理分配上面三种log表现形式:控制台,文件,web。其中控制台是必须的,文件和web是根据配置来产生的
var _createTransports = function (args) {
var transports = [];
var consoleLogLevel = null,
fileLogLevel = null;
if (args.loglevel && args.loglevel.match(":")) {
// --log-level arg can optionally provide diff logging levels for console and file separated by a colon
var lvlPair = args.loglevel.split(':');
consoleLogLevel = lvlPair[0] || consoleLogLevel;
fileLogLevel = lvlPair[1] || fileLogLevel;
} else {
consoleLogLevel = fileLogLevel = args.loglevel;
}
transports.push(_createConsoleTransport(args, consoleLogLevel));
if (args.log) {
try {
// if we don't delete the log file, winston will always append and it will grow infinitely large;
// winston allows for limiting log file size, but as of 9.2.14 there's a serious bug when using
// maxFiles and maxSize together. https://github.com/flatiron/winston/issues/397
if (fs.existsSync(args.log)) {
fs.unlinkSync(args.log);
}
transports.push(_createFileTransport(args, fileLogLevel));
} catch (e) {
console.log("Tried to attach logging to file " + args.log +
" but an error occurred: " + e.msg);
}
}
if (args.webhook) {
try {
transports.push(_createWebhookTransport(args, fileLogLevel));
} catch (e) {
console.log("Tried to attach logging to webhook at " + args.webhook +
" but an error occurred. " + e.msg);
}
}
return transports;
};
var _appDir = path.dirname(require.main.filename);
//将堆栈信息转化为字符串
var _stackToString = function (stack) {
var str = os.EOL + " [------TRACE------]" + os.EOL;
var len = stack.length < 15 ? stack.length : 15;
for (var i = 0; i < len; i++) {
var fileName = stack[i].getFileName();
// ignore calls from this file
if (fileName === __filename) continue;
var substr = " at ";
try {
var typeName = stack[i].getTypeName();
substr += util.format("%s.%s (%s:%d:%d)" + os.EOL, typeName, stack[i].getFunctionName(),
path.relative(_appDir, stack[i].getFileName()), stack[i].getLineNumber(),
stack[i].getColumnNumber());
str += substr;
} catch (e) { }
}
return str;
};
var _addStackTrace = function (fn, stackTrace) {
var _fn = fn;
return function (msg) {
//os.EOL为行结束符
_fn(msg + os.EOL + _stackToString(stackTrace.get()) + os.EOL);
console.log("==========_addStackTrace=================");
};
};
//log系统初始化
module.exports.init = function (args) {
// set de facto param passed to timestamp function
timeZone = args.localTimezone;
// by not adding colors here and not setting 'colorize' in transports
// when logNoColors === true, console output is fully stripped of color.
if (!args.logNoColors) {
winston.addColors(colors);
}
//初始化log器,准备控制台输出(文件和webhook都准备好)
logger = new (winston.Logger)({
transports: _createTransports(args)
});
//设置log等级
logger.setLevels(levels);
// 8/19/14 this is a hack to force Winston to print debug messages to stdout rather than stderr.
// TODO: remove this if winston provides an API for directing streams.
//将debug的信息转化为info等级的lgo
if (levels[logger.transports.console.level] === levels.debug) {
logger.debug = function (msg) { logger.info('[TesterHome] ' + msg); };
}
//如果允许同步堆栈信息,就要在相应的log输出信息中追加这些信息
if (args.asyncTrace) {
stackTrace = require('stack-trace');
//将堆栈信息追加到等级为info的log信息中,后面两个类似
logger.info = _addStackTrace(logger.info, stackTrace);
logger.warn = _addStackTrace(logger.warn, stackTrace);
logger.error = _addStackTrace(logger.error, stackTrace);
}
};
//以get函数对外提供
module.exports.get = function () {
if (logger === null) {
exports.init({});
}
return logger;
};
手机自动化测试:Appium源码分析之跟踪代码分析九的更多相关文章
- 手机自动化测试:Appium源码分析之跟踪代码分析八
手机自动化测试:Appium源码分析之跟踪代码分析八 poptest是国内唯一一家培养测试开发工程师的培训机构,以学员能胜任自动化测试,性能测试,测试工具开发等工作为目标.如果对课程感兴趣,请大家 ...
- 手机自动化测试:Appium源码分析之跟踪代码分析七
手机自动化测试:Appium源码分析之跟踪代码分析七 poptest是国内唯一一家培养测试开发工程师的培训机构,以学员能胜任自动化测试,性能测试,测试工具开发等工作为目标.poptest推出手机自 ...
- 手机自动化测试:Appium源码分析之跟踪代码分析六
手机自动化测试:Appium源码分析之跟踪代码分析六 poptest是国内唯一一家培养测试开发工程师的培训机构,以学员能胜任自动化测试,性能测试,测试工具开发等工作为目标.poptest推出手机自 ...
- 手机自动化测试:Appium源码分析之跟踪代码分析五
手机自动化测试:Appium源码分析之跟踪代码分析五 手机自动化测试是未来很重要的测试技术,作为一名测试人员应该熟练掌握,POPTEST举行手机自动化测试的课程,希望可以训练出优秀的手机测试开发工 ...
- 手机自动化测试:appium源码分析之bootstrap三
手机自动化测试:appium源码分析之bootstrap三 研究bootstrap源码,我们可以通过代码的结构,可以看出来appium的扩展思路和实现方式,从中可以添加我们自己要的功能,针对app ...
- 手机自动化测试:appium源码分析之bootstrap二
手机自动化测试:appium源码分析之bootstrap二 在bootstrap项目中的io.appium.android.bootstrap.handler包中的类都是对应的指令类, priva ...
- 手机自动化测试:appium源码分析之bootstrap一
手机自动化测试:appium源码分析之bootstrap一 前言: poptest是国内唯一一家培养测试开发工程师的培训机构,以学员能胜任自动化测试,性能测试,测试工具开发等工作为目标.popte ...
- 手机自动化测试:appium源码分析之bootstrap十七
手机自动化测试:appium源码分析之bootstrap十七 poptest是国内唯一一家培养测试开发工程师的培训机构,以学员能胜任自动化测试,性能测试,测试工具开发等工作为目标.如果对课程感兴趣 ...
- 手机自动化测试:appium源码分析之bootstrap十六
手机自动化测试:appium源码分析之bootstrap十六 poptest是国内唯一一家培养测试开发工程师的培训机构,以学员能胜任自动化测试,性能测试,测试工具开发等工作为目标.如果对课程感兴趣 ...
随机推荐
- object c入门
无意间看到Object C编写的程序,感觉蛮有意思的,记载下来,慢慢品味,也许会有用得上的时候.吼吼~~ 大部分有一点其他平台开发基础的初学者看到XCode,第一感想是磨拳擦掌,看到 Interfac ...
- warshall、
#include<iostream> int mian() { ][],b[][],c[][]; int i,j,k; cout<<"input the Boolea ...
- GridControl基础设置(一)
1. 如何解决单击记录整行选中的问题 View->OptionsBehavior->EditorShowMode 设置为:Click 2. 如何新增一条记录 (1).gridView.Ad ...
- Spring源码解析三:IOC容器的依赖注入
一般情况下,依赖注入的过程是发生在用户第一次向容器索要Bean是触发的,而触发依赖注入的地方就是BeanFactory的getBean方法. 这里以DefaultListableBeanFactory ...
- js:如何在循环异步请求的每次返回中添加想要的值
先看一个场景 var arr = ["a","b","c"]; for (var i in arr) { $.get(&qu ...
- 在Windows平台搭建轻巧的Python开发环境——面向工程和科研的扩展包配置
首先,下载最新版本的Python. 为什么强调最新版本呢,因为新版本的漏洞通常会少得多,而且反映了未来的趋势. 既然要学,何不起点高一点? 官方下载地址:https://www.python.org/ ...
- ROJ 1166 超级贞鱼
1166: 超级贞鱼 Time Limit: 1 Sec Memory Limit: 128 MB [Submit][Status] 传送门 Description 马达加斯加贞鱼是一种神奇的双脚贞 ...
- 【2017-03-09】SQL Server 数据库基础、四种约束
一.数据库和内存的区别 数据库:一些存储在硬盘上的数据文件 内存:计算机临时存储的一些数据 二.常用数据库 .Net - SQL Server PHP - MySql Java - Oreacl 三. ...
- 【解题报告】pojP1436 Horizontally Visible Segments
http://poj.org/problem?id=1436 题目大意:有n条平行于x轴的线段,每条线段有y坐标,如果两条线段有一段x坐标数值相等,且中间没有其它线段阻隔,则称这两条线段"照 ...
- (五)CSS和JavaScript基础
DHTML :制作动态HTML页面的技术 DHTML=HTML+层叠样式表CSS+脚本语言javascript 一.CSS 1.1 CSS样式的分类: 行内样式:只影响一行,其他相同标签也不影响.如下 ...