org.alephium.protocol.model.TxOutput.scala Maven / Gradle / Ivy
The newest version!
// Copyright 2018 The Alephium Authors
// This file is part of the alephium project.
//
// The library 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.
//
// The library 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 the library. If not, see .
package org.alephium.protocol.model
import akka.util.ByteString
import org.alephium.protocol.ALPH
import org.alephium.protocol.config.GroupConfig
import org.alephium.protocol.vm.LockupScript
import org.alephium.serde._
import org.alephium.util.{AVector, Duration, TimeStamp, U256}
sealed trait TxOutput {
def amount: U256
def lockupScript: LockupScript
def tokens: AVector[(TokenId, U256)]
def hint: Hint
def isAsset: Boolean
def isContract: Boolean = !isAsset
def payGasUnsafe(fee: U256): TxOutput
}
object TxOutput {
implicit val serde: Serde[TxOutput] = eitherSerde[AssetOutput, ContractOutput].xmap(
{
case Left(assetOutput) => assetOutput
case Right(contractOutput) => contractOutput
},
{
case output: AssetOutput => Left(output)
case output: ContractOutput => Right(output)
}
)
def from(
amount: U256,
tokens: AVector[(TokenId, U256)],
lockupScript: LockupScript.Asset
): Option[AVector[TxOutput]] = {
from(amount, tokens, lockupScript, TimeStamp.zero)
}
def from(
amount: U256,
tokens: AVector[(TokenId, U256)],
lockupScript: LockupScript.Asset,
lockTime: TimeStamp
): Option[AVector[TxOutput]] = {
val outputs = Array.ofDim[TxOutput](tokens.length + 1)
tokens.foreachWithIndex { case (token, index) =>
outputs(index) =
AssetOutput(dustUtxoAmount, lockupScript, lockTime, AVector(token), ByteString.empty)
}
val totalTokenDustAmount = dustUtxoAmount.mulUnsafe(U256.unsafe(tokens.length))
if (amount == totalTokenDustAmount) {
Some(AVector.unsafe(outputs, 0, tokens.length))
} else if (amount >= totalTokenDustAmount.addUnsafe(dustUtxoAmount)) {
val alphRemaining = amount.subUnsafe(totalTokenDustAmount)
outputs(tokens.length) =
AssetOutput(alphRemaining, lockupScript, lockTime, AVector.empty, ByteString.empty)
Some(AVector.unsafe(outputs))
} else {
None
}
}
def fromDeprecated(
amount: U256,
tokens: AVector[(TokenId, U256)],
lockupScript: LockupScript
): TxOutput = {
lockupScript match {
case e: LockupScript.P2C => ContractOutput(amount, e, tokens)
case e: LockupScript.Asset => AssetOutput(amount, e, TimeStamp.zero, tokens, ByteString.empty)
}
}
def asset(amount: U256, lockupScript: LockupScript.Asset): AssetOutput = {
asset(amount, AVector.empty, lockupScript)
}
def asset(
amount: U256,
tokens: AVector[(TokenId, U256)],
lockupScript: LockupScript.Asset
): AssetOutput = {
asset(amount, lockupScript, tokens, TimeStamp.zero)
}
def asset(
amount: U256,
lockupScript: LockupScript.Asset,
tokens: AVector[(TokenId, U256)],
lockTimeOpt: Option[TimeStamp]
): AssetOutput = {
asset(amount, lockupScript, tokens, lockTimeOpt.getOrElse(TimeStamp.zero))
}
def asset(
amount: U256,
lockupScript: LockupScript.Asset,
tokens: AVector[(TokenId, U256)],
lockTime: TimeStamp
): AssetOutput = {
AssetOutput(amount, lockupScript, lockTime, tokens, ByteString.empty)
}
def contract(amount: U256, lockupScript: LockupScript.P2C): ContractOutput = {
ContractOutput(amount, lockupScript, AVector.empty)
}
def genesis(
amount: U256,
lockupScript: LockupScript.Asset,
lockupDuration: Duration,
data: ByteString
): AssetOutput = {
val lockTime = ALPH.LaunchTimestamp.plusUnsafe(lockupDuration)
AssetOutput(amount, lockupScript, lockTime, AVector.empty, data)
}
// TODO: improve this when vm is mature
def forSMT: TxOutput =
ContractOutput(U256.One, LockupScript.p2c(ContractId.zero), AVector.empty)
}
/** @param amount
* the number of ALPH in the output
* @param lockupScript
* guarding script for unspent output
* @param lockTime
* the timestamp until when the tx can be used. it's zero by default, and will be replaced with
* block timestamp in worldstate if it's zero we could implement relative time lock based on
* block timestamp
* @param tokens
* secondary tokens in the output
* @param additionalData
* data payload for additional information
*/
final case class AssetOutput(
amount: U256,
lockupScript: LockupScript.Asset,
lockTime: TimeStamp,
tokens: AVector[(TokenId, U256)],
additionalData: ByteString
) extends TxOutput {
def isAsset: Boolean = true
def hint: Hint = Hint.from(this)
def toGroup(implicit config: GroupConfig): GroupIndex = lockupScript.groupIndex
def payGasUnsafe(fee: U256): AssetOutput =
AssetOutput(amount.subUnsafe(fee), lockupScript, lockTime, tokens, additionalData)
}
object AssetOutput {
implicit private[model] val tokenSerde: Serde[(TokenId, U256)] = Serde.tuple2[TokenId, U256]
implicit val serde: Serde[AssetOutput] =
Serde.forProduct5(
AssetOutput.apply,
t => (t.amount, t.lockupScript, t.lockTime, t.tokens, t.additionalData)
)
}
final case class ContractOutput(
amount: U256,
lockupScript: LockupScript.P2C,
tokens: AVector[(TokenId, U256)]
) extends TxOutput {
def isAsset: Boolean = false
def hint: Hint = Hint.from(this)
def payGasUnsafe(fee: U256): ContractOutput =
ContractOutput(amount.subUnsafe(fee), lockupScript, tokens)
}
object ContractOutput {
import AssetOutput.tokenSerde
implicit val serde: Serde[ContractOutput] =
Serde.forProduct3(ContractOutput.apply, t => (t.amount, t.lockupScript, t.tokens))
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy