手机自动化测试:Appium源码分析之跟踪代码分析八

 

poptest是国内唯一一家培养测试开发工程师的培训机构,以学员能胜任自动化测试,性能测试,测试工具开发等工作为目标。如果对课程感兴趣,请大家咨询qq:908821478。

lib/server/helpers.js模块, 研究之前,我们不防猜测一下这个模块的作用,然后在研究完成时在总结里面重新定义一下这个模块的作用。我猜测这个模块的作用就是提供了一些独立的方法,作为一些工具方法供其他模块使用

加载其他模块

var _ = require("underscore")

, gridRegister = require('./grid-register.js')

, logger = require('./logger.js').get('appium')

, status = require('./status.js')

, io = require('socket.io')

, mkdirp = require('mkdirp')

, bytes = require('bytes')

, domain = require('domain')

, format = require('util').format

, Args = require("vargs").Constructor;

模块

意义

gridRegister

暂时不解释,等我学习到的时候再来解释

logger

本地模块,这个模块已经说烂了,简单的来说提供日志输出的

status

本地模块,请求返回码定义模块

socket.io

是跨平台,多种连接方式自动切换,做即时通讯方面的开发很方便

mkdirp

扩展模块,mkdir -p变异体,提供了创建目录的方法。

bytes

公共模块,转化字符串的bytes值到数字型(比如1TB转换成1099511627776)

domain

捕捉异步回调中出现的异常

util

核心模块,提供工具方法的模块

vargs

参数模块

函数

allowCrossDomain

说这个函数之前,我们先来看看CORS协议

module.exports.allowCrossDomain = function (req, res, next) {

safely(req, function () {

//定义了资源能被所有域使用

res.header('Access-Control-Allow-Origin', '*');

//指示存取資源所允許的方法,用來回應先導請求

res.header('Access-Control-Allow-Methods', 'GET,POST,PUT,OPTIONS,DELETE');

//指示那些HTTP header可以出現在真實請求,用於回應先導請求

res.header('Access-Control-Allow-Headers', 'origin, content-type, accept');

});

// need to respond 200 to OPTIONS

//如果接受的方法是OPTIONS方法,会返回200状态码

if ('OPTIONS' === req.method) {

safely(req, function () {

res.sendStatus(200);

});

} else {

//然后调用传入的next函数

next();

}

};

winstonStream

module.exports.winstonStream = {

write: function (msg) {

msg = msg.replace(/$\s*$/m, "");

msg = msg.replace(/\[[^\]]+\] /, "");

logger.log('debug', msg);

}

};

这个是函数么?不太清楚,我只能暂时理解function(msg)内部的东西(这里其实是一个json字符串,只包含一个write元素,它对应了一个处理函数) 
将msg经过2层过滤,去除一些特殊字符,然后打印出来

catchAllHandler

module.exports.catchAllHandler = function (e, req, res, next) {

safely(req, function () {

res.status(500).send({

status: status.codes.UnknownError.code

, value: "ERROR running Appium command: " + e.message

});

});

next(e);

};

这个方法其实就是简单的在函数调用前,添加一个函数。我们要执行的函数是next,而在next函数执行执行的函数是safely,这个safely我们在Appium源码分析(6)-responses模块解释过。其实就是调用了传入的嵌套函数,将response的状态码设置为500,然后发送一个json字符串。

checkArgs

checkArgs的函数分几部分分析

//检查args的有效参数

