Files
GitBucket/src/main/scala/plugin/PluginSystem.scala
2014-06-10 10:44:36 +09:00

107 lines
6.0 KiB
Scala

package plugin
import app.Context
import javax.servlet.http.{HttpServletResponse, HttpServletRequest}
import javax.script.ScriptEngineManager
import scala.collection.mutable.ListBuffer
import org.slf4j.LoggerFactory
import jdk.nashorn.api.scripting.ScriptObjectMirror
/**
* Provides extension points to plug-ins.
*/
object PluginSystem {
private val logger = LoggerFactory.getLogger(PluginSystem.getClass)
private val pluginsMap = scala.collection.mutable.Map[String, Plugin]()
def install(plugin: Plugin): Unit = {
pluginsMap.put(plugin.id, plugin)
}
def plugins: List[Plugin] = pluginsMap.values.toList
def uninstall(id: String): Unit = {
pluginsMap.remove(id)
}
def repositoryMenus : List[RepositoryMenu] = pluginsMap.values.flatMap(_.repositoryMenuList).toList
def globalMenus : List[GlobalMenu] = pluginsMap.values.flatMap(_.globalMenuList).toList
def repositoryActions : List[Action] = pluginsMap.values.flatMap(_.repositoryActionList).toList
def globalActions : List[Action] = pluginsMap.values.flatMap(_.globalActionList).toList
// Case classes to hold plug-ins information internally in GitBucket
case class GlobalMenu(label: String, url: String, icon: String, condition: Context => Boolean)
case class RepositoryMenu(label: String, name: String, url: String, icon: String, condition: Context => Boolean)
case class Action(path: String, function: (HttpServletRequest, HttpServletResponse) => Any)
/**
* This is a plug-in definition class.
*/
class Plugin(val id: String, val author: String, val url: String, val description: String) {
private[PluginSystem] val repositoryMenuList = ListBuffer[RepositoryMenu]()
private[PluginSystem] val globalMenuList = ListBuffer[GlobalMenu]()
private[PluginSystem] val repositoryActionList = ListBuffer[Action]()
private[PluginSystem] val globalActionList = ListBuffer[Action]()
def addRepositoryMenu(label: String, name: String, url: String, icon: String)(condition: Context => Boolean): Unit = {
repositoryMenuList += RepositoryMenu(label, name, url, icon, condition)
}
def addRepositoryMenu(label: String, name: String, url: String, icon: String, condition: ScriptObjectMirror): Unit = {
repositoryMenuList += RepositoryMenu(label, name, url, icon, (context) => condition.call(this, context).asInstanceOf[Boolean])
}
def addGlobalMenu(label: String, url: String, icon: String)(condition: Context => Boolean): Unit = {
globalMenuList += GlobalMenu(label, url, icon, condition)
}
def addGlobalMenu(label: String, url: String, icon: String, condition: ScriptObjectMirror): Unit = {
globalMenuList += GlobalMenu(label, url, icon, (context) => condition.call(this, context).asInstanceOf[Boolean])
}
def addGlobalAction(path: String)(function: (HttpServletRequest, HttpServletResponse) => Any): Unit = {
globalActionList += Action(path, function)
}
def addGlobalAction(path: String, function: ScriptObjectMirror): Unit = {
globalActionList += Action(path, (request, response) => function.call(this, request, response))
}
def addRepositoryAction(path: String)(function: (HttpServletRequest, HttpServletResponse) => Any): Unit = {
repositoryActionList += Action(path, function)
}
def addRepositoryAction(path: String, function: ScriptObjectMirror): Unit = {
repositoryActionList += Action(path, (request, response) => function.call(this, request, response))
}
}
def evaluateJavaScript(script: String): Any = {
val engine = new ScriptEngineManager().getEngineByName("JavaScript")
logger.debug("Script: " + script)
engine.put("PluginSystem", this)
// TODO Support both of Nashorn and Rhino!
val result = engine.eval("var Plugin = Java.type(\"plugin.PluginSystem.Plugin\"); " + script)
logger.debug("Result: " + result)
result
}
// TODO This is a test
// addGlobalMenu("Google", "http://www.google.co.jp/", "")
// { context => context.loginAccount.isDefined }
//
// addRepositoryMenu("Board", "board", "/board", "")
// { context => true}
//
// addGlobalAction("/hello"){ (request, response) =>
// "Hello World!"
// }
}