加速Robolectric下载依赖库及原理剖析

加速Robolectric下载依赖库及原理剖析_第1张图片

前言

春节后,事情比较多,没太多写作灵感。之前在《App组件化与业务拆分那些事》说过要写一篇怎么Android怎么做业务拆分的技术文,由于开发中遇到一些繁琐问题,打算延后一点再写。

为了及时给点干货读者们,今天笔者写写如雷贯耳的 Robolectric 吧!

给Robolectric**的第一次

从我做单元测试开始,一直有小伙伴在群上反映第一次robolectric运行太慢了,大半天都更新不完依赖库。

上两天把项目的robolectric从3.1.2升到3.2.2,本来已经下好的第三方依赖库,3.2.2要求更高版本,只能再下更高版本的库。用过robolectric都懂的,如下图(gif):

笔者的第一次用robolectric,翻了墙,大概用了半小时下载依赖库。之前除了翻墙,也没什么好办法,后来研究一下,解决的办法还不止一种,接下来分析一下。

Robolectric到底在做什么?

简单的robolectric test case:

@RunWith(RobolectricTestRunner.class)
public class RoboTest {

    @Test
    public void firstTest() {
        System.out.println("first test");
    }
}

分析日志

截取其中一部分日志:

WARNING: No manifest file found at .\AndroidManifest.xml.
Falling back to the Android OS resources only.
...

Downloading: org/robolectric/android-all/4.1.2_r1-robolectric-0/android-all-4.1.2_r1-robolectric-0.pom from repository sonatype at https://oss.sonatype.org/content/groups/public/
Transferring 1K from sonatype
Downloading: org/robolectric/android-all/4.1.2_r1-robolectric-0/android-all-4.1.2_r1-robolectric-0.jar from repository sonatype at https://oss.sonatype.org/content/groups/public/
Transferring 30702K from sonatype
...

只要英文不太烂,都知道日志说“正在从 https://oss.sonatype.org/content/groups/public/ 下载 android-all-4.1.2_r1-robolectric-0.pom、android-all-4.1.2_r1-robolectric-0.jar ...”

oss.sonatype.org 是什么?

(已科学上网)

在浏览器输入https://oss.sonatype.org/ :

加速Robolectric下载依赖库及原理剖析_第2张图片
oss.sonatype.org

综合判断:

oss.sonatype.org是一个Nexus搭建的maven仓库。robolectric第一次运行,从https://oss.sonatype.org/ 下载一些必要的依赖包。

oss.sonatype.org服务器在哪?

ping oss.sonatype.org:

C:\Users\kkmike999>ping oss.sonatype.org

正在 Ping oss.sonatype.org [52.22.249.229] 具有 32 字节的数据:
请求超时。
请求超时。
请求超时。
请求超时。

52.22.249.229 的 Ping 统计信息:
数据包: 已发送 = 4,已接收 = 0,丢失 = 4 (100% 丢失),

没错,oss.sonatype.org是外国的网站,百度一下52.22.249.229这个IP:

IP地址: 52.22.249.229美国

笔者甚至用国外的vp*服务器(vultr.com)来ping oss.sonatype.org,也一直超时。

迅雷下载......想太多

那我们找“4.1.2_r1-robolectric-0”在oss.sonatype.org上的路径,浏览https://oss.sonatype.org/content/groups/public/org/robolectric/android-all/4.1.2_r1-robolectric-0/,如下图:

加速Robolectric下载依赖库及原理剖析_第3张图片

可以看到 android-all-4.1.2_r1-robolectric-0.pom、android-all-4.1.2_r1-robolectric-0.jar等文件。

小白:“既然知道android-all-4.1.2_r1-robolectric-0.jar网址,直接下载吧,我有迅雷会员,离线下载,妥妥的!”

1小时后,小白下载并看完两集 波多野老师。再看看android-all-4.1.2_r1-robolectric-0.jar的迅雷任务,呃...

加速Robolectric下载依赖库及原理剖析_第4张图片

Gradle、Jcenter、第三方库

gradle从哪里下载第三方库

我们尝试用gradle下载android-all-4.1.2_r1-robolectric-0。在http://mvnrepository.com/ 找到 android-all-4.1.2_r1-robolectric-0,找到gradle引用它的语句testCompile group: 'org.robolectric', name: 'android-all', version: '4.1.2_r1-robolectric-0'

在app/build.gradle加入引用:

dependencies {
    compile fileTree(dir: 'libs', include: ['*.jar'])
    compile 'com.android.support:appcompat-v7:25.1.1'
    testCompile 'junit:junit:4.12'

    testCompile "org.robolectric:robolectric:3.2.2"
    testCompile group: 'org.robolectric', name: 'android-all', version: '4.1.2_r1-robolectric-0'
}

Sync gradle后,Android Studio底部显示下载进度:

