Hot Patch工具Rollout

最近在iOS上又出现了一个热更新的工具,rollout,小小的体验了一番。

简介

它在编译期间执行一个upload_dsym的脚本,将收集到的文件符号dsym上传到其服务器,这样我们就可以在它提供的后台操作dashboard中,选择对应的app,选择已存在的方法,进行编辑,更改返回值,替换参数,添加逻辑等等。还可以添加新的方法,但是类必须存在,还能为缺失的方法提供实现。

主要技术也是JavaScriptCore和Runtime。在dashboard中添加hot patch之后,客户端重新启动,会拉取新的配置。但在试验中,不是很及时,要启动几次才能切到新的配置,可能跟服务器在国外有关,还有可能就是拉取到配置之后,要patch的动作已经完成,这种在下次启动就可以了。当在工程中新添加类/方法时,在启动后也会将符号上传。但是不能立马在dashboard中看到对应的类。问了rollout的工作人员,说是会有几分钟的延迟,可以过段时间刷新页面看看。

安装

可以通过Cocoapods或者下载他们提供的工具来安装SDK。

我是通过Cocoapods。

另外,我们可以添加options,输出调试信息。

RolloutOptions *option = [[RolloutOptions alloc] init];
option.verbose = RolloutOptionsVerboseLevelDebug;

[Rollout setupWithKey:@"5714eaa036053a902868e088"
    #ifdef DEBUG
        developmentDevice:YES
    #endif
     options:option];

下午操作的时候按照它的提示,在前5步完成之后,compile的时候出现了错误。Build Phase-->Rollout.io dsym upload中路径不存在的问题,后来跟rollout的人反馈了这个问题,回复说是个bug,会在下次release中fix。但是现在晚上再重试已经修复了。真有效率啊。
在运行起来之后,Waiting For App会变成Done。可以进行下一步添加hot patch了。

添加hot patch

1、你可以搜索想要的方法名。

为了方便,我们选择[ViewController viewDidLoad]方法。

  • toggleJS是让我们自己添加js代码,写逻辑。
  • Before the function有2种操作,一种是弹个alert,可以自己设置title,msg等。另一种是进行事件统计。
  • ConditionalPatch是添加patch触发的条件,可设置系统版本,手机型号。
  • HotPatch Type是是否允许使用patch。

下面我们就简单的选择Before the function添加个alert。


然后,重新运行。有时候并不能马上看到效果,因为它是异步去拉取新的配置,配置拉取回来之后,在下次启动时生效。部分配置如下。


{
    "__v" = 0;
    "_id" = 5714f968cfbc4acb63bdc419;
    "app_version" = 5714eeb692e27ab318a043ba;
    bucket = sandbox;
    "creation_date" = "2016-04-18T15:12:40.656Z";
    data =     (
                {
            class = Test;
            configuration = "Ui5OU0NsYXNzRnJvbVN0cmluZygnVUlBcHBsaWNhdGlvbicpLmNhbGwoJ3NoYXJlZEFwcGxpY2F0aW9uJykuY2FsbCgnb3BlblVSTDonLCBSLk5TQ2xhc3NGcm9tU3RyaW5nKCdOU1VSTCcpLmNhbGwoJ1VSTFdpdGhTdHJpbmc6JywgJ2h0dHA6Ly93d3cuYmFpZHUuY29tJykpOwo=";
            configurationType = javascript;
            methodType = instance;
            selector = test;
            signature = "()->Void";
            swizzlingType = createImplementation;
        },
                {
            class = ViewController;
            configuration = Ui5OU0NsYXNzRnJvbVN0cmluZygnTXlPYmplY3QnKS5jYWxsKCduZXcnKS5jYWxsKCdzYXknKTsK;
            configurationType = javascript;
            methodType = instance;
            selector = viewDidLoad;
            signature = "()->Void";
            swizzlingType = replaceImplementation;
        }
    );
}

运行结果如下:


2、Polyfille method
可以理解为方法替代。只能给已存在的类添加方法。可以添加新的方法,或者是给已声明未实现的方法添加实现。
下面的例子中因为[MyObject test]未实现,在MyViewController里面ViewDidload方法调用了[MyObject test]方法,正常情况下是跳转到MyViewController页面就会崩溃的。等下用hot patch来解决。经测试,如果实现[MyObject test],就会走原有逻辑,patch就不起作用了。如果要修改已存在方法,可参照第三条。

@interface MyObject : NSObject

- (void)say;

// 未实现,之后会用hot patch实现
- (void)test;

- (void)calculate:(int)a;

@end

@implementation MyObject

- (void)say {
    NSLog(@"I am MyObject");
}

- (void)calculate:(int)a {
    NSLog(@"calculate:%d", a);
}
@end

我们添加了[MyObject test]方法。

然后添加实现。这里我们打开百度页面。js的写法也比较简单。

R.NSClassFromString('UIApplication').call('sharedApplication').call('openURL:', R.NSClassFromString('NSURL').call('URLWithString:', 'http://www.baidu.com'));

调用方法就用call,传参数,直接跟在后面。

运行,跳转到MyViewController,会跳转到web。

3、修改已存在的方法

还是拿[ViewController viewDidLoad]来说,在里面调用方法[MyObject say]。选择Toggler JS。[MyObject say]是在工程中的方法。

var obj = R.NSClassFromString('MyObject').call('new')
obj.call('say')
self.originalImplementation();

再次运行。

4、修改参数值
将calculate参数为10,修改成20。

在ViewController里

- (void)viewDidLoad {
    [super viewDidLoad];
    // Do any additional setup after loading the view, typically from a nib.
    
    MyObject *obj = [[MyObject alloc] init];
    
    [obj calculate:10];
}

运行几次,发现的确是输出了20。

你可能感兴趣的:(Hot Patch工具Rollout)