- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
在使用数据存储一段时间并用它编写一些业务逻辑之后,我想测试我的使用数据存储的类/存储库。我遇到的问题是,我该怎么做?我试图创建一个 context.createDatastore(...)
使用测试上下文,但我遇到了一些错误。没有关于此的官方文档,或者至少我找不到。
这是我目前的方法:
界面
interface ShopFilterDataStoreRepository {
val shopFilterProduct: Flow<ShopFilterProduct>
val shopFilterList: Flow<ShopFilterList>
suspend fun setValueProduct(newProduct: ShopFilterTempHolder)
suspend fun setValueList(newList: ShopFilterTempHolder)
}
实现(生产中使用的默认实现)
class ShopFilterDataStoreRepositoryImpl @Inject constructor(
@ApplicationContext private val context: Context,
) : ShopFilterDataStoreRepository {
private companion object {
private const val SHOP_FILTER_PRODUCT_DATASTORE: String = "shop_filter_product_datastore"
private const val SHOP_FILTER_LIST_DATASTORE: String = "shop_filter_list_datastore"
}
// this causes error because test context != applicationcontext
private val nonVolatileProductDataStore = context.createDataStore(
fileName = SHOP_FILTER_PRODUCT_DATASTORE,
serializer = ShopFilterProductSerializer
)
// this causes error because test context != applicationcontext
private val nonVolatileListDataStore = context.createDataStore(
fileName = SHOP_FILTER_LIST_DATASTORE,
serializer = ShopFilterListSerializer
)
override val shopFilterProduct: Flow<ShopFilterProduct>
get() = nonVolatileProductDataStore.data
override val shopFilterList: Flow<ShopFilterList>
get() = nonVolatileListDataStore.data
override suspend fun setValueProduct(newProduct: ShopFilterTempHolder) {
if (newProduct.id == null || newProduct.mQuery == null) return
nonVolatileProductDataStore.updateData { preferences ->
preferences.toBuilder().apply {
positionId = newProduct.id
query = newProduct.mQuery
}.build()
}
}
override suspend fun setValueList(newList: ShopFilterTempHolder) {
if (newList.id == null || newList.mQuery == null) return
nonVolatileListDataStore.updateData { preferences ->
preferences.toBuilder().apply {
positionId = newList.id
query = newList.mQuery
mQueryDirection = newList.mQueryDirection
}.build()
}
}
}
我的测试
@RunWith(RobolectricTestRunner::class)
@Config(maxSdk = Build.VERSION_CODES.P, minSdk = Build.VERSION_CODES.P)
class ShopFilterValidatorTest {
@get:Rule
var instantExecutorRule = InstantTaskExecutorRule()
private val context = InstrumentationRegistry.getInstrumentation().targetContext
private lateinit var shopFilterValidator: ShopFilterValidator
private lateinit var shopFilterDataStoreRepository: ShopFilterDataStoreRepository
@Before
fun setup() {
shopFilterDataStoreRepository = ShopFilterDataStoreRepositoryImpl(context)
shopFilterValidator = ShopFilterValidator(shopFilterDataStoreRepository)
}
@Test
fun `test should start`() {
}
}
错误堆栈跟踪
java.lang.ExceptionInInitializerError
at androidx.datastore.core.DataStoreFactory.create(DataStoreFactory.kt:66)
at androidx.datastore.DataStoreFactoryKt.createDataStore(DataStoreFactory.kt:58)
at androidx.datastore.DataStoreFactoryKt.createDataStore$default(DataStoreFactory.kt:56)
at com.example.app.framework.datasource.repository.shop.filter.ShopFilterDataStoreRepositoryImpl.<init>(ShopFilterDataStoreRepositoryImpl.kt:28)
at com.example.app.validator.ShopFilterValidatorTest.setup(ShopFilterValidatorTest.kt:30)
at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:59)
at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12)
at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:56)
at org.junit.internal.runners.statements.RunBefores.invokeMethod(RunBefores.java:33)
at org.junit.internal.runners.statements.RunBefores.evaluate(RunBefores.java:24)
at org.junit.rules.TestWatcher$1.evaluate(TestWatcher.java:61)
at org.junit.runners.ParentRunner$3.evaluate(ParentRunner.java:306)
at org.robolectric.RobolectricTestRunner$HelperTestRunner$1.evaluate(RobolectricTestRunner.java:575)
at org.robolectric.internal.SandboxTestRunner$2.lambda$evaluate$0(SandboxTestRunner.java:263)
at org.robolectric.internal.bytecode.Sandbox.lambda$runOnMainThread$0(Sandbox.java:89)
at java.util.concurrent.FutureTask.run(FutureTask.java:266)
at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1149)
at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:624)
at java.lang.Thread.run(Thread.java:748)
Caused by: java.lang.RuntimeException: java.lang.reflect.InvocationTargetException
at org.robolectric.internal.bytecode.ShadowWrangler.classInitializing(ShadowWrangler.java:166)
at org.robolectric.internal.bytecode.RobolectricInternals.classInitializing(RobolectricInternals.java:21)
at androidx.datastore.core.DataMigrationInitializer.<clinit>(DataMigrationInitializer.kt)
at androidx.datastore.core.DataStoreFactory.$$robo$$androidx_datastore_core_DataStoreFactory$create(DataStoreFactory.kt:66)
at androidx.datastore.core.DataStoreFactory.create(DataStoreFactory.kt)
at androidx.datastore.DataStoreFactoryKt.$$robo$$androidx_datastore_DataStoreFactoryKt$createDataStore(DataStoreFactory.kt:58)
at androidx.datastore.DataStoreFactoryKt.createDataStore(DataStoreFactory.kt)
at androidx.datastore.DataStoreFactoryKt.createDataStore$default(DataStoreFactory.kt:56)
at com.example.app.framework.datasource.repository.shop.filter.ShopFilterDataStoreRepositoryImpl.<init>(ShopFilterDataStoreRepositoryImpl.kt:28)
at com.example.app.validator.ShopFilterValidatorTest.setup(ShopFilterValidatorTest.kt:30)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:498)
... 14 more
Caused by: java.lang.reflect.InvocationTargetException
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:498)
at org.robolectric.internal.bytecode.RobolectricInternals.performStaticInitialization(RobolectricInternals.java:61)
at org.robolectric.internal.bytecode.ShadowWrangler.classInitializing(ShadowWrangler.java:163)
... 27 more
Caused by: java.lang.BootstrapMethodError: java.lang.IllegalAccessError: tried to access class kotlin.jvm.internal.DefaultConstructorMarker from class androidx.datastore.core.DataMigrationInitializer$Companion
at androidx.datastore.core.DataMigrationInitializer$Companion.<init>(DataMigrationInitializer.kt)
at androidx.datastore.core.DataMigrationInitializer.__staticInitializer__(DataMigrationInitializer.kt)
... 33 more
Caused by: java.lang.IllegalAccessError: tried to access class kotlin.jvm.internal.DefaultConstructorMarker from class androidx.datastore.core.DataMigrationInitializer$Companion
... 35 more
最佳答案
两种选择:
RobolectricTestRunner
并手动添加androidx.datastore
将包裹发送至不是 乐器 :class DataStoreRobolectricTestRunner : RobolectricTestRunner {
constructor(testClass: Class<*>) : super(testClass) {}
constructor(testClass: Class<*>, injector: Injector) : super(testClass, injector) {}
override fun createClassLoaderConfig(method: FrameworkMethod?): InstrumentationConfiguration {
val parentClassLoaderConfig = super.createClassLoaderConfig(method)
val builder = InstrumentationConfiguration.Builder(parentClassLoaderConfig)
builder.doNotInstrumentPackage("androidx.datastore")
return builder.build()
}
}
关于android - 如何测试 androidx (proto) 数据存储,java.lang.ExceptionInInitializerError,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/65685099/
专家: 我有两个 .proto 文件(如下所示),第二个依赖于第一个。 ImageMessage.proto: package dvr; message ImageMessage { re
我按照找到的例子 here .但是每当我输入命令“C:/Program Files/protoc/bin/protoc”object_detection/protos/.proto --python_
我有一个名为 myProtos 的库,它看起来像这样 . |-- proto |---- hello.proto | |-- generated └---- hello.pb.go 我在外部有一个名为
我想知道如何正确引用外部原型(prototype)文件。假设我有一个引用标准 protobuf 类型的 .proto 文件,例如 Timestamp : syntax = "proto3"; pack
我是 Google Protobuf 新手。我想解决一个问题,但在互联网上找不到答案,也许我问错了问题.. 我想构建一个使用 .proto 模式(和生成的类)的工具,并将输入字符串从可读字符串转换为
我有这个问题: grpc_tools._protoc_compiler.ProtocErrors: google/protobuf/timestamp.proto:-1:0 error: File n
软件环境:Win10,Goland。 去版本:go1.9.1 windows/amd64.协议(protocol)——版本:libprotoc 3.5.1当我执行命令时。结果如下 protoc -I
我正在使用 Android Studio 来处理 tomahawk-android项目,所以我使用 git 将该项目直接克隆到 Android Studio 中,但无法构建,并显示; Could no
一 点睛 可以在 proto 中定义多个类,然后从中选择任意一个类,可以在 proto 中定义多个 message,然后再将多个 message 以枚举的形式供使用时选择。 二 实战 1 proto
我在不同的 go 包下编译了 2 个 proto,但是当我在服务器中注册它们并运行它时,我得到: panic: proto: file "common.proto" is already regist
perl6 如何决定哪个 proto token先匹配? 下面的代码按预期工作,它匹配字符串 1234 , 和 Grammar::Tracer显示匹配的第一个标记是 s:sym ,这是有道理的,因为它
我有 2 个 grpc 服务( service1 和 service2 ),它们相互交互,在某些情况下,service1 的 rpc 响应will 由 service2 中定义的结构组成,在经历了不可
我有一个 MyResponse.proto 文件,它导入两个 .proto 文件 Alternative.proto 和 Index.proto >. MyResponse.proto 文件: imp
我有一个非常简单的 android 应用程序,它使用 protobuf 从服务器获取数据,然后让用户浏览数据树(简化) 现在我只想在我的结构中使用这些数据,所以我使用 protobuf 回复(通过自动
我有两个服务:Story 和 Tag。这些文件的结构如下: Story |-- StoryService | `-- proto | `-- storyservice.proto `-
对于 CPP 编程,我在 .proto 文件中定义了一个 enum,我必须在另一个 .proto 文件中使用相同的枚举。 //first.proto package A; enum foo {
如果 protobuf 消息中的属性是第一个成员或第二个成员,有什么区别? 我拥有的 Request proto 消息有 2 个字段。我被要求交换属性的位置 message SomeRequest {
我正在使用 protobuf-net 的 ProtoBufTool 自定义工具从 .proto 文件生成 C# 类。到目前为止,我一直无法让原型(prototype)文件中的 import 语句使用相
我正在从事 AI 项目,但我在 Python 方面仍然不是很有经验。 我正在尝试 build and test this project . 我遵循了所有说明,但是当我尝试启动 python 脚本时仍
我有一个名为 Game 的对象,它具有名称、编号、团队等属性。它的许多属性本身就是对象:团队是团队对象的列表。例如,所有这些对象都具有使用 proto 定义的函数(不确定实际术语) Game.prot
我是一名优秀的程序员,十分优秀!