我正在编写一个Scala应用程序(应该使用Spark在Hadoop上运行),我的用户将执行他们上传的JavaScript代码段,并且我希望提供对Scala编写的某些帮助程序函数的访问权限(例如"向这些JavaScript用户发出HTTP调用等等。所以我要做的是编写一个大的JavaScriptHelpers
对象,然后使用
engine = scriptEngineManager.getEngineByName("JavaScript")
engine.put("jql", JavaScriptHelpers)
因此用户可以在JavaScript中说出jql.httpPost(...)
。使这成为可能的Scala代码如下所示:
def httpPost(where: String, params: Object): Try[String] = {
params match {
// JavaScript string becomes Java's String:
case body: String =>
// ...
// JavaScript object becomes Java's ScriptableObject
case obj: ScriptableObject =>
val params = convertToMap(obj)
// ...
}
}
protected def convertToMap(obj: ScriptableObject): Map[String, String] = {
(for (key <- obj.getIds().toList) yield {
(key.toString, obj.get(key) match {
case s: String =>
s
case d: java.lang.Double if d.toString.endsWith(".0") =>
d.toInt.toString
case other => other.toString
})
}).toMap
}
我发现访问存储在JavaScript对象中的信息的唯一方法是将它们视为sun.org.mozilla.javascript.ScriptableObject
的实例。现在这就像我当地的OpenJDK安装上的魅力一样
java version "1.7.0_75"
OpenJDK Runtime Environment (fedora-2.5.4.2.fc20-x86_64 u75-b13)
OpenJDK 64-Bit Server VM (build 24.75-b04, mixed mode)
但是当我在运行
的Hadoop集群上运行相同的代码时java version "1.7.0_67"
Java(TM) SE Runtime Environment (build 1.7.0_67-b01)
Java HotSpot(TM) 64-Bit Server VM (build 24.65-b04, mixed mode)
然后我得到:
java.lang.NoClassDefFoundError: sun/org/mozilla/javascript/ScriptableObject
sun.org.mozilla.javascript.internal.JavaMembers.discoverAccessibleMethods(JavaMembers.java:383)
sun.org.mozilla.javascript.internal.JavaMembers.discoverAccessibleMethods(JavaMembers.java:335)
sun.org.mozilla.javascript.internal.JavaMembers.reflect(JavaMembers.java:455)
sun.org.mozilla.javascript.internal.JavaMembers.<init>(JavaMembers.java:76)
sun.org.mozilla.javascript.internal.JavaMembers.lookupClass(JavaMembers.java:847)
sun.org.mozilla.javascript.internal.NativeJavaObject.initMembers(NativeJavaObject.java:88)
sun.org.mozilla.javascript.internal.NativeJavaObject.<init>(NativeJavaObject.java:78)
sun.org.mozilla.javascript.internal.NativeJavaObject.<init>(NativeJavaObject.java:68)
...
并查看Oracle与JDK 7捆绑在一起的Rhino版本,可以从http://www.oracle.com/technetwork/opensource/jdk7-source-1634015.html下载,似乎所有sun.org.mozilla.javascript.*
类都已移至sun.org.mozilla.javascript.internal.*
。
现在我该如何应对这种情况?是否有任何与Rhino无关的方式来访问Java中JavaScript对象的字段?或者,如何在我的本地环境中使用...javascript.internal.ScriptableObject
时强制Oracle JVM使用...javascript.ScriptableObject
?
任何帮助都非常感激。
答案 0 :(得分:1)
您可以使用函数重载。
// `params` matches a JavaScript string
def httpPost(where: String, params: String): Try[String] = {
// ...
}
// `params` matches a JavaScript object
def httpPost(where: String, params: java.util.Map[_, _]): Try[String] = {
// ...
}
此解决方案适用于我的环境(Oracle JDK 8和OpenJDK 1.7)。