3MNcxHZ6om5XLCGP7SZUcpcCkTRsEZ9rLWv · data · RAW_CODE
{
"RAW_CODE": "{-# STDLIB_VERSION 6 #-}
{-# CONTENT_TYPE DAPP #-}
{-# SCRIPT_TYPE ACCOUNT #-}


let VERSION = "1.0.3"

### KEYS ###
let SEP = "_"
let BIGSEP = "__"
let k_ADMINS = "ADMINS_PBKS"
let k_SUPERADMINS = "SUPERADMINS_PBKS"
let k_MANAGERS = "MANAGERS_PBKS"
let k_LIMIT = "VOTES_QUORUM"
let k_SCRIPT_TYPES = "SCRIPT_TYPES"
let k_ITERATION = "ITERATION"
let k_INITED = "INITED"
let k_SCRIPTS_COOLDOWN = "SCRIPTSCOOLDOWN"

let k_scriptType = "scriptType"
let k_scriptHash = "scriptHash"
let k_storageAddress = "storageAddress"
let k_desc = "desc"
let k_iter = "iteration"
let k_initiator = "initiator"
let k_confirmations = "confirmations"
let k_validBlocks = "validBlocks"
let k_validTo = "validTo"
let k_validFrom = "validFrom"

func _validatePbk(pbk: String) =  {
    let keyBytes = pbk.fromBase58String()
    strict checkLength = if (size(keyBytes) != 32) then throw("Invalid public key") else true
    strict address = addressFromPublicKey(keyBytes)
    strict addressStr = address.toString()
    size(addressStr) > 0
}

func _validateAddress(address: String) = {
    strict addr = addressFromStringValue(address)
    true
}

func _validateTs(ts: Int) = {
    let currTs = match blockInfoByHeight(height) {
        case block:BlockInfo =>
            block.timestamp
        case _ => throw("Can't find block")
    }
    
    # Only restrict timestamps from being too old (1 hour ago), allow any future time
    let oneHour = 60 * 60 * 1000  # 1 hour in milliseconds
    let minTs = currTs - oneHour
    
    ts >= minTs
}


func _getLimit() = {
    getInteger(k_LIMIT).valueOrErrorMessage("Quorum limit not defined")
}

func _isAdmin(pbk: String) = {
    let adminsStr = getString(k_ADMINS).valueOrElse("")
    let admins = adminsStr.split(",")
    indexOf(admins, pbk) != unit
}

func _isSuperAdmin(pbk: String) = {
    let superAdminsStr = getString(k_SUPERADMINS).valueOrElse("")
    let superAdmins = superAdminsStr.split(",")
    indexOf(superAdmins, pbk) != unit
}

func _isManager(pbk: String) = {
    let managersStr = getString(k_MANAGERS).valueOrElse("")
    let managers = managersStr.split(",")
    indexOf(managers, pbk) != unit
}

func _isValidOperator(pbk: String) = {
    let managersStr = getString(k_MANAGERS).valueOrElse("")
    let managers = split(managersStr, ",")
    let isAdmin = _isAdmin(pbk)
    let isManager = _isManager(pbk)
    isAdmin || isManager
}

func _isInited() = {
    getBoolean(k_INITED).valueOrElse(false)
}

### UTILITY ###


func saveTxData(taskId: String, data: String) = {
    let indices = [1,2,3,4,5,6,7,8,9,10]
    let dataSize = size(data)
    strict isNotEmpty = if (dataSize != 0) then true else throw("saveTxData: Empty data passed")
    let chunkSize = 30000
    let chunksCount = if (dataSize % chunkSize == 0) then dataSize / chunkSize else dataSize / chunkSize + 1

    func makeChunks(accum: (String, List[StringEntry]), index: Int) = {
        let chunk = take(accum._1, chunkSize)
        if (size(chunk) == 0) then {accum} else {
            let nextData = drop(accum._1, chunkSize)
            (nextData, accum._2 :+ StringEntry(taskId + "__txdata_" + toString(index), chunk))
        }
    }
    
    # Save each chunk using FOLD
    let entries = FOLD<10>(indices, (data, [StringEntry(taskId + "__txdata_chunks", toString(chunksCount))]), makeChunks)
    
    entries._2
}

### CALLABLES ###
@Callable(i)
func init(admins_pbks: List[String], limit: Int, scriptsCooldown: Int) = {
    strict checkInited = if (_isInited()) then throw("Already inited") else true
    strict size_admins = if (size(admins_pbks) != 0) then true else throw("At least one admin pbk should be passed")
    strict check_admins = if (!isDefined(getString(k_ADMINS))) then true else throw("Already inited")
    strict zeroLimit = if (limit > 0) then true else throw("Limit should be > 0")
    strict checkScriptsCooldown = if (scriptsCooldown > 0) then true else throw("Scripts cooldown should be > 0")
    strict limit_ok = if (size(admins_pbks) >= limit) then true else throw("Limit quorum should be <= size of list of admins, limit: " + limit.toString() + ", current size: " + size(admins_pbks).toString())
    func validate(accum: String, next: String) = {
        _validatePbk(next)
    }
    strict validRes = FOLD<10>(admins_pbks, "", validate)
    let adminsStr = makeString(admins_pbks, ",")
    [
        StringEntry(k_ADMINS, adminsStr),
        IntegerEntry(k_LIMIT, limit),
        IntegerEntry(k_SCRIPTS_COOLDOWN, scriptsCooldown),
        BooleanEntry(k_INITED, true)
    ]
}

## MANAGE ##
@Callable(i)
func addAdmin(pbk: String) = {
    let callerPbk = toBase58String(i.callerPublicKey)
    strict isSelf = if (addressFromPublicKey(i.callerPublicKey) == this) then true else throw("Only self invoke")
    strict isValid = if (_validatePbk(pbk)) then true else throw("Invalid admin pbk")
    let admins = getStringValue(k_ADMINS)
    strict checkPresent = if (indexOf(admins, pbk) != unit) then throw("This admin already present") else true
    let newAdminsStr = admins + "," + pbk

    [
        StringEntry(k_ADMINS, newAdminsStr)
    ]
}

@Callable(i)
func removeAdmin(pbk: String) = {
    let callerPbk = toBase58String(i.callerPublicKey)
    strict isSelf = if (addressFromPublicKey(i.callerPublicKey) == this) then true else throw("Only self invoke")
    strict isValid = if (_validatePbk(pbk)) then true else throw("Invalid admin pbk")
    let oldAdminsStr = getStringValue(k_ADMINS)
    let oldAdmins = oldAdminsStr.split(",")
    strict checkCount = if (size(oldAdmins) <= 1) then throw("Cannot remove, at least one admin should remain") else true
    let index = indexOf(oldAdmins, pbk).valueOrErrorMessage("This pbk is not an admin")
    let newAdmins = removeByIndex(oldAdmins, index)
    let newAdminsStr = makeString(newAdmins, ",")
    [
        StringEntry(k_ADMINS, newAdminsStr)
    ]
}

@Callable(i)
func addScriptType(scriptType: String) = {
    
    let callerPbk = toBase58String(i.callerPublicKey)
    strict isSelf = if (addressFromPublicKey(i.callerPublicKey) == this) then true else throw("Only self invoke")
    strict checkEmpty = if (size(scriptType) == 0) then throw("Script type can not be empty") else true
    let scriptTypesStr = getString(k_SCRIPT_TYPES).valueOrElse("")
    let scriptTypes = split_4C(scriptTypesStr, ",")
    strict checkExists = if (indexOf(scriptTypes, scriptType) != unit) then throw("Script type <" + scriptType + "> already added") else true
    let newScriptTypes = if (scriptTypesStr != "") then scriptTypes :+ scriptType else [scriptType]
    let newScriptTypesStr = makeString(newScriptTypes, ",")
    [
        StringEntry(k_SCRIPT_TYPES, newScriptTypesStr)
    ]
}

@Callable(i)
func removeScriptType(scriptType: String) = {
    let callerPbk = toBase58String(i.callerPublicKey)
    strict isSelf = if (addressFromPublicKey(i.callerPublicKey) == this) then true else throw("Only self invoke")
    let scriptTypesStr = getString(k_SCRIPT_TYPES).valueOrElse("")
    let scriptTypes = split_4C(scriptTypesStr, ",")
    let index = indexOf(scriptTypes, scriptType).valueOrErrorMessage("Script type <" + scriptType + "> is not present")
    
    let newScriptTypes = removeByIndex(scriptTypes, index)
    let newScriptTypesStr = makeString(newScriptTypes, ",")
    [
        StringEntry(k_SCRIPT_TYPES, newScriptTypesStr)
    ]
}

@Callable(i)
func setLimit(limit: Int) = {
    let callerPbk = toBase58String(i.callerPublicKey)
    strict isSelf = if (addressFromPublicKey(i.callerPublicKey) == this) then true else throw("Only self invoke")
    strict zeroLimit = if (limit > 0) then true else throw("Limit should be > 0")
    let admins = getStringValue(k_ADMINS)
    if (size(admins) < limit) then throw("Limit quorum should be <= size of list of admins, limit: " + limit.toString() + ", current size: " + size(admins).toString())
    else {
        [
            IntegerEntry(k_LIMIT, limit)
        ]
    }
}

@Callable(i)
func setCooldownBlocks(scriptsCooldown: Int) = {
    let callerPbk = toBase58String(i.callerPublicKey)
    strict isSelf = if (addressFromPublicKey(i.callerPublicKey) == this) then true else throw("Only self invoke")
    strict checkScriptsCooldown = if (scriptsCooldown > 0) then true else throw("Scripts cooldown should be > 0")
    [
        IntegerEntry(k_SCRIPTS_COOLDOWN, scriptsCooldown)
    ]
}

## MAIN OPERATIONS ##
@Callable(i)
func addScript(scriptType: String, storageAddress: String, _scriptHash: String, desc: String, initiatorAdr: String, validBlocks: Int) = {
    let callerPbk = toBase58String(i.callerPublicKey)
    let storageScriptHash = addressFromStringValue(storageAddress).scriptHash().value().toBase58String()
    let scriptTypesStr = getStringValue(k_SCRIPT_TYPES)
   
    # validations
    strict isAllowed = if (!_isValidOperator(callerPbk)) then throw("You are not allowed") else true
    strict _checkScriptTypes = if (scriptTypesStr.indexOf(scriptType) == unit) then throw("invalid script type or it is not present in storage: " + scriptType) else true
    strict _scriptStorageHashCheck = if (storageScriptHash != _scriptHash) then throw("Stored and passed as argument scriptHashes not match") else true
    strict _descCheck = if (size(desc) == 0) then throw("Description can't be empty") else true
    strict _initiatorCheck = if (addressFromPublicKey(i.callerPublicKey).toString() != initiatorAdr) then throw("Initiator address doesn't match caller") else true
    strict _checkValidBlocks = if (validBlocks > 0) then true else throw("Valid blocks should be > 0")
    
    let iteration = getInteger(k_ITERATION + SEP + scriptType).valueOrElse(0) + 1
    let scriptId = "script%%" + scriptType + "%%" + _scriptHash
    strict _checkPresent = if (isDefined(getString(scriptId + BIGSEP + k_scriptHash))) then throw("Script with this hash already present") else true
    [
        StringEntry(scriptId + BIGSEP + k_scriptType, scriptType),
        StringEntry(scriptId + BIGSEP + k_storageAddress, storageAddress),
        StringEntry(scriptId + BIGSEP + k_scriptHash, storageScriptHash),
        StringEntry(scriptId + BIGSEP + k_desc, desc),
        IntegerEntry(scriptId + BIGSEP + k_iter, iteration),
        IntegerEntry(scriptId + BIGSEP + k_validBlocks, validBlocks),
        StringEntry(scriptId + BIGSEP + k_initiator, initiatorAdr),
        StringEntry(scriptType + BIGSEP + k_iter + BIGSEP + iteration.toString() + "__hash", _scriptHash),
        IntegerEntry(k_ITERATION + SEP + scriptType, iteration)
    ]
}

@Callable(i)
func removeScript(scriptId: String) = {
    let callerPbk = toBase58String(i.callerPublicKey)
    if (!_isValidOperator(callerPbk)) then {
        throw("Not allowed")
    } else {
        let initiatorAdr = getStringValue(scriptId + "__initiator")
        if (initiatorAdr != addressFromPublicKey(i.callerPublicKey).toString()) then {
            throw("Only creator can remove")
        } else {
            let scriptType = getStringValue(scriptId + BIGSEP + k_scriptType)
            let _scriptHash = getStringValue(scriptId + BIGSEP + k_scriptHash)
            let iter = getIntegerValue(scriptId + BIGSEP + k_iter)
            let approvedHashes = getString("script_" + scriptType + "__approved").valueOrElse("")
            strict _checkApproved = if (indexOf(approvedHashes, _scriptHash) != unit) then throw("Cant remove approved script, use revoke") else true
            [
                DeleteEntry(scriptId + BIGSEP + k_scriptType),
                DeleteEntry(scriptId + BIGSEP + k_storageAddress),
                DeleteEntry(scriptId + BIGSEP + k_scriptHash),
                DeleteEntry(scriptId + BIGSEP + k_desc),
                DeleteEntry(scriptId + BIGSEP + k_iter),
                DeleteEntry(scriptId + BIGSEP + k_validBlocks),
                DeleteEntry(scriptType + "__iteration__" + iter.toString() + "__hash"),
                DeleteEntry(scriptId + BIGSEP + k_initiator),
                DeleteEntry(scriptId + BIGSEP + k_confirmations),
                DeleteEntry("script_" + scriptType + SEP + _scriptHash + SEP + k_validFrom),
                DeleteEntry("script_" + scriptType + SEP + _scriptHash + SEP + k_validTo)
            ]
        }
    }
}

@Callable(i)
func addConfirmationScript(scriptId: String) = {
    let callerPbk = toBase58String(i.callerPublicKey)
    let limit = _getLimit()
    strict checkAdmin = if (!_isAdmin(callerPbk)) then throw("Only admins can confirm") else true
    strict _scriptHash = getString(scriptId + BIGSEP + k_scriptHash).valueOrErrorMessage("Ivalid scriptId")
    let scriptType = getStringValue(scriptId + BIGSEP + k_scriptType)
    let validBlocks = getIntegerValue(scriptId + BIGSEP + k_validBlocks)
    # let approvedHashesStr = getString("script_" + scriptType + "__approved").valueOrElse("")
    # strict _checkApproved = if (indexOf(approvedHashesStr, _scriptHash) != unit) then throw("This script is approved already") else true 
    let globalIteration = getInteger(k_ITERATION + SEP + scriptType).valueOrElse(0)
    let thisIteration = getInteger(scriptId + "__iteration").value()
    let confirmationsStr = getString(scriptId + "__confirmations").valueOrElse("")
    let confirmations = confirmationsStr.split(",")
    if (indexOf(confirmations, callerPbk) != unit) then {
        throw("You already confirmed this script")
    } else {
        let newList = if (confirmationsStr != "") then confirmations :+ callerPbk else [callerPbk]
        let newListStr = makeString(newList, ",")
        let approved = size(newList) >= limit
        let result = if (approved) then {
            [
                IntegerEntry("script_" + scriptType + SEP + _scriptHash + SEP + k_validFrom, height + getIntegerValue(k_SCRIPTS_COOLDOWN)),
                IntegerEntry("script_" + scriptType + SEP + _scriptHash + SEP + k_validTo, height + validBlocks)
            ]
        } else {
            []
        }
        result ++ [
                StringEntry(scriptId + "__confirmations", newListStr)
            ]
    }
}

@Callable(i)
func revokeConfirmationScript(scriptId: String) = {
    let callerPbk = toBase58String(i.callerPublicKey)
    let limit = _getLimit()
    if (!_isAdmin(callerPbk)) then {
        throw("Only admins can revoke")
    } else {
        strict _scriptHash = getString(scriptId + "__scriptHash").valueOrErrorMessage("Invalid scriptId")
        let scriptType = getStringValue(scriptId + "__scriptType")
        let confirmationsStr = getString(scriptId + "__confirmations").valueOrElse("")
        let confirmations = confirmationsStr.split(",")
        let confirmIndex = indexOf(confirmations, callerPbk)
        let isExpired = height > getInteger("script_" + scriptType + SEP + _scriptHash + SEP + k_validTo).valueOrElse(height+1)
        strict checkExpired = if (!isExpired) then true else throw("Expired scripts can't be revoked")
        if (confirmIndex == unit) then {
            throw("You didn't confirm this script, nothing to revoke")
        } else {
            # Remove caller's confirmation
            let newConfirmations = confirmations.removeByIndex(confirmIndex.value())
            let newConfirmationsStr = makeString(newConfirmations, ",")
            
            # Check if script was previously approved
            let wasApproved = isDefined(getInteger(_scriptHash + SEP + k_validFrom)) && isDefined(getInteger(_scriptHash + SEP + k_validTo))
            let stillApproved = size(newConfirmations) >= limit
            
            # Determine what state changes are needed
            let stateChanges = if (wasApproved && !stillApproved) then {
                # Script was approved but now falls below threshold - remove from approved list
                [
                    DeleteEntry("script_" + scriptType + SEP + _scriptHash + SEP + k_validFrom),
                    DeleteEntry("script_" + scriptType + SEP + _scriptHash + SEP + k_validTo)
                ]
            } else {
                # No change in approval status needed
                []
            }
            
            # Return all state changes
            stateChanges ++ [
                StringEntry(scriptId + "__confirmations", newConfirmationsStr)
            ]
        }
    }
}

@Callable(i)
func addTask(name: String, dapp: String, txid: String, executeTs: Int, desc: String, initiatorAdr: String, txdata: String) = {
    let callerPbk = toBase58String(i.callerPublicKey)
    if (!_isValidOperator(callerPbk)) then {
        throw("Not allowed")
    } else {
        strict _nameCheck = if (size(name) == 0) then throw("Name can't be empty") else true
        strict _dappCheck = if (!_validateAddress(dapp)) then throw("Incorrect dapp") else true
        strict _txidCheck = if (size(txid) == 0) then throw("txid can't be empty") else true
        strict _timestampCheck = if (!_validateTs(executeTs)) then throw("Invalid execution timestamp - cannot be more than 1 hour in the past") else true
        strict _descCheck = if (size(desc) == 0) then throw("Description can't be empty") else true
        strict _initiatorCheck = if (addressFromPublicKey(i.callerPublicKey).toString() != initiatorAdr) then throw("Initiator address doesn't match caller") else true
        strict _txdataCheck = if (size(txdata) == 0) then throw("Tx data can't be empty") else true
        let taskId = "tx_" + dapp + SEP + txid
        [
            StringEntry(taskId + "__name", name),
            StringEntry(taskId + "__dapp", dapp),
            StringEntry(taskId + "__txId", txid),
            StringEntry(taskId + "__desc", desc),
            StringEntry(taskId + "__txdata", txdata),
            StringEntry(taskId + "__initiator", initiatorAdr),
            IntegerEntry(taskId + "__ts", executeTs)
        ]
    }
}

@Callable(i)
func removeTask(taskId: String) = {
    let callerPbk = toBase58String(i.callerPublicKey)
    if (!_isValidOperator(callerPbk)) then {
        throw("Not allowed")
    } else {
        let initiatorAdr = getStringValue(taskId + "__initiator")
        if (initiatorAdr != addressFromPublicKey(i.callerPublicKey).toString()) then {
            throw("Only creator can remove")
        } else {
            strict txId = getString(taskId + "__txId").valueOrErrorMessage("Ivalid task")
            let dapp = getStringValue(taskId + "__dapp")
            [
                DeleteEntry(taskId + "__name"),
                DeleteEntry(taskId + "__dapp"),
                DeleteEntry(taskId + "__txId"),
                DeleteEntry(taskId + "__desc"),
                DeleteEntry(taskId + "__txdata"),
                DeleteEntry(taskId + "__initiator"),
                DeleteEntry(taskId + "__ts"),
                DeleteEntry(taskId + "__confirmations"),
                DeleteEntry(dapp + SEP + txId + "__approved")
            ]
        }
    }
}

@Callable(i)
func addConfirmation(taskId: String) = {
    let callerPbk = toBase58String(i.callerPublicKey)
    let limit = _getLimit()
    if (!_isAdmin(callerPbk)) then {
        throw("Only admins can confirm")
    } else {
        strict txId = getString(taskId + "__txId").valueOrErrorMessage("Ivalid task")
        let dapp = getStringValue(taskId + "__dapp")
        let confirmationsStr = getString(taskId + "__confirmations").valueOrElse("")
        let confirmations = confirmationsStr.split(",")
        if (indexOf(confirmations, callerPbk) != unit) then {
            throw("You already confirmed this task")
        } else {
            let newList = if (confirmationsStr != "") then confirmations :+ callerPbk else [callerPbk]
            let newListStr = makeString(newList, ",")
            let approved = size(newList) >= limit
            [
                StringEntry(taskId + "__confirmations", newListStr),
                BooleanEntry(dapp + SEP + txId + "__approved", approved)
            ]
        }
    }
}

@Callable(i)
func revokeTask(taskId: String) = {
    let callerPbk = toBase58String(i.callerPublicKey)
    let limit = _getLimit()
    if (!_isAdmin(callerPbk)) then {
        throw("Only admins can call")
    } else {
        strict txId = getString(taskId + "__txId").valueOrErrorMessage("Ivalid task")
        let confirmationsStr = getString(taskId + "__confirmations").valueOrElse("")
        let confirmations = confirmationsStr.split(",")
        let indexMy = indexOf(confirmations, callerPbk)
        if (indexMy == unit) then {
            throw("You did not confirm this task, nothing to revoke")
        } else {
            let dapp = getStringValue(taskId + "__dapp")
            let newList = confirmations.removeByIndex(indexMy.value())
            let newListStr = makeString(newList, ",")
            let approved = size(newList) >= limit
            [
                StringEntry(taskId + "__confirmations", newListStr),
                BooleanEntry(dapp + SEP + txId + "__approved", approved)
            ]
        }
    }
}

@Verifier(tx)
func verify() = {
    ### Can be moved out from Verifier or to datastorage
  let CONTRACT_TYPE = "APPROVER_CONTRACT"

    if (_isInited()) then {
      match tx {
      case ttx:SetScriptTransaction => {
        let scriptHasha = blake2b256(ttx.script.value()).value().toBase58String()

        let validFrom = getInteger(this, "script_" + CONTRACT_TYPE + "_" + scriptHasha + "_validFrom").valueOrErrorMessage("validFrom not set")
        let validTo = getInteger(this, "script_" + CONTRACT_TYPE + "_" + scriptHasha + "_validTo").valueOrErrorMessage("validTo not set")
        height >= validFrom && height <= validTo
      }
      case otx => {
        getBoolean(this.value(), this.toString() + "_" + toBase58String(otx.id) + "__approved").valueOrElse(false)
      }
    } 
  } else {
      sigVerify(tx.bodyBytes, tx.proofs[0], tx.senderPublicKey)
    }
}"
}
Changelog:
{
"2025.11.14 00:47": "{-# STDLIB_VERSION 6 #-}
{-# CONTENT_TYPE DAPP #-}
{-# SCRIPT_TYPE ACCOUNT #-}


let VERSION = "1.0.3"

### KEYS ###
let SEP = "_"
let BIGSEP = "__"
let k_ADMINS = "ADMINS_PBKS"
let k_SUPERADMINS = "SUPERADMINS_PBKS"
let k_MANAGERS = "MANAGERS_PBKS"
let k_LIMIT = "VOTES_QUORUM"
let k_SCRIPT_TYPES = "SCRIPT_TYPES"
let k_ITERATION = "ITERATION"
let k_INITED = "INITED"
let k_SCRIPTS_COOLDOWN = "SCRIPTSCOOLDOWN"

let k_scriptType = "scriptType"
let k_scriptHash = "scriptHash"
let k_storageAddress = "storageAddress"
let k_desc = "desc"
let k_iter = "iteration"
let k_initiator = "initiator"
let k_confirmations = "confirmations"
let k_validBlocks = "validBlocks"
let k_validTo = "validTo"
let k_validFrom = "validFrom"

func _validatePbk(pbk: String) =  {
    let keyBytes = pbk.fromBase58String()
    strict checkLength = if (size(keyBytes) != 32) then throw("Invalid public key") else true
    strict address = addressFromPublicKey(keyBytes)
    strict addressStr = address.toString()
    size(addressStr) > 0
}

func _validateAddress(address: String) = {
    strict addr = addressFromStringValue(address)
    true
}

func _validateTs(ts: Int) = {
    let currTs = match blockInfoByHeight(height) {
        case block:BlockInfo =>
            block.timestamp
        case _ => throw("Can't find block")
    }
    
    # Only restrict timestamps from being too old (1 hour ago), allow any future time
    let oneHour = 60 * 60 * 1000  # 1 hour in milliseconds
    let minTs = currTs - oneHour
    
    ts >= minTs
}


func _getLimit() = {
    getInteger(k_LIMIT).valueOrErrorMessage("Quorum limit not defined")
}

func _isAdmin(pbk: String) = {
    let adminsStr = getString(k_ADMINS).valueOrElse("")
    let admins = adminsStr.split(",")
    indexOf(admins, pbk) != unit
}

func _isSuperAdmin(pbk: String) = {
    let superAdminsStr = getString(k_SUPERADMINS).valueOrElse("")
    let superAdmins = superAdminsStr.split(",")
    indexOf(superAdmins, pbk) != unit
}

func _isManager(pbk: String) = {
    let managersStr = getString(k_MANAGERS).valueOrElse("")
    let managers = managersStr.split(",")
    indexOf(managers, pbk) != unit
}

func _isValidOperator(pbk: String) = {
    let managersStr = getString(k_MANAGERS).valueOrElse("")
    let managers = split(managersStr, ",")
    let isAdmin = _isAdmin(pbk)
    let isManager = _isManager(pbk)
    isAdmin || isManager
}

func _isInited() = {
    getBoolean(k_INITED).valueOrElse(false)
}

### UTILITY ###


func saveTxData(taskId: String, data: String) = {
    let indices = [1,2,3,4,5,6,7,8,9,10]
    let dataSize = size(data)
    strict isNotEmpty = if (dataSize != 0) then true else throw("saveTxData: Empty data passed")
    let chunkSize = 30000
    let chunksCount = if (dataSize % chunkSize == 0) then dataSize / chunkSize else dataSize / chunkSize + 1

    func makeChunks(accum: (String, List[StringEntry]), index: Int) = {
        let chunk = take(accum._1, chunkSize)
        if (size(chunk) == 0) then {accum} else {
            let nextData = drop(accum._1, chunkSize)
            (nextData, accum._2 :+ StringEntry(taskId + "__txdata_" + toString(index), chunk))
        }
    }
    
    # Save each chunk using FOLD
    let entries = FOLD<10>(indices, (data, [StringEntry(taskId + "__txdata_chunks", toString(chunksCount))]), makeChunks)
    
    entries._2
}

### CALLABLES ###
@Callable(i)
func init(admins_pbks: List[String], limit: Int, scriptsCooldown: Int) = {
    strict checkInited = if (_isInited()) then throw("Already inited") else true
    strict size_admins = if (size(admins_pbks) != 0) then true else throw("At least one admin pbk should be passed")
    strict check_admins = if (!isDefined(getString(k_ADMINS))) then true else throw("Already inited")
    strict zeroLimit = if (limit > 0) then true else throw("Limit should be > 0")
    strict checkScriptsCooldown = if (scriptsCooldown > 0) then true else throw("Scripts cooldown should be > 0")
    strict limit_ok = if (size(admins_pbks) >= limit) then true else throw("Limit quorum should be <= size of list of admins, limit: " + limit.toString() + ", current size: " + size(admins_pbks).toString())
    func validate(accum: String, next: String) = {
        _validatePbk(next)
    }
    strict validRes = FOLD<10>(admins_pbks, "", validate)
    let adminsStr = makeString(admins_pbks, ",")
    [
        StringEntry(k_ADMINS, adminsStr),
        IntegerEntry(k_LIMIT, limit),
        IntegerEntry(k_SCRIPTS_COOLDOWN, scriptsCooldown),
        BooleanEntry(k_INITED, true)
    ]
}

## MANAGE ##
@Callable(i)
func addAdmin(pbk: String) = {
    let callerPbk = toBase58String(i.callerPublicKey)
    strict isSelf = if (addressFromPublicKey(i.callerPublicKey) == this) then true else throw("Only self invoke")
    strict isValid = if (_validatePbk(pbk)) then true else throw("Invalid admin pbk")
    let admins = getStringValue(k_ADMINS)
    strict checkPresent = if (indexOf(admins, pbk) != unit) then throw("This admin already present") else true
    let newAdminsStr = admins + "," + pbk

    [
        StringEntry(k_ADMINS, newAdminsStr)
    ]
}

@Callable(i)
func removeAdmin(pbk: String) = {
    let callerPbk = toBase58String(i.callerPublicKey)
    strict isSelf = if (addressFromPublicKey(i.callerPublicKey) == this) then true else throw("Only self invoke")
    strict isValid = if (_validatePbk(pbk)) then true else throw("Invalid admin pbk")
    let oldAdminsStr = getStringValue(k_ADMINS)
    let oldAdmins = oldAdminsStr.split(",")
    strict checkCount = if (size(oldAdmins) <= 1) then throw("Cannot remove, at least one admin should remain") else true
    let index = indexOf(oldAdmins, pbk).valueOrErrorMessage("This pbk is not an admin")
    let newAdmins = removeByIndex(oldAdmins, index)
    let newAdminsStr = makeString(newAdmins, ",")
    [
        StringEntry(k_ADMINS, newAdminsStr)
    ]
}

@Callable(i)
func addScriptType(scriptType: String) = {
    
    let callerPbk = toBase58String(i.callerPublicKey)
    strict isSelf = if (addressFromPublicKey(i.callerPublicKey) == this) then true else throw("Only self invoke")
    strict checkEmpty = if (size(scriptType) == 0) then throw("Script type can not be empty") else true
    let scriptTypesStr = getString(k_SCRIPT_TYPES).valueOrElse("")
    let scriptTypes = split_4C(scriptTypesStr, ",")
    strict checkExists = if (indexOf(scriptTypes, scriptType) != unit) then throw("Script type <" + scriptType + "> already added") else true
    let newScriptTypes = if (scriptTypesStr != "") then scriptTypes :+ scriptType else [scriptType]
    let newScriptTypesStr = makeString(newScriptTypes, ",")
    [
        StringEntry(k_SCRIPT_TYPES, newScriptTypesStr)
    ]
}

@Callable(i)
func removeScriptType(scriptType: String) = {
    let callerPbk = toBase58String(i.callerPublicKey)
    strict isSelf = if (addressFromPublicKey(i.callerPublicKey) == this) then true else throw("Only self invoke")
    let scriptTypesStr = getString(k_SCRIPT_TYPES).valueOrElse("")
    let scriptTypes = split_4C(scriptTypesStr, ",")
    let index = indexOf(scriptTypes, scriptType).valueOrErrorMessage("Script type <" + scriptType + "> is not present")
    
    let newScriptTypes = removeByIndex(scriptTypes, index)
    let newScriptTypesStr = makeString(newScriptTypes, ",")
    [
        StringEntry(k_SCRIPT_TYPES, newScriptTypesStr)
    ]
}

@Callable(i)
func setLimit(limit: Int) = {
    let callerPbk = toBase58String(i.callerPublicKey)
    strict isSelf = if (addressFromPublicKey(i.callerPublicKey) == this) then true else throw("Only self invoke")
    strict zeroLimit = if (limit > 0) then true else throw("Limit should be > 0")
    let admins = getStringValue(k_ADMINS)
    if (size(admins) < limit) then throw("Limit quorum should be <= size of list of admins, limit: " + limit.toString() + ", current size: " + size(admins).toString())
    else {
        [
            IntegerEntry(k_LIMIT, limit)
        ]
    }
}

@Callable(i)
func setCooldownBlocks(scriptsCooldown: Int) = {
    let callerPbk = toBase58String(i.callerPublicKey)
    strict isSelf = if (addressFromPublicKey(i.callerPublicKey) == this) then true else throw("Only self invoke")
    strict checkScriptsCooldown = if (scriptsCooldown > 0) then true else throw("Scripts cooldown should be > 0")
    [
        IntegerEntry(k_SCRIPTS_COOLDOWN, scriptsCooldown)
    ]
}

## MAIN OPERATIONS ##
@Callable(i)
func addScript(scriptType: String, storageAddress: String, _scriptHash: String, desc: String, initiatorAdr: String, validBlocks: Int) = {
    let callerPbk = toBase58String(i.callerPublicKey)
    let storageScriptHash = addressFromStringValue(storageAddress).scriptHash().value().toBase58String()
    let scriptTypesStr = getStringValue(k_SCRIPT_TYPES)
   
    # validations
    strict isAllowed = if (!_isValidOperator(callerPbk)) then throw("You are not allowed") else true
    strict _checkScriptTypes = if (scriptTypesStr.indexOf(scriptType) == unit) then throw("invalid script type or it is not present in storage: " + scriptType) else true
    strict _scriptStorageHashCheck = if (storageScriptHash != _scriptHash) then throw("Stored and passed as argument scriptHashes not match") else true
    strict _descCheck = if (size(desc) == 0) then throw("Description can't be empty") else true
    strict _initiatorCheck = if (addressFromPublicKey(i.callerPublicKey).toString() != initiatorAdr) then throw("Initiator address doesn't match caller") else true
    strict _checkValidBlocks = if (validBlocks > 0) then true else throw("Valid blocks should be > 0")
    
    let iteration = getInteger(k_ITERATION + SEP + scriptType).valueOrElse(0) + 1
    let scriptId = "script%%" + scriptType + "%%" + _scriptHash
    strict _checkPresent = if (isDefined(getString(scriptId + BIGSEP + k_scriptHash))) then throw("Script with this hash already present") else true
    [
        StringEntry(scriptId + BIGSEP + k_scriptType, scriptType),
        StringEntry(scriptId + BIGSEP + k_storageAddress, storageAddress),
        StringEntry(scriptId + BIGSEP + k_scriptHash, storageScriptHash),
        StringEntry(scriptId + BIGSEP + k_desc, desc),
        IntegerEntry(scriptId + BIGSEP + k_iter, iteration),
        IntegerEntry(scriptId + BIGSEP + k_validBlocks, validBlocks),
        StringEntry(scriptId + BIGSEP + k_initiator, initiatorAdr),
        StringEntry(scriptType + BIGSEP + k_iter + BIGSEP + iteration.toString() + "__hash", _scriptHash),
        IntegerEntry(k_ITERATION + SEP + scriptType, iteration)
    ]
}

@Callable(i)
func removeScript(scriptId: String) = {
    let callerPbk = toBase58String(i.callerPublicKey)
    if (!_isValidOperator(callerPbk)) then {
        throw("Not allowed")
    } else {
        let initiatorAdr = getStringValue(scriptId + "__initiator")
        if (initiatorAdr != addressFromPublicKey(i.callerPublicKey).toString()) then {
            throw("Only creator can remove")
        } else {
            let scriptType = getStringValue(scriptId + BIGSEP + k_scriptType)
            let _scriptHash = getStringValue(scriptId + BIGSEP + k_scriptHash)
            let iter = getIntegerValue(scriptId + BIGSEP + k_iter)
            let approvedHashes = getString("script_" + scriptType + "__approved").valueOrElse("")
            strict _checkApproved = if (indexOf(approvedHashes, _scriptHash) != unit) then throw("Cant remove approved script, use revoke") else true
            [
                DeleteEntry(scriptId + BIGSEP + k_scriptType),
                DeleteEntry(scriptId + BIGSEP + k_storageAddress),
                DeleteEntry(scriptId + BIGSEP + k_scriptHash),
                DeleteEntry(scriptId + BIGSEP + k_desc),
                DeleteEntry(scriptId + BIGSEP + k_iter),
                DeleteEntry(scriptId + BIGSEP + k_validBlocks),
                DeleteEntry(scriptType + "__iteration__" + iter.toString() + "__hash"),
                DeleteEntry(scriptId + BIGSEP + k_initiator),
                DeleteEntry(scriptId + BIGSEP + k_confirmations),
                DeleteEntry("script_" + scriptType + SEP + _scriptHash + SEP + k_validFrom),
                DeleteEntry("script_" + scriptType + SEP + _scriptHash + SEP + k_validTo)
            ]
        }
    }
}

@Callable(i)
func addConfirmationScript(scriptId: String) = {
    let callerPbk = toBase58String(i.callerPublicKey)
    let limit = _getLimit()
    strict checkAdmin = if (!_isAdmin(callerPbk)) then throw("Only admins can confirm") else true
    strict _scriptHash = getString(scriptId + BIGSEP + k_scriptHash).valueOrErrorMessage("Ivalid scriptId")
    let scriptType = getStringValue(scriptId + BIGSEP + k_scriptType)
    let validBlocks = getIntegerValue(scriptId + BIGSEP + k_validBlocks)
    # let approvedHashesStr = getString("script_" + scriptType + "__approved").valueOrElse("")
    # strict _checkApproved = if (indexOf(approvedHashesStr, _scriptHash) != unit) then throw("This script is approved already") else true 
    let globalIteration = getInteger(k_ITERATION + SEP + scriptType).valueOrElse(0)
    let thisIteration = getInteger(scriptId + "__iteration").value()
    let confirmationsStr = getString(scriptId + "__confirmations").valueOrElse("")
    let confirmations = confirmationsStr.split(",")
    if (indexOf(confirmations, callerPbk) != unit) then {
        throw("You already confirmed this script")
    } else {
        let newList = if (confirmationsStr != "") then confirmations :+ callerPbk else [callerPbk]
        let newListStr = makeString(newList, ",")
        let approved = size(newList) >= limit
        let result = if (approved) then {
            [
                IntegerEntry("script_" + scriptType + SEP + _scriptHash + SEP + k_validFrom, height + getIntegerValue(k_SCRIPTS_COOLDOWN)),
                IntegerEntry("script_" + scriptType + SEP + _scriptHash + SEP + k_validTo, height + validBlocks)
            ]
        } else {
            []
        }
        result ++ [
                StringEntry(scriptId + "__confirmations", newListStr)
            ]
    }
}

@Callable(i)
func revokeConfirmationScript(scriptId: String) = {
    let callerPbk = toBase58String(i.callerPublicKey)
    let limit = _getLimit()
    if (!_isAdmin(callerPbk)) then {
        throw("Only admins can revoke")
    } else {
        strict _scriptHash = getString(scriptId + "__scriptHash").valueOrErrorMessage("Invalid scriptId")
        let scriptType = getStringValue(scriptId + "__scriptType")
        let confirmationsStr = getString(scriptId + "__confirmations").valueOrElse("")
        let confirmations = confirmationsStr.split(",")
        let confirmIndex = indexOf(confirmations, callerPbk)
        let isExpired = height > getInteger("script_" + scriptType + SEP + _scriptHash + SEP + k_validTo).valueOrElse(height+1)
        strict checkExpired = if (!isExpired) then true else throw("Expired scripts can't be revoked")
        if (confirmIndex == unit) then {
            throw("You didn't confirm this script, nothing to revoke")
        } else {
            # Remove caller's confirmation
            let newConfirmations = confirmations.removeByIndex(confirmIndex.value())
            let newConfirmationsStr = makeString(newConfirmations, ",")
            
            # Check if script was previously approved
            let wasApproved = isDefined(getInteger(_scriptHash + SEP + k_validFrom)) && isDefined(getInteger(_scriptHash + SEP + k_validTo))
            let stillApproved = size(newConfirmations) >= limit
            
            # Determine what state changes are needed
            let stateChanges = if (wasApproved && !stillApproved) then {
                # Script was approved but now falls below threshold - remove from approved list
                [
                    DeleteEntry("script_" + scriptType + SEP + _scriptHash + SEP + k_validFrom),
                    DeleteEntry("script_" + scriptType + SEP + _scriptHash + SEP + k_validTo)
                ]
            } else {
                # No change in approval status needed
                []
            }
            
            # Return all state changes
            stateChanges ++ [
                StringEntry(scriptId + "__confirmations", newConfirmationsStr)
            ]
        }
    }
}

@Callable(i)
func addTask(name: String, dapp: String, txid: String, executeTs: Int, desc: String, initiatorAdr: String, txdata: String) = {
    let callerPbk = toBase58String(i.callerPublicKey)
    if (!_isValidOperator(callerPbk)) then {
        throw("Not allowed")
    } else {
        strict _nameCheck = if (size(name) == 0) then throw("Name can't be empty") else true
        strict _dappCheck = if (!_validateAddress(dapp)) then throw("Incorrect dapp") else true
        strict _txidCheck = if (size(txid) == 0) then throw("txid can't be empty") else true
        strict _timestampCheck = if (!_validateTs(executeTs)) then throw("Invalid execution timestamp - cannot be more than 1 hour in the past") else true
        strict _descCheck = if (size(desc) == 0) then throw("Description can't be empty") else true
        strict _initiatorCheck = if (addressFromPublicKey(i.callerPublicKey).toString() != initiatorAdr) then throw("Initiator address doesn't match caller") else true
        strict _txdataCheck = if (size(txdata) == 0) then throw("Tx data can't be empty") else true
        let taskId = "tx_" + dapp + SEP + txid
        [
            StringEntry(taskId + "__name", name),
            StringEntry(taskId + "__dapp", dapp),
            StringEntry(taskId + "__txId", txid),
            StringEntry(taskId + "__desc", desc),
            StringEntry(taskId + "__txdata", txdata),
            StringEntry(taskId + "__initiator", initiatorAdr),
            IntegerEntry(taskId + "__ts", executeTs)
        ]
    }
}

@Callable(i)
func removeTask(taskId: String) = {
    let callerPbk = toBase58String(i.callerPublicKey)
    if (!_isValidOperator(callerPbk)) then {
        throw("Not allowed")
    } else {
        let initiatorAdr = getStringValue(taskId + "__initiator")
        if (initiatorAdr != addressFromPublicKey(i.callerPublicKey).toString()) then {
            throw("Only creator can remove")
        } else {
            strict txId = getString(taskId + "__txId").valueOrErrorMessage("Ivalid task")
            let dapp = getStringValue(taskId + "__dapp")
            [
                DeleteEntry(taskId + "__name"),
                DeleteEntry(taskId + "__dapp"),
                DeleteEntry(taskId + "__txId"),
                DeleteEntry(taskId + "__desc"),
                DeleteEntry(taskId + "__txdata"),
                DeleteEntry(taskId + "__initiator"),
                DeleteEntry(taskId + "__ts"),
                DeleteEntry(taskId + "__confirmations"),
                DeleteEntry(dapp + SEP + txId + "__approved")
            ]
        }
    }
}

@Callable(i)
func addConfirmation(taskId: String) = {
    let callerPbk = toBase58String(i.callerPublicKey)
    let limit = _getLimit()
    if (!_isAdmin(callerPbk)) then {
        throw("Only admins can confirm")
    } else {
        strict txId = getString(taskId + "__txId").valueOrErrorMessage("Ivalid task")
        let dapp = getStringValue(taskId + "__dapp")
        let confirmationsStr = getString(taskId + "__confirmations").valueOrElse("")
        let confirmations = confirmationsStr.split(",")
        if (indexOf(confirmations, callerPbk) != unit) then {
            throw("You already confirmed this task")
        } else {
            let newList = if (confirmationsStr != "") then confirmations :+ callerPbk else [callerPbk]
            let newListStr = makeString(newList, ",")
            let approved = size(newList) >= limit
            [
                StringEntry(taskId + "__confirmations", newListStr),
                BooleanEntry(dapp + SEP + txId + "__approved", approved)
            ]
        }
    }
}

@Callable(i)
func revokeTask(taskId: String) = {
    let callerPbk = toBase58String(i.callerPublicKey)
    let limit = _getLimit()
    if (!_isAdmin(callerPbk)) then {
        throw("Only admins can call")
    } else {
        strict txId = getString(taskId + "__txId").valueOrErrorMessage("Ivalid task")
        let confirmationsStr = getString(taskId + "__confirmations").valueOrElse("")
        let confirmations = confirmationsStr.split(",")
        let indexMy = indexOf(confirmations, callerPbk)
        if (indexMy == unit) then {
            throw("You did not confirm this task, nothing to revoke")
        } else {
            let dapp = getStringValue(taskId + "__dapp")
            let newList = confirmations.removeByIndex(indexMy.value())
            let newListStr = makeString(newList, ",")
            let approved = size(newList) >= limit
            [
                StringEntry(taskId + "__confirmations", newListStr),
                BooleanEntry(dapp + SEP + txId + "__approved", approved)
            ]
        }
    }
}

@Verifier(tx)
func verify() = {
    ### Can be moved out from Verifier or to datastorage
  let CONTRACT_TYPE = "APPROVER_CONTRACT"

    if (_isInited()) then {
      match tx {
      case ttx:SetScriptTransaction => {
        let scriptHasha = blake2b256(ttx.script.value()).value().toBase58String()

        let validFrom = getInteger(this, "script_" + CONTRACT_TYPE + "_" + scriptHasha + "_validFrom").valueOrErrorMessage("validFrom not set")
        let validTo = getInteger(this, "script_" + CONTRACT_TYPE + "_" + scriptHasha + "_validTo").valueOrErrorMessage("validTo not set")
        height >= validFrom && height <= validTo
      }
      case otx => {
        getBoolean(this.value(), this.toString() + "_" + toBase58String(otx.id) + "__approved").valueOrElse(false)
      }
    } 
  } else {
      sigVerify(tx.bodyBytes, tx.proofs[0], tx.senderPublicKey)
    }
}"
}