Unit Test 探求記は、unit test に関してまったりと実験しつつその過程を綴ってみるというものです。
今回のお題
Instrumented Unit Test 専用の resource を追加してみます。
方法
普通に src/androidTest/res 以下に配置してあげれば OK です。
テスト対象
View のコンストラクタ引数の AttributeSet から属性情報を取得するためのメソッドをテスト対象とします。
Resource Files
◆ layout
- レイアウトファイルには、テスト対象となる attribute の情報が入っています。。
- ファイル名は名前空間の衝突を避けるため、unit test class のFQCNに近い名称を用いています。
android_view_get_attr_instrumented_test_activity.xml
<?xml version="1.0" encoding="utf-8"?>
<com.objectfanatics.commons.android.view.GetAttrInstrumentedTestView
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
android:id="@+id/range_number_text_view"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
app:nonNullBooleanValue="true"
app:nonNullIntegerValue="20"
app:nonNullFloatValue="20.0"
app:nonNullStringValue="あ"
app:nonNullLongValue="9223372036854775807"
app:nonNullIntegerArrayValue="@array/android_view_get_attr_instrumented_test_integer_array"
app:nonNullStringArrayValue="@array/android_view_get_attr_instrumented_test_string_array"
app:nullValue="@null" />
◆ values
values 以下の各種リソースは、種類の軸ではなく unit test という軸でまとめたほうが保守しやすいと考え、unit test class のFQCNに近い名称の単一ファイルにまとめました。
android_view_get_attr_instrumented_test.xml
<?xml version="1.0" encoding="utf-8"?>
<resources>
<declare-styleable name="GetAttrInstrumentedTestView">
<attr name="nonNullIntegerValue" format="integer" />
<attr name="nonDefIntegerValue" format="integer" />
<attr name="nonNullStringValue" format="string" />
<attr name="nonDefStringValue" format="string" />
<attr name="nonNullBooleanValue" format="boolean" />
<attr name="nonDefBooleanValue" format="boolean" />
<attr name="nonNullFloatValue" format="float" />
<attr name="nonDefFloatValue" format="float" />
<attr name="nonNullLongValue" format="string" />
<attr name="nonDefLongValue" format="string" />
<attr name="nonNullIntegerArrayValue" format="reference" />
<attr name="nonDefIntegerArrayValue" format="reference" />
<attr name="nonNullStringArrayValue" format="reference" />
<attr name="nonDefStringArrayValue" format="reference" />
<!-- Attribute for null. It doesn't necessarily has to be string. -->
<attr name="nullValue" format="string" />
</declare-styleable>
<string-array name="android_view_get_attr_instrumented_test_string_array">
<item>a</item>
<item>b</item>
<item>c</item>
</string-array>
<integer-array name="android_view_get_attr_instrumented_test_integer_array">
<item>1</item>
<item>2</item>
<item>3</item>
</integer-array>
</resources>
Unit Test
View
のコンストラクタ内部にテストのロジックが入るという変則的な形になりました。1
package com.objectfanatics.commons.android.view
import android.app.Activity
import android.content.Context
import android.os.Bundle
import android.util.AttributeSet
import android.view.View
import androidx.test.core.app.launchActivity
import androidx.test.ext.junit.rules.ActivityScenarioRule
import com.objectfanatics.commons.android.test.R
import org.junit.Assert.*
import org.junit.Rule
import org.junit.Test
import com.objectfanatics.commons.android.test.R.styleable.GetAttrInstrumentedTestView as styleable
import com.objectfanatics.commons.android.test.R.styleable.GetAttrInstrumentedTestView_nonDefBooleanValue as nonDefBooleanValueIndex
import com.objectfanatics.commons.android.test.R.styleable.GetAttrInstrumentedTestView_nonDefFloatValue as nonDefFloatValueIndex
import com.objectfanatics.commons.android.test.R.styleable.GetAttrInstrumentedTestView_nonDefIntegerArrayValue as nonDefIntegerArrayValueIndex
import com.objectfanatics.commons.android.test.R.styleable.GetAttrInstrumentedTestView_nonDefIntegerValue as nonDefIntegerValueIndex
import com.objectfanatics.commons.android.test.R.styleable.GetAttrInstrumentedTestView_nonDefLongValue as nonDefLongValueIndex
import com.objectfanatics.commons.android.test.R.styleable.GetAttrInstrumentedTestView_nonDefStringArrayValue as nonDefStringArrayValueIndex
import com.objectfanatics.commons.android.test.R.styleable.GetAttrInstrumentedTestView_nonDefStringValue as nonDefStringValueIndex
import com.objectfanatics.commons.android.test.R.styleable.GetAttrInstrumentedTestView_nonNullBooleanValue as nonNullBooleanValueIndex
import com.objectfanatics.commons.android.test.R.styleable.GetAttrInstrumentedTestView_nonNullFloatValue as nonNullFloatValueIndex
import com.objectfanatics.commons.android.test.R.styleable.GetAttrInstrumentedTestView_nonNullIntegerArrayValue as nonNullIntegerArrayValueIndex
import com.objectfanatics.commons.android.test.R.styleable.GetAttrInstrumentedTestView_nonNullIntegerValue as nonNullIntegerValueIndex
import com.objectfanatics.commons.android.test.R.styleable.GetAttrInstrumentedTestView_nonNullLongValue as nonNullLongValueIndex
import com.objectfanatics.commons.android.test.R.styleable.GetAttrInstrumentedTestView_nonNullStringArrayValue as nonNullStringArrayValueIndex
import com.objectfanatics.commons.android.test.R.styleable.GetAttrInstrumentedTestView_nonNullStringValue as nonNullStringValueIndex
import com.objectfanatics.commons.android.test.R.styleable.GetAttrInstrumentedTestView_nullValue as nullValueIndex
class GetAttrInstrumentedTest {
@get:Rule
var activityRule: ActivityScenarioRule<GetAttrInstrumentedTestActivity> =
ActivityScenarioRule(GetAttrInstrumentedTestActivity::class.java)
@Test
fun getXxxAttrTest() {
launchActivity<GetAttrInstrumentedTestActivity>()
}
}
class GetAttrInstrumentedTestActivity : Activity() {
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.android_view_get_attr_instrumented_test_activity)
}
}
class GetAttrInstrumentedTestView : View {
constructor(context: Context?) : super(context)
constructor(context: Context?, attrs: AttributeSet?) : super(context, attrs) {
getAttrs(attrs)
}
constructor(context: Context?, attrs: AttributeSet?, defStyleAttr: Int) : super(context, attrs, defStyleAttr) {
getAttrs(attrs)
}
constructor(context: Context?, attrs: AttributeSet?, defStyleAttr: Int, defStyleRes: Int) : super(context, attrs, defStyleAttr, defStyleRes) {
getAttrs(attrs)
}
/**
* 各型に対して、以下のパターンのテストを実行します。
* (1) getXxxAttr(): デフォルト値が指定されているが、値がセットされているため、セットされた値が返る。
* (2) getXxxAttr(): デフォルト値が指定されており、値に @null がセットされているため、デフォルト値が返る。
* (3) getXxxAttr(): デフォルト値が指定されており、値がセットされていないため、デフォルト値が返る。
* (4) getNullableXxxAttr(): デフォルト値が指定されておらず、値がセットされているため、セットされた値が返る。
* (5) getNullableXxxAttr(): デフォルト値が指定されておらず、値に @null がセットされているため、null が返る。
* (6) getNullableXxxAttr(): デフォルト値が指定されておらず、値がセットされていないため、null が返る。
* (7) getXxxAttr(): デフォルト値が指定されていないが、値がセットされているため、セットされた値が返る。
* (8) getXxxAttr(): デフォルト値が指定されておらず、値に @null がセットされているため、AttributeValueNotFoundException がスローされる。
* (9) getXxxAttr(): デフォルト値が指定されておらず、値がセットされていないため、AttributeValueNotFoundException がスローされる。
*/
private fun getAttrs(attrs: AttributeSet?) {
// @formatter:off
// boolean
assertObject(
attrs,
DEF_BOOLEAN_VALUE, SET_BOOLEAN_VALUE,
this::getBooleanAttr, this::getNullableBooleanAttr, this::getBooleanAttr,
nonNullBooleanValueIndex, nonDefBooleanValueIndex
)
// integer
assertObject(
attrs,
DEF_INTEGER_VALUE, SET_INTEGER_VALUE,
this::getIntegerAttr, this::getNullableIntegerAttr, this::getIntegerAttr,
nonNullIntegerValueIndex, nonDefIntegerValueIndex
)
// float
assertObject(
attrs,
DEF_FLOAT_VALUE, SET_FLOAT_VALUE,
this::getFloatAttr, this::getNullableFloatAttr, this::getFloatAttr,
nonNullFloatValueIndex, nonDefFloatValueIndex
)
// string
assertObject(
attrs,
DEF_STRING_VALUE, SET_STRING_VALUE,
this::getStringAttr, this::getNullableStringAttr, this::getStringAttr,
nonNullStringValueIndex, nonDefStringValueIndex
)
// long
assertObject(
attrs,
DEF_LONG_VALUE, SET_LONG_VALUE,
this::getLongAttr, this::getNullableLongAttr, this::getLongAttr,
nonNullLongValueIndex, nonDefLongValueIndex
)
// integer array
assertIntArray(
attrs,
DEF_INTEGER_ARRAY_VALUE, SET_INTEGER_ARRAY_VALUE,
this::getIntegerArrayAttr, this::getNullableIntegerArrayAttr, this::getIntegerArrayAttr,
nonNullIntegerArrayValueIndex, nonDefIntegerArrayValueIndex
)
// string array
assertArray(
attrs,
DEF_STRING_ARRAY_VALUE, SET_STRING_ARRAY_VALUE,
this::getStringArrayAttr, this::getNullableStringArrayAttr, this::getStringArrayAttr,
nonNullStringArrayValueIndex, nonDefStringArrayValueIndex
)
// @formatter:on
}
private fun <T> assertObject(
attrs: AttributeSet?,
defValue: T,
setValue: T,
getAttrWithDef: (attrs: AttributeSet?, styleable: IntArray, index: Int, defValue: T) -> T,
getNullableAttr: (attrs: AttributeSet?, styleable: IntArray, index: Int) -> T?,
getAttrWithoutDef: (attrs: AttributeSet?, styleable: IntArray, index: Int) -> T,
nonNullValueIndex: Int,
nonDefValueIndex: Int
) {
assertEquals(setValue, getAttrWithDef(attrs, styleable, nonNullValueIndex, defValue))
assertEquals(defValue, getAttrWithDef(attrs, styleable, nullValueIndex, defValue))
assertEquals(defValue, getAttrWithDef(attrs, styleable, nonDefValueIndex, defValue))
assertEquals(setValue, getNullableAttr(attrs, styleable, nonNullValueIndex))
assertEquals(null, getNullableAttr(attrs, styleable, nullValueIndex))
assertEquals(null, getNullableAttr(attrs, styleable, nonDefValueIndex))
assertEquals(setValue, getAttrWithoutDef(attrs, styleable, nonNullValueIndex))
assertAttributeValueNotFound { getAttrWithoutDef(attrs, styleable, nullValueIndex) }
assertAttributeValueNotFound { getAttrWithoutDef(attrs, styleable, nonDefValueIndex) }
}
@Suppress("SameParameterValue")
private fun assertIntArray(
attrs: AttributeSet?,
defValue: IntArray,
setValue: IntArray,
getAttrWithDef: (attrs: AttributeSet?, styleable: IntArray, index: Int, defValue: IntArray) -> IntArray,
getNullableAttr: (attrs: AttributeSet?, styleable: IntArray, index: Int) -> IntArray?,
getAttrWithoutDef: (attrs: AttributeSet?, styleable: IntArray, index: Int) -> IntArray,
nonNullValueIndex: Int,
nonDefValueIndex: Int
) {
assertArrayEquals(setValue, getAttrWithDef(attrs, styleable, nonNullValueIndex, defValue))
assertArrayEquals(defValue, getAttrWithDef(attrs, styleable, nullValueIndex, defValue))
assertArrayEquals(defValue, getAttrWithDef(attrs, styleable, nonDefValueIndex, defValue))
assertArrayEquals(setValue, getNullableAttr(attrs, styleable, nonNullValueIndex))
assertEquals(null, getNullableAttr(attrs, styleable, nullValueIndex))
assertEquals(null, getNullableAttr(attrs, styleable, nonDefValueIndex))
assertArrayEquals(setValue, getAttrWithoutDef(attrs, styleable, nonNullValueIndex))
assertAttributeValueNotFound { getAttrWithoutDef(attrs, styleable, nullValueIndex) }
assertAttributeValueNotFound { getAttrWithoutDef(attrs, styleable, nonDefValueIndex) }
}
@Suppress("SameParameterValue")
private fun <T> assertArray(
attrs: AttributeSet?,
defValue: Array<T>,
setValue: Array<T>,
getAttrWithDef: (attrs: AttributeSet?, styleable: IntArray, index: Int, defValue: Array<T>) -> Array<T>,
getNullableAttr: (attrs: AttributeSet?, styleable: IntArray, index: Int) -> Array<T>?,
getAttrWithoutDef: (attrs: AttributeSet?, styleable: IntArray, index: Int) -> Array<T>,
nonNullValueIndex: Int,
nonDefValueIndex: Int
) {
assertArrayEquals(setValue, getAttrWithDef(attrs, styleable, nonNullValueIndex, defValue))
assertArrayEquals(defValue, getAttrWithDef(attrs, styleable, nullValueIndex, defValue))
assertArrayEquals(defValue, getAttrWithDef(attrs, styleable, nonDefValueIndex, defValue))
assertArrayEquals(setValue, getNullableAttr(attrs, styleable, nonNullValueIndex))
assertArrayEquals(null, getNullableAttr(attrs, styleable, nullValueIndex))
assertArrayEquals(null, getNullableAttr(attrs, styleable, nonDefValueIndex))
assertArrayEquals(setValue, getAttrWithoutDef(attrs, styleable, nonNullValueIndex))
assertAttributeValueNotFound { getAttrWithoutDef(attrs, styleable, nullValueIndex) }
assertAttributeValueNotFound { getAttrWithoutDef(attrs, styleable, nonDefValueIndex) }
}
private fun <T> assertAttributeValueNotFound(function: () -> T) {
try {
function()
fail("${AttributeValueNotFoundException::class.simpleName} must be thrown")
} catch (e: AttributeValueNotFoundException) {
// success
}
}
companion object {
private const val DEF_BOOLEAN_VALUE: Boolean = false
private const val SET_BOOLEAN_VALUE: Boolean = true
private const val DEF_INTEGER_VALUE: Int = 10
private const val SET_INTEGER_VALUE: Int = 20
private const val DEF_FLOAT_VALUE: Float = 10f
private const val SET_FLOAT_VALUE: Float = 20f
private const val DEF_STRING_VALUE: String = "い"
private const val SET_STRING_VALUE: String = "あ"
private const val DEF_LONG_VALUE: Long = Long.MAX_VALUE / 2 + 1
private const val SET_LONG_VALUE: Long = Long.MAX_VALUE
private val DEF_STRING_ARRAY_VALUE: Array<String> = arrayOf("d", "e", "f")
private val SET_STRING_ARRAY_VALUE: Array<String> = arrayOf("a", "b", "c")
private val DEF_INTEGER_ARRAY_VALUE: IntArray = intArrayOf(4, 5, 6)
private val SET_INTEGER_ARRAY_VALUE: IntArray = intArrayOf(1, 2, 3)
}
}
まとめ
Instrumented Unit Test 専用の resource を追加してみました。
-
AttributeSet を外部公開するような方法も考えたが、View の 生成完了後に AttributeSet の内容が保持される保証が無く、また、現実に即していない形をわざわざ作ってのテストというのも無駄にリスクが増えるだけなので、実際に利用する場所でテストを実行することにした。 ↩