Newer
Older
GitBucket / src / main / scala / plugin / PluginSystem.scala
package plugin

import app.Context
import javax.servlet.http.{HttpServletResponse, HttpServletRequest}
import scala.collection.mutable.ListBuffer
import org.slf4j.LoggerFactory
import org.mozilla.javascript.{Context => JsContext}
import org.mozilla.javascript.{Function => JsFunction}

/**
 * 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: JsFunction): Unit = {
      repositoryMenuList += RepositoryMenu(label, name, url, icon, (context) => {
        val context = JsContext.enter()
        try {
          condition.call(context, condition, condition, Array(context)).asInstanceOf[Boolean]
        } finally {
          JsContext.exit()
        }
      })
    }

//    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: JsFunction): Unit = {
      globalMenuList += GlobalMenu(label, url, icon, (context) => {
        val context = JsContext.enter()
        try {
          condition.call(context, condition, condition, Array(context)).asInstanceOf[Boolean]
        } finally {
          JsContext.exit()
        }
      })
    }

//    def addGlobalAction(path: String)(function: (HttpServletRequest, HttpServletResponse) => Any): Unit = {
//      globalActionList += Action(path, function)
//    }

    def addGlobalAction(path: String, function: JsFunction): Unit = {
      globalActionList += Action(path, (request, response) => {
        val context = JsContext.enter()
        try {
          function.call(context, function, function, Array(request, response))
        } finally {
          JsContext.exit()
        }
      })
    }

//    def addRepositoryAction(path: String)(function: (HttpServletRequest, HttpServletResponse) => Any): Unit = {
//      repositoryActionList += Action(path, function)
//    }

    def addRepositoryAction(path: String, function: JsFunction): Unit = {
      repositoryActionList += Action(path, (request, response) => {
        val context = JsContext.enter()
        try {
          function.call(context, function, function, Array(request, response))
        } finally {
          JsContext.exit()
        }
      })
    }

  }

  def definePlugin(id: String, author: String, url: String, description: String): Plugin = new Plugin(id, author, url, description)

  def evaluateJavaScript(script: String): Any = {
    val context = JsContext.enter()
    try {
      val scope = context.initStandardObjects()
      scope.put("PluginSystem", scope, this)
      val result = context.evaluateString(scope, script, "<cmd>", 1, null)
      result
    } finally {
      JsContext.exit
    }


//    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!"
//  }

}