Search code examples
androidjunitmockitoreact-nativeunsatisfiedlinkerror

UnsatisfiedLinkError when unit testing WritableNativeMap


I am currently creating an Android library for use in a react native project. I need to emit a map to javascript, so I'm using react native's WriteableMap class. Unfortunately, the class loads the reactnativejni SO in a static block, which leads to an UnsatisfiedLinkError during unit tests. I'm using JUnit and Mockito to test.

My code:

@Override
public void onSomething() {
    WritableMap params = Arguments.createMap();

    //fill map

    sendEvent("onChange", params);
}

The error:

java.lang.UnsatisfiedLinkError: no reactnativejni in java.library.path
  at java.lang.ClassLoader.loadLibrary(ClassLoader.java:1857)
  at java.lang.Runtime.loadLibrary0(Runtime.java:870)
  at java.lang.System.loadLibrary(System.java:1119)
  at com.facebook.soloader.SoLoader.loadLibrary(SoLoader.java:172)
  at com.facebook.react.bridge.NativeMap.<clinit>(NativeMap.java:23)
  at com.facebook.react.bridge.Arguments.createMap(Arguments.java:29)
  at me.MyClass.onSomething(myClass.java:23)

I started using the Arguments.createMap() method after seeing a comment about stubbing WriteableMap for unit tests, but it's static and I would prefer to not have to stub a static method.

Is there some way to get rid of this error when running unit tests?


Solution

  • I don't think you can avoid mocking the Arguments methods in a unit test (though I believe you do not need to do so in an instrumentation test).

    In Facebook's own tests, they use PowerMockito to mock them: https://github.com/facebook/react-native/blob/master/ReactAndroid/src/test/java/com/facebook/react/RootViewTest.java#L67

    Interesting bits:

    PowerMockito.mockStatic(Arguments.class);
    PowerMockito.when(Arguments.createArray()).thenAnswer(new Answer<Object>() {
      @Override
      public Object answer(InvocationOnMock invocation) throws Throwable {
        return new JavaOnlyArray();
      }
    });
    PowerMockito.when(Arguments.createMap()).thenAnswer(new Answer<Object>() {
      @Override
      public Object answer(InvocationOnMock invocation) throws Throwable {
        return new JavaOnlyMap();
      }
    });
    

    Also note that this requires you to modify your build.gradle to include these mocking tools: https://github.com/facebook/react-native/blob/master/ReactAndroid/build.gradle#L266

    dependencies {
      ...
      testCompile "org.powermock:powermock-api-mockito:${POWERMOCK_VERSION}"
      testCompile "org.powermock:powermock-module-junit4-rule:${POWERMOCK_VERSION}"
      testCompile "org.powermock:powermock-classloading-xstream:${POWERMOCK_VERSION}"
      testCompile "org.mockito:mockito-core:${MOCKITO_CORE_VERSION}"
      testCompile "org.easytesting:fest-assert-core:${FEST_ASSERT_CORE_VERSION}"
      testCompile "org.robolectric:robolectric:${ROBOLECTRIC_VERSION}"
      ...
    }
    

    The versions they use can be found in their gradle.properties file.

    I don't know how stable these test configurations will be over the long run, but this configuration has let me work with ReadableMap/Array and WritableMap/Array in unit tests.