org.alephium.protocol.model.UnsignedTransaction.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 scala.collection.immutable.ListMap
import akka.util.ByteString
import org.alephium.protocol.ALPH
import org.alephium.protocol.config.{GroupConfig, NetworkConfig}
import org.alephium.protocol.vm._
import org.alephium.serde._
import org.alephium.util.{AVector, EitherF, Math, TimeStamp, U256}
// scalastyle:off number.of.methods
/** Up to one new token might be issued in each transaction exception for the coinbase transaction
* The id of the new token will be hash of the first input
*
* @param version
* the version of the tx
* @param networkId
* the id of the chain which can accept the tx
* @param scriptOpt
* optional script for invoking stateful contracts
* @param gasAmount
* the amount of gas can be used for tx execution
* @param inputs
* a vector of TxInput
* @param fixedOutputs
* a vector of TxOutput. ContractOutput are put in front of AssetOutput
*/
final case class UnsignedTransaction(
version: Byte,
networkId: NetworkId,
scriptOpt: Option[StatefulScript],
gasAmount: GasBox,
gasPrice: GasPrice,
inputs: AVector[TxInput],
fixedOutputs: AVector[AssetOutput]
) extends AnyRef {
lazy val id: TransactionId = TransactionId.hash(serialize(this))
// this might only works for validated tx
def fromGroup(implicit config: GroupConfig): GroupIndex = {
inputs.head.fromGroup
}
// this might only works for validated tx
def toGroup(implicit config: GroupConfig): GroupIndex = {
val from = fromGroup
val outputs = fixedOutputs
if (outputs.isEmpty) {
from
} else {
val index = outputs.indexWhere(_.toGroup != from)
if (index == -1) {
from
} else {
outputs(index).toGroup
}
}
}
// this might only works for validated tx
def chainIndex(implicit config: GroupConfig): ChainIndex = ChainIndex(fromGroup, toGroup)
lazy val fixedOutputRefs: AVector[AssetOutputRef] = fixedOutputs.mapWithIndex {
case (output, outputIndex) =>
AssetOutputRef.from(output, TxOutputRef.key(id, outputIndex))
}
}
object UnsignedTransaction {
implicit val serde: Serde[UnsignedTransaction] = Serde.forProduct7(
UnsignedTransaction.apply,
t => (t.version, t.networkId, t.scriptOpt, t.gasAmount, t.gasPrice, t.inputs, t.fixedOutputs)
)
def apply(
scriptOpt: Option[StatefulScript],
startGas: GasBox,
gasPrice: GasPrice,
inputs: AVector[TxInput],
fixedOutputs: AVector[AssetOutput]
)(implicit networkConfig: NetworkConfig): UnsignedTransaction = {
new UnsignedTransaction(
DefaultTxVersion,
networkConfig.networkId,
scriptOpt,
startGas,
gasPrice,
inputs,
fixedOutputs
)
}
def apply(
txScriptOpt: Option[StatefulScript],
inputs: AVector[TxInput],
fixedOutputs: AVector[AssetOutput]
)(implicit networkConfig: NetworkConfig): UnsignedTransaction = {
UnsignedTransaction(
DefaultTxVersion,
networkConfig.networkId,
txScriptOpt,
minimalGas,
nonCoinbaseMinGasPrice,
inputs,
fixedOutputs
)
}
def apply(inputs: AVector[TxInput], fixedOutputs: AVector[AssetOutput])(implicit
networkConfig: NetworkConfig
): UnsignedTransaction = {
UnsignedTransaction(
DefaultTxVersion,
networkConfig.networkId,
None,
minimalGas,
nonCoinbaseMinGasPrice,
inputs,
fixedOutputs
)
}
private def buildInputs(
fromUnlockScript: UnlockScript,
inputs: AVector[(AssetOutputRef, AssetOutput)]
): AVector[TxInput] = {
assume(fromUnlockScript != UnlockScript.SameAsPrevious)
inputs.mapWithIndex { case ((outputRef, _), index) =>
if (index == 0) {
TxInput(outputRef, fromUnlockScript)
} else {
TxInput(outputRef, UnlockScript.SameAsPrevious)
}
}
}
// scalastyle:off parameter.number
def buildScriptTx(
script: StatefulScript,
fromLockupScript: LockupScript.Asset,
fromUnlockScript: UnlockScript,
inputs: AVector[(AssetOutputRef, AssetOutput)],
approvedAttoAlphAmount: U256,
approvedTokens: AVector[(TokenId, U256)],
gasAmount: GasBox,
gasPrice: GasPrice
)(implicit networkConfig: NetworkConfig): Either[String, UnsignedTransaction] = {
val approved =
TxOutputInfo(fromLockupScript, approvedAttoAlphAmount, approvedTokens, None, None)
val approvedAsOutput = buildOutputs(approved)
for {
gasFee <- preCheckBuildTx(inputs, gasAmount, gasPrice)
fixedOutputs <- calculateChangeOutputs(
fromLockupScript,
inputs,
approvedAsOutput,
gasFee
)
} yield {
UnsignedTransaction(
Some(script),
gasAmount,
gasPrice,
buildInputs(fromUnlockScript, inputs),
fixedOutputs
)
}
}
// scalastyle:on parameter.number
def coinbase(inputs: AVector[TxInput], fixedOutputs: AVector[AssetOutput])(implicit
networkConfig: NetworkConfig
): UnsignedTransaction = {
UnsignedTransaction(
DefaultTxVersion,
networkConfig.networkId,
None,
minimalGas,
coinbaseGasPrice,
inputs,
fixedOutputs
)
}
@inline private def calculateChangeOutputs(
fromLockupScript: LockupScript.Asset,
inputs: AVector[(AssetOutputRef, AssetOutput)],
txOutputs: AVector[AssetOutput],
gasFee: U256
): Either[String, AVector[AssetOutput]] = {
val inputUTXOView = inputs.map(_._2)
for {
alphRemainder <- calculateAlphRemainder(inputUTXOView, txOutputs, gasFee)
tokensRemainder <- calculateTokensRemainder(inputUTXOView, txOutputs)
changeOutputs <- calculateChangeOutputs(alphRemainder, tokensRemainder, fromLockupScript)
} yield changeOutputs
}
@inline private def preCheckBuildTx(
inputs: AVector[(AssetOutputRef, AssetOutput)],
gas: GasBox,
gasPrice: GasPrice
): Either[String, U256] = {
assume(gas >= minimalGas)
assume(gasPrice.value <= ALPH.MaxALPHValue)
for {
_ <- checkWithMaxTxInputNum(inputs)
_ <- checkUniqueInputs(inputs)
} yield gasPrice * gas
}
def buildTxOutputs(
fromLockupScript: LockupScript.Asset,
inputs: AVector[(AssetOutputRef, AssetOutput)],
outputInfos: AVector[TxOutputInfo],
gas: GasBox,
gasPrice: GasPrice
): Either[String, (AVector[AssetOutput], AVector[AssetOutput])] = {
for {
gasFee <- preCheckBuildTx(inputs, gas, gasPrice)
_ <- checkMinimalAlphPerOutput(outputInfos)
_ <- checkTokenValuesNonZero(outputInfos)
txOutputs = buildOutputs(outputInfos)
changeOutputs <- calculateChangeOutputs(fromLockupScript, inputs, txOutputs, gasFee)
} yield txOutputs -> changeOutputs
}
def buildTransferTx(
fromLockupScript: LockupScript.Asset,
fromUnlockScript: UnlockScript,
inputs: AVector[(AssetOutputRef, AssetOutput)],
outputInfos: AVector[TxOutputInfo],
gas: GasBox,
gasPrice: GasPrice
)(implicit networkConfig: NetworkConfig): Either[String, UnsignedTransaction] = {
buildTxOutputs(fromLockupScript, inputs, outputInfos, gas, gasPrice)
.map { case (txOutputs, changeOutputs) =>
UnsignedTransaction(
None,
gas,
gasPrice,
buildInputs(fromUnlockScript, inputs),
txOutputs ++ changeOutputs
)
}
}
def buildTransferTxAndReturnChange(
fromLockupScript: LockupScript.Asset,
fromUnlockScript: UnlockScript,
inputs: AVector[(AssetOutputRef, AssetOutput)],
outputInfos: AVector[TxOutputInfo],
gas: GasBox,
gasPrice: GasPrice
)(implicit
networkConfig: NetworkConfig
): Either[String, (UnsignedTransaction, AVector[(AssetOutputRef, AssetOutput)])] = {
buildTxOutputs(fromLockupScript, inputs, outputInfos, gas, gasPrice)
.map { case (txOutputs, changeOutputs) =>
val outputs = txOutputs ++ changeOutputs
val tx =
UnsignedTransaction(
None,
gas,
gasPrice,
buildInputs(fromUnlockScript, inputs),
outputs
)
var changeOutputIndex = txOutputs.length
val changeOutputsRefs = changeOutputs.map { changeOutput =>
val ref = AssetOutputRef.from(
changeOutput,
TxOutputRef.key(tx.id, changeOutputIndex)
)
changeOutputIndex += 1
ref -> changeOutput
}
tx -> changeOutputsRefs
}
}
def buildOutputs(outputInfos: AVector[TxOutputInfo]): AVector[AssetOutput] = {
outputInfos.flatMap(buildOutputs)
}
def buildOutputs(outputInfo: TxOutputInfo): AVector[AssetOutput] = {
val TxOutputInfo(toLockupScript, attoAlphAmount, tokens, lockTimeOpt, additionalDataOpt) =
outputInfo
val tokenOutputs = tokens.map { token =>
AssetOutput(
dustUtxoAmount,
toLockupScript,
lockTimeOpt.getOrElse(TimeStamp.zero),
AVector(token),
additionalDataOpt.getOrElse(ByteString.empty)
)
}
val alphRemaining = attoAlphAmount
.sub(dustUtxoAmount.mulUnsafe(U256.unsafe(tokens.length)))
.getOrElse(U256.Zero)
if (alphRemaining == U256.Zero) {
tokenOutputs
} else {
val alphOutput = AssetOutput(
Math.max(alphRemaining, dustUtxoAmount),
toLockupScript,
lockTimeOpt.getOrElse(TimeStamp.zero),
AVector.empty,
additionalDataOpt.getOrElse(ByteString.empty)
)
tokenOutputs :+ alphOutput
}
}
def buildGeneric(
from: AVector[UnlockScriptWithAssets],
outputInfos: AVector[TxOutputInfo],
gas: GasBox,
gasPrice: GasPrice
)(implicit networkConfig: NetworkConfig): Either[String, UnsignedTransaction] = {
assume(gas >= minimalGas)
assume(gasPrice.value <= ALPH.MaxALPHValue)
val gasFee = gasPrice * gas
val inputs = from.flatMap(_.assets)
val inputRefs = from.flatMap { _.assets.map { case (_, o) => o } }
for {
_ <- checkWithMaxTxInputNum(inputs)
_ <- checkUniqueInputs(inputs)
outputs = buildOutputs(outputInfos)
_ <- checkMinimalAlphPerOutput(outputInfos)
_ <- checkTokenValuesNonZero(outputInfos)
alphRemainder <- calculateAlphRemainder(inputRefs, outputs, gasFee)
_ <- checkNoAlphRemainder(alphRemainder)
tokensRemainder <- calculateTokensRemainder(inputRefs, outputs)
_ <- checkNoTokensRemainder(tokensRemainder)
} yield {
UnsignedTransaction(
DefaultTxVersion,
networkConfig.networkId,
scriptOpt = None,
gas,
gasPrice,
from.flatMap(in => buildInputs(in.fromUnlockScript, in.assets)),
outputs
)
}
}
def checkNoAlphRemainder(alphRemainder: U256): Either[String, Unit] = {
if (alphRemainder != U256.Zero) {
Left("Inputs' Alph don't sum up to outputs and gas fee")
} else {
Right(())
}
}
def checkNoTokensRemainder(tokensRemainder: AVector[(TokenId, U256)]): Either[String, Unit] = {
if (tokensRemainder.exists { case (_, value) => value != U256.Zero }) {
Left("Inputs' tokens don't sum up to outputs' tokens")
} else {
Right(())
}
}
def checkUniqueInputs(
assets: AVector[(AssetOutputRef, AssetOutput)]
): Either[String, Unit] = {
check(
failCondition = assets.length > assets.map(_._1).toSet.size,
"Inputs not unique"
)
}
def checkWithMaxTxInputNum(
assets: AVector[(AssetOutputRef, AssetOutput)]
): Either[String, Unit] = {
check(
failCondition = assets.length > ALPH.MaxTxInputNum,
"Too many inputs for the transfer, consider to reduce the amount to send, or use the `sweep-address` endpoint to consolidate the inputs first"
)
}
private def calculateAlphRemainder(
inputs: AVector[AssetOutput],
outputs: AVector[AssetOutput],
gasFee: U256
): Either[String, U256] = {
calculateAlphRemainder(
inputs.map(_.amount),
outputs.map(_.amount),
gasFee
)
}
def calculateAlphRemainder(
inputs: AVector[U256],
outputs: AVector[U256],
gasFee: U256
): Either[String, U256] = {
for {
inputSum <- EitherF.foldTry(inputs, U256.Zero)(_ add _ toRight "Input amount overflow")
outputAmount <- outputs.foldE(U256.Zero)(
_ add _ toRight "Output amount overflow"
)
remainder0 <- inputSum.sub(outputAmount).toRight("Not enough balance")
remainder <- remainder0.sub(gasFee).toRight("Not enough balance for gas fee")
} yield remainder
}
private def calculateTokensRemainder(
inputs: AVector[AssetOutput],
outputs: AVector[AssetOutput]
): Either[String, AVector[(TokenId, U256)]] = {
calculateTokensRemainder(
inputs.flatMap(_.tokens),
outputs.flatMap(_.tokens)
)
}
def calculateTokensRemainder(
inputs: AVector[(TokenId, U256)],
outputs: AVector[(TokenId, U256)]
): Either[String, AVector[(TokenId, U256)]] = {
for {
inputs <- calculateTotalAmountPerToken(inputs)
outputs <- calculateTotalAmountPerToken(outputs)
_ <- checkNoNewTokensInOutputs(inputs, outputs)
remainder <- calculateRemainingTokens(inputs, outputs)
} yield {
remainder.filterNot(_._2 == U256.Zero)
}
}
def calculateChangeOutputs(
alphRemainder: U256,
tokensRemainder: AVector[(TokenId, U256)],
fromLockupScript: LockupScript.Asset
): Either[String, AVector[AssetOutput]] = {
if (alphRemainder == U256.Zero && tokensRemainder.isEmpty) {
Right(AVector.empty)
} else {
val tokenDustAmount = dustUtxoAmount.mulUnsafe(U256.unsafe(tokensRemainder.length))
val totalDustAmount = tokenDustAmount.addUnsafe(dustUtxoAmount)
if ((alphRemainder == tokenDustAmount) || (alphRemainder >= totalDustAmount)) {
Right(
buildOutputs(TxOutputInfo(fromLockupScript, alphRemainder, tokensRemainder, None, None))
)
} else if (tokensRemainder.isEmpty) {
Left(
s"Not enough ALPH for ALPH change output, expected $dustUtxoAmount, got $alphRemainder"
)
} else if (alphRemainder < tokenDustAmount) {
Left(
s"Not enough ALPH for token change output, expected $tokenDustAmount, got $alphRemainder"
)
} else {
Left(
s"Not enough ALPH for ALPH and token change output, expected $totalDustAmount, got $alphRemainder"
)
}
}
}
private def checkMinimalAlphPerOutput(
outputs: AVector[TxOutputInfo]
): Either[String, Unit] = {
check(
failCondition = outputs.exists { output =>
output.attoAlphAmount < dustUtxoAmount
},
"Tx output value is too small, avoid spreading dust"
)
}
private def checkTokenValuesNonZero(
outputs: AVector[TxOutputInfo]
): Either[String, Unit] = {
check(
failCondition = outputs.exists(_.tokens.exists(_._2.isZero)),
"Value is Zero for one or many tokens in the transaction output"
)
}
// Note: this would calculate excess dustAmount to cover the complicated cases
def calculateTotalAmountNeeded(
outputInfos: AVector[TxOutputInfo]
): Either[String, (U256, AVector[(TokenId, U256)], Int)] = {
outputInfos
.foldE((U256.Zero, ListMap.empty[TokenId, U256], 0)) {
case ((totalAlphAmount, totalTokens, totalOutputLength), outputInfo) =>
val tokenDustAmount = dustUtxoAmount.mulUnsafe(U256.unsafe(outputInfo.tokens.length))
val outputLength = outputInfo.tokens.length + // UTXOs for token
(if (outputInfo.attoAlphAmount <= tokenDustAmount) 0 else 1) // UTXO for ALPH
val alphAmount =
Math.max(outputInfo.attoAlphAmount, dustUtxoAmount.mulUnsafe(U256.unsafe(outputLength)))
for {
newAlphAmount <- totalAlphAmount.add(alphAmount).toRight("ALPH amount overflow")
newTotalTokens <- updateTokens(totalTokens, outputInfo.tokens)
} yield (newAlphAmount, newTotalTokens, totalOutputLength + outputLength)
}
.flatMap { case ((totalAlphAmount, totalTokens, totalOutputLength)) =>
val outputLengthSender = totalTokens.size + 1
val alphAmountSender = dustUtxoAmount.mulUnsafe(U256.unsafe(outputLengthSender))
totalAlphAmount.add(alphAmountSender).toRight("ALPH amount overflow").map {
finalAlphAmount =>
(
finalAlphAmount,
AVector.from(totalTokens.iterator),
totalOutputLength + outputLengthSender
)
}
}
}
private def updateTokens(
totalTokens: ListMap[TokenId, U256],
newTokens: AVector[(TokenId, U256)]
): Either[String, ListMap[TokenId, U256]] = {
newTokens.foldE(totalTokens) { case (acc, (tokenId, amount)) =>
acc.get(tokenId) match {
case Some(totalAmount) =>
totalAmount.add(amount) match {
case Some(newAmount) => Right(acc + (tokenId -> newAmount))
case None => Left(s"Amount overflow for token $tokenId")
}
case None => Right(acc + (tokenId -> amount))
}
}
}
def calculateTotalAmountPerToken(
tokens: AVector[(TokenId, U256)]
): Either[String, AVector[(TokenId, U256)]] = {
tokens.foldE(AVector.empty[(TokenId, U256)]) { case (acc, (id, amount)) =>
val index = acc.indexWhere(_._1 == id)
if (index == -1) {
Right(acc :+ (id -> amount))
} else {
acc(index)._2.add(amount).toRight(s"Amount overflow for token $id").map { amt =>
acc.replace(index, (id, amt))
}
}
}
}
private def checkNoNewTokensInOutputs(
inputs: AVector[(TokenId, U256)],
outputs: AVector[(TokenId, U256)]
): Either[String, Unit] = {
val newTokens = outputs.map(_._1).toSet -- inputs.map(_._1).toSet
check(
failCondition = newTokens.nonEmpty,
s"New tokens found in outputs: $newTokens"
)
}
private def calculateRemainingTokens(
inputTokens: AVector[(TokenId, U256)],
outputTokens: AVector[(TokenId, U256)]
): Either[String, AVector[(TokenId, U256)]] = {
inputTokens.foldE(AVector.empty[(TokenId, U256)]) { case (acc, (inputId, inputAmount)) =>
val outputAmount = outputTokens.find(_._1 == inputId).fold(U256.Zero)(_._2)
inputAmount.sub(outputAmount).toRight(s"Not enough balance for token $inputId").map {
remainder =>
acc :+ (inputId -> remainder)
}
}
}
@inline private def check(failCondition: Boolean, errorMessage: String): Either[String, Unit] = {
Either.cond(!failCondition, (), errorMessage)
}
final case class TxOutputInfo(
lockupScript: LockupScript.Asset,
attoAlphAmount: U256,
tokens: AVector[(TokenId, U256)],
lockTime: Option[TimeStamp],
additionalDataOpt: Option[ByteString]
)
object TxOutputInfo {
def apply(
lockupScript: LockupScript.Asset,
attoAlphAmount: U256,
tokens: AVector[(TokenId, U256)],
lockTime: Option[TimeStamp]
): TxOutputInfo = {
TxOutputInfo(lockupScript, attoAlphAmount, tokens, lockTime, None)
}
}
final case class UnlockScriptWithAssets(
fromUnlockScript: UnlockScript,
assets: AVector[(AssetOutputRef, AssetOutput)]
)
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy