diff --git a/ast/inspector_test.go b/ast/inspector_test.go index 17979ef95f..5b1ee8c0c8 100644 --- a/ast/inspector_test.go +++ b/ast/inspector_test.go @@ -25,7 +25,6 @@ import ( "github.com/onflow/cadence/ast" "github.com/onflow/cadence/parser" - "github.com/onflow/cadence/tests/examples" ) // TestInspector_Elements compares Inspector against Inspect. @@ -33,9 +32,30 @@ func TestInspector_Elements(t *testing.T) { t.Parallel() + const code = ` + access(all) contract interface FungibleToken { + + access(all) resource interface Provider { + access(all) fun withdraw(amount: Int): @Vault + } + + access(all) resource interface Receiver { + access(all) fun deposit(vault: @Vault) + } + + access(all) resource interface Vault: Provider, Receiver { + access(all) balance: Int + } + + access(all) fun absorb(vault: @Vault) + + access(all) fun sprout(balance: Int): @Vault + } + ` + program, err := parser.ParseProgram( nil, - []byte(examples.FungibleTokenContractInterface), + []byte(code), parser.Config{}, ) require.NoError(t, err) diff --git a/tests/checker/interface_test.go b/tests/checker/interface_test.go index d0943d0fb0..fd453b789a 100644 --- a/tests/checker/interface_test.go +++ b/tests/checker/interface_test.go @@ -27,11 +27,7 @@ import ( "github.com/onflow/cadence/common" "github.com/onflow/cadence/errors" - "github.com/onflow/cadence/parser" "github.com/onflow/cadence/sema" - "github.com/onflow/cadence/stdlib" - "github.com/onflow/cadence/tests/examples" - . "github.com/onflow/cadence/tests/utils" ) func constructorArguments(compositeKind common.CompositeKind) string { @@ -1630,128 +1626,6 @@ func TestCheckInvalidTypeRequirementDeclaration(t *testing.T) { }) } -// TODO: re-enable this test with the v2 fungible token contract -/* func TestCheckContractInterfaceFungibleToken(t *testing.T) { - - t.Parallel() - - const code = examples.FungibleTokenContractInterface - - _, err := ParseAndCheck(t, code) - require.NoError(t, err) -} */ - -// TODO: re-enable this test with the v2 fungible token contract -/* func TestCheckContractInterfaceFungibleTokenConformance(t *testing.T) { - - t.Parallel() - - code := examples.FungibleTokenContractInterface + "\n" + examples.ExampleFungibleTokenContract - - _, err := ParseAndCheckWithPanic(t, code) - require.NoError(t, err) -} */ - -func BenchmarkContractInterfaceFungibleToken(b *testing.B) { - - const code = examples.FungibleTokenContractInterface - - program, err := parser.ParseProgram(nil, []byte(code), parser.Config{}) - if err != nil { - b.Fatal(err) - } - - b.ReportAllocs() - b.ResetTimer() - - for i := 0; i < b.N; i++ { - checker, err := sema.NewChecker( - program, - TestLocation, - nil, - &sema.Config{ - AccessCheckMode: sema.AccessCheckModeNotSpecifiedUnrestricted, - }, - ) - if err != nil { - b.Fatal(err) - } - err = checker.Check() - if err != nil { - b.Fatal(err) - } - } -} - -func BenchmarkCheckContractInterfaceFungibleTokenConformance(b *testing.B) { - - code := examples.FungibleTokenContractInterface + "\n" + examples.ExampleFungibleTokenContract - - program, err := parser.ParseProgram(nil, []byte(code), parser.Config{}) - if err != nil { - b.Fatal(err) - } - - baseValueActivation := sema.NewVariableActivation(sema.BaseValueActivation) - baseValueActivation.DeclareValue(stdlib.PanicFunction) - - config := &sema.Config{ - AccessCheckMode: sema.AccessCheckModeNotSpecifiedUnrestricted, - BaseValueActivationHandler: func(_ common.Location) *sema.VariableActivation { - return baseValueActivation - }, - } - - b.ReportAllocs() - b.ResetTimer() - - for i := 0; i < b.N; i++ { - checker, err := sema.NewChecker( - program, - TestLocation, - nil, - config, - ) - if err != nil { - b.Fatal(err) - } - err = checker.Check() - if err != nil { - b.Fatal(err) - } - } -} - -// TODO: re-enable this test with the v2 fungible token contract -/* func TestCheckContractInterfaceFungibleTokenUse(t *testing.T) { - - t.Parallel() - - code := examples.FungibleTokenContractInterface + "\n" + - examples.ExampleFungibleTokenContract + "\n" + ` - - fun test(): Int { - let publisher <- ExampleToken.sprout(balance: 100) - let receiver <- ExampleToken.sprout(balance: 0) - - let withdrawn <- publisher.withdraw(amount: 60) - receiver.deposit(vault: <-withdrawn) - - let publisherBalance = publisher.balance - let receiverBalance = receiver.balance - - destroy publisher - destroy receiver - - return receiverBalance - } - ` - - _, err := ParseAndCheckWithPanic(t, code) - - require.NoError(t, err) -} */ - // TestCheckInvalidInterfaceUseAsTypeSuggestion tests that an interface // can not be used as a type, and the suggestion to fix it is correct func TestCheckInvalidInterfaceUseAsTypeSuggestion(t *testing.T) { diff --git a/tests/checker/nft_test.go b/tests/checker/nft_test.go deleted file mode 100644 index 41374e7903..0000000000 --- a/tests/checker/nft_test.go +++ /dev/null @@ -1,992 +0,0 @@ -/* - * Cadence - The resource-oriented smart contract programming language - * - * Copyright Flow Foundation - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -package checker - -// TODO: re-enable this test with the v2 fungible token contract -/* -const realNonFungibleTokenContractInterface = ` - -// The main NFT contract interface. Other NFT contracts will -// import and implement this interface -// -access(all) contract interface NonFungibleToken { - - // The total number of tokens of this type in existence - access(all) var totalSupply: UInt64 - - // Event that emitted when the NFT contract is initialized - // - access(all) event ContractInitialized() - - // 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, from: Address?) - - // 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, to: Address?) - - // Interface that the NFTs have to conform to - // - access(all) resource interface INFT { - // The unique ID that each NFT has - access(all) let id: UInt64 - } - - // Requirement that all conforming NFT smart contracts have - // to define a resource called NFT that conforms to INFT - access(all) resource NFT: INFT { - access(all) let id: UInt64 - } - - // entitles references to withdraw: - // TODO: /~https://github.com/onflow/cadence/issues/2503 - entitlement Withdrawable - - // Interface to mediate withdraws from the Collection - // - access(all) resource interface Provider { - // withdraw removes an NFT from the collection and moves it to the caller - access(Withdrawable) fun withdraw(withdrawID: UInt64): @NFT { - post { - result.id == withdrawID: "The ID of the withdrawn token must be the same as the requested ID" - } - } - } - - // 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) - } - - // Interface that an account would commonly - // publish for their collection - access(all) resource interface CollectionPublic { - access(all) fun deposit(token: @NFT) - access(all) fun getIDs(): [UInt64] - access(all) fun borrowNFT(id: UInt64): &NFT - } - - // Requirement for the the concrete resource type - // to be declared in the implementing contract - // - access(all) resource Collection: Provider, Receiver, CollectionPublic { - - // Dictionary to hold the NFTs in the Collection - access(all) var ownedNFTs: @{UInt64: NFT} - - // withdraw removes an NFT from the collection and moves it to the caller - access(Withdrawable) fun withdraw(withdrawID: UInt64): @NFT - - // deposit takes a NFT and adds it to the collections dictionary - // and adds the ID to the id array - access(all) fun deposit(token: @NFT) - - // getIDs returns an array of the IDs that are in the collection - access(all) fun getIDs(): [UInt64] - - // Returns a borrowed reference to an NFT in the collection - // so that the caller can read data and call methods from it - access(all) fun borrowNFT(id: UInt64): &NFT { - pre { - self.ownedNFTs[id] != nil: "NFT does not exist in the collection!" - } - } - } - - // createEmptyCollection creates an empty Collection - // and returns it to the caller so that they can own NFTs - access(all) fun createEmptyCollection(): @Collection { - post { - result.ownedNFTs.length == 0: "The created collection must be empty!" - } - } -} -` -const topShotContract = ` - -import NonFungibleToken from 0x1 - -access(all) contract TopShot: NonFungibleToken { - - // ----------------------------------------------------------------------- - // TopShot contract Events - // ----------------------------------------------------------------------- - - // Emitted when the TopShot contract is created - access(all) event ContractInitialized() - - // Emitted when a new Play struct is created - access(all) event PlayCreated(id: UInt32, metadata: {String:String}) - // Emitted when a new series has been triggered by an admin - access(all) event NewSeriesStarted(newCurrentSeries: UInt32) - - // Events for Set-Related actions - // - // Emitted when a new Set is created - access(all) event SetCreated(setID: UInt32, series: UInt32) - // Emitted when a new Play is added to a Set - access(all) event PlayAddedToSet(setID: UInt32, playID: UInt32) - // Emitted when a Play is retired from a Set and cannot be used to mint - access(all) event PlayRetiredFromSet(setID: UInt32, playID: UInt32, numMoments: UInt32) - // Emitted when a Set is locked, meaning Plays cannot be added - access(all) event SetLocked(setID: UInt32) - // Emitted when a Moment is minted from a Set - access(all) event MomentMinted(momentID: UInt64, playID: UInt32, setID: UInt32, serialNumber: UInt32) - - // Events for Collection-related actions - // - // Emitted when a moment is withdrawn from a Collection - access(all) event Withdraw(id: UInt64, from: Address?) - // Emitted when a moment is deposited into a Collection - access(all) event Deposit(id: UInt64, to: Address?) - - // Emitted when a Moment is destroyed - access(all) event MomentDestroyed(id: UInt64) - - // ----------------------------------------------------------------------- - // TopShot contract-level fields. - // These contain actual values that are stored in the smart contract. - // ----------------------------------------------------------------------- - - // Series that this Set belongs to. - // Series is a concept that indicates a group of Sets through time. - // Many Sets can exist at a time, but only one series. - access(all) var currentSeries: UInt32 - - // Variable size dictionary of Play structs - access(self) var playDatas: {UInt32: Play} - - // Variable size dictionary of SetData structs - access(self) var setDatas: {UInt32: SetData} - - // Variable size dictionary of Set resources - access(self) var sets: @{UInt32: Set} - - // The ID that is used to create Plays. - // Every time a Play is created, playID is assigne - // to the new Play's ID and then is incremented by 1. - access(all) var nextPlayID: UInt32 - - // The ID that is used to create Sets. Every time a Set is created - // setID is assigned to the new set's ID and then is incremented by 1. - access(all) var nextSetID: UInt32 - - // The total number of Top shot Moment NFTs that have been created - // Because NFTs can be destroyed, it doesn't necessarily mean that this - // reflects the total number of NFTs in existence, just the number that - // have been minted to date. Also used as global moment IDs for minting. - access(all) var totalSupply: UInt64 - - // ----------------------------------------------------------------------- - // TopShot contract-level Composite Type definitions - // ----------------------------------------------------------------------- - // These are just *definitions* for Types that this contract - // and other accounts can use. These definitions do not contain - // actual stored values, but an instance (or object) of one of these Types - // can be created by this contract that contains stored values. - // ----------------------------------------------------------------------- - - // Play is a Struct that holds metadata associated - // with a specific NBA play, like the legendary moment when - // Ray Allen hit the 3 to tie the Heat and Spurs in the 2013 finals game 6 - // or when Lance Stephenson blew in the ear of Lebron James. - // - // Moment NFTs will all reference a single play as the owner of - // its metadata. The plays are publicly accessible, so anyone can - // read the metadata associated with a specific play ID - // - access(all) struct Play { - - // The unique ID for the Play - access(all) let playID: UInt32 - - // Stores all the metadata about the play as a string mapping - // This is not the long term way NFT metadata will be stored. It's a temporary - // construct while we figure out a better way to do metadata. - // - access(all) let metadata: {String: String} - - init(metadata: {String: String}) { - pre { - metadata.length != 0: "New Play metadata cannot be empty" - } - self.playID = TopShot.nextPlayID - self.metadata = metadata - - // Increment the ID so that it isn't used again - TopShot.nextPlayID = TopShot.nextPlayID + UInt32(1) - - emit PlayCreated(id: self.playID, metadata: metadata) - } - } - - // A Set is a grouping of Plays that have occurred in the real world - // that make up a related group of collectibles, like sets of baseball - // or Magic cards. A Play can exist in multiple different sets. - // - // SetData is a struct that is stored in a field of the contract. - // Anyone can query the constant information - // about a set by calling various getters located - // at the end of the contract. Only the admin has the ability - // to modify any data in the private Set resource. - // - access(all) struct SetData { - - // Unique ID for the Set - access(all) let setID: UInt32 - - // Name of the Set - // ex. "Times when the Toronto Raptors choked in the playoffs" - access(all) let name: String - - // Series that this Set belongs to. - // Series is a concept that indicates a group of Sets through time. - // Many Sets can exist at a time, but only one series. - access(all) let series: UInt32 - - init(name: String) { - pre { - name.length > 0: "New Set name cannot be empty" - } - self.setID = TopShot.nextSetID - self.name = name - self.series = TopShot.currentSeries - - // Increment the setID so that it isn't used again - TopShot.nextSetID = TopShot.nextSetID + UInt32(1) - - emit SetCreated(setID: self.setID, series: self.series) - } - } - - // Set is a resource type that contains the functions to add and remove - // Plays from a set and mint Moments. - // - // It is stored in a private field in the contract so that - // the admin resource can call its methods. - // - // The admin can add Plays to a Set so that the set can mint Moments - // that reference that playdata. - // The Moments that are minted by a Set will be listed as belonging to - // the Set that minted it, as well as the Play it references. - // - // Admin can also retire Plays from the Set, meaning that the retired - // Play can no longer have Moments minted from it. - // - // If the admin locks the Set, no more Plays can be added to it, but - // Moments can still be minted. - // - // If retireAll() and lock() are called back-to-back, - // the Set is closed off forever and nothing more can be done with it. - access(all) resource Set { - - // Unique ID for the set - access(all) let setID: UInt32 - - // Array of plays that are a part of this set. - // When a play is added to the set, its ID gets appended here. - // The ID does not get removed from this array when a Play is retired. - access(all) var plays: [UInt32] - - // Map of Play IDs that Indicates if a Play in this Set can be minted. - // When a Play is added to a Set, it is mapped to false (not retired). - // When a Play is retired, this is set to true and cannot be changed. - access(all) var retired: {UInt32: Bool} - - // Indicates if the Set is currently locked. - // When a Set is created, it is unlocked - // and Plays are allowed to be added to it. - // When a set is locked, Plays cannot be added. - // A Set can never be changed from locked to unlocked, - // the decision to lock a Set it is final. - // If a Set is locked, Plays cannot be added, but - // Moments can still be minted from Plays - // that exist in the Set. - access(all) var locked: Bool - - // Mapping of Play IDs that indicates the number of Moments - // that have been minted for specific Plays in this Set. - // When a Moment is minted, this value is stored in the Moment to - // show its place in the Set, eg. 13 of 60. - access(all) var numberMintedPerPlay: {UInt32: UInt32} - - init(name: String) { - self.setID = TopShot.nextSetID - self.plays = [] - self.retired = {} - self.locked = false - self.numberMintedPerPlay = {} - - // Create a new SetData for this Set and store it in contract storage - TopShot.setDatas[self.setID] = SetData(name: name) - } - - // addPlay adds a play to the set - // - // Parameters: playID: The ID of the Play that is being added - // - // Pre-Conditions: - // The Play needs to be an existing play - // The Set needs to be not locked - // The Play can't have already been added to the Set - // - access(all) fun addPlay(playID: UInt32) { - pre { - TopShot.playDatas[playID] != nil: "Cannot add the Play to Set: Play doesn't exist." - !self.locked: "Cannot add the play to the Set after the set has been locked." - self.numberMintedPerPlay[playID] == nil: "The play has already beed added to the set." - } - - // Add the Play to the array of Plays - self.plays.append(playID) - - // Open the Play up for minting - self.retired[playID] = false - - // Initialize the Moment count to zero - self.numberMintedPerPlay[playID] = 0 - - emit PlayAddedToSet(setID: self.setID, playID: playID) - } - - // addPlays adds multiple Plays to the Set - // - // Parameters: playIDs: The IDs of the Plays that are being added - // as an array - // - access(all) fun addPlays(playIDs: [UInt32]) { - for play in playIDs { - self.addPlay(playID: play) - } - } - - // retirePlay retires a Play from the Set so that it can't mint new Moments - // - // Parameters: playID: The ID of the Play that is being retired - // - // Pre-Conditions: - // The Play is part of the Set and not retired (available for minting). - // - access(all) fun retirePlay(playID: UInt32) { - pre { - self.retired[playID] != nil: "Cannot retire the Play: Play doesn't exist in this set!" - } - - if !self.retired[playID]! { - self.retired[playID] = true - - emit PlayRetiredFromSet(setID: self.setID, playID: playID, numMoments: self.numberMintedPerPlay[playID]!) - } - } - - // retireAll retires all the plays in the Set - // Afterwards, none of the retired Plays will be able to mint new Moments - // - access(all) fun retireAll() { - for play in self.plays { - self.retirePlay(playID: play) - } - } - - // lock() locks the Set so that no more Plays can be added to it - // - // Pre-Conditions: - // The Set should not be locked - access(all) fun lock() { - if !self.locked { - self.locked = true - emit SetLocked(setID: self.setID) - } - } - - // mintMoment mints a new Moment and returns the newly minted Moment - // - // Parameters: playID: The ID of the Play that the Moment references - // - // Pre-Conditions: - // The Play must exist in the Set and be allowed to mint new Moments - // - // Returns: The NFT that was minted - // - access(all) fun mintMoment(playID: UInt32): @NFT { - pre { - self.retired[playID] != nil: "Cannot mint the moment: This play doesn't exist." - !self.retired[playID]!: "Cannot mint the moment from this play: This play has been retired." - } - - // Gets the number of Moments that have been minted for this Play - // to use as this Moment's serial number - let numInPlay = self.numberMintedPerPlay[playID]! - - // Mint the new moment - let newMoment: @NFT <- create NFT(serialNumber: numInPlay + UInt32(1), - playID: playID, - setID: self.setID) - - // Increment the count of Moments minted for this Play - self.numberMintedPerPlay[playID] = numInPlay + UInt32(1) - - return <-newMoment - } - - // batchMintMoment mints an arbitrary quantity of Moments - // and returns them as a Collection - // - // Parameters: playID: the ID of the Play that the Moments are minted for - // quantity: The quantity of Moments to be minted - // - // Returns: Collection object that contains all the Moments that were minted - // - access(all) fun batchMintMoment(playID: UInt32, quantity: UInt64): @Collection { - let newCollection <- create Collection() - - var i: UInt64 = 0 - while i < quantity { - newCollection.deposit(token: <-self.mintMoment(playID: playID)) - i = i + UInt64(1) - } - - return <-newCollection - } - } - - access(all) struct MomentData { - - // The ID of the Set that the Moment comes from - access(all) let setID: UInt32 - - // The ID of the Play that the Moment references - access(all) let playID: UInt32 - - // The place in the edition that this Moment was minted - // Otherwise know as the serial number - access(all) let serialNumber: UInt32 - - init(setID: UInt32, playID: UInt32, serialNumber: UInt32) { - self.setID = setID - self.playID = playID - self.serialNumber = serialNumber - } - - } - - // The resource that represents the Moment NFTs - // - access(all) resource NFT: NonFungibleToken.INFT { - - // Global unique moment ID - access(all) let id: UInt64 - - // Struct of Moment metadata - access(all) let data: MomentData - - init(serialNumber: UInt32, playID: UInt32, setID: UInt32) { - // Increment the global Moment IDs - TopShot.totalSupply = TopShot.totalSupply + UInt64(1) - - self.id = TopShot.totalSupply - - // Set the metadata struct - self.data = MomentData(setID: setID, playID: playID, serialNumber: serialNumber) - - emit MomentMinted(momentID: self.id, playID: playID, setID: self.data.setID, serialNumber: self.data.serialNumber) - } - } - - // Admin is a special authorization resource that - // allows the owner to perform important functions to modify the - // various aspects of the Plays, Sets, and Moments - // - access(all) resource Admin { - - // createPlay creates a new Play struct - // and stores it in the Plays dictionary in the TopShot smart contract - // - // Parameters: metadata: A dictionary mapping metadata titles to their data - // example: {"Player Name": "Kevin Durant", "Height": "7 feet"} - // (because we all know Kevin Durant is not 6'9") - // - // Returns: the ID of the new Play object - // - access(all) fun createPlay(metadata: {String: String}): UInt32 { - // Create the new Play - var newPlay = Play(metadata: metadata) - let newID = newPlay.playID - - // Store it in the contract storage - TopShot.playDatas[newID] = newPlay - - return newID - } - - // createSet creates a new Set resource and stores it - // in the sets mapping in the TopShot contract - // - // Parameters: name: The name of the Set - // - access(all) fun createSet(name: String) { - // Create the new Set - var newSet <- create Set(name: name) - - // Store it in the sets mapping field - TopShot.sets[newSet.setID] <-! newSet - } - - // borrowSet returns a reference to a set in the TopShot - // contract so that the admin can call methods on it - // - // Parameters: setID: The ID of the Set that you want to - // get a reference to - // - // Returns: A reference to the Set with all of the fields - // and methods exposed - // - access(all) fun borrowSet(setID: UInt32): &Set { - pre { - TopShot.sets[setID] != nil: "Cannot borrow Set: The Set doesn't exist" - } - - // Get a reference to the Set and return it - // use & to indicate the reference to the object and type - return (&TopShot.sets[setID] as &Set?)! - } - - // startNewSeries ends the current series by incrementing - // the series number, meaning that Moments minted after this - // will use the new series number - // - // Returns: The new series number - // - access(all) fun startNewSeries(): UInt32 { - // End the current series and start a new one - // by incrementing the TopShot series number - TopShot.currentSeries = TopShot.currentSeries + UInt32(1) - - emit NewSeriesStarted(newCurrentSeries: TopShot.currentSeries) - - return TopShot.currentSeries - } - - // createNewAdmin creates a new Admin resource - // - access(all) fun createNewAdmin(): @Admin { - return <-create Admin() - } - } - - // This is the interface that users can cast their Moment Collection as - // to allow others to deposit Moments into their Collection. It also allows for reading - // the IDs of Moments in the Collection. - access(all) resource interface MomentCollectionPublic { - access(all) fun deposit(token: @NonFungibleToken.NFT) - access(all) fun batchDeposit(tokens: @NonFungibleToken.Collection) - access(all) fun getIDs(): [UInt64] - access(all) fun borrowNFT(id: UInt64): &NonFungibleToken.NFT - access(all) fun borrowMoment(id: UInt64): &TopShot.NFT? { - // If the result isn't nil, the id of the returned reference - // should be the same as the argument to the function - post { - (result == nil) || (result?.id == id): - "Cannot borrow Moment reference: The ID of the returned reference is incorrect" - } - } - } - - // Collection is a resource that every user who owns NFTs - // will store in their account to manage their NFTS - // - access(all) resource Collection: MomentCollectionPublic, NonFungibleToken.Provider, NonFungibleToken.Receiver, NonFungibleToken.CollectionPublic { - // Dictionary of Moment conforming tokens - // NFT is a resource type with a UInt64 ID field - access(all) var ownedNFTs: @{UInt64: NonFungibleToken.NFT} - - init() { - self.ownedNFTs <- {} - } - - // withdraw removes an Moment from the Collection and moves it to the caller - // - // Parameters: withdrawID: The ID of the NFT - // that is to be removed from the Collection - // - // returns: @NonFungibleToken.NFT the token that was withdrawn - access(NonFungibleToken.Withdrawable) fun withdraw(withdrawID: UInt64): @NonFungibleToken.NFT { - - // Remove the nft from the Collection - let token <- self.ownedNFTs.remove(key: withdrawID) - ?? panic("Cannot withdraw: Moment does not exist in the collection") - - emit Withdraw(id: token.id, from: self.owner?.address) - - // Return the withdrawn token - return <-token - } - - // batchWithdraw withdraws multiple tokens and returns them as a Collection - // - // Parameters: ids: An array of IDs to withdraw - // - // Returns: @NonFungibleToken.Collection: A collection that contains - // the withdrawn moments - // - access(all) fun batchWithdraw(ids: [UInt64]): @NonFungibleToken.Collection { - // Create a new empty Collection - var batchCollection <- create Collection() - - // Iterate through the ids and withdraw them from the Collection - for id in ids { - batchCollection.deposit(token: <-self.withdraw(withdrawID: id)) - } - - // Return the withdrawn tokens - return <-batchCollection - } - - // deposit takes a Moment and adds it to the Collections dictionary - // - // Parameters: token: the NFT to be deposited in the collection - // - access(all) fun deposit(token: @NonFungibleToken.NFT) { - - // Cast the deposited token as a TopShot NFT to make sure - // it is the correct type - let token <- token as! @TopShot.NFT - - // Get the token's ID - let id = token.id - - // Add the new token to the dictionary - let oldToken <- self.ownedNFTs[id] <- token - - // Only emit a deposit event if the Collection - // is in an account's storage - if self.owner?.address != nil { - emit Deposit(id: id, to: self.owner?.address) - } - - // Destroy the empty old token that was "removed" - destroy oldToken - } - - // batchDeposit takes a Collection object as an argument - // and deposits each contained NFT into this Collection - access(all) fun batchDeposit(tokens: @NonFungibleToken.Collection) { - - // Get an array of the IDs to be deposited - let keys = tokens.getIDs() - - // Iterate through the keys in the collection and deposit each one - for key in keys { - self.deposit(token: <-tokens.withdraw(withdrawID: key)) - } - - // Destroy the empty Collection - destroy tokens - } - - // getIDs returns an array of the IDs that are in the Collection - access(all) fun getIDs(): [UInt64] { - return self.ownedNFTs.keys - } - - // borrowNFT Returns a borrowed reference to a Moment in the Collection - // so that the caller can read its ID - // - // Parameters: id: The ID of the NFT to get the reference for - // - // Returns: A reference to the NFT - // - // Note: This only allows the caller to read the ID of the NFT, - // not any topshot specific data. Please use borrowMoment to - // read Moment data. - // - access(all) fun borrowNFT(id: UInt64): &NonFungibleToken.NFT { - return (&self.ownedNFTs[id] as &NonFungibleToken.NFT?)! - } - - // borrowMoment returns a borrowed reference to a Moment - // so that the caller can read data and call methods from it. - // They can use this to read its setID, playID, serialNumber, - // or any of the setData or Play data associated with it by - // getting the setID or playID and reading those fields from - // the smart contract. - // - // Parameters: id: The ID of the NFT to get the reference for - // - // Returns: A reference to the NFT - access(all) fun borrowMoment(id: UInt64): &TopShot.NFT? { - if self.ownedNFTs[id] != nil { - let ref = (&self.ownedNFTs[id] as &NonFungibleToken.NFT?)! - return ref as! &TopShot.NFT - } else { - return nil - } - } - } - - // ----------------------------------------------------------------------- - // TopShot contract-level function definitions - // ----------------------------------------------------------------------- - - // createEmptyCollection creates a new, empty Collection object so that - // a user can store it in their account storage. - // Once they have a Collection in their storage, they are able to receive - // Moments in transactions. - // - access(all) fun createEmptyCollection(): @NonFungibleToken.Collection { - return <-create TopShot.Collection() - } - - // getAllPlays returns all the plays in topshot - // - // Returns: An array of all the plays that have been created - access(all) fun getAllPlays(): [TopShot.Play] { - return TopShot.playDatas.values - } - - // getPlayMetaData returns all the metadata associated with a specific Play - // - // Parameters: playID: The id of the Play that is being searched - // - // Returns: The metadata as a String to String mapping optional - access(all) fun getPlayMetaData(playID: UInt32): {String: String}? { - return self.playDatas[playID]?.metadata - } - - // getPlayMetaDataByField returns the metadata associated with a - // specific field of the metadata - // Ex: field: "Team" will return something - // like "Memphis Grizzlies" - // - // Parameters: playID: The id of the Play that is being searched - // field: The field to search for - // - // Returns: The metadata field as a String Optional - access(all) fun getPlayMetaDataByField(playID: UInt32, field: String): String? { - // Don't force a revert if the playID or field is invalid - if let play = TopShot.playDatas[playID] { - return play.metadata[field] - } else { - return nil - } - } - - // getSetName returns the name that the specified Set - // is associated with. - // - // Parameters: setID: The id of the Set that is being searched - // - // Returns: The name of the Set - access(all) fun getSetName(setID: UInt32): String? { - // Don't force a revert if the setID is invalid - return TopShot.setDatas[setID]?.name - } - - // getSetSeries returns the series that the specified Set - // is associated with. - // - // Parameters: setID: The id of the Set that is being searched - // - // Returns: The series that the Set belongs to - access(all) fun getSetSeries(setID: UInt32): UInt32? { - // Don't force a revert if the setID is invalid - return TopShot.setDatas[setID]?.series - } - - // getSetIDsByName returns the IDs that the specified Set name - // is associated with. - // - // Parameters: setName: The name of the Set that is being searched - // - // Returns: An array of the IDs of the Set if it exists, or nil if doesn't - access(all) fun getSetIDsByName(setName: String): [UInt32]? { - var setIDs: [UInt32] = [] - - // Iterate through all the setDatas and search for the name - for setData in TopShot.setDatas.values { - if setName == setData.name { - // If the name is found, return the ID - setIDs.append(setData.setID) - } - } - - // If the name isn't found, return nil - // Don't force a revert if the setName is invalid - if setIDs.length == 0 { - return nil - } else { - return setIDs - } - } - - // getPlaysInSet returns the list of Play IDs that are in the Set - // - // Parameters: setID: The id of the Set that is being searched - // - // Returns: An array of Play IDs - access(all) fun getPlaysInSet(setID: UInt32): [UInt32]? { - // Don't force a revert if the setID is invalid - return TopShot.sets[setID]?.plays - } - - // isEditionRetired returns a boolean that indicates if a Set/Play combo - // (otherwise known as an edition) is retired. - // If an edition is retired, it still remains in the Set, - // but Moments can no longer be minted from it. - // - // Parameters: setID: The id of the Set that is being searched - // playID: The id of the Play that is being searched - // - // Returns: Boolean indicating if the edition is retired or not - access(all) fun isEditionRetired(setID: UInt32, playID: UInt32): Bool? { - // Don't force a revert if the set or play ID is invalid - // Remove the set from the dictionary to get its field - if let setToRead <- TopShot.sets.remove(key: setID) { - - // See if the Play is retired from this Set - let retired = setToRead.retired[playID] - - // Put the Set back in the contract storage - TopShot.sets[setID] <-! setToRead - - // Return the retired status - return retired - } else { - - // If the Set wasn't found, return nil - return nil - } - } - - // isSetLocked returns a boolean that indicates if a Set - // is locked. If it's locked, - // new Plays can no longer be added to it, - // but Moments can still be minted from Plays the set contains. - // - // Parameters: setID: The id of the Set that is being searched - // - // Returns: Boolean indicating if the Set is locked or not - access(all) fun isSetLocked(setID: UInt32): Bool? { - // Don't force a revert if the setID is invalid - return TopShot.sets[setID]?.locked - } - - // getNumMomentsInEdition return the number of Moments that have been - // minted from a certain edition. - // - // Parameters: setID: The id of the Set that is being searched - // playID: The id of the Play that is being searched - // - // Returns: The total number of Moments - // that have been minted from an edition - access(all) fun getNumMomentsInEdition(setID: UInt32, playID: UInt32): UInt32? { - // Don't force a revert if the Set or play ID is invalid - // Remove the Set from the dictionary to get its field - if let setToRead <- TopShot.sets.remove(key: setID) { - - // Read the numMintedPerPlay - let amount = setToRead.numberMintedPerPlay[playID] - - // Put the Set back into the Sets dictionary - TopShot.sets[setID] <-! setToRead - - return amount - } else { - // If the set wasn't found return nil - return nil - } - } - - // ----------------------------------------------------------------------- - // TopShot initialization function - // ----------------------------------------------------------------------- - // - init() { - // Initialize contract fields - self.currentSeries = 0 - self.playDatas = {} - self.setDatas = {} - self.sets <- {} - self.nextPlayID = 1 - self.nextSetID = 1 - self.totalSupply = 0 - - // Put a new Collection in storage - self.account.storage.save<@Collection>(<- create Collection(), to: /storage/MomentCollection) - - // Create a public capability for the Collection - let cap = self.account.capabilities.storage.issue<&{MomentCollectionPublic}>(/storage/MomentCollection) - self.account.capabilities.publish(cap, at: /public/MomentCollection) - - // Put the Minter in storage - self.account.storage.save<@Admin>(<- create Admin(), to: /storage/TopShotAdmin) - - emit ContractInitialized() - } -} -` - func TestCheckTopShotContract(t *testing.T) { - - t.Parallel() - - nftChecker, err := ParseAndCheckWithOptions(t, - realNonFungibleTokenContractInterface, - ParseAndCheckOptions{ - Location: common.AddressLocation{ - Address: common.MustBytesToAddress([]byte{0x1}), - Name: "NonFungibleToken", - }, - }, - ) - require.NoError(t, err) - - baseValueActivation := sema.NewVariableActivation(sema.BaseValueActivation) - baseValueActivation.DeclareValue(stdlib.PanicFunction) - - _, err = ParseAndCheckWithOptions(t, - topShotContract, - ParseAndCheckOptions{ - Location: common.AddressLocation{ - Address: common.MustBytesToAddress([]byte{0x2}), - Name: "TopShot", - }, - Config: &sema.Config{ - ImportHandler: func(_ *sema.Checker, _ common.Location, _ ast.Range) (sema.Import, error) { - return sema.ElaborationImport{ - Elaboration: nftChecker.Elaboration, - }, nil - }, - BaseValueActivationHandler: func(_ common.Location) *sema.VariableActivation { - return baseValueActivation - }, - }, - }, - ) - require.NoError(t, err) -} */ diff --git a/tests/examples/examples.go b/tests/examples/examples.go deleted file mode 100644 index 7a96b68337..0000000000 --- a/tests/examples/examples.go +++ /dev/null @@ -1,82 +0,0 @@ -/* - * Cadence - The resource-oriented smart contract programming language - * - * Copyright Flow Foundation - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -package examples - -const FungibleTokenContractInterface = ` - access(all) contract interface FungibleToken { - - access(all) resource interface Provider { - - access(all) fun withdraw(amount: Int): @Vault - } - - access(all) resource interface Receiver { - - access(all) fun deposit(vault: @Vault) - } - - access(all) resource Vault: Provider, Receiver { - - access(all) balance: Int - - init(balance: Int) - } - - access(all) fun absorb(vault: @Vault) - - access(all) fun sprout(balance: Int): @Vault - } -` - -const ExampleFungibleTokenContract = ` - access(all) contract ExampleToken: FungibleToken { - - access(all) resource Vault: FungibleToken.Receiver, FungibleToken.Provider { - - access(all) var balance: Int - - init(balance: Int) { - self.balance = balance - } - - access(all) fun withdraw(amount: Int): @FungibleToken.Vault { - self.balance = self.balance - amount - return <-create Vault(balance: amount) - } - - access(all) fun deposit(vault: @FungibleToken.Vault) { - if let exampleVault <- vault as? @Vault { - self.balance = self.balance + exampleVault.balance - destroy exampleVault - } else { - destroy vault - panic("deposited vault is not an ExampleToken.Vault") - } - } - } - - access(all) fun absorb(vault: @FungibleToken.Vault) { - destroy vault - } - - access(all) fun sprout(balance: Int): @FungibleToken.Vault { - return <-create Vault(balance: balance) - } - } -` diff --git a/tests/interpreter/interpreter_test.go b/tests/interpreter/interpreter_test.go index 29d4f11823..7c4c6d762f 100644 --- a/tests/interpreter/interpreter_test.go +++ b/tests/interpreter/interpreter_test.go @@ -8530,82 +8530,6 @@ func TestInterpretCompositeDeclarationNestedConstructor(t *testing.T) { ) } -// TODO: re-enable this test with the v2 fungible token contract -/* func TestInterpretFungibleTokenContract(t *testing.T) { - - t.Parallel() - - code := strings.Join( - []string{ - examples.FungibleTokenContractInterface, - examples.ExampleFungibleTokenContract, - ` - access(all) fun test(): [Int; 2] { - - let publisher <- ExampleToken.sprout(balance: 100) - let receiver <- ExampleToken.sprout(balance: 0) - - let withdrawn <- publisher.withdraw(amount: 60) - receiver.deposit(vault: <-withdrawn) - - let publisherBalance = publisher.balance - let receiverBalance = receiver.balance - - destroy publisher - destroy receiver - - return [publisherBalance, receiverBalance] - } - `, - }, - "\n", - ) - - baseValueActivation := sema.NewVariableActivation(sema.BaseValueActivation) - baseValueActivation.DeclareValue(stdlib.PanicFunction) - - baseActivation := activations.NewActivation(nil, interpreter.BaseActivation) - interpreter.Declare(baseActivation, stdlib.PanicFunction) - - inter, err := parseCheckAndInterpretWithOptions(t, - code, - ParseCheckAndInterpretOptions{ - Config: &interpreter.Config{ - BaseActivationHandler: func(_ common.Location) *interpreter.VariableActivation { - return baseActivation - }, - ContractValueHandler: makeContractValueHandler(nil, nil, nil), - }, - CheckerConfig: &sema.Config{ - BaseValueActivationHandler: func(_ common.Location) *sema.VariableActivation { - return baseValueActivation - }, - }, - }, - ) - require.NoError(t, err) - - value, err := inter.Invoke("test") - require.NoError(t, err) - - AssertValuesEqual( - t, - inter, - interpreter.NewArrayValue( - inter, - interpreter.EmptyLocationRange, - interpreter.ConstantSizedStaticType{ - Type: interpreter.PrimitiveStaticTypeInt, - Size: 2, - }, - common.ZeroAddress, - interpreter.NewUnmeteredIntValueFromInt64(40), - interpreter.NewUnmeteredIntValueFromInt64(60), - ), - value, - ) -} */ - func TestInterpretContractAccountFieldUse(t *testing.T) { t.Parallel()