加速Robolectric下载依赖库及原理剖析_第5张图片

可以看到gradle从 https://jcenter.bintray.com下载android-all-4.1.2_r1-robolectric-0依赖库。这个jar有30MB,jcenter有几十KB速度,需要点时间才能下载完。

从jcenter下载的库本地目录

Android Studio project窗口,External Libraries已经有android-all-4.1.2_r1-robolectric-0,证明已经把库下载到本地。

加速Robolectric下载依赖库及原理剖析_第6张图片

右键->Library Properties

加速Robolectric下载依赖库及原理剖析_第7张图片

原来jar保存在 C:\Users\{User Name}\.gradle\caches\modules-2\files-2.1\目录下。

再次运行robolectric单元测试

小白:“既然gradle从jcenter下好了android-all-4.1.2_r1-robolectric-0.jar,那这下robolectric就能依赖了吧!?”
于是,小白跑一次刚才的test case...

加速Robolectric下载依赖库及原理剖析_第8张图片

非常遗憾!robolectric显然不认~/.gradle/的账。

robolectric依赖的本地目录 与 gradle依赖的本地目录 不相同

robolectric的依赖库,本地放在哪?

用过eclipse或者inteliJ的同学应该知道,从maven仓库同步回来的库,会存在本地一个目录,这个目录就是~/.m2/

默认情况:

windows:C:\Users{用户名}.m2\repository
mac:\Users{用户名}.m2\repository\

如果你自定义了maven本地路径,那就找到设置后的~/.m2/目录。

如果刚才通过gradle从oss.sonatype.org同步了一点点文件回来,这时应该存在 C:\Users\{用户名}\.m2\repository\org\robolectric\android-all\4.1.2_r1-robolectric-0\,目录下有几个文件:

加速Robolectric下载依赖库及原理剖析_第9张图片

android-all-4.1.2_r1-robolectric-0.jar.tmp
android-all-4.1.2_r1-robolectric-0.pom
android-all-4.1.2_r1-robolectric-0.pom.sha1
android-all-4.1.2_r1-robolectric-0.pom.tmp.sha1.tmp

结论:

robolectric下载的库放在本地目录 ~/.m2/repository/

至于为什么robolectric会依赖~/.m2/,在下一节源码剖析,会说明一下。

robolectric源代码

RobolectricTestRunner

public class RobolectricTestRunner extends BlockJUnit4ClassRunner {

    private DependencyResolver dependencyResolver;

    protected DependencyResolver getJarResolver() {
        if (dependencyResolver == null) {
        if (Boolean.getBoolean("robolectric.offline")) {
            String dependencyDir = System.getProperty("robolectric.dependency.dir", ".");
            dependencyResolver = new LocalDependencyResolver(new File(dependencyDir));
        } else {
            File cacheDir = new File(new File(System.getProperty("java.io.tmpdir")), "robolectric");

            if (cacheDir.exists() || cacheDir.mkdir()) {
              Logger.info("Dependency cache location: %s", cacheDir.getAbsolutePath());
              dependencyResolver = new CachedDependencyResolver(new MavenDependencyResolver(), cacheDir, 60 * 60 * 24 * 1000);
            } else {
              dependencyResolver = new MavenDependencyResolver();
            }
        }

        URL buildPathPropertiesUrl = getClass().getClassLoader().getResource("robolectric-deps.properties");
        if (buildPathPropertiesUrl != null) {
            Logger.info("Using Robolectric classes from %s", buildPathPropertiesUrl.getPath());

            FsFile propertiesFile = Fs.fileFromPath(buildPathPropertiesUrl.getFile());
            try {
              dependencyResolver = new PropertiesDependencyResolver(propertiesFile, dependencyResolver);
            } catch (IOException e) {
                throw new RuntimeException("couldn't read " + buildPathPropertiesUrl, e);
            }
        }
    }

    return dependencyResolver;
  }
}

我们找到DependencyResolver dependencyResolver成员和跟dependencyResolver密切相关的getJarResolver()方法。

debug一下test case,并在getJarResolver()里面打Breakpoints:

加速Robolectric下载依赖库及原理剖析_第10张图片

你发现调用了:

dependencyResolver = new CachedDependencyResolver(new MavenDependencyResolver(), cacheDir, 60 * 60 * 24 * 1000);

运行完getJarResolver(),在Android Studio Debug工具查看RobolectricTestRunner的变量:

加速Robolectric下载依赖库及原理剖析_第11张图片

关键的东西在这里,CachedDependencyResolver dependencyResolver里面还有一个变量MavenDependencyResolver dependencyResolver,这个MavenDependencyResolver有变量及其值:

repositoryUrl = https://oss.sonatype.org/content/groups/public
repositoryId = sonatype