module.exports.checkArgs = function (parser, args) {

//定义一个二维数组

var exclusives = [

['noReset', 'fullReset']

, ['ipa', 'safari']

, ['app', 'safari']

, ['forceIphone', 'forceIpad']

, ['deviceName', 'defaultDevice']

];

//遍历二维数组exclusives,遍历的元素的设置为exSet,为一维数组

_.each(exclusives, function (exSet) {

var numFoundInArgs = 0;

//遍历一维数组,遍历的元素设置为opt

_.each(exSet, function (opt) {

if (_.has(args, opt) && args[opt]) {

//如果args含有exclusives定义的参数,numFoundInArgs加1

numFoundInArgs++;

}

});

if (numFoundInArgs > 1) {

//如果超过一个参数,打印错误信息

console.error(("You can't pass in more than one argument from the set " +

JSON.stringify(exSet) + ", since they are mutually exclusive").red);

process.exit(1);

}

});

checkValidPort

checkArgs嵌套函数:

//检查端口的有效性

var checkValidPort = function (port) {

//限定端口号范围为(0,6536)

if (port > 0 && port < 65536) return true;

//超出范围,报错,返回false

console.error("Port must be greater than 0 and less than 65536");

return false;

};

 

validations字符串

checkArgs函数中定义的json字符串,定义了一些变量和对应的处理函数:

var validations = {

port: checkValidPort

, callbackPort: checkValidPort

, bootstrapPort: checkValidPort

, selendroidPort: checkValidPort

, chromedriverPort: checkValidPort

, robotPort: checkValidPort

, backendRetries: function (r) { return r >= 0; }

};

checkValidPort函数我们刚讲过,就不多说了。json字符串的最后一个元素,指定了一个匿名函数,就是判断传入的参数是否非负。

检查非默认参数的有效性

//解析出非默认参数的参数,是一个json字符串

var nonDefaultArgs = getNonDefaultArgs(parser, args);

//遍历validations字符串,元素定义为map的key值为args,value值为validator

_.each(validations, function (validator, arg) {

//是否为非默认的参数

if (_.has(nonDefaultArgs, arg)) {

//检查是否通过

if (!validator(args[arg])) {

//不通过检查的打印错误信息并退出

console.error("Invalid argument for param " + arg + ": " + args[arg]);

process.exit(1);

}

}

});

到此checkArgs函数解释完毕,看得出来这个函数的工作很多,但是总结下来就是:检查一些非默认参数的有效性。

getNonDefaultArgs

这个函数是找到非默认参数的,参照物就是parser.js模块中定义的一系列参数,我们来先看看parser.js模块:

var args = [

[['--shell'], {

required: false

, defaultValue: null

, help: 'Enter REPL mode'

, nargs: 0

}],

[['--localizable-strings-dir'], {

required: false

, dest: 'localizableStringsDir'

, defaultValue: 'en.lproj'

, help: 'IOS only: the relative path of the dir where Localizable.strings file resides '

, example: "en.lproj"

}],

[['--app'], {

看一下就知道了,一会我们会用到这个。

//得到非默认的参数

var getNonDefaultArgs = function (parser, args) {

var nonDefaults = {};

//遍历parser中定义的参数,遍历的元素设置为rawArg

_.each(parser.rawArgs, function (rawArg) {

//取每一个rawArg的第二个元素的值,取得dest属性值

var arg = rawArg[1].dest;

if (args[arg] !== rawArg[1].defaultValue) {

//如果args相同字段对应的值不是我们默认值,将该值添加到nonDefaults,以属性值为下标。

nonDefaults[arg] = args[arg];

}

});

return nonDefaults;

};

上面的解释用到parser.js中json字符串args,该args是由多个json字符串组成,所以rawArg[1]取的就是其中遍历到的json字符串的值,比如第二次便利的时候,rawArg的值为:

[['--localizable-strings-dir'], {

required: false

, dest: 'localizableStringsDir'

, defaultValue: 'en.lproj'

, help: 'IOS only: the relative path of the dir where Localizable.strings file resides '

, example: "en.lproj"

}]

那么rawArg[1]指的就是:

{

required: false

, dest: 'localizableStringsDir'

, defaultValue: 'en.lproj'

, help: 'IOS only: the relative path of the dir where Localizable.strings file resides '

, example: "en.lproj"

}

那么rawArg[1].defaultValue的值自然就是en.lproj

noColorLogger

module.exports.noColorLogger = function (tokens, req, res) {

//得到response的Header部分,属性为Content-Length的值,将其转化为10进制的整数

var len = parseInt(res.getHeader('Content-Length'), 10);

//如果len不是整形,赋值为空字符串,如果是整形,在后面追加横岗(-)和字节数

len = isNaN(len) ? '' : ' - ' + bytes(len);

//组装字符串返回,该字符串有请求的方法,url地址,状态码,时间,毫秒单位,长度

return req.method + ' ' + req.originalUrl + ' ' +

res.statusCode + ' ' + (new Date() - req._startTime) + 'ms' + len;

};

configureServer

该方法在main.js模块中有调用。

module.exports.configureServer = function (rawConfig, appiumVer, appiumServer,

cb) {

//定义一个变量

var appiumRev;

//判断配置数据是否定义,未定义的话就直接报错,cb代表回调函数callback

if (!rawConfig) {

return cb(new Error('config data required'));

}

//定义一个空json字符串

var versionMismatches = {};

//定义一个数组excludedKeys并赋值

var excludedKeys = ["git-sha", "node_bin", "built"];

//遍历rawConfig

_.each(rawConfig, function (deviceConfig, key) {

//如果配置的元素中,版本的信息不等于appium的版本,而且属性的name值不在excludedKeys内,说明

//这个配置项是不匹配的,需要保存在versionMismatches中。

if (deviceConfig.version !== appiumVer && !_.contains(excludedKeys, key)) {

versionMismatches[key] = deviceConfig.version;

} else if (key === "git-sha") {

//如果上面的条件不满足,但是key值等于git-sha,将appiumRev的值设置为rawConfig中的git-sha指代的值

appiumRev = rawConfig['git-sha'];

}

});

//keys为遍历所有json字符串中的key值,组成一个数组,判断该数组的长度

if (_.keys(versionMismatches).length) {

//如果不匹配的配置项的个数大于0,输出一些错误的提示信息

logger.error("Got some configuration version mismatches. Appium is " +

"at " + appiumVer + ".");

_.each(versionMismatches, function (mismatchedVer, key) {

logger.error(key + " configured at " + mismatchedVer);

});

logger.error("Please re-run reset.sh or config");

return cb(new Error("Appium / config version mismatch"));

} else {

//如果配置都是正确的,调用registerConfig开始设置

appiumServer.registerConfig(rawConfig);

cb(null, appiumRev);

}

};

conditionallyPreLaunch

//预加载模式

module.exports.conditionallyPreLaunch = function (args, appiumServer, cb) {

//判断args.launch的属性是否为true

if (args.launch) {

logger.debug("Starting Appium in pre-launch mode");

//调用appium.js模块的预加载函数preLaunch,传入的参数为一个回调函数

appiumServer.preLaunch(function (err) {

if (err) {

logger.error("Could not pre-launch appium: " + err);

cb(err);

} else {

cb(null);

}

});

} else {

cb(null);

}

};

prepareTmpDir

//创建临时目录

module.exports.prepareTmpDir = function (args, cb) {

if (args.tmpDir === null) return cb();

//调用mkdirp模块的的方法创建目录

mkdirp(args.tmpDir, function (err) {

if (err) {

logger.error("Could not ensure tmp dir '" + args.tmpDir + "' exists");

logger.error(err);

}

cb(err);

});

};

startAlertSocket

var startAlertSocket = function (restServer, appiumServer) {

var alerts = io(restServer, {

'flash policy port': -1,

'logger': logger,

'log level': 1,

'polling duration': 10,

'transports': ['websocket', 'flashsocket']

});

//连接服务器,回调函数为连接后进行的处理

alerts.sockets.on("connection", function (socket) {

logger.debug("Client connected: " + (socket.id).toString());

//监听disconnect事件,当断开连接后,调用回调函数

socket.on('disconnect', function (data) {

logger.debug("Client disconnected: " + data);

});

});

// add web socket so we can emit events

//将该alerts时间添加到web socket中

appiumServer.attachSocket(alerts);

};

getDeprecatedArgs

 

//得到废弃的参数

var getDeprecatedArgs = function (parser, args) {

var deprecated = {};

//遍历parser中定义的参数,设置遍历的元素为rawArg

_.each(parser.rawArgs, function (rawArg) {

//获取元素的dest的值

var arg = rawArg[1].dest;

//如果dest指代的值存在,且rawArg[1]的属性deprecatedFor也存在

//将该值添加到json字符串deprecated中

if (args[arg] && rawArg[1].deprecatedFor) {

deprecated[rawArg[0]] = "use instead: " + rawArg[1].deprecatedFor;

}

});

return deprecated;

};

startListening

module.exports.startListening = function (server, args, parser, appiumVer, appiumRev, appiumServer, cb) {

//声明变量alreadyReturned并赋值为false

var alreadyReturned = false;

//监听某个url下的某个端口的消息,回调函数为连接成功后处理函数

server.listen(args.port, args.address, function () {

//欢迎信息,这些信息相信用过appium的人都见过,首先打印欢迎信息

var welcome = "Welcome to Appium v" + appiumVer;

//如果git-sha的赋值给了appiumRev,就将其追加到欢迎信息中

if (appiumRev) {

welcome += " (REV " + appiumRev + ")";

}

//打印welcome信息

logger.info(welcome);

var logMessage = "Appium REST http interface listener started on " +

args.address + ":" + args.port;

//打印ip地址和端口号

logger.info(logMessage);

//调用startAlertSocket启动连接socket

startAlertSocket(server, appiumServer);

if (args.nodeconfig !== null) {

//如果node配置信息不为null,那么就配置node信息

gridRegister.registerNode(args.nodeconfig, args.address, args.port);

}

var showArgs = getNonDefaultArgs(parser, args);

if (_.size(showArgs)) {

//如果非默认参数的个数大于0,需要打印出来

logger.debug("Non-default server args: " + JSON.stringify(showArgs));

}

var deprecatedArgs = getDeprecatedArgs(parser, args);

if (_.size(deprecatedArgs)) {

//如果有废弃的参数,也需要打印出来

logger.warn("Deprecated server args: " + JSON.stringify(deprecatedArgs));

}

//这个应该很熟悉,我们经常启动的时候就能看见这个,输出log的等级,大于等于这个等级的log才会输出

logger.info('Console LogLevel: ' + logger.transports.console.level);

//文件log的等级,大于等于这个等级的信息才会保存

if (logger.transports.file) {

logger.info('File LogLevel: ' + logger.transports.file.level);

}

});

//监听error事件

server.on('error', function (err) {

if (err.code === 'EADDRNOTAVAIL') {

logger.error("Couldn't start Appium REST http interface listener. Requested address is not available.");

} else {

logger.error("Couldn't start Appium REST http interface listener. Requested port is already in use. Please make sure there's no other instance of Appium running already.");

}

if (!alreadyReturned) {

alreadyReturned = true;

cb(err);

}

});

//设置超时连接时间为10分钟

server.on('connection', function (socket) {

socket.setTimeout(600 * 1000); // 10 minute timeout

});

//延迟任务,在1秒后执行函数

setTimeout(function () {

if (!alreadyReturned) {

alreadyReturned = true;

cb(null);

}

}, 1000);

};

调用该方法,一般会打印如下信息:

nfo: Welcome to Appium v1.3.7 (REV 72fbfaa116d3d9f6a862600ee99cf02f6d0e2182)

info: Appium REST http interface listener started on 0.0.0.0:4723

info: [debug] Non-default server args: {"platformName":"Android","platformVersion":"4.4","automationName":"Appium","defaultCommandTimeout":7200}

info: Console LogLevel: debug

compile

function compile(fmt) {

fmt = fmt.replace(/"/g, '\\"');

var js = '  return "' + fmt.replace(/:([-\w]{2,})(?:\[([^\]]+)\])?/g,

function (_, name, arg) {

return '"\n    + (tokens["' + name + '"](req, res, "' + arg + '") || "-") + "';

}) + '";';

// jshint evil:true

return new Function('tokens, req, res', js);

}

这个函数是设置打印的字符串格式

requestStartLoggingFormat

//一次请求的开始,输出的log格式,格式类似 info: --> GET /wd/hub/status {}
module.exports.requestStartLoggingFormat = compile('-->'.white + ' ' + ':method'.white + ' ' +
  ':url'.white);
 
requestEndLoggingFormat
//一次请求的结束,输出log格式,格式类似:info: <-- GET /wd/hub/status 200 4.102 ms - 104 
//{"status":0,"value":{"build":{"version":"1.3.7","revision":"72fbfaa116d3d9f6a862600ee99cf02f6d0e2182"}}}
module.exports.requestEndLoggingFormat = function (tokens, req, res) {
  var status = res.statusCode;
  var statusStr = ':status';
  //状态码大于500的话,状态码为红色
  if (status >= 500) statusStr = statusStr.red;
  //状态码大于400小于500,状态码为黄色
  else if (status >= 400) statusStr = statusStr.yellow;
  //状态码大于300小于400,状态码为蓝绿色
  else if (status >= 300) statusStr = statusStr.cyan;
  //小于300,状态码为绿色
  else statusStr = statusStr.green;
  var fn = compile('<-- :method :url '.white + statusStr +
    ' :response-time ms - :res[content-length]'.grey);
  return fn(tokens, req, res);
};
 
getRequestContext

//得到请求的内容

function getRequestContext(req) {

//如果req未定义,直接返回空字符串

if (!req) return '';

var data = '';

try {

//截取req的body部分,索引0到200的字符串

if (req.body) data = JSON.stringify(req.body).substring(0, 200);

} catch (ign) {}

return format('context: [%s %s %s]', req.method, req.url, data).replace(/ ]$/, '');

}

safely

var safely = function () {

//获取传递进来的参数

var args = new (Args)(arguments);

//获得第一个参数

var req = args.all[0];

//回调函数

var fn = args.callback;

try {

//调用回调函数

fn();

} catch (err) {

logger.error('Unexpected error:', err.stack, getRequestContext(req));

}

};

module.exports.safely = safely;

domainMiddleware

module.exports.domainMiddleware = function () {

return function (req, res, next) {

//创建一个域记录

var reqDomain = domain.create();

//将req和res添加到域管理

reqDomain.add(req);

reqDomain.add(res);

//res监听close事件

res.on('close', function () {

//延迟事件,5秒后关闭域

setTimeout(function () {

reqDomain.dispose();

}, 5000);

});

//reqDomain添加error事件

reqDomain.on('error', function (err) {

logger.error('Unhandled error:', err.stack, getRequestContext(req));

});

//执行next函数

reqDomain.run(next);

};

};

手机自动化测试:Appium源码分析之跟踪代码分析八的更多相关文章

  1. 手机自动化测试:Appium源码分析之跟踪代码分析九

    手机自动化测试:Appium源码分析之跟踪代码分析九   poptest是国内唯一一家培养测试开发工程师的培训机构,以学员能胜任自动化测试,性能测试,测试工具开发等工作为目标.如果对课程感兴趣,请大家 ...

  2. 手机自动化测试:Appium源码分析之跟踪代码分析七

    手机自动化测试:Appium源码分析之跟踪代码分析七   poptest是国内唯一一家培养测试开发工程师的培训机构,以学员能胜任自动化测试,性能测试,测试工具开发等工作为目标.poptest推出手机自 ...

  3. 手机自动化测试:Appium源码分析之跟踪代码分析六

    手机自动化测试:Appium源码分析之跟踪代码分析六   poptest是国内唯一一家培养测试开发工程师的培训机构,以学员能胜任自动化测试,性能测试,测试工具开发等工作为目标.poptest推出手机自 ...

  4. 手机自动化测试:Appium源码分析之跟踪代码分析五

    手机自动化测试:Appium源码分析之跟踪代码分析五   手机自动化测试是未来很重要的测试技术,作为一名测试人员应该熟练掌握,POPTEST举行手机自动化测试的课程,希望可以训练出优秀的手机测试开发工 ...

  5. 手机自动化测试:appium源码分析之bootstrap三

    手机自动化测试:appium源码分析之bootstrap三   研究bootstrap源码,我们可以通过代码的结构,可以看出来appium的扩展思路和实现方式,从中可以添加我们自己要的功能,针对app ...

  6. 手机自动化测试:appium源码分析之bootstrap二

    手机自动化测试:appium源码分析之bootstrap二   在bootstrap项目中的io.appium.android.bootstrap.handler包中的类都是对应的指令类, priva ...

  7. 手机自动化测试:appium源码分析之bootstrap一

    手机自动化测试:appium源码分析之bootstrap一   前言: poptest是国内唯一一家培养测试开发工程师的培训机构,以学员能胜任自动化测试,性能测试,测试工具开发等工作为目标.popte ...

  8. 手机自动化测试:appium源码分析之bootstrap十七

    手机自动化测试:appium源码分析之bootstrap十七   poptest是国内唯一一家培养测试开发工程师的培训机构,以学员能胜任自动化测试,性能测试,测试工具开发等工作为目标.如果对课程感兴趣 ...

  9. 手机自动化测试:appium源码分析之bootstrap十六

    手机自动化测试:appium源码分析之bootstrap十六   poptest是国内唯一一家培养测试开发工程师的培训机构,以学员能胜任自动化测试,性能测试,测试工具开发等工作为目标.如果对课程感兴趣 ...

随机推荐

  1. EF CodeFirst下数据库更新

    用EF Code first模式来开发系统,可使用Migrations命令来让数据库自动更新 1.在VS->工具->库程序包管理器->程序包管理控制台 中执行 Enable-Migr ...

  2. 第22篇 js中的this指针的用法

    前面把js的相关知识总结了下,今天把js中的上下文的this,对于强类型语言,this的用法非常的单一,因为他们没有js特有的动态绑定. 首先看下面代码: function funcA() { thi ...

  3. 第八篇 一个用JS写的省市县三级联动

    前些天,做网站用需要用到一个省市县的三级联动,数据要从数据库里面读取,我想了下思路,动手写了下来.    一.思路           js利用Ajax读取控制器里面的函数,利用函数读取存储过程,返回 ...

  4. Asp.Net MVC学习总结(三)——过滤器你怎么看?

    一.过滤器简介 1.1.理解什么是过滤器 1.过滤器(Filters)就是向请求处理管道中注入额外的逻辑.提供了一个简单而优雅的方式来实现横切关注点. 2.所谓的过滤器(Filters),MVC框架里 ...

  5. php与mysql之间操作原理

    php和mysql相关扩展有:mysql.mysqli和pdo三种 mysql扩展从php5.5.0被废弃,并且从从php7.0.0开始被废除 mysql之前的使用---几个基本的函数:mysql_c ...

  6. (原创)我对未来的人类的发展,以及AI技术发展的一些思考。

    最近AI非常的火,不仅仅是阿尔法狗的成功,因为它击败了人类最强的大脑,颠覆了人类几千年来的对传统的认识,也让人类意识 到了一个问题:天外有天,人外有AI. 那么AI究竟会对人类的未来造成什么深远的影响 ...

  7. 消消乐、candy crush类三消游戏程序逻辑分析

    最近在开发一款类似消消乐的三消游戏,在碰到实现斜方向下落的时候卡住了很长时间.好几天没有思路,原本的思路是一次性预判多个宝石的一连串运动路径,运用缓动运动队列来实现宝石运动路径,例如 下落->滑 ...

  8. 转换器4:手写PHP转Python编译器,语法解析部分

    写完词法部分,又有很多杂事,周末终于有空来实现伟大的语法解析部分了. 撸完代码之后发现,程序太短了,不算上状态机,才186行(含注释),关键代码不到100行.运行调试过后,发现还行.居然可以解析One ...

  9. Servlet 学习简介

    一.Servlet简介 Java Servlet 是运行在 Web 服务器或应用服务器上的程序,它是作为来自 Web 浏览器或其他 HTTP 客户端的请求和 HTTP 服务器上的数据库或应用程序之间的 ...

  10. 记 suds 模块循环依赖的坑-RuntimeError: maximum recursion depth exceeded

    下面是soa接口调用的核心代码 #! /usr/bin/python # coding:utf-8 from suds.client import Clientdef SoaRequest(wsdl, ...