1

我正在尝试使用 sbt 和jberkel/android-plugin在 Scala 和 Akka 中设置简单的 Android 项目。我成功地根据android-plugin 的入门教程创建了简单的应用程序。我还设法组装了 ProGuard 配置,使我能够将 Akka 混合到项目中。

现在我想添加 Akka 配置文件,但我无法做到。我在src/main/resources文件夹中创建了它,并希望它能够正常工作。不幸的是,在 APK 组装过程中,sbt 失败并出现以下错误:

[trace] Stack trace suppressed: run last Akkdroid/android:package-debug for the full output.
[error] (Akkdroid/android:package-debug) 
[error] Using keystore: /home/ghik/.android/debug.keystore
[error] Packaging akkdroid-0.1.apk
[error] /data/Studia/S10/Mobilne/akkdroid/target/resources.apk:
[error] => res/layout/main.xml
[error] => AndroidManifest.xml
[error] => resources.arsc
[error] /data/Studia/S10/Mobilne/akkdroid/target/classes.dex => classes.dex
[error] /data/Studia/S10/Mobilne/akkdroid/target/classes.min.jar:
[error] => akkdroid.conf
[error] => library.properties
[error] => reference.conf
[error] => org/jboss/netty/container/spring/beans.xml
[error] /data/Studia/S10/Mobilne/akkdroid/src/main/resources/akkdroid.conf => akkdroid.conf
[error] 
[error] Error packaging /data/Studia/S10/Mobilne/akkdroid/target/akkdroid-0.1.apk: Duplicate files at the same path inside the APK
[error] Total time: 20 s, completed 2013-04-24 20:32:47

从这些消息中,我推断我的akkdroid.conf文件首先被复制(由 ProGuard?)classes.min.jar,然后sbt尝试构建包,包括来自两者的资源,classes.min.jarsrc/main/resources最终看到我的文件两次。

不幸的是,我不知道为什么会发生这种情况以及如何解决它。

如果有帮助,这是我的sbt构建文件(主要从jberkel 的模板生成):

import sbt._

import Keys._
import AndroidKeys._

object General {
  val settings = Defaults.defaultSettings ++ Seq(
    name := "Akkdroid",
    version := "0.1",
    versionCode := 0,
    scalaVersion := "2.10.1",
    platformName in Android := "android-10",
    javacOptions ++= Seq("-source", "1.6", "-target", "1.6")
  )

  val proguardSettings = Seq(
    useProguard in Android := true,
    proguardOption in Android :=
      """-keepclassmembers class * {
        |  ** MODULE$;
        | }
        |-keep public class akka.actor.LocalActorRefProvider {
        |public <init>(...);
        |}
        |-keep public class akka.remote.RemoteActorRefProvider {
        |public <init>(...);
        |}
        |-keep class akka.actor.SerializedActorRef {
        |*;
        |}
        |-keep class akka.remote.netty.NettyRemoteTransport {
        |*;
        |}
        |-keep class akka.serialization.JavaSerializer {
        |*;
        |}
        |-keep class akka.serialization.ProtobufSerializer {
        |*;
        |}
        |-keep class com.google.protobuf.GeneratedMessage {
        |*;
        |}
        |-keep class akka.event.Logging*
        |-keep class akka.event.Logging$LogExt{
        |*;
        |}
        |-keep class scala.Option
        |-keep class scala.Function1
        |-keep class scala.PartialFunction
        |-keep class scala.collection.SeqLike {
        |public protected *;
        |}
        |-keep class akka.**
        |-keepclassmembers class akka.**
        |-keep class org.omg.**
        |-keep class scala.Tuple2
        |-dontskipnonpubliclibraryclassmembers
        |-dontskipnonpubliclibraryclasses
      """.stripMargin
  )

  lazy val fullAndroidSettings =
    General.settings ++
      AndroidProject.androidSettings ++
      TypedResources.settings ++
      proguardSettings ++
      AndroidManifestGenerator.settings ++
      AndroidMarketPublish.settings ++ Seq(
      keyalias in Android := "change-me",
      libraryDependencies += "org.scalatest" %% "scalatest" % "1.9.1" % "test"
    )
}