这个就是robolectric为什么从https://oss.sonatype.org下载依赖库的原因,只要把repositoryUrl替换其他url,就可以改变maven仓库网址了。

CachedDependencyResolver、MavenDependencyResolver

CachedDependencyResolver:

public class CachedDependencyResolver implements DependencyResolver {

  private final DependencyResolver dependencyResolver;// MavenDependencyResolver
  
  @Override
  public URL[] getLocalArtifactUrls(DependencyJar... dependencies) {
    ...
    final URL[] urls = dependencyResolver.getLocalArtifactUrls(dependencies);
    ...
    return urls;
  }

  @Override
  public URL getLocalArtifactUrl(DependencyJar dependency) {
    ...
    final URL url = dependencyResolver.getLocalArtifactUrl(dependency);
    ...
    return url;
  }
}

MavenDependencyResolver(重点):

public class MavenDependencyResolver implements DependencyResolver {

  private final String repositoryUrl;
  private final String repositoryId;

  // 默认从RoboSetting获取repositoryUrl和repositoryId,RoboSettings相当于Hook
  public MavenDependencyResolver() {
    this(RoboSettings.getMavenRepositoryUrl(), RoboSettings.getMavenRepositoryId());
  }

  public MavenDependencyResolver(String repositoryUrl, String repositoryId) {
    this.repositoryUrl = repositoryUrl;
    this.repositoryId = repositoryId;
  }

  @Override
  public URL[] getLocalArtifactUrls(DependencyJar... dependencies) {
    DependenciesTask dependenciesTask = createDependenciesTask();// AbstractArtifactTask子类
    ...
    RemoteRepository remoteRepository = new RemoteRepository();
    remoteRepository.setUrl(repositoryUrl);// 默认https://oss.sonatype.org/content/groups/public/
    remoteRepository.setId(repositoryId);// 默认sonatype
    dependenciesTask.addConfiguredRemoteRepository(remoteRepository);
    ...
    
    dependenciesTask.execute(); // 调用AbstractArtifactTask.execute()
    ...
  }
}

RoboSettings :

public class RoboSettings {

  private static String mavenRepositoryId;
  private static String mavenRepositoryUrl;

  static {
    mavenRepositoryId = System.getProperty("robolectric.dependency.repo.id", "sonatype");
    mavenRepositoryUrl = System.getProperty("robolectric.dependency.repo.url", "https://oss.sonatype.org/content/groups/public/");// 看到默认以https://oss.sonatype.org/content/groups/public/为resitoryUrl
  }

  public static String getMavenRepositoryId() {
    return mavenRepositoryId;
  }

  public static void setMavenRepositoryId(String mavenRepositoryId) {
    RoboSettings.mavenRepositoryId = mavenRepositoryId;
  }

  public static String getMavenRepositoryUrl() {
    return mavenRepositoryUrl;
  }

  public static void setMavenRepositoryUrl(String mavenRepositoryUrl) {
    RoboSettings.mavenRepositoryUrl = mavenRepositoryUrl;
  }
}

AbstractArtifactTask:

public abstract class AbstractArtifactTask extends Task{

    public void execute()
    {
        ...
        initSettings();
        doExecute(); // 下载或从本地读取依赖库
       ...
    }
    
    private File newFile( String parent, String subdir, String filename )
    {
        return new File( new File( parent, subdir ), filename );
    }
    
    private void initSettings()
    {
        if ( userSettingsFile == null )
        {
            File tempSettingsFile = newFile( System.getProperty( "user.home" ), ".ant", "settings.xml" );
            if ( tempSettingsFile.exists() )
            {
                userSettingsFile = tempSettingsFile;
            }
            else
            {
                tempSettingsFile = newFile( System.getProperty( "user.home" ), ".m2", "settings.xml" );
                if ( tempSettingsFile.exists() )
                {
                    userSettingsFile = tempSettingsFile;
                }
            }
        }
        if ( globalSettingsFile == null )
        {
            File tempSettingsFile = newFile( System.getProperty( "ant.home" ), "etc", "settings.xml" );
            if ( tempSettingsFile.exists() )
            {
                globalSettingsFile = tempSettingsFile;
            }
            else
            {
                // look in ${M2_HOME}/conf
                List env = Execute.getProcEnvironment();
                for ( String var: env )
                {
                    if ( var.startsWith( "M2_HOME=" ) )
                    {
                        String m2Home = var.substring( "M2_HOME=".length() );
                        tempSettingsFile = newFile( m2Home, "conf", "settings.xml" );
                        if ( tempSettingsFile.exists() )
                        {
                            globalSettingsFile = tempSettingsFile;
                        }
                        break;
                    }
                }
            }
        }

        Settings userSettings = loadSettings( userSettingsFile );// 读取并解析配置
        Settings globalSettings = loadSettings( globalSettingsFile );// 读取并解析配置

        SettingsUtils.merge( userSettings, globalSettings, TrackableBase.GLOBAL_LEVEL );
        settings = userSettings;

        if ( StringUtils.isEmpty( settings.getLocalRepository() ) )
        {
            String location = newFile( System.getProperty( "user.home" ), ".m2", "repository" ).getAbsolutePath();// 默认maven目录
            settings.setLocalRepository( location );// 设置默认maven目录
        }
        ...
    }
}

