blob: 5dd625e15d0e0eaae097efc53b643d666bda48f9 [file] [log] [blame] [raw]
package li.cil.oc.server.component
import li.cil.oc.Settings
import li.cil.oc.api.Network
import li.cil.oc.api.network.{Arguments, Context, Callback, Visibility}
import net.minecraft.nbt.NBTTagCompound
import scala.Some
import scala.collection.convert.WrapAsScala._
import stargatetech2.api.StargateTechAPI
import stargatetech2.api.bus._
class AbstractBus(val owner: Context with IBusDevice) extends ManagedComponent with IBusDriver {
val node = Network.newNode(this, Visibility.Neighbors).
withComponent("abstract_bus").
withConnector().
create()
val busInterface: IBusInterface = StargateTechAPI.api.getFactory.getIBusInterface(owner, this)
protected var isEnabled = true
protected var address = 0
protected var sendQueue: Option[QueuedPacket] = None
// ----------------------------------------------------------------------- //
def canHandlePacket(sender: Short, protocolID: Int, hasLIP: Boolean) = hasLIP
def handlePacket(packet: BusPacket) {
val lip = packet.getPlainText
val data = Map(lip.getEntryList.map(key => (key, lip.get(key))): _*)
val metadata = Map("mod" -> lip.getMetadata.modID, "device" -> lip.getMetadata.deviceName, "player" -> lip.getMetadata.playerName)
owner.signal("bus_message", Int.box(packet.getProtocolID), Int.box(packet.getSender), Int.box(packet.getTarget), data, metadata)
}
def getNextPacketToSend = if (sendQueue.isDefined) {
val info = sendQueue.get
sendQueue = None
val packet = new BusPacketLIP(info.sender, info.target)
for ((key, value) <- info.data) {
packet.set(key, value)
}
packet.setMetadata(new BusPacketLIP.LIPMetadata("OpenComputers", node.address, null))
packet.finish()
packet
}
else null
def isInterfaceEnabled = isEnabled
def getInterfaceAddress = address.toShort
// ----------------------------------------------------------------------- //
@Callback
def getEnabled(context: Context, args: Arguments): Array[AnyRef] = result(isEnabled)
@Callback
def setEnabled(context: Context, args: Arguments): Array[AnyRef] = {
isEnabled = args.checkBoolean(0)
result(isEnabled)
}
@Callback
def getAddress(context: Context, args: Arguments): Array[AnyRef] = result(address)
@Callback
def setAddress(context: Context, args: Arguments): Array[AnyRef] = {
address = args.checkInteger(0) & 0xFFFF
result(address)
}
@Callback
def send(context: Context, args: Arguments): Array[AnyRef] = {
val target = args.checkInteger(0) & 0xFFFF
val data = args.checkTable(1)
if (node.tryChangeBuffer(-Settings.get.abstractBusPacketCost)) {
sendQueue = Some(new QueuedPacket(address.toShort, target.toShort, Map(data.toSeq.map(entry => (entry._1.toString, entry._2.toString)): _*)))
busInterface.sendAllPackets()
result(true)
}
else result(false, "not enough energy")
}
@Callback(direct = true)
def maxPacketSize(context: Context, args: Arguments): Array[AnyRef] = result(Settings.get.maxNetworkPacketSize)
// ----------------------------------------------------------------------- //
override def load(nbt: NBTTagCompound) {
super.load(nbt)
busInterface.readFromNBT(nbt, "bus")
// Don't default to false.
if (nbt.hasKey("enabled")) {
isEnabled = nbt.getBoolean("enabled")
}
address = nbt.getInteger("address") & 0xFFFF
}
override def save(nbt: NBTTagCompound) {
super.save(nbt)
busInterface.writeToNBT(nbt, "bus")
nbt.setBoolean("enabled", isEnabled)
nbt.setInteger("address", address)
}
protected class QueuedPacket(val sender: Short, val target: Short, val data: Map[String, String]) {
// Extra braces because we don't want/have to keep size as a field.
{
val size = data.foldLeft(0)((acc, arg) => {
acc + arg._1.length + arg._2.length
})
if (size > Settings.get.maxNetworkPacketSize) {
throw new IllegalArgumentException("packet too big (max " + Settings.get.maxNetworkPacketSize + ")")
}
}
}
}