Skip to content

Commit 15dd8f3

Browse files
committed
Tentatively, dynamically loaded plugins seem to work. (re: #288)
- Added a 'Plugin' manager object that dynamically loads a jar, parses out plugin metadata, and in - Added --plugin/-P command-line option to load plugins at startup - Added a vizier-plugins option to the Vizier properties file to load plugins at startup - Fixed typo in capitalization of Class__L__oader in ClassLoaderUtils
1 parent 3d73d5b commit 15dd8f3

File tree

8 files changed

+142
-3
lines changed

8 files changed

+142
-3
lines changed

plugins/mimir-pip.jar

116 KB
Binary file not shown.

vizier/backend/src/info/vizierdb/Config.scala

+10
Original file line numberDiff line numberDiff line change
@@ -137,6 +137,16 @@ class Config(arguments: Seq[String])
137137
descr = "Set the SparkSQL warehouse directory (default: {cache-dir}/spark-warehouse)"
138138
)
139139

140+
val extraPlugins = opt[List[File]]("plugin",
141+
short = 'P',
142+
descr = "Enable a plugin for this session"
143+
)
144+
145+
lazy val plugins:Seq[File] =
146+
Option(defaults.getProperty("vizier-plugins"))
147+
.map { _.split(":").map { new File(_) }.toSeq }
148+
.getOrElse { Seq() } ++ extraPlugins()
149+
140150
def workingDirectoryFile: File =
141151
new File(
142152
workingDirectory
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,85 @@
1+
package info.vizierdb
2+
3+
import play.api.libs.json._
4+
import java.net.URLClassLoader
5+
import java.net.URL
6+
import java.io.File
7+
import java.io.FileNotFoundException
8+
import com.typesafe.scalalogging.LazyLogging
9+
import scala.collection.mutable
10+
import info.vizierdb.util.ClassLoaderUtils
11+
12+
case class Plugin(
13+
name: String,
14+
schema_version: Int,
15+
plugin_class: String,
16+
description: String,
17+
documentation: Option[String],
18+
)
19+
{
20+
}
21+
22+
object Plugin
23+
extends LazyLogging
24+
{
25+
26+
val loaded = mutable.Map[String, Plugin]()
27+
28+
29+
implicit val pluginFormat: Format[Plugin] = Json.format
30+
31+
def load(file: File): Plugin =
32+
{
33+
val jar =
34+
if(file.isAbsolute){ file }
35+
else {
36+
Vizier.config.workingDirectoryFile.toPath
37+
.resolve(file.toPath)
38+
.toFile
39+
}
40+
41+
if(!jar.exists()){ throw new FileNotFoundException(jar.getAbsoluteFile.toString) }
42+
val url = jar.getAbsoluteFile().toURI().toURL()
43+
44+
// It feels like a bit of a hack to just stack plugins onto the currently running
45+
// ClassLoader... but this seems to be the only way to make the loaded classes visible
46+
// to the running spark instance.
47+
val loader = new URLClassLoader(
48+
Array(jar.toURI.toURL),
49+
Thread.currentThread().getContextClassLoader()
50+
)
51+
Thread.currentThread().setContextClassLoader(loader)
52+
53+
val plugin =
54+
Json.parse(
55+
loader.getResourceAsStream("vizier-plugin.json")
56+
).asOpt[Plugin]
57+
.getOrElse {
58+
throw new RuntimeException(s"$jar is not a valid Vizier Plugin")
59+
}
60+
61+
val detail = s"${plugin.name} [$jar]"
62+
63+
assert(
64+
plugin.schema_version > 0 && plugin.schema_version <= 1,
65+
s"Unsupported version '${plugin.schema_version} for plugin $detail"
66+
)
67+
68+
assert(
69+
!loaded.contains(plugin.name),
70+
s"Plugin ${plugin.name} is already loaded."
71+
)
72+
73+
loaded.put(plugin.name, plugin)
74+
75+
val clazz = Class.forName(plugin.plugin_class, true, loader)
76+
val singleton = clazz.getDeclaredField("MODULE$").get()
77+
val initMethod = clazz.getMethod("init", Vizier.sparkSession.getClass)
78+
79+
ClassLoaderUtils.withContextClassLoader(loader) {
80+
initMethod.invoke(singleton, Vizier.sparkSession)
81+
}
82+
83+
return plugin
84+
}
85+
}

vizier/backend/src/info/vizierdb/Vizier.scala

+15
Original file line numberDiff line numberDiff line change
@@ -159,6 +159,15 @@ object Vizier
159159
}
160160
}
161161

