需要开发一个特殊的app。所以需要用到networkExtension。但在使用过程中遇到了很多问题,在这里记录一下。因为我自己也是新手在做这个的时候到处找不到资料很痛苦啊。如果中间有错误的地方还请大佬们指正。
1.为什么要用networkExtension
苹果巴巴为了省电等需求在我们的app退到后台后,是不会让我们的app活多久的。但是要做一个代理全部手机上网络流量的app就需要能够在我们的app退出后依然能够工作。networkExtension就是苹果给我们的方案。
2.创建target
在我们是用networkExtension的时候,它已经不是我们当前的app了而是一个独立的app。所以就需要在当前项目中创建一个netwokrExtension的target。如截图中箭头指示的
第一步选择添加一个target。
第二步选择NetworExtension。然后输入项目名称就好了。
第三步 需要配置Netwokr Extensions 和 Personal XXX 这两个权限在以前是需要去申请的,现在只要你的账号是开发者账号就可以直接选择添加,选择好了后是如p-2这样的。但是有一点不要忘了就我们的主app和扩展app的target这里都需要添加这两个权限。
3.扩展app的调试及打印
调试
扩展app的调试根据扩展app的类型来区分有2种方式。
* 一种是普通类型的APP 直接运行扩展app就好了。
* 另一种是通知类 需要启动主工程,然后选择Xcode -> Debug -> Attach to process by pID or name 这里有个坑就是你需要输入的是扩展app的 Display Name 而不是Bundie Idetifier。我当时就被这个坑惨了。
打印
扩展的打印一直困扰了我很久。网上看到了很多方案,但是我都没有跑通。我这里就说一下我使用的方案。就是利用系统日志。在console中输出打印。打印的方法在iOS10以前可以用NSLog() 方法直接就能够把打印输出到console中,在iOS10以后需要使用os_log()方法,如果你找不到这个方法,可以试着引入 import os库。当然os_log方法功能有限,这里推荐一个第三方库CocoaLumberjack OC和swift都支持。也是可以输出到console中的。
4.扩展程序中添加桥接文件
在扩展程序中可能会用到OC/C/C++这些语言写的类。我们就需要用到桥接文件了。和我们正常使用桥接文件是一样的,你可以在扩展程序中新建一个桥接文件 然后在扩展程序的target中添加这个桥接文件。就可以正常引用C/C++/OC的方法了。
这里有一个建议就是在扩展程序中单独建一个桥接文件,不和主app公用一个桥接文件。这样桥接文件在引用头文件时不会出现头文件路径不对,找不到的问题。
5.流量拦截及如何处理不希望拦截的流量
本身这个问题是和使用networkExtension没有关系的,因为我自己脑子转不过来在这个问题上卡了很久所以才拿出来说。当我们使用packetFlow.readPackets读取到ip数据包后,我们把分析了数据后,一部分数据我们会代理,但是有一些数据我们不想处理怎么办?我最后的解决方法是把这些数据在发到这些数据的目标服务器去,是tcp就建立tcp连接发送,是udp就建立udp连接发送。五元组的数据是能够在ip数据包中拿到的。
6.一个可用的封装的tun2socks ios库
ios这边我找了很久都没有找到一个可以使用的tun2socks的库,最后找到了这个库。感觉很厉害
https://github.com/zapcannon87/ZPTCPIPStack