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

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