162+
def loadPlugins(plugins: Seq[File]): Unit =
163+
{
164+
for(p <- plugins){
165+
println(s" ...loading plugin $p")
166+
val plugin = Plugin.load(p)
167+
println(s" ...loaded ${plugin.name}")
168+
}
169+
}
170+
162171
def setWorkingDirectory(): Unit =
163172
{
164173
if(config.workingDirectory.isDefined){
@@ -200,6 +209,12 @@ object Vizier
200209
println("Starting Spark...")
201210
initSpark()
202211

212+
// Set up plugins
213+
if(!config.plugins.isEmpty){
214+
println("Loading plugins...")
215+
loadPlugins(config.plugins)
216+
}
217+
203218
config.subcommand match {
204219
//////////////// HANDLE SPECIAL COMMANDS //////////////////
205220
case Some(subcommand) =>

vizier/backend/src/info/vizierdb/util/ClassLoaderUtils.scala

+1-1
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,7 @@ package info.vizierdb.util
1616

1717
object ClassLoaderUtils
1818
{
19-
def withContextClassloader[T](classloader: ClassLoader)(f: => T): T =
19+
def withContextClassLoader[T](classloader: ClassLoader)(f: => T): T =
2020
{
2121
val originalClassloader = Thread.currentThread().getContextClassLoader()
2222
Thread.currentThread().setContextClassLoader(classloader)

vizier/backend/src/info/vizierdb/viztrails/RunningCell.scala

+1-1
Original file line numberDiff line numberDiff line change
@@ -80,7 +80,7 @@ class RunningCell(
8080

8181
def exec(): Boolean =
8282
{
83-
ClassLoaderUtils.withContextClassloader(workflowTask.classloader) {
83+
ClassLoaderUtils.withContextClassLoader(workflowTask.classloader) {
8484
try {
8585
processSynchronously match {
8686
case _ if aborted.get =>

vizier/backend/src/info/vizierdb/viztrails/RunningWorkflow.scala

+1-1
Original file line numberDiff line numberDiff line change
@@ -62,7 +62,7 @@ class RunningWorkflow(workflow: Workflow, val classloader: ClassLoader)
6262

6363
def exec: Boolean =
6464
{
65-
ClassLoaderUtils.withContextClassloader[Boolean](classloader){
65+
ClassLoaderUtils.withContextClassLoader[Boolean](classloader){
6666
val ret =
6767
try {
6868
logger.info(s"Starting execution of Workflow ${workflow.id}")
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,29 @@
1+
package info.vizierdb
2+
3+
import org.specs2.mutable.Specification
4+
import org.specs2.specification.BeforeAll
5+
import info.vizierdb.test.SharedTestResources
6+
import java.io.File
7+
import info.vizierdb.util.ClassLoaderUtils
8+
9+
class PluginTest extends Specification with BeforeAll
10+
{
11+
def beforeAll(): Unit =
12+
{
13+
SharedTestResources.init()
14+
}
15+
16+
"Load Class Jars" >>
17+
{
18+
val jar = new File("plugins/mimir-pip.jar")
19+
20+
val plugin = Plugin.load(jar)
21+
22+
val df = Vizier.sparkSession.sql("""
23+
SELECT gaussian(5.0, 1.0)
24+
""")
25+
26+
ok
27+
}
28+
29+
}

0 commit comments

Comments
 (0)