MockNet -- Android网络接口开发与测试神器

   <p>最近写了一个轻量级的服务器框架 MockNet,可以在 Android 上或者 Java 平台快速搭建服务器,方便开发和测试网络接口,不需要服务器知识就可以使用。</p>    <h3>写这个框架的原因</h3>    <p>平时写项目的时候,总会遇到这些情况</p>    <ol>     <li>服务器接口还没有写好,Android 客户端只能先做一些界面的开发工作,遇到需要展示数据的地方只能先留着或者用一些假数据填充,服务器接口写好以后,这些代码又得删掉或者再改。</li>     <li>在调试网络接口的时候,往往需要后台客户端一起配合,有时候会花不少时间确认是服务器还是客户端的问题,之后联调也很花时间。</li>     <li>在没有网络但是又需要展示应用功能的时候,只能用一堆 if else 来添加假数据或者数据库里的数据。代码写起来很不方便。</li>    </ol>    <p>写这个框架就是想解决这些问题。</p>    <ul>     <li>首先是网络接口开发的问题,通过 MockNet 这个库,无需后台接口完成就可以进行网络接口的开发,开发流程要更顺畅一点。</li>     <li>其次是网络接口调试的问题,通过 MockNet 这个库,在没有后台同学的帮助下也可以对 Android 客户端网络接口进行调试。</li>     <li>最后是没有网络需要展示功能的问题,通过 MockNet 这个库,没有网络一样可以展示应用的网络功能。</li>    </ul>    <h3>框架介绍</h3>    <p>MockNet 是为了方便客户端网络接口的开发和测试而写的,简单来说就是在本地启动服务器用来响应客户端的网络请求。但是不需要有服务器开发知识,几行代码就可以添加此功能。</p>    <p>代码上传到了 github 上,网址: <a href="/misc/goto?guid=4959749307234612484" rel="nofollow,noindex">https://github.com/5A59/MockNet</a></p>    <h3>库下载</h3>    <p>Gradle</p>    <pre>  <code class="language-java">// 在模块的 gradle 文件中添加下面的代码    compile 'com.zy.mocknet:mocknet:1.0'</code></pre>    <p>Maven</p>    <pre>  <code class="language-java"><dependency>    <groupId>com.zy.mocknet</groupId>    <artifactId>mocknet</artifactId>    <version>1.0</version>    <type>pom</type>  </dependency></code></pre>    <p>jar包下载</p>    <p><a href="/misc/goto?guid=4959749307342995132" rel="nofollow,noindex">mocknet_1_0_0.jar</a></p>    <h3>框架使用</h3>    <p><a href="/misc/goto?guid=4959749307435543602" rel="nofollow,noindex">使用示例</a></p>    <p>MockNet 使用简单,步骤如下:</p>    <ol>     <li>修改网络访问 ip 为本地 ip (127.0.0.1:port 或 本地真实ip:port),因为服务器是在本地搭建的。</li>     <li>初始化。 <pre>  <code class="language-java">// 创建 MockNet  MockNet mockNet = MockNet.create();</code></pre> </li>     <li>添加对请求的处理<br> MockNet 中对每一个 request 都会对应一个 response,request 和 response 和起来称为一个 MockConnection,添加对请求的处理就是添加 MockConnection 实例。 <pre>  <code class="language-java">MockConnection conn = MockConnectionFactory.getInstance()   .createGeneralConnection("/*", "general connection");  mockNet.addConnection(conn);</code></pre> createGeneralConnection(String, String) 第一个参数是请求 url,例:/test,第二个参数是返回内容。这是最简单的创建方式,更多的 MockConnection 创建方式可以参看 <a href="/misc/goto?guid=4959749307234612484" rel="nofollow,noindex">MockNet介绍</a> 或者 <a href="/misc/goto?guid=4959749307540840701" rel="nofollow,noindex">java文档</a></li>     <li>启动服务 <pre>  <code class="language-java">// 默认使用 8088 端口  mockNet.start();  // 使用指定的端口  mockNet.start(int port);</code></pre> </li>     <li>关闭服务 <pre>  <code class="language-java">mockNet.stop();</code></pre> </li>    </ol>    <p>以上就是 MockNet 的使用方法。还可以通过链式调用来写,代码更简洁。</p>    <pre>  <code class="language-java">MockNet mockNet = MockNet.create()                  .addConnection(MockConnectionFactory.getInstance()                          .createGeneralConnection("/test", "{'res':'ok'}"))                  .addConnection(MockConnectionFactory.getInstance()                          .createGeneralConnection("/*", "{'res':'ok'}"))                  .start();</code></pre>    <h3>进阶使用</h3>    <p>上面是 MockNet 的简单使用,还有更多的扩展用法。</p>    <ol>     <li> <p>自定义 MockConnection</p> <p>通过 MockConnection.Builder 生成 Builder,并通过 Builder 相关方法构建 MockConnection。Builder 常用的方法可以参看文档: <a href="/misc/goto?guid=4959749307540840701" rel="nofollow,noindex">Builder</a></p> </li>     <li> <p>随机对请求做出响应</p> <p>MockNet 内部通过 url 和 method(GET,POST等)来对请求做出区分,在 addConnection 时如果添加了多个相同的 url 和 method 的 MockConnection,会通过 IConnectionSelector 来选择其中之一进行返回,默认使用 RandomSelector 随机返回。</p> <p>可以通过 实现 IConnectionSelector 接口并通过 MockNet.setSelector() 设置返回规则。</p> </li>     <li> <p>Log 设置</p> <p>默认对每个 MockConnection 都会输出 Log 以帮助调试,如果想关闭 Log,可以在构建 MockConnection 时设置 isLog(false)。</p> <p>MockNet 的 Log 输出是由 Logger 和 Printer 完成的,默认设置了 AndroidPrinter 和 JavaPrinter,如果想自定义 Log,可以实现 Printer 接口,并调用 Logger.init(yourPrinter) 设置,但是要在调用了 MockConnection.create() 之后,否则设置会被覆盖为默认设置</p> </li>     <li> <p>自定义 Handler 对请求响应进行处理</p> <p>MockNet 中对 Request 和 Response 的处理采用了责任链模式(可查看后面的框架介绍),通过添加 Handler 增加处理环节,框架自带的 Handler 有 BlockHandler,LogHandler,VerifyHeaderHandler,VerifyParamHandler,ConnectionHandler。</p> <p>如果想增加自己的处理环节,请实现 Handler 接口,并通过 MockNet.addHandler(Handler h) 来设置。具体实现方法可参照框架默认实现的 Handler 代码。</p> </li>     <li> <p>支持 https</p> <p>支持 https 可以通过下面代码来开启:</p> <pre>  <code class="language-java">MockNet mockNet = MockNet.create();  mockNet.start(ServerSocketFactory.createHttpsServerSocket(int port, String jksPath, String storePwd));</code></pre> </li>     <li> <p>动态处理数据</p> <p>为了简化使用方法和加快开发速度,默认只支持了返回静态数据,暂时没有对数据进行动态处理。如果想动态处理请求数据,可以继承 RequestExecutor 接口并实现 execute 方法。在 execute 方法中对请求进行动态处理,并创建 Reponse 返回。</p> <p>之后通过 Server 构造函数构造 Server 对象并传入实现 RequestExecutor 接口的类对象。</p> <p>具体可以参考 MockRequestExecutor 的实现以及 Server 构造函数。</p> </li>    </ol>    <p>更多使用方法可查看 <a href="/misc/goto?guid=4959749307234612484" rel="nofollow,noindex">MockNet</a> 和 <a href="/misc/goto?guid=4959749307540840701" rel="nofollow,noindex">java文档</a></p>    <h3>框架架构</h3>    <p>上面介绍了框架的使用,这里对框架的实现做一些介绍。</p>    <p>先上一张架构图。</p>    <p><img src="https://simg.open-open.com/show/cf212307dd942c323582394c634e0a1f.png"></p>    <p>mocknet.png</p>    <p>自己认为整个架构还是不错的(-_-)。</p>    <p>整体可分为两层, Server 层和 Application 层。</p>    <p>Server 层</p>    <p>Server 层主要做的工作是 socket 通信和对 Request,Response 的解析。</p>    <p>Server 类主要是对端口的监听,接受 socket 请求并创建 RequestRunnable 处理请求。</p>    <p>RequestRunnable 创建后会加入到线程池中。RequestRunnable 类中对请求进行解析,并把请求发送到 RequestExecutor 去处理,RequestExecutor 会返回 Response,RequestRunnable 再负责把 Response 写入到 socket 中。</p>    <p>Application 层</p>    <p>Application 层工作是对 Request 的处理。通过实现 RequestExecutor 接口处理 Request 并返回 Response。</p>    <p>框架中默认实现 RequestRunnable 接口的是 MockRequestExecutor 类,用来返回静态消息,如果想动态处理请求,需要自己实现 RequestExecutor 接口,框架后续看情况也可能会增加动态处理请求的功能。</p>    <p>MockRequestExecutor 中主要采用责任链模式处理请求,每一个 Handler 都是一个处理环节。这个链式的实现是之前看过 OkHttp 代码学到的。这里贴代码来看一下。</p>    <pre>  <code class="language-java">HandlerChain chain = new RealHandlerChain();    public Response execute(Request request) {      for (Handler h : userHandlers) {          chain.addHandler(h);      }        if (initHandler) {          chain.addHandler(new BlockHandler());          chain.addHandler(new VerifyParamHandler());          chain.addHandler(new VerifyHeaderHandler());          chain.addHandler(new LogHandler());          chain.addHandler(new ConnectionHandler());      }        return chain.start(request);  }</code></pre>    <p>上面是 MockRequestExecutor 的 execute() 函数的实现,首先将 Handler 加入到 chain 中,之后调用 chain.start() 开始调用 Handler 进行处理。</p>    <p>再看一下 HandlerChain 的 start() 方法实现。</p>    <pre>  <code class="language-java">public Response start(Request request) {      Handler handler = handlers.get(index);      Response response = handler.handle(request, this, index);      return response;  }</code></pre>    <p>直接调用了 handler.handle()进行处理,在 Handler 的 handle 方法中需要调用下一个 Handler。</p>    <p>框架默认实现了几个 Handler:BlockHandler,VerifyParamHandler,VeriffyHeaderHandler,LogHandler,ConnectionHandler。</p>    <p>这些通过名字可以看出功能,这里重点看一下 ConnectionHandler。</p>    <p>ConnectionHandler 是处理链调用的终点,负责生成 Response。</p>    <p>ConnectionHandler 的 handle 方法如下:</p>    <pre>  <code class="language-java">String method = request.getMethod();  String url = request.getRequestUri();  MockConnection connection =      ConnectionStore.getInstance().getConnection(method, url);  if (connection == null) {      return Response.create404Response();  }  Response response = new Response();  // 省略对 Response 的设置  return response;</code></pre>    <p>通过请求的 url 和 method 从 ConnectionStore 中获取到 MockConnection,之后用 MockConnection 中的响应信息创建并填充 Response。</p>    <p>那 ConnectionStore 中的 MockConnection 是什么时候放进去的呢?就是在我们开始通过 MockNet.addConnection() 添加的。</p>    <p>这样整个框架就差不多讲完了,具体的细节就请查看 <a href="/misc/goto?guid=4959749307234612484" rel="nofollow,noindex">代码</a> 吧。感觉整个框架实现不难,模块化和扩展性都还不错(自夸一下)。</p>    <h3>其他</h3>    <p>如果有什么问题或者想法可以上 <a href="/misc/goto?guid=4959749307234612484" rel="nofollow,noindex">github</a> 提 issue,或者可以邮箱交流 zy5a59@outlook.com。</p>    <p>  </p>   <p>项目主页:<a href="http://www.open-open.com/lib/view/home/1495805036512">http://www.open-open.com/lib/view/home/1495805036512</a></p>    <p></p>