initSetting()主要任务,就是找到默认或setting.xml配置的maven目录,代码大致意思是:

1.加载 $user.home/.ant/setting.xml$user.home/.m2/setting.xml$M2_HOME/conf/setting.xml ,读取并解析配置文件,获取配置的maven目录;
2.如果没找到setting.xml,则默认$user.home/.m2/repository/为maven本地目录。

$user.home变量对应windows默认是C:\Users\{用户名}\,mac默认\Users\{用户名}\。这就知道默认.m2目录是C:\Users\{用户名}\.m2\repository\\Users\{用户名}\.m2\repository\了。

DependenciesTask.doExecute()处理从maven服务器下载依赖库到本地,读取本地依赖库等逻辑,本文不详述了,有兴趣的读者自己看看源码。


加速终极大招

大招1——把依赖文件拷贝到maven目录

既然我们知道robolectric依赖$user.home\.m2\repository\,那直接把下载好的jar拷贝到该目录。例如4.1.2_r1-robolectric-0:

拷贝C:\Users\kkmike999\.gradle\caches\modules-2\files-2.1\org.robolectric\android-all\4.1.2_r1-robolectric-0\aecc8ce5119a25fcea1cdf8285469c9d1261a352\android-all-4.1.2_r1-robolectric-0.jar$user.home\.m2\repository\org\robolectric\android-all\4.1.2_r1-robolectric-0\

或者到http://mvnrepository.com/artifact/org.robolectric/android-all/4.1.2_r1-robolectric-0下载android-all-4.1.2_r1-robolectric-0.jar,再拷贝到该目录。

robolectric有好几个依赖,必须把所有依赖都拷全。笔者不推荐这种做法。

大招2——把oss.sonatype.org改成阿里云maven仓库(推荐)

(2017.3.5更新)

先把$user.home\.m2\repository\org\robolectric\里面未下载完的目录删掉。因为这里可能有pom配置文件,里面的配置还是指向oss.sonatype.org,所以必须删除。

MyRobolectricTestRunner:

public class MyRobolectricTestRunner extends RobolectricTestRunner {
    static {
        // 从源码知道MavenDependencyResolver默认以RoboSettings的repositoryUrl和repositoryId为默认值,因此只需要对RoboSetting进行赋值即可
        MavenRoboSettings.setMavenRepositoryUserName("");
        MavenRoboSettings.setMavenRepositoryPassword("");
        MavenRoboSettings.setMavenRepositoryId("alimaven");
        MavenRoboSettings.setMavenRepositoryUrl("http://maven.aliyun.com/nexus/content/groups/public/");
    }

    public MyRobolectricTestRunner(Class testClass) throws InitializationError {
        super(testClass);
    }
}

test case:

@Config(manifest = "./src/main/AndroidManifest.xml")
@RunWith(MyRobolectricTestRunner.class)
public class RoboTest {

    @Test
    public void firstTest() {
        System.out.println("first test");
    }
}

运行单元测试:

速度2M/s左右,有时更快。依赖库下载完,并完成单元测试,耗时17s:

加速Robolectric下载依赖库及原理剖析_第12张图片

(注意,这个速度测试,笔者仅删掉android-all-4.1.2_r1-robolectric-0.jar,实际robolectric还有好些依赖包,实际耗时要更长一些)

启发

可以在project/build.gradle添加阿里云maven仓库:

build.gradle

allprojects {
    repositories {
        maven { url 'http://maven.aliyun.com/nexus/content/groups/public/' }
        jcenter()
    }
}

速度扛扛的!


小结

Robolectric确实是不错的android单元测试第三方库,尽管运行起来有点慢。它能做挺多事情,例如直接测试sqlite(《Android单元测试 - Sqlite、SharedPreference、Assets、文件操作 怎么测?》)。

笔者写本文时,曾反复琢磨,究竟要慢慢分析问题,以实验形式来引出解决方法,还是剖析源码中,寻找解决方法呢?最终平衡了两个需求,成了本文这个样子。

希望更多的同学,在第一次做robolectric单元测试时,阅读本文,避免浪费时间。


关于作者

我是键盘男。
在广州生活,在互联网公司上班,猥琐文艺码农。跑步、喜欢科学、历史,玩玩投资,偶尔旅行。

你可能感兴趣的:(加速Robolectric下载依赖库及原理剖析)