From 4121feb3474255897887318e553b801e27c0ef83 Mon Sep 17 00:00:00 2001 From: Jordan Ribbink Date: Wed, 10 Jan 2024 22:18:41 -0800 Subject: [PATCH] Update Cadence code to stable cadence --- .vscode/settings.json | 3 + cadence/contracts/FCL.cdc | 64 +- cadence/contracts/FCLCrypto.cdc | 51 +- cadence/contracts/FungibleToken.cdc | 199 ----- cadence/contracts/utility/FlowToken.cdc | 311 ++++++++ cadence/contracts/utility/FungibleToken.cdc | 201 +++++ .../utility/FungibleTokenMetadataViews.cdc | 192 +++++ cadence/contracts/utility/MetadataViews.cdc | 722 ++++++++++++++++++ .../contracts/utility/NonFungibleToken.cdc | 191 +++++ cadence/contracts/utility/ViewResolver.cdc | 55 ++ cadence/scripts/getAccount.cdc | 2 +- cadence/scripts/getAccounts.cdc | 4 +- cadence/transactions/fundFLOW.cdc | 12 +- cadence/transactions/init.cdc | 2 +- cadence/transactions/newAccount.cdc | 14 +- cadence/transactions/updateAccount.cdc | 2 +- contexts/ConfigContext.tsx | 17 +- flow.json | 43 +- src/accounts.ts | 2 + src/fclConfig.ts | 17 +- src/harness/cmds/q1.js | 2 +- src/harness/cmds/q2.js | 2 +- 22 files changed, 1823 insertions(+), 285 deletions(-) create mode 100644 .vscode/settings.json delete mode 100644 cadence/contracts/FungibleToken.cdc create mode 100644 cadence/contracts/utility/FlowToken.cdc create mode 100644 cadence/contracts/utility/FungibleToken.cdc create mode 100644 cadence/contracts/utility/FungibleTokenMetadataViews.cdc create mode 100644 cadence/contracts/utility/MetadataViews.cdc create mode 100644 cadence/contracts/utility/NonFungibleToken.cdc create mode 100644 cadence/contracts/utility/ViewResolver.cdc diff --git a/.vscode/settings.json b/.vscode/settings.json new file mode 100644 index 0000000..cac0e10 --- /dev/null +++ b/.vscode/settings.json @@ -0,0 +1,3 @@ +{ + "editor.formatOnSave": true +} \ No newline at end of file diff --git a/cadence/contracts/FCL.cdc b/cadence/contracts/FCL.cdc index 56950f3..c6af0f6 100644 --- a/cadence/contracts/FCL.cdc +++ b/cadence/contracts/FCL.cdc @@ -1,12 +1,12 @@ -pub contract FCL { - pub let storagePath: StoragePath +access(all) contract FCL { + access(all) let storagePath: StoragePath - pub struct Account { - pub let type: String - pub let address: Address - pub let keyId: Int - pub var label: String - pub var scopes: [String] + access(all) struct FCLAccount { + access(all) let type: String + access(all) let address: Address + access(all) let keyId: Int + access(all) var label: String + access(all) var scopes: [String] init(address: Address, label: String, scopes: [String]) { self.type = "ACCOUNT" @@ -16,59 +16,73 @@ pub contract FCL { self.scopes = scopes } - pub fun update(label: String, scopes: [String]) { + access(all) fun update(label: String, scopes: [String]) { self.label = label self.scopes = scopes } } - pub resource Root { - pub let key: [UInt8] - pub let accounts: {Address: Account} + access(all) resource Root { + access(all) let key: [UInt8] + access(all) let accounts: {Address: FCLAccount} init (_ key: String) { self.key = key.decodeHex() self.accounts = {} } - pub fun add(_ acct: Account) { + access(all) fun add(_ acct: FCLAccount) { self.accounts[acct.address] = acct } - pub fun update(address: Address, label: String, scopes: [String]) { + access(all) fun update(address: Address, label: String, scopes: [String]) { let acct = self.accounts[address] acct!.update(label: label, scopes: scopes) self.accounts[address] = acct } } - pub fun accounts(): {Address: Account} { - return self.account.borrow<&Root>(from: self.storagePath)!.accounts + access(all) fun accounts(): &{Address: FCLAccount} { + return self.account.storage.borrow<&Root>(from: self.storagePath)!.accounts } - pub fun getServiceKey(): [UInt8] { - return self.account.borrow<&Root>(from: self.storagePath)!.key + access(all) fun getServiceKey(): [UInt8] { + let keyRef = self.account.storage.borrow<&Root>(from: self.storagePath)!.key + return keyRef.map(fun (_ x: UInt8): UInt8 { + return x + }) } - pub fun new(label: String, scopes: [String], address: Address?): AuthAccount { - let acct = AuthAccount(payer: self.account) - acct.addPublicKey(self.getServiceKey()) + access(all) fun new(label: String, scopes: [String], address: Address?): &Account { + let acct = Account(payer: self.account) + let rawKey = self.getServiceKey() + let decodedKey = RLP.decodeList(rawKey) + + acct.keys.add( + publicKey: PublicKey( + publicKey: decodedKey[0].slice(from: 2, upTo: decodedKey[0].length), + signatureAlgorithm: SignatureAlgorithm.ECDSA_P256, + ), + hashAlgorithm: HashAlgorithm(decodedKey[2][0])!, + weight: UFix64.fromBigEndianBytes(decodedKey[3])! + ) self.account + .storage .borrow<&Root>(from: self.storagePath)! - .add(Account(address: address ?? acct.address, label: label, scopes: scopes)) + .add(FCLAccount(address: address ?? acct.address, label: label, scopes: scopes)) return acct } - pub fun update(address: Address, label: String, scopes: [String]) { - self.account.borrow<&Root>(from: self.storagePath)! + access(all) fun update(address: Address, label: String, scopes: [String]) { + self.account.storage.borrow<&Root>(from: self.storagePath)! .update(address: address, label: label, scopes: scopes) } init (key: String, initAccountsLabels: [String]) { self.storagePath = /storage/FCL_DEV_WALLET - self.account.save(<- create Root(key), to: self.storagePath) + self.account.storage.save(<- create Root(key), to: self.storagePath) self.new(label: initAccountsLabels[0], scopes: [], address: self.account.address) var acctInitIndex = 1 diff --git a/cadence/contracts/FCLCrypto.cdc b/cadence/contracts/FCLCrypto.cdc index ebebb20..8fde29c 100644 --- a/cadence/contracts/FCLCrypto.cdc +++ b/cadence/contracts/FCLCrypto.cdc @@ -1,6 +1,21 @@ -pub contract FCLCrypto { - - pub fun verifyUserSignatures( +/* + FCLCrypto + + The FCLCrypto contract provides functions which allow to verify signatures and check for signing power. +*/ + +access(all) contract FCLCrypto { + + /// verifyUserSignatures allows to verify the user signatures for the given account. + /// + /// @param address: The address of the account + /// @param message: The signed data + /// @param keyIndices: This integer array maps the signatures to the account keys by index + /// @param signatures: The signatures belonging to the account keys + /// + /// @return Whether all signatures are valid and the combined total key weight reaches signing power + /// + access(all) fun verifyUserSignatures( address: Address, message: String, keyIndices: [Int], @@ -15,7 +30,16 @@ pub contract FCLCrypto { ) } - pub fun verifyAccountProofSignatures( + /// verifyAccountProofSignatures allows to verify the account proof signatures for the given account. + /// + /// @param address: The address of the account + /// @param message: The signed data + /// @param keyIndices: This integer array maps the signatures to the account keys by index + /// @param signatures: The signatures belonging to the account keys + /// + /// @return Whether all signatures are valid and the combined total key weight reaches signing power + /// + access(all) fun verifyAccountProofSignatures( address: Address, message: String, keyIndices: [Int], @@ -37,7 +61,18 @@ pub contract FCLCrypto { ) } - priv fun verifySignatures( + /// verifySignatures is a private function which provides the functionality to verify + /// signatures for the public functions. + /// + /// @param address: The address of the account + /// @param message: The signed data + /// @param keyIndices: This integer array maps the signatures to the account keys by index + /// @param signatures: The signatures belonging to the account keys + /// @param domainSeparationTag: The domain tag originally used for the signatures + /// + /// @return Whether all signatures are valid and the combined total key weight reaches signing power + /// + access(self) fun verifySignatures( address: Address, message: String, keyIndices: [Int], @@ -96,9 +131,9 @@ pub contract FCLCrypto { return totalWeight >= 1000.0 } - priv let domainSeparationTagFlowUser: String - priv let domainSeparationTagFCLUser: String - priv let domainSeparationTagAccountProof: String + access(self) let domainSeparationTagFlowUser: String + access(self) let domainSeparationTagFCLUser: String + access(self) let domainSeparationTagAccountProof: String init() { self.domainSeparationTagFlowUser = "FLOW-V0.0-user" diff --git a/cadence/contracts/FungibleToken.cdc b/cadence/contracts/FungibleToken.cdc deleted file mode 100644 index 6fddb0e..0000000 --- a/cadence/contracts/FungibleToken.cdc +++ /dev/null @@ -1,199 +0,0 @@ -/** - -# The Flow Fungible Token standard - -## `FungibleToken` contract interface - -The interface that all fungible token contracts would have to conform to. -If a users wants to deploy a new token contract, their contract -would need to implement the FungibleToken interface. - -Their contract would have to follow all the rules and naming -that the interface specifies. - -## `Vault` resource - -Each account that owns tokens would need to have an instance -of the Vault resource stored in their account storage. - -The Vault resource has methods that the owner and other users can call. - -## `Provider`, `Receiver`, and `Balance` resource interfaces - -These interfaces declare pre-conditions and post-conditions that restrict -the execution of the functions in the Vault. - -They are separate because it gives the user the ability to share -a reference to their Vault that only exposes the fields functions -in one or more of the interfaces. - -It also gives users the ability to make custom resources that implement -these interfaces to do various things with the tokens. -For example, a faucet can be implemented by conforming -to the Provider interface. - -By using resources and interfaces, users of FungibleToken contracts -can send and receive tokens peer-to-peer, without having to interact -with a central ledger smart contract. To send tokens to another user, -a user would simply withdraw the tokens from their Vault, then call -the deposit function on another user's Vault to complete the transfer. - -*/ - -/// FungibleToken -/// -/// The interface that fungible token contracts implement. -/// -pub contract interface FungibleToken { - - /// The total number of tokens in existence. - /// It is up to the implementer to ensure that the total supply - /// stays accurate and up to date - /// - pub var totalSupply: UFix64 - - /// TokensInitialized - /// - /// The event that is emitted when the contract is created - /// - pub event TokensInitialized(initialSupply: UFix64) - - /// TokensWithdrawn - /// - /// The event that is emitted when tokens are withdrawn from a Vault - /// - pub event TokensWithdrawn(amount: UFix64, from: Address?) - - /// TokensDeposited - /// - /// The event that is emitted when tokens are deposited into a Vault - /// - pub event TokensDeposited(amount: UFix64, to: Address?) - - /// Provider - /// - /// The interface that enforces the requirements for withdrawing - /// tokens from the implementing type. - /// - /// It does not enforce requirements on `balance` here, - /// because it leaves open the possibility of creating custom providers - /// that do not necessarily need their own balance. - /// - pub resource interface Provider { - - /// withdraw subtracts tokens from the owner's Vault - /// and returns a Vault with the removed tokens. - /// - /// The function's access level is public, but this is not a problem - /// because only the owner storing the resource in their account - /// can initially call this function. - /// - /// The owner may grant other accounts access by creating a private - /// capability that allows specific other users to access - /// the provider resource through a reference. - /// - /// The owner may also grant all accounts access by creating a public - /// capability that allows all users to access the provider - /// resource through a reference. - /// - pub fun withdraw(amount: UFix64): @Vault { - post { - // `result` refers to the return value - result.balance == amount: - "Withdrawal amount must be the same as the balance of the withdrawn Vault" - } - } - } - - /// Receiver - /// - /// The interface that enforces the requirements for depositing - /// tokens into the implementing type. - /// - /// We do not include a condition that checks the balance because - /// we want to give users the ability to make custom receivers that - /// can do custom things with the tokens, like split them up and - /// send them to different places. - /// - pub resource interface Receiver { - - /// deposit takes a Vault and deposits it into the implementing resource type - /// - pub fun deposit(from: @Vault) - } - - /// Balance - /// - /// The interface that contains the `balance` field of the Vault - /// and enforces that when new Vaults are created, the balance - /// is initialized correctly. - /// - pub resource interface Balance { - - /// The total balance of a vault - /// - pub var balance: UFix64 - - init(balance: UFix64) { - post { - self.balance == balance: - "Balance must be initialized to the initial balance" - } - } - } - - /// Vault - /// - /// The resource that contains the functions to send and receive tokens. - /// - pub resource Vault: Provider, Receiver, Balance { - - // The declaration of a concrete type in a contract interface means that - // every Fungible Token contract that implements the FungibleToken interface - // must define a concrete `Vault` resource that conforms to the `Provider`, `Receiver`, - // and `Balance` interfaces, and declares their required fields and functions - - /// The total balance of the vault - /// - pub var balance: UFix64 - - // The conforming type must declare an initializer - // that allows prioviding the initial balance of the Vault - // - init(balance: UFix64) - - /// withdraw subtracts `amount` from the Vault's balance - /// and returns a new Vault with the subtracted balance - /// - pub fun withdraw(amount: UFix64): @Vault { - pre { - self.balance >= amount: - "Amount withdrawn must be less than or equal than the balance of the Vault" - } - post { - // use the special function `before` to get the value of the `balance` field - // at the beginning of the function execution - // - self.balance == before(self.balance) - amount: - "New Vault balance must be the difference of the previous balance and the withdrawn Vault" - } - } - - /// deposit takes a Vault and adds its balance to the balance of this Vault - /// - pub fun deposit(from: @Vault) { - post { - self.balance == before(self.balance) + before(from.balance): - "New Vault balance must be the sum of the previous balance and the deposited Vault" - } - } - } - - /// createEmptyVault allows any user to create a new Vault that has a zero balance - /// - pub fun createEmptyVault(): @Vault { - post { - result.balance == 0.0: "The newly created Vault must have zero balance" - } - } -} diff --git a/cadence/contracts/utility/FlowToken.cdc b/cadence/contracts/utility/FlowToken.cdc new file mode 100644 index 0000000..eea8022 --- /dev/null +++ b/cadence/contracts/utility/FlowToken.cdc @@ -0,0 +1,311 @@ +import "FungibleToken" +import "MetadataViews" +import "FungibleTokenMetadataViews" +import "ViewResolver" + +access(all) contract FlowToken: ViewResolver { + + // Total supply of Flow tokens in existence + access(all) var totalSupply: UFix64 + + // Event that is emitted when the contract is created + access(all) event TokensInitialized(initialSupply: UFix64) + + // Event that is emitted when tokens are withdrawn from a Vault + access(all) event TokensWithdrawn(amount: UFix64, from: Address?) + + // Event that is emitted when tokens are deposited to a Vault + access(all) event TokensDeposited(amount: UFix64, to: Address?) + + // Event that is emitted when new tokens are minted + access(all) event TokensMinted(amount: UFix64) + + // Event that is emitted when tokens are destroyed + access(all) event TokensBurned(amount: UFix64) + + // Event that is emitted when a new minter resource is created + access(all) event MinterCreated(allowedAmount: UFix64) + + // Event that is emitted when a new burner resource is created + access(all) event BurnerCreated() + + // Vault + // + // Each user stores an instance of only the Vault in their storage + // The functions in the Vault and governed by the pre and post conditions + // in FungibleToken when they are called. + // The checks happen at runtime whenever a function is called. + // + // Resources can only be created in the context of the contract that they + // are defined in, so there is no way for a malicious user to create Vaults + // out of thin air. A special Minter resource needs to be defined to mint + // new tokens. + // + access(all) resource Vault: FungibleToken.Vault, FungibleToken.Provider, FungibleToken.Receiver, ViewResolver.Resolver { + + // holds the balance of a users tokens + access(all) var balance: UFix64 + + access(all) view fun getBalance(): UFix64 { + return self.balance + } + + // initialize the balance at resource creation time + init(balance: UFix64) { + self.balance = balance + } + + /// getSupportedVaultTypes optionally returns a list of vault types that this receiver accepts + access(all) view fun getSupportedVaultTypes(): {Type: Bool} { + return {self.getType(): true} + } + + access(all) view fun isSupportedVaultType(type: Type): Bool { + if (type == self.getType()) { return true } else { return false } + } + + /// Returns the storage path where the vault should typically be stored + access(all) view fun getDefaultStoragePath(): StoragePath? { + return /storage/flowTokenVault + } + + /// Returns the public path where this vault should have a public capability + access(all) view fun getDefaultPublicPath(): PublicPath? { + return /public/flowTokenReceiver + } + + // withdraw + // + // Function that takes an integer amount as an argument + // and withdraws that amount from the Vault. + // It creates a new temporary Vault that is used to hold + // the money that is being transferred. It returns the newly + // created Vault to the context that called so it can be deposited + // elsewhere. + // + access(FungibleToken.Withdrawable) fun withdraw(amount: UFix64): @{FungibleToken.Vault} { + self.balance = self.balance - amount + emit TokensWithdrawn(amount: amount, from: self.owner?.address) + return <-create Vault(balance: amount) + } + + // deposit + // + // Function that takes a Vault object as an argument and adds + // its balance to the balance of the owners Vault. + // It is allowed to destroy the sent Vault because the Vault + // was a temporary holder of the tokens. The Vault's balance has + // been consumed and therefore can be destroyed. + access(all) fun deposit(from: @{FungibleToken.Vault}) { + let vault <- from as! @FlowToken.Vault + self.balance = self.balance + vault.balance + emit TokensDeposited(amount: vault.balance, to: self.owner?.address) + vault.balance = 0.0 + destroy vault + } + + /// Get all the Metadata Views implemented by FlowToken + /// + /// @return An array of Types defining the implemented views. This value will be used by + /// developers to know which parameter to pass to the resolveView() method. + /// + access(all) view fun getViews(): [Type]{ + return FlowToken.getViews() + } + + /// Get a Metadata View from FlowToken + /// + /// @param view: The Type of the desired view. + /// @return A structure representing the requested view. + /// + access(all) fun resolveView(_ view: Type): AnyStruct? { + return FlowToken.resolveView(view) + } + + access(all) fun createEmptyVault(): @{FungibleToken.Vault} { + return <-create Vault(balance: 0.0) + } + } + + // createEmptyVault + // + // Function that creates a new Vault with a balance of zero + // and returns it to the calling context. A user must call this function + // and store the returned Vault in their storage in order to allow their + // account to be able to receive deposits of this token type. + // + access(all) fun createEmptyVault(): @FlowToken.Vault { + return <-create Vault(balance: 0.0) + } + + access(all) view fun getViews(): [Type] { + return [Type(), + Type(), + Type(), + Type()] + } + + /// Get a Metadata View from FlowToken + /// + /// @param view: The Type of the desired view. + /// @return A structure representing the requested view. + /// + access(all) fun resolveView(_ view: Type): AnyStruct? { + switch view { + case Type(): + return FungibleTokenMetadataViews.FTView( + ftDisplay: self.resolveView(Type()) as! FungibleTokenMetadataViews.FTDisplay?, + ftVaultData: self.resolveView(Type()) as! FungibleTokenMetadataViews.FTVaultData? + ) + case Type(): + let media = MetadataViews.Media( + file: MetadataViews.HTTPFile( + url: FlowToken.getLogoURI() + ), + mediaType: "image/svg+xml" + ) + let medias = MetadataViews.Medias([media]) + return FungibleTokenMetadataViews.FTDisplay( + name: "FLOW Network Token", + symbol: "FLOW", + description: "FLOW is the native token for the Flow blockchain. It is required for securing the network, transaction fees, storage fees, staking, FLIP voting and may be used by applications built on the Flow Blockchain", + externalURL: MetadataViews.ExternalURL("https://flow.com"), + logos: medias, + socials: { + "twitter": MetadataViews.ExternalURL("https://twitter.com/flow_blockchain") + } + ) + case Type(): + return FungibleTokenMetadataViews.FTVaultData( + storagePath: /storage/flowTokenVault, + receiverPath: /public/flowTokenReceiver, + metadataPath: /public/flowTokenBalance, + providerPath: /private/flowTokenVault, + receiverLinkedType: Type<&FlowToken.Vault>(), + metadataLinkedType: Type<&FlowToken.Vault>(), + providerLinkedType: Type<&FlowToken.Vault>(), + createEmptyVaultFunction: (fun (): @{FungibleToken.Vault} { + return <-FlowToken.createEmptyVault() + }) + ) + case Type(): + return FungibleTokenMetadataViews.TotalSupply(totalSupply: FlowToken.totalSupply) + } + return nil + } + + access(all) resource Administrator { + // createNewMinter + // + // Function that creates and returns a new minter resource + // + access(all) fun createNewMinter(allowedAmount: UFix64): @Minter { + emit MinterCreated(allowedAmount: allowedAmount) + return <-create Minter(allowedAmount: allowedAmount) + } + + // createNewBurner + // + // Function that creates and returns a new burner resource + // + access(all) fun createNewBurner(): @Burner { + emit BurnerCreated() + return <-create Burner() + } + } + + // Minter + // + // Resource object that token admin accounts can hold to mint new tokens. + // + access(all) resource Minter { + + // the amount of tokens that the minter is allowed to mint + access(all) var allowedAmount: UFix64 + + // mintTokens + // + // Function that mints new tokens, adds them to the total supply, + // and returns them to the calling context. + // + access(all) fun mintTokens(amount: UFix64): @FlowToken.Vault { + pre { + amount > UFix64(0): "Amount minted must be greater than zero" + amount <= self.allowedAmount: "Amount minted must be less than the allowed amount" + } + FlowToken.totalSupply = FlowToken.totalSupply + amount + self.allowedAmount = self.allowedAmount - amount + emit TokensMinted(amount: amount) + return <-create Vault(balance: amount) + } + + init(allowedAmount: UFix64) { + self.allowedAmount = allowedAmount + } + } + + // Burner + // + // Resource object that token admin accounts can hold to burn tokens. + // + access(all) resource Burner { + + // burnTokens + // + // Function that destroys a Vault instance, effectively burning the tokens. + // + // Note: the burned tokens are automatically subtracted from the + // total supply in the Vault destructor. + // + access(all) fun burnTokens(from: @FlowToken.Vault) { + FlowToken.burnTokens(from: <-from) + } + } + + access(all) fun burnTokens(from: @FlowToken.Vault) { + let vault <- from as! @FlowToken.Vault + let amount = vault.balance + destroy vault + if amount > 0.0 { + FlowToken.totalSupply = FlowToken.totalSupply - amount + emit TokensBurned(amount: amount) + } + } + + /// Gets the Flow Logo XML URI from storage + access(all) fun getLogoURI(): String { + return FlowToken.account.storage.copy(from: /storage/flowTokenLogoURI) ?? "" + } + + init(adminAccount: auth(Storage, Capabilities) &Account) { + self.totalSupply = 0.0 + + // Create the Vault with the total supply of tokens and save it in storage + // + let vault <- create Vault(balance: self.totalSupply) + + // Example of how to resolve a metadata view for a Vault + let ftView = vault.resolveView(Type()) + + adminAccount.storage.save(<-vault, to: /storage/flowTokenVault) + + // Create a public capability to the stored Vault that only exposes + // the `deposit` method through the `Receiver` interface + // + let receiverCapability = adminAccount.capabilities.storage.issue<&FlowToken.Vault>(/storage/flowTokenVault) + adminAccount.capabilities.publish(receiverCapability, at: /public/flowTokenReceiver) + + // Create a public capability to the stored Vault that only exposes + // the `balance` field through the `Balance` interface + // + let balanceCapability = adminAccount.capabilities.storage.issue<&FlowToken.Vault>(/storage/flowTokenVault) + adminAccount.capabilities.publish(balanceCapability, at: /public/flowTokenBalance) + + let admin <- create Administrator() + adminAccount.storage.save(<-admin, to: /storage/flowTokenAdmin) + + // Emit an event that shows that the contract was initialized + emit TokensInitialized(initialSupply: self.totalSupply) + + } +} diff --git a/cadence/contracts/utility/FungibleToken.cdc b/cadence/contracts/utility/FungibleToken.cdc new file mode 100644 index 0000000..f8d2e84 --- /dev/null +++ b/cadence/contracts/utility/FungibleToken.cdc @@ -0,0 +1,201 @@ +/** + +# The Flow Fungible Token standard + +## `FungibleToken` contract + +The Fungible Token standard is no longer an interface +that all fungible token contracts would have to conform to. + +If a users wants to deploy a new token contract, their contract +does not need to implement the FungibleToken interface, but their tokens +do need to implement the interfaces defined in this contract. + +## `Vault` resource interface + +Each fungible token resource type needs to implement the `Vault` resource interface. + +## `Provider`, `Receiver`, and `Balance` resource interfaces + +These interfaces declare pre-conditions and post-conditions that restrict +the execution of the functions in the Vault. + +They are separate because it gives the user the ability to share +a reference to their Vault that only exposes the fields functions +in one or more of the interfaces. + +It also gives users the ability to make custom resources that implement +these interfaces to do various things with the tokens. +For example, a faucet can be implemented by conforming +to the Provider interface. + +*/ + +import "ViewResolver" + +/// FungibleToken +/// +/// Fungible Token implementations are no longer required to implement the fungible token +/// interface. We still have it as an interface here because there are some useful +/// utility methods that many projects will still want to have on their contracts, +/// but they are by no means required. all that is required is that the token +/// implements the `Vault` interface +access(all) contract FungibleToken { + + // An entitlement for allowing the withdrawal of tokens from a Vault + access(all) entitlement Withdrawable + + /// The event that is emitted when tokens are withdrawn from a Vault + access(all) event Withdraw(amount: UFix64, from: Address?, type: String) + + /// The event that is emitted when tokens are deposited to a Vault + access(all) event Deposit(amount: UFix64, to: Address?, type: String) + + /// Provider + /// + /// The interface that enforces the requirements for withdrawing + /// tokens from the implementing type. + /// + /// It does not enforce requirements on `balance` here, + /// because it leaves open the possibility of creating custom providers + /// that do not necessarily need their own balance. + /// + access(all) resource interface Provider { + + /// withdraw subtracts tokens from the owner's Vault + /// and returns a Vault with the removed tokens. + /// + /// The function's access level is public, but this is not a problem + /// because only the owner storing the resource in their account + /// can initially call this function. + /// + /// The owner may grant other accounts access by creating a private + /// capability that allows specific other users to access + /// the provider resource through a reference. + /// + /// The owner may also grant all accounts access by creating a public + /// capability that allows all users to access the provider + /// resource through a reference. + /// + access(Withdrawable) fun withdraw(amount: UFix64): @{Vault} { + post { + // `result` refers to the return value + result.getBalance() == amount: + "Withdrawal amount must be the same as the balance of the withdrawn Vault" + emit Withdraw(amount: amount, from: self.owner?.address, type: self.getType().identifier) + } + } + } + + /// Receiver + /// + /// The interface that enforces the requirements for depositing + /// tokens into the implementing type. + /// + /// We do not include a condition that checks the balance because + /// we want to give users the ability to make custom receivers that + /// can do custom things with the tokens, like split them up and + /// send them to different places. + /// + access(all) resource interface Receiver { + + /// deposit takes a Vault and deposits it into the implementing resource type + /// + access(all) fun deposit(from: @{Vault}) + + /// getSupportedVaultTypes optionally returns a list of vault types that this receiver accepts + access(all) view fun getSupportedVaultTypes(): {Type: Bool} { + pre { true: "dummy" } + } + + /// Returns whether or not the given type is accepted by the Receiver + /// A vault that can accept any type should just return true by default + access(all) view fun isSupportedVaultType(type: Type): Bool { + pre { true: "dummy" } + } + } + + /// Vault + /// + /// Ideally, this interface would also conform to Receiver, Balance, Transferor, Provider, and Resolver + /// but that is not supported yet + /// + access(all) resource interface Vault: Receiver, Provider, ViewResolver.Resolver { + + //access(all) event ResourceDestroyed(balance: UFix64 = self.getBalance()) + + /// Get the balance of the vault + access(all) view fun getBalance(): UFix64 + + /// getSupportedVaultTypes optionally returns a list of vault types that this receiver accepts + access(all) view fun getSupportedVaultTypes(): {Type: Bool} { + // Below check is implemented to make sure that run-time type would + // only get returned when the parent resource conforms with `FungibleToken.Vault`. + if self.getType().isSubtype(of: Type<@{FungibleToken.Vault}>()) { + return {self.getType(): true} + } else { + // Return an empty dictionary as the default value for resource who don't + // implement `FungibleToken.Vault`, such as `FungibleTokenSwitchboard`, `TokenForwarder` etc. + return {} + } + } + + access(all) view fun isSupportedVaultType(type: Type): Bool { + return self.getSupportedVaultTypes()[type] ?? false + } + + /// Returns the storage path where the vault should typically be stored + access(all) view fun getDefaultStoragePath(): StoragePath? + + /// Returns the public path where this vault should have a public capability + access(all) view fun getDefaultPublicPath(): PublicPath? + + /// Returns the public path where this vault's Receiver should have a public capability + /// Publishing a Receiver Capability at a different path enables alternate Receiver implementations to be used + /// in the same canonical namespace as the underlying Vault. + access(all) view fun getDefaultReceiverPath(): PublicPath? { + return nil + } + + /// withdraw subtracts `amount` from the Vault's balance + /// and returns a new Vault with the subtracted balance + /// + access(Withdrawable) fun withdraw(amount: UFix64): @{Vault} { + pre { + self.getBalance() >= amount: + "Amount withdrawn must be less than or equal than the balance of the Vault" + } + post { + // use the special function `before` to get the value of the `balance` field + // at the beginning of the function execution + // + self.getBalance() == before(self.getBalance()) - amount: + "New Vault balance must be the difference of the previous balance and the withdrawn Vault balance" + } + } + + /// deposit takes a Vault and adds its balance to the balance of this Vault + /// + access(all) fun deposit(from: @{FungibleToken.Vault}) { + // Assert that the concrete type of the deposited vault is the same + // as the vault that is accepting the deposit + pre { + from.isInstance(self.getType()): + "Cannot deposit an incompatible token type" + emit Deposit(amount: from.getBalance(), to: self.owner?.address, type: from.getType().identifier) + } + post { + self.getBalance() == before(self.getBalance()) + before(from.getBalance()): + "New Vault balance must be the sum of the previous balance and the deposited Vault" + } + } + + /// createEmptyVault allows any user to create a new Vault that has a zero balance + /// + access(all) fun createEmptyVault(): @{Vault} { + post { + result.getBalance() == 0.0: "The newly created Vault must have zero balance" + } + } + } +} \ No newline at end of file diff --git a/cadence/contracts/utility/FungibleTokenMetadataViews.cdc b/cadence/contracts/utility/FungibleTokenMetadataViews.cdc new file mode 100644 index 0000000..c424258 --- /dev/null +++ b/cadence/contracts/utility/FungibleTokenMetadataViews.cdc @@ -0,0 +1,192 @@ +import FungibleToken from "FungibleToken" +import MetadataViews from "MetadataViews" +import ViewResolver from "ViewResolver" + +/// This contract implements the metadata standard proposed +/// in FLIP-1087. +/// +/// Ref: https://github.com/onflow/flips/blob/main/application/20220811-fungible-tokens-metadata.md +/// +/// Structs and resources can implement one or more +/// metadata types, called views. Each view type represents +/// a different kind of metadata. +/// +access(all) contract FungibleTokenMetadataViews { + + /// FTView wraps FTDisplay and FTVaultData, and is used to give a complete + /// picture of a Fungible Token. Most Fungible Token contracts should + /// implement this view. + /// + access(all) struct FTView { + access(all) let ftDisplay: FTDisplay? + access(all) let ftVaultData: FTVaultData? + view init( + ftDisplay: FTDisplay?, + ftVaultData: FTVaultData? + ) { + self.ftDisplay = ftDisplay + self.ftVaultData = ftVaultData + } + } + + /// Helper to get a FT view. + /// + /// @param viewResolver: A reference to the resolver resource + /// @return A FTView struct + /// + access(all) fun getFTView(viewResolver: &{ViewResolver.Resolver}): FTView { + let maybeFTView = viewResolver.resolveView(Type()) + if let ftView = maybeFTView { + return ftView as! FTView + } + return FTView( + ftDisplay: self.getFTDisplay(viewResolver), + ftVaultData: self.getFTVaultData(viewResolver) + ) + } + + /// View to expose the information needed to showcase this FT. + /// This can be used by applications to give an overview and + /// graphics of the FT. + /// + access(all) struct FTDisplay { + /// The display name for this token. + /// + /// Example: "Flow" + /// + access(all) let name: String + + /// The abbreviated symbol for this token. + /// + /// Example: "FLOW" + access(all) let symbol: String + + /// A description the provides an overview of this token. + /// + /// Example: "The FLOW token is the native currency of the Flow network." + access(all) let description: String + + /// External link to a URL to view more information about the fungible token. + access(all) let externalURL: MetadataViews.ExternalURL + + /// One or more versions of the fungible token logo. + access(all) let logos: MetadataViews.Medias + + /// Social links to reach the fungible token's social homepages. + /// Possible keys may be "instagram", "twitter", "discord", etc. + access(all) let socials: {String: MetadataViews.ExternalURL} + + view init( + name: String, + symbol: String, + description: String, + externalURL: MetadataViews.ExternalURL, + logos: MetadataViews.Medias, + socials: {String: MetadataViews.ExternalURL} + ) { + self.name = name + self.symbol = symbol + self.description = description + self.externalURL = externalURL + self.logos = logos + self.socials = socials + } + } + + /// Helper to get FTDisplay in a way that will return a typed optional. + /// + /// @param viewResolver: A reference to the resolver resource + /// @return An optional FTDisplay struct + /// + access(all) fun getFTDisplay(_ viewResolver: &{ViewResolver.Resolver}): FTDisplay? { + if let maybeDisplayView = viewResolver.resolveView(Type()) { + if let displayView = maybeDisplayView as? FTDisplay { + return displayView + } + } + return nil + } + + /// View to expose the information needed store and interact with a FT vault. + /// This can be used by applications to setup a FT vault with proper + /// storage and public capabilities. + /// + access(all) struct FTVaultData { + /// Path in storage where this FT vault is recommended to be stored. + access(all) let storagePath: StoragePath + + /// Public path which must be linked to expose the public receiver capability. + access(all) let receiverPath: PublicPath + + /// Public path which must be linked to expose the balance and resolver public capabilities. + access(all) let metadataPath: PublicPath + + /// Private path which should be linked to expose the provider capability to withdraw funds + /// from the vault. + access(all) let providerPath: PrivatePath + + /// Type that should be linked at the `receiverPath`. This is a restricted type requiring + /// the `FungibleToken.Receiver` interface. + access(all) let receiverLinkedType: Type + + /// Type that should be linked at the `receiverPath`. This is a restricted type requiring + /// the `ViewResolver.Resolver` interfaces. + access(all) let metadataLinkedType: Type + + /// Type that should be linked at the aforementioned private path. This + /// is normally a restricted type with at a minimum the `FungibleToken.Provider` interface. + access(all) let providerLinkedType: Type + + /// Function that allows creation of an empty FT vault that is intended + /// to store the funds. + access(all) let createEmptyVault: fun(): @{FungibleToken.Vault} + + view init( + storagePath: StoragePath, + receiverPath: PublicPath, + metadataPath: PublicPath, + providerPath: PrivatePath, + receiverLinkedType: Type, + metadataLinkedType: Type, + providerLinkedType: Type, + createEmptyVaultFunction: fun(): @{FungibleToken.Vault} + ) { + pre { + receiverLinkedType.isSubtype(of: Type<&{FungibleToken.Receiver}>()): "Receiver public type must include FungibleToken.Receiver." + metadataLinkedType.isSubtype(of: Type<&{ViewResolver.Resolver}>()): "Metadata public type must include ViewResolver.Resolver interfaces." + providerLinkedType.isSubtype(of: Type<&{FungibleToken.Provider}>()): "Provider type must include FungibleToken.Provider interface." + } + self.storagePath = storagePath + self.receiverPath = receiverPath + self.metadataPath = metadataPath + self.providerPath = providerPath + self.receiverLinkedType = receiverLinkedType + self.metadataLinkedType = metadataLinkedType + self.providerLinkedType = providerLinkedType + self.createEmptyVault = createEmptyVaultFunction + } + } + + /// Helper to get FTVaultData in a way that will return a typed Optional. + /// + /// @param viewResolver: A reference to the resolver resource + /// @return A optional FTVaultData struct + /// + access(all) fun getFTVaultData(_ viewResolver: &{ViewResolver.Resolver}): FTVaultData? { + if let view = viewResolver.resolveView(Type()) { + if let v = view as? FTVaultData { + return v + } + } + return nil + } + + /// View to expose the total supply of the Vault's token + access(all) struct TotalSupply { + access(all) let supply: UFix64 + + view init(totalSupply: UFix64) { + self.supply = totalSupply + } + } +} \ No newline at end of file diff --git a/cadence/contracts/utility/MetadataViews.cdc b/cadence/contracts/utility/MetadataViews.cdc new file mode 100644 index 0000000..8207326 --- /dev/null +++ b/cadence/contracts/utility/MetadataViews.cdc @@ -0,0 +1,722 @@ +import FungibleToken from "FungibleToken" +import NonFungibleToken from "NonFungibleToken" +import ViewResolver from "ViewResolver" + +/// This contract implements the metadata standard proposed +/// in FLIP-0636. +/// +/// Ref: https://github.com/onflow/flips/blob/main/application/20210916-nft-metadata.md +/// +/// Structs and resources can implement one or more +/// metadata types, called views. Each view type represents +/// a different kind of metadata, such as a creator biography +/// or a JPEG image file. +/// +access(all) contract MetadataViews { + + /// Display is a basic view that includes the name, description and + /// thumbnail for an object. Most objects should implement this view. + /// + access(all) struct Display { + + /// The name of the object. + /// + /// This field will be displayed in lists and therefore should + /// be short an concise. + /// + access(all) let name: String + + /// A written description of the object. + /// + /// This field will be displayed in a detailed view of the object, + /// so can be more verbose (e.g. a paragraph instead of a single line). + /// + access(all) let description: String + + /// A small thumbnail representation of the object. + /// + /// This field should be a web-friendly file (i.e JPEG, PNG) + /// that can be displayed in lists, link previews, etc. + /// + access(all) let thumbnail: {File} + + view init( + name: String, + description: String, + thumbnail: {File} + ) { + self.name = name + self.description = description + self.thumbnail = thumbnail + } + } + + /// Helper to get Display in a typesafe way + /// + /// @param viewResolver: A reference to the resolver resource + /// @return An optional Display struct + /// + access(all) fun getDisplay(_ viewResolver: &{ViewResolver.Resolver}) : Display? { + if let view = viewResolver.resolveView(Type()) { + if let v = view as? Display { + return v + } + } + return nil + } + + /// Generic interface that represents a file stored on or off chain. Files + /// can be used to references images, videos and other media. + /// + access(all) struct interface File { + access(all) view fun uri(): String + } + + /// View to expose a file that is accessible at an HTTP (or HTTPS) URL. + /// + access(all) struct HTTPFile: File { + access(all) let url: String + + view init(url: String) { + self.url = url + } + + access(all) view fun uri(): String { + return self.url + } + } + + /// View to expose a file stored on IPFS. + /// IPFS images are referenced by their content identifier (CID) + /// rather than a direct URI. A client application can use this CID + /// to find and load the image via an IPFS gateway. + /// + access(all) struct IPFSFile: File { + + /// CID is the content identifier for this IPFS file. + /// + /// Ref: https://docs.ipfs.io/concepts/content-addressing/ + /// + access(all) let cid: String + + /// Path is an optional path to the file resource in an IPFS directory. + /// + /// This field is only needed if the file is inside a directory. + /// + /// Ref: https://docs.ipfs.io/concepts/file-systems/ + /// + access(all) let path: String? + + view init(cid: String, path: String?) { + self.cid = cid + self.path = path + } + + /// This function returns the IPFS native URL for this file. + /// Ref: https://docs.ipfs.io/how-to/address-ipfs-on-web/#native-urls + /// + /// @return The string containing the file uri + /// + access(all) view fun uri(): String { + if let path = self.path { + return "ipfs://".concat(self.cid).concat("/").concat(path) + } + + return "ipfs://".concat(self.cid) + } + } + + /// View to represent a file with an correspoiding mediaType. + /// + access(all) struct Media { + + /// File for the media + /// + access(all) let file: {File} + + /// media-type comes on the form of type/subtype as described here + /// https://developer.mozilla.org/en-US/docs/Web/HTTP/Basics_of_HTTP/MIME_types + /// + access(all) let mediaType: String + + view init(file: {File}, mediaType: String) { + self.file=file + self.mediaType=mediaType + } + } + + /// Wrapper view for multiple media views + /// + access(all) struct Medias { + + /// An arbitrary-sized list for any number of Media items + access(all) let items: [Media] + + view init(_ items: [Media]) { + self.items = items + } + } + + /// Helper to get Medias in a typesafe way + /// + /// @param viewResolver: A reference to the resolver resource + /// @return A optional Medias struct + /// + access(all) fun getMedias(_ viewResolver: &{ViewResolver.Resolver}) : Medias? { + if let view = viewResolver.resolveView(Type()) { + if let v = view as? Medias { + return v + } + } + return nil + } + + /// View to represent a license according to https://spdx.org/licenses/ + /// This view can be used if the content of an NFT is licensed. + /// + access(all) struct License { + access(all) let spdxIdentifier: String + + view init(_ identifier: String) { + self.spdxIdentifier = identifier + } + } + + /// Helper to get License in a typesafe way + /// + /// @param viewResolver: A reference to the resolver resource + /// @return A optional License struct + /// + access(all) fun getLicense(_ viewResolver: &{ViewResolver.Resolver}) : License? { + if let view = viewResolver.resolveView(Type()) { + if let v = view as? License { + return v + } + } + return nil + } + + /// View to expose a URL to this item on an external site. + /// This can be used by applications like .find and Blocto to direct users + /// to the original link for an NFT or a project page that describes the NFT collection. + /// eg https://www.my-nft-project.com/overview-of-nft-collection + /// + access(all) struct ExternalURL { + access(all) let url: String + + view init(_ url: String) { + self.url=url + } + } + + /// Helper to get ExternalURL in a typesafe way + /// + /// @param viewResolver: A reference to the resolver resource + /// @return A optional ExternalURL struct + /// + access(all) fun getExternalURL(_ viewResolver: &{ViewResolver.Resolver}) : ExternalURL? { + if let view = viewResolver.resolveView(Type()) { + if let v = view as? ExternalURL { + return v + } + } + return nil + } + + /// View that defines the composable royalty standard that gives marketplaces a + /// unified interface to support NFT royalties. + /// + access(all) struct Royalty { + + /// Generic FungibleToken Receiver for the beneficiary of the royalty + /// Can get the concrete type of the receiver with receiver.getType() + /// Recommendation - Users should create a new link for a FlowToken + /// receiver for this using `getRoyaltyReceiverPublicPath()`, and not + /// use the default FlowToken receiver. This will allow users to update + /// the capability in the future to use a more generic capability + access(all) let receiver: Capability<&{FungibleToken.Receiver}> + + /// Multiplier used to calculate the amount of sale value transferred to + /// royalty receiver. Note - It should be between 0.0 and 1.0 + /// Ex - If the sale value is x and multiplier is 0.56 then the royalty + /// value would be 0.56 * x. + /// Generally percentage get represented in terms of basis points + /// in solidity based smart contracts while cadence offers `UFix64` + /// that already supports the basis points use case because its + /// operations are entirely deterministic integer operations and support + /// up to 8 points of precision. + access(all) let cut: UFix64 + + /// Optional description: This can be the cause of paying the royalty, + /// the relationship between the `wallet` and the NFT, or anything else + /// that the owner might want to specify. + access(all) let description: String + + view init(receiver: Capability<&{FungibleToken.Receiver}>, cut: UFix64, description: String) { + pre { + cut >= 0.0 && cut <= 1.0 : "Cut value should be in valid range i.e [0,1]" + } + self.receiver = receiver + self.cut = cut + self.description = description + } + } + + /// Wrapper view for multiple Royalty views. + /// Marketplaces can query this `Royalties` struct from NFTs + /// and are expected to pay royalties based on these specifications. + /// + access(all) struct Royalties { + + /// Array that tracks the individual royalties + access(self) let cutInfos: [Royalty] + + access(all) view init(_ cutInfos: [Royalty]) { + // Validate that sum of all cut multipliers should not be greater than 1.0 + var totalCut = 0.0 + for royalty in cutInfos { + totalCut = totalCut + royalty.cut + } + assert(totalCut <= 1.0, message: "Sum of cutInfos multipliers should not be greater than 1.0") + // Assign the cutInfos + self.cutInfos = cutInfos + } + + /// Return the cutInfos list + /// + /// @return An array containing all the royalties structs + /// + access(all) view fun getRoyalties(): [Royalty] { + return self.cutInfos + } + } + + /// Helper to get Royalties in a typesafe way + /// + /// @param viewResolver: A reference to the resolver resource + /// @return A optional Royalties struct + /// + access(all) fun getRoyalties(_ viewResolver: &{ViewResolver.Resolver}) : Royalties? { + if let view = viewResolver.resolveView(Type()) { + if let v = view as? Royalties { + return v + } + } + return nil + } + + /// Get the path that should be used for receiving royalties + /// This is a path that will eventually be used for a generic switchboard receiver, + /// hence the name but will only be used for royalties for now. + /// + /// @return The PublicPath for the generic FT receiver + /// + access(all) view fun getRoyaltyReceiverPublicPath(): PublicPath { + return /public/GenericFTReceiver + } + + /// View to represent a single field of metadata on an NFT. + /// This is used to get traits of individual key/value pairs along with some + /// contextualized data about the trait + /// + access(all) struct Trait { + // The name of the trait. Like Background, Eyes, Hair, etc. + access(all) let name: String + + // The underlying value of the trait, the rest of the fields of a trait provide context to the value. + access(all) let value: AnyStruct + + // displayType is used to show some context about what this name and value represent + // for instance, you could set value to a unix timestamp, and specify displayType as "Date" to tell + // platforms to consume this trait as a date and not a number + access(all) let displayType: String? + + // Rarity can also be used directly on an attribute. + // + // This is optional because not all attributes need to contribute to the NFT's rarity. + access(all) let rarity: Rarity? + + view init(name: String, value: AnyStruct, displayType: String?, rarity: Rarity?) { + self.name = name + self.value = value + self.displayType = displayType + self.rarity = rarity + } + } + + /// Wrapper view to return all the traits on an NFT. + /// This is used to return traits as individual key/value pairs along with + /// some contextualized data about each trait. + access(all) struct Traits { + access(all) let traits: [Trait] + + view init(_ traits: [Trait]) { + self.traits = traits + } + + /// Adds a single Trait to the Traits view + /// + /// @param Trait: The trait struct to be added + /// + access(all) fun addTrait(_ t: Trait) { + self.traits.append(t) + } + } + + /// Helper to get Traits view in a typesafe way + /// + /// @param viewResolver: A reference to the resolver resource + /// @return A optional Traits struct + /// + access(all) fun getTraits(_ viewResolver: &{ViewResolver.Resolver}) : Traits? { + if let view = viewResolver.resolveView(Type()) { + if let v = view as? Traits { + return v + } + } + return nil + } + + /// Helper function to easily convert a dictionary to traits. For NFT + /// collections that do not need either of the optional values of a Trait, + /// this method should suffice to give them an array of valid traits. + /// + /// @param dict: The dictionary to be converted to Traits + /// @param excludedNames: An optional String array specifying the `dict` + /// keys that are not wanted to become `Traits` + /// @return The generated Traits view + /// + access(all) fun dictToTraits(dict: {String: AnyStruct}, excludedNames: [String]?): Traits { + // Collection owners might not want all the fields in their metadata included. + // They might want to handle some specially, or they might just not want them included at all. + if excludedNames != nil { + for k in excludedNames! { + dict.remove(key: k) + } + } + + let traits: [Trait] = [] + for k in dict.keys { + let trait = Trait(name: k, value: dict[k]!, displayType: nil, rarity: nil) + traits.append(trait) + } + + return Traits(traits) + } + + /// Optional view for collections that issue multiple objects + /// with the same or similar metadata, for example an X of 100 set. This + /// information is useful for wallets and marketplaces. + /// An NFT might be part of multiple editions, which is why the edition + /// information is returned as an arbitrary sized array + /// + access(all) struct Edition { + + /// The name of the edition + /// For example, this could be Set, Play, Series, + /// or any other way a project could classify its editions + access(all) let name: String? + + /// The edition number of the object. + /// For an "24 of 100 (#24/100)" item, the number is 24. + access(all) let number: UInt64 + + /// The max edition number of this type of objects. + /// This field should only be provided for limited-editioned objects. + /// For an "24 of 100 (#24/100)" item, max is 100. + /// For an item with unlimited edition, max should be set to nil. + /// + access(all) let max: UInt64? + + view init(name: String?, number: UInt64, max: UInt64?) { + if max != nil { + assert(number <= max!, message: "The number cannot be greater than the max number!") + } + self.name = name + self.number = number + self.max = max + } + } + + /// Wrapper view for multiple Edition views + /// + access(all) struct Editions { + + /// An arbitrary-sized list for any number of editions + /// that the NFT might be a part of + access(all) let infoList: [Edition] + + view init(_ infoList: [Edition]) { + self.infoList = infoList + } + } + + /// Helper to get Editions in a typesafe way + /// + /// @param viewResolver: A reference to the resolver resource + /// @return An optional Editions struct + /// + access(all) fun getEditions(_ viewResolver: &{ViewResolver.Resolver}) : Editions? { + if let view = viewResolver.resolveView(Type()) { + if let v = view as? Editions { + return v + } + } + return nil + } + + /// View representing a project-defined serial number for a specific NFT + /// Projects have different definitions for what a serial number should be + /// Some may use the NFTs regular ID and some may use a different + /// classification system. The serial number is expected to be unique among + /// other NFTs within that project + /// + access(all) struct Serial { + access(all) let number: UInt64 + + view init(_ number: UInt64) { + self.number = number + } + } + + /// Helper to get Serial in a typesafe way + /// + /// @param viewResolver: A reference to the resolver resource + /// @return An optional Serial struct + /// + access(all) fun getSerial(_ viewResolver: &{ViewResolver.Resolver}) : Serial? { + if let view = viewResolver.resolveView(Type()) { + if let v = view as? Serial { + return v + } + } + return nil + } + + /// View to expose rarity information for a single rarity + /// Note that a rarity needs to have either score or description but it can + /// have both + /// + access(all) struct Rarity { + /// The score of the rarity as a number + access(all) let score: UFix64? + + /// The maximum value of score + access(all) let max: UFix64? + + /// The description of the rarity as a string. + /// + /// This could be Legendary, Epic, Rare, Uncommon, Common or any other string value + access(all) let description: String? + + view init(score: UFix64?, max: UFix64?, description: String?) { + if score == nil && description == nil { + panic("A Rarity needs to set score, description or both") + } + + self.score = score + self.max = max + self.description = description + } + } + + /// Helper to get Rarity view in a typesafe way + /// + /// @param viewResolver: A reference to the resolver resource + /// @return A optional Rarity struct + /// + access(all) fun getRarity(_ viewResolver: &{ViewResolver.Resolver}) : Rarity? { + if let view = viewResolver.resolveView(Type()) { + if let v = view as? Rarity { + return v + } + } + return nil + } + + /// NFTView wraps all Core views along `id` and `uuid` fields, and is used + /// to give a complete picture of an NFT. Most NFTs should implement this + /// view. + /// + access(all) struct NFTView { + access(all) let id: UInt64 + access(all) let uuid: UInt64 + access(all) let display: MetadataViews.Display? + access(all) let externalURL: MetadataViews.ExternalURL? + access(all) let collectionData: NFTCollectionData? + access(all) let collectionDisplay: NFTCollectionDisplay? + access(all) let royalties: Royalties? + access(all) let traits: Traits? + + view init( + id : UInt64, + uuid : UInt64, + display : MetadataViews.Display?, + externalURL : MetadataViews.ExternalURL?, + collectionData : NFTCollectionData?, + collectionDisplay : NFTCollectionDisplay?, + royalties : Royalties?, + traits: Traits? + ) { + self.id = id + self.uuid = uuid + self.display = display + self.externalURL = externalURL + self.collectionData = collectionData + self.collectionDisplay = collectionDisplay + self.royalties = royalties + self.traits = traits + } + } + + /// Helper to get an NFT view + /// + /// @param id: The NFT id + /// @param viewResolver: A reference to the resolver resource + /// @return A NFTView struct + /// + access(all) fun getNFTView(id: UInt64, viewResolver: &{ViewResolver.Resolver}) : NFTView { + let nftView = viewResolver.resolveView(Type()) + if nftView != nil { + return nftView! as! NFTView + } + + return NFTView( + id : id, + uuid: viewResolver.uuid, + display: MetadataViews.getDisplay(viewResolver), + externalURL : MetadataViews.getExternalURL(viewResolver), + collectionData : self.getNFTCollectionData(viewResolver), + collectionDisplay : self.getNFTCollectionDisplay(viewResolver), + royalties : self.getRoyalties(viewResolver), + traits : self.getTraits(viewResolver) + ) + } + + /// View to expose the information needed store and retrieve an NFT. + /// This can be used by applications to setup a NFT collection with proper + /// storage and public capabilities. + /// + access(all) struct NFTCollectionData { + /// Path in storage where this NFT is recommended to be stored. + access(all) let storagePath: StoragePath + + /// Public path which must be linked to expose public capabilities of this NFT + /// including standard NFT interfaces and metadataviews interfaces + access(all) let publicPath: PublicPath + + /// Private path which should be linked to expose the provider + /// capability to withdraw NFTs from the collection holding NFTs + access(all) let providerPath: PrivatePath + + /// Public collection type that is expected to provide sufficient read-only access to standard + /// functions (deposit + getIDs + borrowNFT). For new + /// collections, this may be set to be equal to the type specified in `publicLinkedType`. + access(all) let publicCollection: Type + + /// Type that should be linked at the aforementioned public path. This is normally a + /// restricted type with many interfaces. Notably the + /// `NFT.Receiver`, and `ViewResolver.ResolverCollection` interfaces are required. + access(all) let publicLinkedType: Type + + /// Type that should be linked at the aforementioned private path. This is normally + /// a restricted type with at a minimum the `NFT.Provider` interface + access(all) let providerLinkedType: Type + + /// Function that allows creation of an empty NFT collection that is intended to store + /// this NFT. + access(all) let createEmptyCollection: fun(): @{NonFungibleToken.Collection} + + view init( + storagePath: StoragePath, + publicPath: PublicPath, + providerPath: PrivatePath, + publicCollection: Type, + publicLinkedType: Type, + providerLinkedType: Type, + createEmptyCollectionFunction: fun(): @{NonFungibleToken.Collection} + ) { + pre { + publicLinkedType.isSubtype(of: Type<&{NonFungibleToken.Receiver, ViewResolver.ResolverCollection}>()): "Public type must include NonFungibleToken.Receiver and ViewResolver.ResolverCollection interfaces." + providerLinkedType.isSubtype(of: Type<&{NonFungibleToken.Provider, ViewResolver.ResolverCollection}>()): "Provider type must include NonFungibleToken.Provider and ViewResolver.ResolverCollection interface." + } + self.storagePath=storagePath + self.publicPath=publicPath + self.providerPath = providerPath + self.publicCollection=publicCollection + self.publicLinkedType=publicLinkedType + self.providerLinkedType = providerLinkedType + self.createEmptyCollection=createEmptyCollectionFunction + } + } + + /// Helper to get NFTCollectionData in a way that will return an typed Optional + /// + /// @param viewResolver: A reference to the resolver resource + /// @return A optional NFTCollectionData struct + /// + access(all) fun getNFTCollectionData(_ viewResolver: &{ViewResolver.Resolver}) : NFTCollectionData? { + if let view = viewResolver.resolveView(Type()) { + if let v = view as? NFTCollectionData { + return v + } + } + return nil + } + + /// View to expose the information needed to showcase this NFT's + /// collection. This can be used by applications to give an overview and + /// graphics of the NFT collection this NFT belongs to. + /// + access(all) struct NFTCollectionDisplay { + // Name that should be used when displaying this NFT collection. + access(all) let name: String + + // Description that should be used to give an overview of this collection. + access(all) let description: String + + // External link to a URL to view more information about this collection. + access(all) let externalURL: MetadataViews.ExternalURL + + // Square-sized image to represent this collection. + access(all) let squareImage: MetadataViews.Media + + // Banner-sized image for this collection, recommended to have a size near 1200x630. + access(all) let bannerImage: MetadataViews.Media + + // Social links to reach this collection's social homepages. + // Possible keys may be "instagram", "twitter", "discord", etc. + access(all) let socials: {String: MetadataViews.ExternalURL} + + view init( + name: String, + description: String, + externalURL: MetadataViews.ExternalURL, + squareImage: MetadataViews.Media, + bannerImage: MetadataViews.Media, + socials: {String: MetadataViews.ExternalURL} + ) { + self.name = name + self.description = description + self.externalURL = externalURL + self.squareImage = squareImage + self.bannerImage = bannerImage + self.socials = socials + } + } + + /// Helper to get NFTCollectionDisplay in a way that will return a typed + /// Optional + /// + /// @param viewResolver: A reference to the resolver resource + /// @return A optional NFTCollection struct + /// + access(all) fun getNFTCollectionDisplay(_ viewResolver: &{ViewResolver.Resolver}) : NFTCollectionDisplay? { + if let view = viewResolver.resolveView(Type()) { + if let v = view as? NFTCollectionDisplay { + return v + } + } + return nil + } +} diff --git a/cadence/contracts/utility/NonFungibleToken.cdc b/cadence/contracts/utility/NonFungibleToken.cdc new file mode 100644 index 0000000..3c92e0f --- /dev/null +++ b/cadence/contracts/utility/NonFungibleToken.cdc @@ -0,0 +1,191 @@ +/** + +## The Flow Non-Fungible Token standard + +## `NonFungibleToken` contract interface + +The interface that all Non-Fungible Token contracts should conform to. +If a user wants to deploy a new NFT contract, their contract would need +to implement the NonFungibleToken interface. + +Their contract must follow all the rules and naming +that the interface specifies. + +## `NFT` resource + +The core resource type that represents an NFT in the smart contract. + +## `Collection` Resource + +The resource that stores a user's NFT collection. +It includes a few functions to allow the owner to easily +move tokens in and out of the collection. + +## `Provider` and `Receiver` resource interfaces + +These interfaces declare functions with some pre and post conditions +that require the Collection to follow certain naming and behavior standards. + +They are separate because it gives the user the ability to share a reference +to their Collection that only exposes the fields and functions in one or more +of the interfaces. It also gives users the ability to make custom resources +that implement these interfaces to do various things with the tokens. + +By using resources and interfaces, users of NFT smart contracts can send +and receive tokens peer-to-peer, without having to interact with a central ledger +smart contract. + +To send an NFT to another user, a user would simply withdraw the NFT +from their Collection, then call the deposit function on another user's +Collection to complete the transfer. + +*/ + +import "ViewResolver" + +/// The main NFT contract interface. Other NFT contracts will +/// import and implement this interface +/// +access(all) contract NonFungibleToken { + + /// An entitlement for allowing the withdrawal of tokens from a Vault + access(all) entitlement Withdrawable + + /// An entitlement for allowing updates and update events for an NFT + access(all) entitlement Updatable + + /// Event that contracts should emit when the metadata of an NFT is updated + /// It can only be emitted by calling the `emitNFTUpdated` function + /// with an `Updatable` entitled reference to the NFT that was updated + /// The entitlement prevents spammers from calling this from other users' collections + /// because only code within a collection or that has special entitled access + /// to the collections methods will be able to get the entitled reference + /// + /// The event makes it so that third-party indexers can monitor the events + /// and query the updated metadata from the owners' collections. + /// + access(all) event Updated(id: UInt64, uuid: UInt64, owner: Address?, type: String) + access(all) view fun emitNFTUpdated(_ nftRef: auth(Updatable) &{NonFungibleToken.NFT}) + { + emit Updated(id: nftRef.getID(), uuid: nftRef.uuid, owner: nftRef.owner?.address, type: nftRef.getType().identifier) + } + + + /// Event that is emitted when a token is withdrawn, + /// indicating the owner of the collection that it was withdrawn from. + /// + /// If the collection is not in an account's storage, `from` will be `nil`. + /// + access(all) event Withdraw(id: UInt64, uuid: UInt64, from: Address?, type: String) + + /// Event that emitted when a token is deposited to a collection. + /// + /// It indicates the owner of the collection that it was deposited to. + /// + access(all) event Deposit(id: UInt64, uuid: UInt64, to: Address?, type: String) + + /// Interface that the NFTs must conform to + /// + access(all) resource interface NFT: ViewResolver.Resolver { + /// The unique ID that each NFT has + access(all) view fun getID(): UInt64 + + // access(all) event ResourceDestroyed(uuid: UInt64 = self.uuid, type: self.getType().identifier) + + /// Get a reference to an NFT that this NFT owns + /// Both arguments are optional to allow the NFT to choose + /// how it returns sub NFTs depending on what arguments are provided + /// For example, if `type` has a value, but `id` doesn't, the NFT + /// can choose which NFT of that type to return if there is a "default" + /// If both are `nil`, then NFTs that only store a single NFT can just return + /// that. This helps callers who aren't sure what they are looking for + /// + /// @param type: The Type of the desired NFT + /// @param id: The id of the NFT to borrow + /// + /// @return A structure representing the requested view. + access(all) fun getSubNFT(type: Type, id: UInt64) : &{NonFungibleToken.NFT}? { + return nil + } + } + + /// Interface to mediate withdraws from the Collection + /// + access(all) resource interface Provider { + + // We emit withdraw events from the provider interface because conficting withdraw + // events aren't as confusing to event listeners as conflicting deposit events + + /// withdraw removes an NFT from the collection and moves it to the caller + /// It does not specify whether the ID is UUID or not + access(Withdrawable) fun withdraw(withdrawID: UInt64): @{NFT} { + post { + result.getID() == withdrawID: "The ID of the withdrawn token must be the same as the requested ID" + emit Withdraw(id: result.getID(), uuid: result.uuid, from: self.owner?.address, type: result.getType().identifier) + } + } + } + + /// Interface to mediate deposits to the Collection + /// + access(all) resource interface Receiver { + + /// deposit takes an NFT as an argument and adds it to the Collection + /// + access(all) fun deposit(token: @{NFT}) + + /// getSupportedNFTTypes returns a list of NFT types that this receiver accepts + access(all) view fun getSupportedNFTTypes(): {Type: Bool} + + /// Returns whether or not the given type is accepted by the collection + /// A collection that can accept any type should just return true by default + access(all) view fun isSupportedNFTType(type: Type): Bool + } + + /// Requirement for the concrete resource type + /// to be declared in the implementing contract + /// + access(all) resource interface Collection: Provider, Receiver, ViewResolver.ResolverCollection { + + /// Return the default storage path for the collection + access(all) view fun getDefaultStoragePath(): StoragePath? + + /// Return the default public path for the collection + access(all) view fun getDefaultPublicPath(): PublicPath? + + /// Normally we would require that the collection specify + /// a specific dictionary to store the NFTs, but this isn't necessary any more + /// as long as all the other functions are there + + /// createEmptyCollection creates an empty Collection + /// and returns it to the caller so that they can own NFTs + access(all) fun createEmptyCollection(): @{Collection} { + post { + result.getLength() == 0: "The created collection must be empty!" + } + } + + /// deposit takes a NFT and adds it to the collections dictionary + /// and adds the ID to the id array + access(all) fun deposit(token: @{NonFungibleToken.NFT}) { + pre { + // We emit the deposit event in the `Collection` interface + // because the `Collection` interface is almost always the final destination + // of tokens and deposit emissions from custom receivers could be confusing + // and hard to reconcile to event listeners + emit Deposit(id: token.getID(), uuid: token.uuid, to: self.owner?.address, type: token.getType().identifier) + } + } + + /// Gets the amount of NFTs stored in the collection + access(all) view fun getLength(): Int + + access(all) view fun borrowNFT(_ id: UInt64): &{NonFungibleToken.NFT}? { + post { + (result == nil) || (result?.getID() == id): + "Cannot borrow NFT reference: The ID of the returned reference does not match the ID that was specified" + } + return nil + } + } +} diff --git a/cadence/contracts/utility/ViewResolver.cdc b/cadence/contracts/utility/ViewResolver.cdc new file mode 100644 index 0000000..6dd962f --- /dev/null +++ b/cadence/contracts/utility/ViewResolver.cdc @@ -0,0 +1,55 @@ +// Taken from the NFT Metadata standard, this contract exposes an interface to let +// anyone borrow a contract and resolve views on it. +// +// This will allow you to obtain information about a contract without necessarily knowing anything about it. +// All you need is its address and name and you're good to go! +access(all) contract interface ViewResolver { + + /// Function that returns all the Metadata Views implemented by the resolving contract + /// + /// @return An array of Types defining the implemented views. This value will be used by + /// developers to know which parameter to pass to the resolveView() method. + /// + access(all) view fun getViews(): [Type] { + return [] + } + + /// Function that resolves a metadata view for this token. + /// + /// @param view: The Type of the desired view. + /// @return A structure representing the requested view. + /// + access(all) fun resolveView(_ view: Type): AnyStruct? { + return nil + } + + /// Provides access to a set of metadata views. A struct or + /// resource (e.g. an NFT) can implement this interface to provide access to + /// the views that it supports. + /// + access(all) resource interface Resolver { + + /// Same as getViews above, but on a specific NFT instead of a contract + access(all) view fun getViews(): [Type] { + return [] + } + + /// Same as resolveView above, but on a specific NFT instead of a contract + access(all) fun resolveView(_ view: Type): AnyStruct? { + return nil + } + } + + /// A group of view resolvers indexed by ID. + /// + access(all) resource interface ResolverCollection { + access(all) view fun borrowViewResolver(id: UInt64): &{Resolver}? { + return nil + } + + access(all) view fun getIDs(): [UInt64] { + return [] + } + } +} + \ No newline at end of file diff --git a/cadence/scripts/getAccount.cdc b/cadence/scripts/getAccount.cdc index 6cae595..aa2c3f1 100644 --- a/cadence/scripts/getAccount.cdc +++ b/cadence/scripts/getAccount.cdc @@ -1,3 +1,3 @@ -pub fun main(address: Address): PublicAccount { +access(all) fun main(address: Address): &Account { return getAccount(address) } diff --git a/cadence/scripts/getAccounts.cdc b/cadence/scripts/getAccounts.cdc index 5e5e1af..fdc3378 100644 --- a/cadence/scripts/getAccounts.cdc +++ b/cadence/scripts/getAccounts.cdc @@ -1,5 +1,5 @@ -import FCL from 0xSERVICE +import "FCL" -pub fun main(): [FCL.Account] { +access(all) fun main(): &[FCL.FCLAccount] { return FCL.accounts().values } diff --git a/cadence/transactions/fundFLOW.cdc b/cadence/transactions/fundFLOW.cdc index caaae92..c94bd71 100644 --- a/cadence/transactions/fundFLOW.cdc +++ b/cadence/transactions/fundFLOW.cdc @@ -1,18 +1,20 @@ -import FlowToken from 0xFLOWTOKENADDRESS -import FungibleToken from 0xFUNGIBLETOKENADDRESS +import "FlowToken" +import "FungibleToken" transaction(address: Address, amount: UFix64) { let tokenAdmin: &FlowToken.Administrator let tokenReceiver: &{FungibleToken.Receiver} - prepare(signer: AuthAccount) { + prepare(signer: auth(Storage, Capabilities) &Account) { self.tokenAdmin = signer + .storage .borrow<&FlowToken.Administrator>(from: /storage/flowTokenAdmin) ?? panic("Signer is not the token admin") self.tokenReceiver = getAccount(address) - .getCapability(/public/flowTokenReceiver)! - .borrow<&{FungibleToken.Receiver}>() + .capabilities + .get<&{FungibleToken.Receiver}>(/public/flowTokenReceiver)! + .borrow() ?? panic("Unable to borrow receiver reference") } diff --git a/cadence/transactions/init.cdc b/cadence/transactions/init.cdc index 33bce21..d9a769e 100644 --- a/cadence/transactions/init.cdc +++ b/cadence/transactions/init.cdc @@ -1,5 +1,5 @@ transaction(code: String, key: String, initAccountsLabels: [String]) { - prepare(acct: AuthAccount) { + prepare(acct: auth(Contracts) &Account) { acct.contracts.add(name: "FCL", code: code.utf8, key: key, initAccountsLabels: initAccountsLabels) } } diff --git a/cadence/transactions/newAccount.cdc b/cadence/transactions/newAccount.cdc index 21a061f..064932e 100644 --- a/cadence/transactions/newAccount.cdc +++ b/cadence/transactions/newAccount.cdc @@ -1,6 +1,6 @@ -import FCL from 0xSERVICE -import FlowToken from 0xFLOWTOKENADDRESS -import FungibleToken from 0xFUNGIBLETOKENADDRESS +import "FCL" +import "FlowToken" +import "FungibleToken" /// This transaction creates a new account and funds it with /// an initial balance of FLOW tokens. @@ -10,16 +10,18 @@ transaction(label: String, scopes: [String], initialBalance: UFix64) { let tokenAdmin: &FlowToken.Administrator let tokenReceiver: &{FungibleToken.Receiver} - prepare(signer: AuthAccount) { + prepare(signer: auth(Storage) &Account) { let account = FCL.new(label: label, scopes: scopes, address: nil) self.tokenAdmin = signer + .storage .borrow<&FlowToken.Administrator>(from: /storage/flowTokenAdmin) ?? panic("Signer is not the token admin") self.tokenReceiver = account - .getCapability(/public/flowTokenReceiver)! - .borrow<&{FungibleToken.Receiver}>() + .capabilities + .get<&{FungibleToken.Receiver}>(/public/flowTokenReceiver)! + .borrow() ?? panic("Unable to borrow receiver reference") } diff --git a/cadence/transactions/updateAccount.cdc b/cadence/transactions/updateAccount.cdc index 641b30c..27bb549 100644 --- a/cadence/transactions/updateAccount.cdc +++ b/cadence/transactions/updateAccount.cdc @@ -1,4 +1,4 @@ -import FCL from 0xSERVICE +import "FCL" transaction(address: Address, label: String, scopes: [String]) { prepare() { diff --git a/contexts/ConfigContext.tsx b/contexts/ConfigContext.tsx index c4cd483..09034ce 100644 --- a/contexts/ConfigContext.tsx +++ b/contexts/ConfigContext.tsx @@ -81,22 +81,9 @@ export function ConfigContextProvider({children}: {children: React.ReactNode}) { async function fetchConfig() { const config = await getConfig() - const { - flowAccessNode, - flowAccountAddress, - contractFungibleToken, - contractFlowToken, - contractFUSD, - } = config - - fclConfig( - flowAccessNode, - flowAccountAddress, - contractFungibleToken, - contractFlowToken, - contractFUSD - ) + const {flowAccessNode} = config + fclConfig(flowAccessNode) setConfig(config) } diff --git a/flow.json b/flow.json index ca9f525..4b833a4 100755 --- a/flow.json +++ b/flow.json @@ -7,20 +7,49 @@ }, "contracts": { "FCL": { - "source": "./cadence/contracts/FCL.cdc" + "source": "./cadence/contracts/FCL.cdc", + "aliases": { + "emulator": "0xf8d6e0586b0a20c7" + } + }, + "FCLCrypto": { + "source": "./cadence/contracts/FCLCrypto.cdc", + "aliases": {} + }, + "FlowToken": { + "source": "./cadence/contracts/utility/FlowToken.cdc", + "aliases": { + "emulator": "0x0ae53cb6e3f42a79" + } }, "FungibleToken": { - "source": "./cadence/contracts/FungibleToken.cdc", + "source": "./cadence/contracts/utility/FungibleToken.cdc", "aliases": { "emulator": "0xee82856bf20e2aa6" } }, - "FCLCrypto": { - "source": "./cadence/contracts/FCLCrypto.cdc", + "FungibleTokenMetadataViews": { + "source": "./cadence/contracts/utility/FungibleTokenMetadataViews.cdc", + "aliases": { + "emulator": "0xee82856bf20e2aa6" + } + }, + "MetadataViews": { + "source": "./cadence/contracts/utility/MetadataViews.cdc", + "aliases": { + "emulator": "0xf8d6e0586b0a20c7" + } + }, + "NonFungibleToken": { + "source": "./cadence/contracts/utility/NonFungibleToken.cdc", + "aliases": { + "emulator": "0xf8d6e0586b0a20c7" + } + }, + "ViewResolver": { + "source": "./cadence/contracts/utility/ViewResolver.cdc", "aliases": { - "emulator": "0xf8d6e0586b0a20c7", - "testnet": "0x74daa6f9c7ef24b1", - "mainnet": "0xb4b82a1c9d21d284" + "emulator": "0xf8d6e0586b0a20c7" } } }, diff --git a/src/accounts.ts b/src/accounts.ts index f9257a0..b76623c 100644 --- a/src/accounts.ts +++ b/src/accounts.ts @@ -47,6 +47,8 @@ export async function getAccount(address: string) { export async function getAccounts(config: {flowAccountAddress: string}) { const {flowAccountAddress} = config + fcl.config().all().then(console.log) + const accounts = await fcl .send([fcl.script(getAccountsScript)]) .then(fcl.decode) diff --git a/src/fclConfig.ts b/src/fclConfig.ts index 835f57e..7def0aa 100644 --- a/src/fclConfig.ts +++ b/src/fclConfig.ts @@ -1,16 +1,7 @@ import {config} from "@onflow/fcl" +import flowJSON from "../flow.json" -export default function fclConfig( - flowAccessNode: string, - flowAccountAddress: string, - contractFungibleToken: string, - contractFlowToken: string, - contractFUSD: string -) { - config() - .put("accessNode.api", flowAccessNode) - .put("0xSERVICE", flowAccountAddress) - .put("0xFUNGIBLETOKENADDRESS", contractFungibleToken) - .put("0xFLOWTOKENADDRESS", contractFlowToken) - .put("0xFUSDADDRESS", contractFUSD) +export default function fclConfig(flowAccessNode: string) { + config().put("accessNode.api", flowAccessNode).put("flow.network", "local") + config().load({flowJSON}) } diff --git a/src/harness/cmds/q1.js b/src/harness/cmds/q1.js index a5e9662..71e951d 100644 --- a/src/harness/cmds/q1.js +++ b/src/harness/cmds/q1.js @@ -6,7 +6,7 @@ export const CMD = async () => { // prettier-ignore return query({ cadence: ` - pub fun main(): Int { + access(all) fun main(): Int { return 7 } `, diff --git a/src/harness/cmds/q2.js b/src/harness/cmds/q2.js index f715a32..57a92b2 100644 --- a/src/harness/cmds/q2.js +++ b/src/harness/cmds/q2.js @@ -6,7 +6,7 @@ export const CMD = async () => { // prettier-ignore return query({ cadence: ` - pub fun main(a: Int, b: Int): Int { + access(all) fun main(a: Int, b: Int): Int { return a + b } `,