All Downloads are FREE. Search and download functionalities are using the official Maven repository.

xml.persistent.CachedFileStorage.scala Maven / Gradle / Ivy

/*                     __                                               *\
**     ________ ___   / /  ___     Scala API                            **
**    / __/ __// _ | / /  / _ |    (c) 2002-2013, LAMP/EPFL             **
**  __\ \/ /__/ __ |/ /__/ __ |    http://scala-lang.org/               **
** /____/\___/_/ |_/____/_/ | |                                         **
**                          |/                                          **
\*                                                                      */

package scala
package xml
package persistent

import java.io.{ File, FileOutputStream }
import java.nio.ByteBuffer
import java.nio.channels.Channels
import java.lang.Thread

import scala.collection.Iterator

/**
 * Mutable storage of immutable xml trees. Everything is kept in memory,
 *  with a thread periodically checking for changes and writing to file.
 *
 *  To ensure atomicity, two files are used, `filename1` and `'$'+filename1`.
 *  The implementation switches between the two, deleting the older one
 *  after a complete dump of the database has been written.
 *
 *  @author Burak Emir
 */
abstract class CachedFileStorage(private val file1: File) extends Thread {

  private val file2 = new File(file1.getParent, file1.getName + "$")

  /**
   * Either equals `file1` or `file2`, references the next file in which
   *  updates will be stored.
   */
  private var theFile: File = null

  private def switch() = { theFile = if (theFile == file1) file2 else file1; }

  /** this storage modified since last modification check */
  protected var dirty = false

  /** period between modification checks, in milliseconds */
  protected val interval = 1000

  /**
   * finds and loads the storage file. subclasses should call this method
   *  prior to any other, but only once, to obtain the initial sequence of nodes.
   */
  protected def initialNodes: Iterator[Node] = (file1.exists, file2.exists) match {
    case (false, false) =>
      theFile = file1
      Iterator.empty
    case (true, true) if (file1.lastModified < file2.lastModified) =>
      theFile = file2
      load
    case (true, _) =>
      theFile = file1
      load
    case _ =>
      theFile = file2
      load
  }

  /** returns an iterator over the nodes in this storage */
  def nodes: Iterator[Node]

  /** adds a node, setting this.dirty to true as a side effect */
  def +=(e: Node): Unit

  /** removes a tree, setting this.dirty to true as a side effect */
  def -=(e: Node): Unit

  /* loads and parses XML from file */
  private def load: Iterator[Node] = {
    import scala.io.Source
    import scala.xml.parsing.ConstructingParser
    log("[load]\nloading " + theFile)
    val src = Source.fromFile(theFile)
    log("parsing " + theFile)
    val res = ConstructingParser.fromSource(src, preserveWS = false).document.docElem(0)
    switch()
    log("[load done]")
    res.child.iterator
  }

  /** saves the XML to file */
  private def save() = if (this.dirty) {
    log("[save]\ndeleting " + theFile)
    theFile.delete()
    log("creating new " + theFile)
    theFile.createNewFile()
    val fos = new FileOutputStream(theFile)
    val c = fos.getChannel()

    // @todo: optimize
    val storageNode = { nodes.toList }
    val w = Channels.newWriter(c, "utf-8")
    XML.write(w, storageNode, "utf-8", xmlDecl = true, doctype = null)

    log("writing to " + theFile)

    w.close
    c.close
    fos.close
    dirty = false
    switch()
    log("[save done]")
  }

  /**
   * Run method of the thread. remember to use `start()` to start a thread,
   * not `run`.
   */
  override def run = {
    log("[run]\nstarting storage thread, checking every " + interval + " ms")
    while (true) {
      Thread.sleep(this.interval.toLong)
      save()
    }
  }

  /**
   * Force writing of contents to the file, even if there has not been any
   * update.
   */
  def flush() = {
    this.dirty = true
    save()
  }

  @deprecated("This method and its usages will be removed. Use a debugger to debug code.", "2.11")
  def log(msg: String): Unit = {}
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy