9

我有一个使用 FakeApplication 和嵌入式 mongodb 数据库的 specs2 测试。

def inMemoryMongoDatabase(name: String = "default"): Map[String, String] = {
    val dbname: String = "play-test-" + scala.util.Random.nextInt
    Map(
        ("mongodb." + name + ".db" -> dbname),
        ("mongodb." + name + ".port" -> EmbeddedMongoTestPort.toString))
}

override def around[T <% Result](t: => T) = {
    running(FakeApplication(additionalConfiguration = inMemoryMongoDatabase(), additionalPlugins = Seq("se.radley.plugin.salat.SalatPlugin"))) {
        t // execute t inside a http session
    }
}

FakeApplication 使用 conf 目录中的默认 application.conf 配置和为每个测试创建的测试数据库的附加配置。
在我们设置 mongodb 副本集之前,这是有效的发现。现在 application.conf 包含此副本集的配置

mongodb.default.replicaset {
host1.host = "localhost"
host1.port = 27017
host2.host = "localhost"
host2.port = 27018
host3.host = "localhost"
host3.port = 27019
}

由于 FakeApplication 使用默认配置,因此测试失败,因为找不到副本集的主机。我想为我的测试设置不同的配置,基本上删除 mongodb.default.replicaset 条目。如果 mongodb.default.replicaset 是一个简单的 Map[String, String] 这将很容易,因为我可以将它添加到 additonalConfiguration 但是当我尝试这样做时它会失败,因为预期的值类型不是字符串而是对象。我还尝试通过路径参数向 FakeApplication 提供单独的 test.conf 文件。

override def around[T <% Result](t: => T) = {
    running(FakeApplication(path = new java.io.File("conf/test.conf"), additionalConfiguration = inMemoryMongoDatabase(), additionalPlugins = Seq("se.radley.plugin.salat.SalatPlugin"))) {
        t // execute t inside a http session
    }
}

这也不起作用,因为它没有加载任何配置。

我将不胜感激任何帮助。谢谢。

克里斯

4

5 回答 5

10

我们在为集成测试加载额外配置时遇到了类似的问题。我们发现手动填充地图很乏味,因此我们使用了以下方法:

private Configuration additionalConfigurations;
@Before
public void initialize(){
    Config additionalConfig = ConfigFactory.parseFile(new File("conf/integration.conf"));
    additionalConfigurations = new Configuration(additionalConfig);
}
@Test
public void testPropertiesGetLoaded() throws Exception{
    running(testServer(3333, fakeApplication(additionalConfigurations.asMap())), HTMLUNIT, new Callback<TestBrowser>(){
        public void invoke(TestBrowser browser){
            String specificProperty = Play.application().configuration().getString("specific.property");
            System.out.println(specificProperty);
        }
    });
}

我不知道在 Scala 方面是否有一个不错的方法,我们正在用 java 编写所有代码。

于 2013-10-04T14:57:54.050 回答
3

问题是如何在使用 Play 的 FakeAppication 运行集成测试时指定 test.conf 文件。在我的集成测试中,我无法调用play -Dconfig.file=conf/test.conf.

我设法做的是:

object FakeSalatApp extends Around {

 def EmbeddedMongoTestPort: Int = 27028

 def inMemoryMongoDatabase(name: String = "default"): Map[String, String] = {
   val dbname: String = "play-test-" + scala.util.Random.nextInt
   Map(
     ("mongodb." + name + ".db" -> dbname),
     ("mongodb." + name + ".port" -> EmbeddedMongoTestPort.toString),
     ("mongodb." + name + ".replicaset.host1.host" -> "localhost"),
     ("mongodb." + name + ".replicaset.host1.port" -> EmbeddedMongoTestPort.toString),
     ("mongodb." + name + ".replicaset.host2.host" -> "localhost"),
     ("mongodb." + name + ".replicaset.host2.port" -> (EmbeddedMongoTestPort + 1).toString),
     ("mongodb." + name + ".replicaset.host3.host" -> "localhost"),
     ("mongodb." + name + ".replicaset.host3.port" -> (EmbeddedMongoTestPort + 2).toString))
  }

 override def around[T <% Result](t: => T) = {
   running(FakeApplication(additionalConfiguration = inMemoryMongoDatabase(), additionalPlugins = Seq("se.radley.plugin.salat.SalatPlugin"))) {
     t // execute t inside a http session
   }
 }
}
于 2012-11-15T09:29:27.867 回答
2

这就是我在 Play 2.3.x 中的做法

  1. GlobalSettings在包中的类中定义我的应用AppGlobal程序(不是根包)

    package configs
    
    class AppGlobal extends GlobalSettings {
      // Your application global settings
      ???
    }
    
  2. 将应用程序全局设置定义为object Global extends AppGlobal您的应用程序使用的设置。

  3. 在测试类中,定义一个测试全局。最后添加测试配置以覆盖或添加到整体应用程序配置:

    object TestGlobal extends AppGlobal {
      override def onLoadConfig(config: Configuration, 
                                path: File, 
                                classloader: ClassLoader, 
                                mode: Mode): Configuration = {
        config ++ configuration ++ 
              Configuration.load(path, mode = Mode.Dev, 
                                 Map("config.file" -> "conf/test.conf"))
        }
    }
    
  4. 使用上述创建假应用程序TestGlobal

    FakeApplication(withGlobal = Some(TestGlobal))
    
于 2014-12-08T02:12:36.017 回答
1

就我而言,我只是创建了一个所有测试都扩展的基类。在创建 FakeApplication 之前,我定义了设置应用程序配置的系统属性config.resource。然后我将我的配置结构如下:

application.conf : 包含 no-env 特定配置

test.conf :包括 application.conf 并定义运行单元测试的配置

env_local.conf :包括 application.conf 并定义配置以在本地运行应用程序

env_prod.conf :类似于 env_local.conf 但用于生产等...

在我的项目中,为方便起见,我创建了一个脚本 local.sh,它只运行 activator -Dconfig.resource=env_local.conf

@RunWith(classOf[JUnitRunner])
class ApplicationTest extends FunSuite with MockitoSugar {
   System.setProperty("config.resource", "test.conf")
   val app = Helpers.fakeApplication()
}
于 2015-07-03T19:11:15.630 回答
0

Usingpath在这里不起作用,因为这是您正在运行的 FakeApplication 的路径(在某些情况下您可能有不同的路径)。

在您的情况下,我的建议是test.conf在运行 Play 以进行测试模式时指定一个,例如

play -Dconfig.file=conf/test.conf

然后test.conf会被接走。然后,您还可以让它包含您的正常application.conf并仅覆盖 mongo 设置。

也许将“单一目标模式”作为连接到 mongodb 的默认方式也是有意义的application.conf,并覆盖 mongob 配置以仅在生产配置中使用副本集。

于 2012-10-25T06:50:46.530 回答