object AndroidBuild extends Build {
  javacOptions ++= Seq("-target", "1.6")

  lazy val main = Project(
    "Akkdroid",
    file("."),
    settings = General.fullAndroidSettings ++ Seq(
      libraryDependencies += "com.typesafe.akka" % "akka-actor_2.10" % "2.1.2",
      libraryDependencies += "com.typesafe.akka" % "akka-remote_2.10" % "2.1.2"
    )
  )

  lazy val tests = Project(
    "tests",
    file("tests"),
    settings = General.settings ++
      AndroidTest.androidSettings ++
      General.proguardSettings ++ Seq(
      name := "AkkdroidTests"
    )
  ) dependsOn main
}

我将非常感谢一些帮助。我几乎没有 Android 或 SBT 方面的经验。

4

2 回答 2

1

这是另一种似乎可行的方法,并且避免了使用 AndroidContext将 Akka 配置加载为原始 android 资源的需要。我不确定它是否比将配置文件视为原始 Android 资源更“正确”。

config我在项目的顶层创建了一个名为的目录sbt,并将我的 Akka 配置文件命名application.conf为 。

myapp/
    config/
        application.conf
    project/
        Build.scala
        plugins.sbt
    src/
        main/
            scala/

然后我将以下设置添加到我的 Build.scala 中:

unmanagedClasspath in Runtime <+= (baseDirectory) map { bd => Attributed.blank(bd / "config") }

我正在使用Jan Berkel 的 Android 应用程序模板,我的Build.scala. 上下文中的相关设置看起来像

object General {
  val settings = Defaults.defaultSettings ++ Seq (
    name := "AkkaWeb",
    version := "0.1",
    <other settings elided>...
    unmanagedClasspath in Runtime <+= (baseDirectory) map { bd => Attributed.blank(bd / "config") }
)

我必须做一个sbtclean和.reloadapplication.conf.apkreference.confActorSystem

值得注意的一点:如果我在我的主线程中构建,那么在严格模式下ActorSystem运行时读取配置文件会导致磁盘访问冲突。这是一个单独的问题,与此问题无关,但需要注意一些事项,尤其是当我在构建.ActorSystem

此解决方案直接从 Akka 文档here复制而来,您可以在此处阅读更多详细信息以了解它到底在做什么。

更新

我刚刚注意到别的东西。Build.scala我的影响中reference.conf出现在.apk文件中的 Maven 工件的顺序。如果我希望 jar 中的文件出现在我的文件中,我必须将其放在其他具有与 Typesafe 配置库一起使用的文件的akka-actor工件之前。例如:reference.confreference.confakka-actorreference.conf.apk

libraryDependencies ++= Seq(
    "com.typesafe.akka" % "akka-actor_2.10" % "2.1.4",
    "io.spray" % "spray-util" % "1.1-M7.1-openssl-M6"
)

如果这两个工件的顺序颠倒了,那么reference.conf结果文件中的.apk文件将是来自 Spray 工件而不是 Akka 工件的文件。

于 2013-05-27T00:09:23.613 回答
1

我有同样的问题。我不知道这是否是最好的处理方法,但我将我的配置文件放在了android项目的raw目录中。然后我将它解析为一个Config对象并将其传递给ActorSystem构造函数。

object AkkaSystem {
  private var systemOption: Option[ActorSystem] = None

  def system(implicit context: Context): ActorSystem = synchronized {
    systemOption match {
      case None =>
        val reader = new InputStreamReader(
          context.getResources.openRawResource(R.raw.configuration)
        )
        systemOption = Option (
          ActorSystem("MySystem", ConfigFactory.parseReader(reader))
        )
        reader.close()
        systemOption.get
      case Some(existingSystem) => existingSystem
    }
  }
}

然后在我的Activity课堂上,我可以使用该AkkaSystem对象来获取配置ActorSystem

class MainActivity extends Activity {
  implicit val context = this

  private def mainActor = AkkaSystem.system.actorOf(Props[MainActor], "main")
}

我同意,如果我可以避免所有这些代码并且需要通过Context以获取 Android 资源,那会更好。

于 2013-05-14T19:48:58.353 回答