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

org.beangle.data.serialize.ReferenceSerializer.scala Maven / Gradle / Ivy

/*
 * Beangle, Agile Development Scaffold and Toolkit
 *
 * Copyright (c) 2005-2016, Beangle Software.
 *
 * Beangle is free software: you can redistribute it and/or modify
 * it under the terms of the GNU Lesser General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * Beangle is distributed in the hope that it will be useful.
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public License
 * along with Beangle.  If not, see .
 */
package org.beangle.data.serialize

import org.beangle.commons.lang.Strings
import org.beangle.data.serialize.marshal.{ Marshaller, MarshallerRegistry }
import org.beangle.data.serialize.io.StreamWriter
import org.beangle.data.serialize.mapper.Mapper
import org.beangle.data.serialize.marshal.{ Id, MarshallingContext }
import org.beangle.data.serialize.io.Path
import org.beangle.data.serialize.io.AbstractWriter
import org.beangle.data.serialize.io.PathStack
import org.beangle.data.serialize.io.StreamDriver

abstract class ReferenceSerializer extends AbstractSerializer {

  override def marshal(item: Any, marshaller: Marshaller[Any], context: MarshallingContext): Unit = {
    val writer = context.writer
    if (marshaller.targetType.scalar) {
      // strings, ints, dates, etc... don't bother using references.
      marshaller.marshal(item, writer, context)
    } else {
      val currentPath = writer.currentPath
      val objectItem =item.asInstanceOf[AnyRef]
      val existed = context.references.get(objectItem)
      if (existed != null) {
        val referAttrName = mapper.aliasForSystemAttribute("reference")
        if (referAttrName != null) writer.addAttribute(referAttrName, createReference(currentPath, existed.key, context))
      } else {
        val newReferKey = fireReference(currentPath, item, context)
        context.references.put(objectItem, new Id(newReferKey, currentPath))
        marshaller.marshal(item, writer, context)
      }
    }
  }

  protected def createReference(currentPath: Path, existedKey: Object, context: MarshallingContext): String
  protected def fireReference(currentPath: Path, item: Any, context: MarshallingContext): Object
}

abstract class ReferenceByIdSerializer extends ReferenceSerializer {

  protected override def createReference(currentPath: Path, existedKey: Object, context: MarshallingContext): String = {
    existedKey.toString
  }

  protected override def fireReference(currentPath: Path, item: Any, context: MarshallingContext): AnyRef = {
    val key = Strings.unCamel(item.getClass().getSimpleName) + "_" + System.identityHashCode(item)
    val attributeName = mapper.aliasForSystemAttribute("id")
    if (attributeName != null) context.writer.addAttribute(attributeName, key.toString())
    key
  }

}

abstract class ReferenceByXPathSerializer(val absolutePath: Boolean, val singleNode: Boolean) extends ReferenceSerializer {

  protected override def createReference(currentPath: Path, existedKey: Object, context: MarshallingContext): String = {
    val existingPath = existedKey.asInstanceOf[Path]
    val referencePath = if (absolutePath) existingPath else currentPath.relativeTo(existingPath)
    return if (singleNode) referencePath.explicit() else referencePath.toString
  }

  protected override def fireReference(currentPath: Path, item: Any, context: MarshallingContext): AnyRef = {
    return currentPath
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy