diff --git a/CONTRIBUTING.md b/CONTRIBUTING
similarity index 95%
rename from CONTRIBUTING.md
rename to CONTRIBUTING
index 57dff0a2f9..585bdf8bc9 100644
--- a/CONTRIBUTING.md
+++ b/CONTRIBUTING
@@ -135,18 +135,23 @@ import { useLocation } from '@docusaurus/router';
```
-
+:::warning
+
The `index` page should only include information that is available elsewhere within the category's other pages.
-
+
+:::
### Using DocCardList
Using `DocCardList` improves the layout of the index page:
-
+:::note Example
+
![edit page](./static/images/contributing/doc-card-list-light.png#gh-light-mode-only)
+
![edit page](./static/images/contributing/doc-card-list-dark.png#gh-dark-mode-only)
-
+
+:::
On the most pages you can use just `` component imported from `'@theme/DocCardList'`
@@ -173,12 +178,13 @@ import { useLocation } from '@docusaurus/router';
!isSamePath(item.href, useLocation().pathname))}/>
```
-
+:::warning
+
If you use `` on the top level category index page (e.g. `./docs/tools/index.mdx`), Docusaurus will throw an error:
> useCurrentSidebarCategory() should only be used on category index pages.
-
+:::
## SEO
@@ -225,9 +231,11 @@ Writing links in your documents is easy if you follow this rule-of-thumb: If it
[link](https://www.google.com)
```
-
+:::tip
+
Use relative links directly to .md/.mdx files
-
+
+:::
### Callouts
@@ -243,12 +251,32 @@ Available types:
Example:
```markdown
-
+:::tip
+
Use relative links directly to .md/.mdx files
-
+:::
```
+### Code references
+
+To include code from a file using a direct URL use `!from` operand inside a code block.
+
+Example:
+
+````markdown
+```
+!from https://raw.githubusercontent.com/onflow/docs/refs/heads/main/docs/evm/about.md
+```
+````
+
+mdx
+Copy code
+```javascript
+{code}
+This method keeps your documentation synchronized with your codebase by pulling the latest code directly into your docs.
+
+
## Content Validation
Content is validated each time a PR is submitted to the `docs` repository.
diff --git a/docs/evm/cadence/vm-bridge.md b/docs/evm/cadence/vm-bridge.md
index bf4b029200..d45d3358ac 100644
--- a/docs/evm/cadence/vm-bridge.md
+++ b/docs/evm/cadence/vm-bridge.md
@@ -87,129 +87,16 @@ Below are transactions relevant to onboarding assets:
onboard_by_type.cdc
-```cadence title="onboard_by_type.cdc"
-// source: https://www.github.com/onflow/flow-evm-bridge/blob/main/cadence/transactions/bridge/onboarding/onboard_by_type.cdc
-
-import "FungibleToken"
-import "FlowToken"
-
-import "ScopedFTProviders"
-
-import "EVM"
-
-import "FlowEVMBridge"
-import "FlowEVMBridgeConfig"
-
-/// This transaction onboards the asset type to the bridge, configuring the bridge to move assets between environments
-/// NOTE: This must be done before bridging a Cadence-native asset to EVM
-///
-/// @param type: The Cadence type of the bridgeable asset to onboard to the bridge
-///
-transaction(type: Type) {
-
- let scopedProvider: @ScopedFTProviders.ScopedFTProvider
-
- prepare(signer: auth(CopyValue, BorrowValue, IssueStorageCapabilityController, PublishCapability, SaveValue) &Account) {
-
- /* --- Configure a ScopedFTProvider --- */
- //
- // Issue and store bridge-dedicated Provider Capability in storage if necessary
- if signer.storage.type(at: FlowEVMBridgeConfig.providerCapabilityStoragePath) == nil {
- let providerCap = signer.capabilities.storage.issue(
- /storage/flowTokenVault
- )
- signer.storage.save(providerCap, to: FlowEVMBridgeConfig.providerCapabilityStoragePath)
- }
- // Copy the stored Provider capability and create a ScopedFTProvider
- let providerCapCopy = signer.storage.copy>(
- from: FlowEVMBridgeConfig.providerCapabilityStoragePath
- ) ?? panic("Invalid Provider Capability found in storage.")
- let providerFilter = ScopedFTProviders.AllowanceFilter(FlowEVMBridgeConfig.onboardFee)
- self.scopedProvider <- ScopedFTProviders.createScopedFTProvider(
- provider: providerCapCopy,
- filters: [ providerFilter ],
- expiration: getCurrentBlock().timestamp + 1.0
- )
- }
-
- execute {
- // Onboard the asset Type
- FlowEVMBridge.onboardByType(
- type,
- feeProvider: &self.scopedProvider as auth(FungibleToken.Withdraw) &{FungibleToken.Provider}
- )
- destroy self.scopedProvider
- }
-
- post {
- FlowEVMBridge.typeRequiresOnboarding(type) == false:
- "Asset ".concat(type.identifier).concat(" was not onboarded to the bridge.")
- }
-}
+```cadence onboard_by_type.cdc
+!from https://www.github.com/onflow/flow-evm-bridge/blob/main/cadence/transactions/bridge/onboarding/onboard_by_type.cdc
```
onboard_by_evm_address.cdc
-```
-// source: https://www.github.com/onflow/flow-evm-bridge/blob/main/cadence/transactions/bridge/onboarding/onboard_by_evm_address.cdc
-
-import "FungibleToken"
-import "FlowToken"
-
-import "ScopedFTProviders"
-
-import "EVM"
-
-import "FlowEVMBridge"
-import "FlowEVMBridgeConfig"
-
-/// This transaction onboards the NFT type to the bridge, configuring the bridge to move NFTs between environments
-/// NOTE: This must be done before bridging a Cadence-native NFT to EVM
-///
-/// @param contractAddressHex: The EVM address of the contract defining the bridgeable asset to be onboarded
-///
-transaction(contractAddressHex: String) {
-
- let contractAddress: EVM.EVMAddress
- let scopedProvider: @ScopedFTProviders.ScopedFTProvider
-
- prepare(signer: auth(CopyValue, BorrowValue, IssueStorageCapabilityController, PublishCapability, SaveValue) &Account) {
- /* --- Construct EVMAddress from hex string (no leading `"0x"`) --- */
- //
- self.contractAddress = EVM.addressFromString(contractAddressHex)
-
- /* --- Configure a ScopedFTProvider --- */
- //
- // Issue and store bridge-dedicated Provider Capability in storage if necessary
- if signer.storage.type(at: FlowEVMBridgeConfig.providerCapabilityStoragePath) == nil {
- let providerCap = signer.capabilities.storage.issue(
- /storage/flowTokenVault
- )
- signer.storage.save(providerCap, to: FlowEVMBridgeConfig.providerCapabilityStoragePath)
- }
- // Copy the stored Provider capability and create a ScopedFTProvider
- let providerCapCopy = signer.storage.copy>(
- from: FlowEVMBridgeConfig.providerCapabilityStoragePath
- ) ?? panic("Invalid Provider Capability found in storage.")
- let providerFilter = ScopedFTProviders.AllowanceFilter(FlowEVMBridgeConfig.onboardFee)
- self.scopedProvider <- ScopedFTProviders.createScopedFTProvider(
- provider: providerCapCopy,
- filters: [ providerFilter ],
- expiration: getCurrentBlock().timestamp + 1.0
- )
- }
-
- execute {
- // Onboard the EVM contract
- FlowEVMBridge.onboardByEVMAddress(
- self.contractAddress,
- feeProvider: &self.scopedProvider as auth(FungibleToken.Withdraw) &{FungibleToken.Provider}
- )
- destroy self.scopedProvider
- }
-}
+```cadence onboard_by_evm_address.cdc
+!from https://www.github.com/onflow/flow-evm-bridge/blob/main/cadence/transactions/bridge/onboarding/onboard_by_evm_address.cdc
```
@@ -234,26 +121,10 @@ You may also retrieve the type associated with a given EVM contract address usin
get_associated_type.cdc
-```cadence title="get_associated_type.cdc"
-// source: https://github.com/onflow/flow-evm-bridge/blob/main/cadence/scripts/bridge/get_associated_type.cdc
-
-import "EVM"
-
-import "FlowEVMBridgeConfig"
-
-/// Returns the Cadence Type associated with the given EVM address (as its hex String)
-///
-/// @param evmAddressHex: The hex-encoded address of the EVM contract as a String
-///
-/// @return The Cadence Type associated with the EVM address or nil if the address is not onboarded. `nil` may also be
-/// returned if the address is not a valid EVM address.
-///
-access(all)
-fun main(addressHex: String): Type? {
- let address = EVM.addressFromString(addressHex)
- return FlowEVMBridgeConfig.getTypeAssociated(with: address)
-}
+```cadence get_associated_type.cdc
+!from https://github.com/onflow/flow-evm-bridge/blob/main/cadence/scripts/bridge/get_associated_type.cdc
```
+
Alternatively, given some onboarded Cadence type, you can retrieve the associated EVM address using the following
@@ -263,28 +134,8 @@ script:
get_associated_address.cdc
-```cadence title="get_associated_address.cdc"
-// source: https://github.com/onflow/flow-evm-bridge/blob/main/cadence/scripts/bridge/get_associated_evm_address.cdc
-
-import "EVM"
-
-import "FlowEVMBridgeConfig"
-
-/// Returns the EVM address associated with the given Cadence type (as its identifier String)
-///
-/// @param typeIdentifier: The Cadence type identifier String
-///
-/// @return The EVM address as a hex string if the type has an associated EVMAddress, otherwise nil
-///
-access(all)
-fun main(identifier: String): String? {
- if let type = CompositeType(identifier) {
- if let address = FlowEVMBridgeConfig.getEVMAddressAssociated(with: type) {
- return address.toString()
- }
- }
- return nil
-}
+```cadence get_associated_address.cdc
+!from https://github.com/onflow/flow-evm-bridge/blob/main/cadence/scripts/bridge/get_associated_evm_address.cdc
```
@@ -300,127 +151,8 @@ Below are transactions relevant to bridging NFTs:
bridge_nft_to_evm.cdc
-```cadence title="bridge_nft_to_evm.cdc"
-// source: https://www.github.com/onflow/flow-evm-bridge/blob/main/cadence/transactions/bridge/nft/bridge_nft_to_evm.cdc
-
-import "FungibleToken"
-import "NonFungibleToken"
-import "ViewResolver"
-import "MetadataViews"
-import "FlowToken"
-
-import "ScopedFTProviders"
-
-import "EVM"
-
-import "FlowEVMBridge"
-import "FlowEVMBridgeConfig"
-import "FlowEVMBridgeUtils"
-
-/// Bridges an NFT from the signer's collection in Cadence to the signer's COA in FlowEVM
-///
-/// NOTE: This transaction also onboards the NFT to the bridge if necessary which may incur additional fees
-/// than bridging an asset that has already been onboarded.
-///
-/// @param nftIdentifier: The Cadence type identifier of the NFT to bridge - e.g. nft.getType().identifier
-/// @param id: The Cadence NFT.id of the NFT to bridge to EVM
-///
-transaction(nftIdentifier: String, id: UInt64) {
-
- let nft: @{NonFungibleToken.NFT}
- let coa: auth(EVM.Bridge) &EVM.CadenceOwnedAccount
- let requiresOnboarding: Bool
- let scopedProvider: @ScopedFTProviders.ScopedFTProvider
-
- prepare(signer: auth(CopyValue, BorrowValue, IssueStorageCapabilityController, PublishCapability, SaveValue) &Account) {
- /* --- Reference the signer's CadenceOwnedAccount --- */
- //
- // Borrow a reference to the signer's COA
- self.coa = signer.storage.borrow(from: /storage/evm)
- ?? panic("Could not borrow COA from provided gateway address")
-
- /* --- Construct the NFT type --- */
- //
- // Construct the NFT type from the provided identifier
- let nftType = CompositeType(nftIdentifier)
- ?? panic("Could not construct NFT type from identifier: ".concat(nftIdentifier))
- // Parse the NFT identifier into its components
- let nftContractAddress = FlowEVMBridgeUtils.getContractAddress(fromType: nftType)
- ?? panic("Could not get contract address from identifier: ".concat(nftIdentifier))
- let nftContractName = FlowEVMBridgeUtils.getContractName(fromType: nftType)
- ?? panic("Could not get contract name from identifier: ".concat(nftIdentifier))
-
- /* --- Retrieve the NFT --- */
- //
- // Borrow a reference to the NFT collection, configuring if necessary
- let viewResolver = getAccount(nftContractAddress).contracts.borrow<&{ViewResolver}>(name: nftContractName)
- ?? panic("Could not borrow ViewResolver from NFT contract")
- let collectionData = viewResolver.resolveContractView(
- resourceType: nftType,
- viewType: Type()
- ) as! MetadataViews.NFTCollectionData? ?? panic("Could not resolve NFTCollectionData view")
- let collection = signer.storage.borrow(
- from: collectionData.storagePath
- ) ?? panic("Could not access signer's NFT Collection")
-
- // Withdraw the requested NFT & calculate the approximate bridge fee based on NFT storage usage
- let currentStorageUsage = signer.storage.used
- self.nft <- collection.withdraw(withdrawID: id)
- let withdrawnStorageUsage = signer.storage.used
- var approxFee = FlowEVMBridgeUtils.calculateBridgeFee(
- bytes: currentStorageUsage - withdrawnStorageUsage
- ) * 1.10
- // Determine if the NFT requires onboarding - this impacts the fee required
- self.requiresOnboarding = FlowEVMBridge.typeRequiresOnboarding(self.nft.getType())
- ?? panic("Bridge does not support this asset type")
- if self.requiresOnboarding {
- approxFee = approxFee + FlowEVMBridgeConfig.onboardFee
- }
-
- /* --- Configure a ScopedFTProvider --- */
- //
- // Issue and store bridge-dedicated Provider Capability in storage if necessary
- if signer.storage.type(at: FlowEVMBridgeConfig.providerCapabilityStoragePath) == nil {
- let providerCap = signer.capabilities.storage.issue(
- /storage/flowTokenVault
- )
- signer.storage.save(providerCap, to: FlowEVMBridgeConfig.providerCapabilityStoragePath)
- }
- // Copy the stored Provider capability and create a ScopedFTProvider
- let providerCapCopy = signer.storage.copy>(
- from: FlowEVMBridgeConfig.providerCapabilityStoragePath
- ) ?? panic("Invalid Provider Capability found in storage.")
- let providerFilter = ScopedFTProviders.AllowanceFilter(approxFee)
- self.scopedProvider <- ScopedFTProviders.createScopedFTProvider(
- provider: providerCapCopy,
- filters: [ providerFilter ],
- expiration: getCurrentBlock().timestamp + 1.0
- )
- }
-
- pre {
- self.nft.getType().identifier == nftIdentifier:
- "Attempting to send invalid nft type - requested: ".concat(nftIdentifier)
- .concat(", sending: ").concat(self.nft.getType().identifier)
- }
-
- execute {
- if self.requiresOnboarding {
- // Onboard the NFT to the bridge
- FlowEVMBridge.onboardByType(
- self.nft.getType(),
- feeProvider: &self.scopedProvider as auth(FungibleToken.Withdraw) &{FungibleToken.Provider}
- )
- }
- // Execute the bridge
- self.coa.depositNFT(
- nft: <-self.nft,
- feeProvider: &self.scopedProvider as auth(FungibleToken.Withdraw) &{FungibleToken.Provider}
- )
- // Destroy the ScopedFTProvider
- destroy self.scopedProvider
- }
-}
+```cadence bridge_nft_to_evm.cdc
+!from https://www.github.com/onflow/flow-evm-bridge/blob/main/cadence/transactions/bridge/nft/bridge_nft_to_evm.cdc
```
@@ -428,115 +160,8 @@ transaction(nftIdentifier: String, id: UInt64) {
bridge_nft_from_evm.cdc
-```cadence title="bridge_nft_from_evm.cdc"
-// source: https://www.github.com/onflow/flow-evm-bridge/blob/main/cadence/transactions/bridge/nft/bridge_nft_from_evm.cdc
-
-import "FungibleToken"
-import "NonFungibleToken"
-import "ViewResolver"
-import "MetadataViews"
-import "FlowToken"
-
-import "ScopedFTProviders"
-
-import "EVM"
-
-import "FlowEVMBridge"
-import "FlowEVMBridgeConfig"
-import "FlowEVMBridgeUtils"
-
-/// This transaction bridges an NFT from EVM to Cadence assuming it has already been onboarded to the FlowEVMBridge
-/// NOTE: The ERC721 must have first been onboarded to the bridge. This can be checked via the method
-/// FlowEVMBridge.evmAddressRequiresOnboarding(address: self.evmContractAddress)
-///
-/// @param nftIdentifier: The Cadence type identifier of the NFT to bridge - e.g. nft.getType().identifier
-/// @param id: The ERC721 id of the NFT to bridge to Cadence from EVM
-///
-transaction(nftIdentifier: String, id: UInt256) {
-
- let nftType: Type
- let collection: &{NonFungibleToken.Collection}
- let scopedProvider: @ScopedFTProviders.ScopedFTProvider
- let coa: auth(EVM.Bridge) &EVM.CadenceOwnedAccount
-
- prepare(signer: auth(BorrowValue, CopyValue, IssueStorageCapabilityController, PublishCapability, SaveValue, UnpublishCapability) &Account) {
- /* --- Reference the signer's CadenceOwnedAccount --- */
- //
- // Borrow a reference to the signer's COA
- self.coa = signer.storage.borrow(from: /storage/evm)
- ?? panic("Could not borrow COA from provided gateway address")
-
- /* --- Construct the NFT type --- */
- //
- // Construct the NFT type from the provided identifier
- self.nftType = CompositeType(nftIdentifier)
- ?? panic("Could not construct NFT type from identifier: ".concat(nftIdentifier))
- // Parse the NFT identifier into its components
- let nftContractAddress = FlowEVMBridgeUtils.getContractAddress(fromType: self.nftType)
- ?? panic("Could not get contract address from identifier: ".concat(nftIdentifier))
- let nftContractName = FlowEVMBridgeUtils.getContractName(fromType: self.nftType)
- ?? panic("Could not get contract name from identifier: ".concat(nftIdentifier))
-
- /* --- Reference the signer's NFT Collection --- */
- //
- // Borrow a reference to the NFT collection, configuring if necessary
- let viewResolver = getAccount(nftContractAddress).contracts.borrow<&{ViewResolver}>(name: nftContractName)
- ?? panic("Could not borrow ViewResolver from NFT contract")
- let collectionData = viewResolver.resolveContractView(
- resourceType: self.nftType,
- viewType: Type()
- ) as! MetadataViews.NFTCollectionData? ?? panic("Could not resolve NFTCollectionData view")
- if signer.storage.borrow<&{NonFungibleToken.Collection}>(from: collectionData.storagePath) == nil {
- signer.storage.save(<-collectionData.createEmptyCollection(), to: collectionData.storagePath)
- signer.capabilities.unpublish(collectionData.publicPath)
- let collectionCap = signer.capabilities.storage.issue<&{NonFungibleToken.Collection}>(collectionData.storagePath)
- signer.capabilities.publish(collectionCap, at: collectionData.publicPath)
- }
- self.collection = signer.storage.borrow<&{NonFungibleToken.Collection}>(from: collectionData.storagePath)
- ?? panic("Could not borrow collection from storage path")
-
- /* --- Configure a ScopedFTProvider --- */
- //
- // Calculate the bridge fee - bridging from EVM consumes no storage, so flat fee
- let approxFee = FlowEVMBridgeUtils.calculateBridgeFee(bytes: 0)
- // Issue and store bridge-dedicated Provider Capability in storage if necessary
- if signer.storage.type(at: FlowEVMBridgeConfig.providerCapabilityStoragePath) == nil {
- let providerCap = signer.capabilities.storage.issue(
- /storage/flowTokenVault
- )
- signer.storage.save(providerCap, to: FlowEVMBridgeConfig.providerCapabilityStoragePath)
- }
- // Copy the stored Provider capability and create a ScopedFTProvider
- let providerCapCopy = signer.storage.copy>(
- from: FlowEVMBridgeConfig.providerCapabilityStoragePath
- ) ?? panic("Invalid Provider Capability found in storage.")
- let providerFilter = ScopedFTProviders.AllowanceFilter(approxFee)
- self.scopedProvider <- ScopedFTProviders.createScopedFTProvider(
- provider: providerCapCopy,
- filters: [ providerFilter ],
- expiration: getCurrentBlock().timestamp + 1.0
- )
- }
-
- execute {
- // Execute the bridge
- let nft: @{NonFungibleToken.NFT} <- self.coa.withdrawNFT(
- type: self.nftType,
- id: id,
- feeProvider: &self.scopedProvider as auth(FungibleToken.Withdraw) &{FungibleToken.Provider}
- )
- // Ensure the bridged nft is the correct type
- assert(
- nft.getType() == self.nftType,
- message: "Bridged nft type mismatch - requeswted: ".concat(self.nftType.identifier)
- .concat(", received: ").concat(nft.getType().identifier)
- )
- // Deposit the bridged NFT into the signer's collection
- self.collection.deposit(token: <-nft)
- // Destroy the ScopedFTProvider
- destroy self.scopedProvider
- }
-}
+```cadence bridge_nft_from_evm.cdc
+!from https://www.github.com/onflow/flow-evm-bridge/blob/main/cadence/transactions/bridge/nft/bridge_nft_from_evm.cdc
```
@@ -560,128 +185,8 @@ Below are transactions relevant to bridging fungible tokens:
bridge_tokens_to_evm.cdc
-```cadence title="bridge_tokens_to_evm.cdc"
-// source: https://www.github.com/onflow/flow-evm-bridge/blob/main/cadence/transactions/bridge/tokens/bridge_tokens_to_evm.cdc
-
-import "FungibleToken"
-import "ViewResolver"
-import "FungibleTokenMetadataViews"
-import "FlowToken"
-
-import "ScopedFTProviders"
-
-import "EVM"
-
-import "FlowEVMBridge"
-import "FlowEVMBridgeConfig"
-import "FlowEVMBridgeUtils"
-
-/// Bridges a Vault from the signer's storage to the signer's COA in EVM.Account.
-///
-/// NOTE: This transaction also onboards the Vault to the bridge if necessary which may incur additional fees
-/// than bridging an asset that has already been onboarded.
-///
-/// @param vaultIdentifier: The Cadence type identifier of the FungibleToken Vault to bridge
-/// - e.g. vault.getType().identifier
-/// @param amount: The amount of tokens to bridge from EVM
-///
-transaction(vaultIdentifier: String, amount: UFix64) {
-
- let sentVault: @{FungibleToken.Vault}
- let coa: auth(EVM.Bridge) &EVM.CadenceOwnedAccount
- let requiresOnboarding: Bool
- let scopedProvider: @ScopedFTProviders.ScopedFTProvider
-
- prepare(signer: auth(CopyValue, BorrowValue, IssueStorageCapabilityController, PublishCapability, SaveValue) &Account) {
- /* --- Reference the signer's CadenceOwnedAccount --- */
- //
- // Borrow a reference to the signer's COA
- self.coa = signer.storage.borrow(from: /storage/evm)
- ?? panic("Could not borrow COA from provided gateway address")
-
- /* --- Construct the Vault type --- */
- //
- // Construct the Vault type from the provided identifier
- let vaultType = CompositeType(vaultIdentifier)
- ?? panic("Could not construct Vault type from identifier: ".concat(vaultIdentifier))
- // Parse the Vault identifier into its components
- let tokenContractAddress = FlowEVMBridgeUtils.getContractAddress(fromType: vaultType)
- ?? panic("Could not get contract address from identifier: ".concat(vaultIdentifier))
- let tokenContractName = FlowEVMBridgeUtils.getContractName(fromType: vaultType)
- ?? panic("Could not get contract name from identifier: ".concat(vaultIdentifier))
-
- /* --- Retrieve the funds --- */
- //
- // Borrow a reference to the FungibleToken Vault
- let viewResolver = getAccount(tokenContractAddress).contracts.borrow<&{ViewResolver}>(name: tokenContractName)
- ?? panic("Could not borrow ViewResolver from FungibleToken contract")
- let vaultData = viewResolver.resolveContractView(
- resourceType: vaultType,
- viewType: Type()
- ) as! FungibleTokenMetadataViews.FTVaultData? ?? panic("Could not resolve FTVaultData view")
- let vault = signer.storage.borrow(
- from: vaultData.storagePath
- ) ?? panic("Could not access signer's FungibleToken Vault")
-
- // Withdraw the requested balance & calculate the approximate bridge fee based on storage usage
- let currentStorageUsage = signer.storage.used
- self.sentVault <- vault.withdraw(amount: amount)
- let withdrawnStorageUsage = signer.storage.used
- // Approximate the bridge fee based on the difference in storage usage with some buffer
- var approxFee = FlowEVMBridgeUtils.calculateBridgeFee(
- bytes: currentStorageUsage - withdrawnStorageUsage
- ) * 1.10
- // Determine if the Vault requires onboarding - this impacts the fee required
- self.requiresOnboarding = FlowEVMBridge.typeRequiresOnboarding(self.sentVault.getType())
- ?? panic("Bridge does not support this asset type")
- if self.requiresOnboarding {
- approxFee = approxFee + FlowEVMBridgeConfig.onboardFee
- }
-
- /* --- Configure a ScopedFTProvider --- */
- //
- // Issue and store bridge-dedicated Provider Capability in storage if necessary
- if signer.storage.type(at: FlowEVMBridgeConfig.providerCapabilityStoragePath) == nil {
- let providerCap = signer.capabilities.storage.issue(
- /storage/flowTokenVault
- )
- signer.storage.save(providerCap, to: FlowEVMBridgeConfig.providerCapabilityStoragePath)
- }
- // Copy the stored Provider capability and create a ScopedFTProvider
- let providerCapCopy = signer.storage.copy>(
- from: FlowEVMBridgeConfig.providerCapabilityStoragePath
- ) ?? panic("Invalid Provider Capability found in storage.")
- let providerFilter = ScopedFTProviders.AllowanceFilter(approxFee)
- self.scopedProvider <- ScopedFTProviders.createScopedFTProvider(
- provider: providerCapCopy,
- filters: [ providerFilter ],
- expiration: getCurrentBlock().timestamp + 1.0
- )
- }
-
- pre {
- self.sentVault.getType().identifier == vaultIdentifier:
- "Attempting to send invalid vault type - requested: ".concat(vaultIdentifier)
- .concat(", sending: ").concat(self.sentVault.getType().identifier)
- }
-
- execute {
- if self.requiresOnboarding {
- // Onboard the Vault to the bridge
- FlowEVMBridge.onboardByType(
- self.sentVault.getType(),
- feeProvider: &self.scopedProvider as auth(FungibleToken.Withdraw) &{FungibleToken.Provider}
- )
- }
- // Execute the bridge
- self.coa.depositTokens(
- vault: <-self.sentVault,
- feeProvider: &self.scopedProvider as auth(FungibleToken.Withdraw) &{FungibleToken.Provider}
- )
- // Destroy the ScopedFTProvider
- destroy self.scopedProvider
- }
-}
+```cadence bridge_tokens_to_evm.cdc
+!from https://www.github.com/onflow/flow-evm-bridge/blob/main/cadence/transactions/bridge/tokens/bridge_tokens_to_evm.cdc
```
@@ -689,121 +194,8 @@ transaction(vaultIdentifier: String, amount: UFix64) {
bridge_tokens_from_evm.cdc
-```cadence title="bridge_tokens_from_evm.cdc"
-// source: https://www.github.com/onflow/flow-evm-bridge/blob/main/cadence/transactions/bridge/tokens/bridge_tokens_from_evm.cdc
-
-import "FungibleToken"
-import "FungibleTokenMetadataViews"
-import "ViewResolver"
-import "MetadataViews"
-import "FlowToken"
-
-import "ScopedFTProviders"
-
-import "EVM"
-
-import "FlowEVMBridge"
-import "FlowEVMBridgeConfig"
-import "FlowEVMBridgeUtils"
-
-/// This transaction bridges fungible tokens from EVM to Cadence assuming it has already been onboarded to the
-/// FlowEVMBridge.
-///
-/// NOTE: The ERC20 must have first been onboarded to the bridge. This can be checked via the method
-/// FlowEVMBridge.evmAddressRequiresOnboarding(address: self.evmContractAddress)
-///
-/// @param vaultIdentifier: The Cadence type identifier of the FungibleToken Vault to bridge
-/// - e.g. vault.getType().identifier
-/// @param amount: The amount of tokens to bridge from EVM
-///
-transaction(vaultIdentifier: String, amount: UInt256) {
-
- let vaultType: Type
- let receiver: &{FungibleToken.Vault}
- let scopedProvider: @ScopedFTProviders.ScopedFTProvider
- let coa: auth(EVM.Bridge) &EVM.CadenceOwnedAccount
-
- prepare(signer: auth(BorrowValue, CopyValue, IssueStorageCapabilityController, PublishCapability, SaveValue, UnpublishCapability) &Account) {
- /* --- Reference the signer's CadenceOwnedAccount --- */
- //
- // Borrow a reference to the signer's COA
- self.coa = signer.storage.borrow(from: /storage/evm)
- ?? panic("Could not borrow COA from provided gateway address")
-
- /* --- Construct the Vault type --- */
- //
- // Construct the Vault type from the provided identifier
- self.vaultType = CompositeType(vaultIdentifier)
- ?? panic("Could not construct Vault type from identifier: ".concat(vaultIdentifier))
- // Parse the Vault identifier into its components
- let tokenContractAddress = FlowEVMBridgeUtils.getContractAddress(fromType: self.vaultType)
- ?? panic("Could not get contract address from identifier: ".concat(vaultIdentifier))
- let tokenContractName = FlowEVMBridgeUtils.getContractName(fromType: self.vaultType)
- ?? panic("Could not get contract name from identifier: ".concat(vaultIdentifier))
-
- /* --- Reference the signer's Vault --- */
- //
- // Borrow a reference to the FungibleToken Vault, configuring if necessary
- let viewResolver = getAccount(tokenContractAddress).contracts.borrow<&{ViewResolver}>(name: tokenContractName)
- ?? panic("Could not borrow ViewResolver from FungibleToken contract")
- let vaultData = viewResolver.resolveContractView(
- resourceType: self.vaultType,
- viewType: Type()
- ) as! FungibleTokenMetadataViews.FTVaultData? ?? panic("Could not resolve FTVaultData view")
- // If the vault does not exist, create it and publish according to the contract's defined configuration
- if signer.storage.borrow<&{FungibleToken.Vault}>(from: vaultData.storagePath) == nil {
- signer.storage.save(<-vaultData.createEmptyVault(), to: vaultData.storagePath)
-
- signer.capabilities.unpublish(vaultData.receiverPath)
- signer.capabilities.unpublish(vaultData.metadataPath)
-
- let receiverCap = signer.capabilities.storage.issue<&{FungibleToken.Vault}>(vaultData.storagePath)
- let metadataCap = signer.capabilities.storage.issue<&{FungibleToken.Vault}>(vaultData.storagePath)
-
- signer.capabilities.publish(receiverCap, at: vaultData.receiverPath)
- signer.capabilities.publish(metadataCap, at: vaultData.metadataPath)
- }
- self.receiver = signer.storage.borrow<&{FungibleToken.Vault}>(from: vaultData.storagePath)
- ?? panic("Could not borrow Vault from storage path")
-
- /* --- Configure a ScopedFTProvider --- */
- //
- // Calculate the bridge fee - bridging from EVM consumes no storage, so flat fee
- let approxFee = FlowEVMBridgeUtils.calculateBridgeFee(bytes: 0)
- // Issue and store bridge-dedicated Provider Capability in storage if necessary
- if signer.storage.type(at: FlowEVMBridgeConfig.providerCapabilityStoragePath) == nil {
- let providerCap = signer.capabilities.storage.issue(
- /storage/flowTokenVault
- )
- signer.storage.save(providerCap, to: FlowEVMBridgeConfig.providerCapabilityStoragePath)
- }
- // Copy the stored Provider capability and create a ScopedFTProvider
- let providerCapCopy = signer.storage.copy>(
- from: FlowEVMBridgeConfig.providerCapabilityStoragePath
- ) ?? panic("Invalid Provider Capability found in storage.")
- let providerFilter = ScopedFTProviders.AllowanceFilter(approxFee)
- self.scopedProvider <- ScopedFTProviders.createScopedFTProvider(
- provider: providerCapCopy,
- filters: [ providerFilter ],
- expiration: getCurrentBlock().timestamp + 1.0
- )
- }
-
- execute {
- // Execute the bridge request
- let vault: @{FungibleToken.Vault} <- self.coa.withdrawTokens(
- type: self.vaultType,
- amount: amount,
- feeProvider: &self.scopedProvider as auth(FungibleToken.Withdraw) &{FungibleToken.Provider}
- )
- // Ensure the bridged vault is the correct type
- assert(vault.getType() == self.vaultType, message: "Bridged vault type mismatch")
- // Deposit the bridged token into the signer's vault
- self.receiver.deposit(from: <-vault)
- // Destroy the ScopedFTProvider
- destroy self.scopedProvider
- }
-}
+```cadence bridge_tokens_from_evm.cdc
+!from https://www.github.com/onflow/flow-evm-bridge/blob/main/cadence/transactions/bridge/tokens/bridge_tokens_from_evm.cdc
```
diff --git a/docusaurus.config.js b/docusaurus.config.js
index 7c32808ab0..ad9e7f4daa 100644
--- a/docusaurus.config.js
+++ b/docusaurus.config.js
@@ -177,6 +177,7 @@ const config = {
({
docs: {
beforeDefaultRemarkPlugins: [
+ require('./src/plugins/code-reference'),
[
remarkCodeHike,
{ theme: 'nord', lineNumbers: true, showCopyButton: true },
diff --git a/package.json b/package.json
index c8889b51bf..b69ed545d1 100644
--- a/package.json
+++ b/package.json
@@ -40,6 +40,7 @@
"dotenv": "16.0.3",
"eslint-config-prettier": "^8.8.0",
"eslint-plugin-prettier": "^4.2.1",
+ "node-fetch": "^3.3.2",
"postcss": "^8.4.31",
"prettier": "^2.8.7",
"prism-react-renderer": "^1.3.5",
@@ -51,6 +52,7 @@
"shiki": "1.14.1",
"tailwind-scrollbar-hide": "1.1.7",
"tailwindcss": "^3.3.5",
+ "unist-util-visit": "^5.0.0",
"webpack-merge": "5.8.0"
},
"devDependencies": {
diff --git a/src/plugins/code-reference.js b/src/plugins/code-reference.js
new file mode 100644
index 0000000000..779f1f13d7
--- /dev/null
+++ b/src/plugins/code-reference.js
@@ -0,0 +1,49 @@
+import { visit } from 'unist-util-visit';
+import fetch from 'node-fetch';
+
+const VALUE_STARTS_WITH = '!from ';
+
+const githubReplace = /^(https:\/\/)(www.)?github.com\/(.+)\/blob\/(.+)/;
+
+const getUrl = (nodeValue) => {
+ const url = nodeValue.replace(VALUE_STARTS_WITH, '').trim();
+
+ return url.replace(githubReplace, '$1raw.githubusercontent.com/$3/$4');
+};
+
+const plugin = () => {
+ const transformer = async (ast) => {
+ const promises = [];
+ visit(ast, 'code', (node) => {
+ if (node.value?.startsWith(VALUE_STARTS_WITH)) {
+ const url = getUrl(node.value);
+ if (!url) {
+ return;
+ }
+ const fetchPromise = fetch(url)
+ .then((res) => {
+ if (!res.ok) {
+ throw new Error(
+ `Failed to fetch code from ${url}: ${res.statusText}`,
+ );
+ }
+ return res.text();
+ })
+ .then((code) => {
+ node.value = code;
+ })
+ .catch((err) => {
+ console.error(err);
+ node.value = `Error fetching code: ${err.message}`;
+ });
+
+ promises.push(fetchPromise);
+ }
+ });
+
+ await Promise.all(promises);
+ };
+ return transformer;
+};
+
+export default plugin;
diff --git a/src/plugins/insert-info-tags-loader.js b/src/plugins/insert-info-tags-loader.js
deleted file mode 100644
index a930d3a095..0000000000
--- a/src/plugins/insert-info-tags-loader.js
+++ /dev/null
@@ -1,32 +0,0 @@
-module.exports = function (source) {
- const infoBannerMessage =
- 'For Cadence 0.42 go to [Legacy Docs](https://legacy.developers.flow.com/)';
- // Function to check if a page content contains "cadence" code block
- const containsCadenceCodeBlock = (content) => {
- return content.includes('```cadence');
- };
-
- // Function to insert :::info::: tag into content
- const insertInfoTag = (content) => {
- const infoBanner = ':::info\n' + infoBannerMessage + '\n:::\n';
- // Check fof front matter
- if (!content.startsWith('---\n')) {
- // Insert :::info::: tag at the beginning of the content
- return infoBanner + content;
- }
- // Split the content into front matter and body
- const [frontMatter, ...body] = content.split('---\n').slice(1);
-
- // Insert :::info::: tag after front matter
- return `---\n${frontMatter}---\n${infoBanner}${body.join('---\n')}`;
- };
-
- // Check if the content contains a "cadence" code block
- if (containsCadenceCodeBlock(source)) {
- // Insert :::info::: tag
- return insertInfoTag(source);
- }
-
- // Return the original content if no changes are needed
- return source;
-};
diff --git a/yarn.lock b/yarn.lock
index 7180ea239e..7a74733951 100644
--- a/yarn.lock
+++ b/yarn.lock
@@ -5414,6 +5414,11 @@ csstype@^3.0.2:
resolved "https://registry.yarnpkg.com/csstype/-/csstype-3.1.2.tgz#1d4bf9d572f11c14031f0436e1c10bc1f571f50b"
integrity sha512-I7K1Uu0MBPzaFKg4nI5Q7Vs2t+3gWWW648spaF+Rg7pI9ds18Ugn+lvg4SHczUdKlHI5LWBXyqfS8+DufyBsgQ==
+data-uri-to-buffer@^4.0.0:
+ version "4.0.1"
+ resolved "https://registry.yarnpkg.com/data-uri-to-buffer/-/data-uri-to-buffer-4.0.1.tgz#d8feb2b2881e6a4f58c2e08acfd0e2834e26222e"
+ integrity sha512-0R9ikRb668HB7QDxT1vkpuUBtqc53YyAwMwGeUFKRojY/NWKvdZ+9UYtRfGmhqNbRkTSVpMbmyhXipFFv2cb/A==
+
date-fns@2.29.3:
version "2.29.3"
resolved "https://registry.yarnpkg.com/date-fns/-/date-fns-2.29.3.tgz#27402d2fc67eb442b511b70bbdf98e6411cd68a8"
@@ -6608,6 +6613,14 @@ feed@^4.2.2:
dependencies:
xml-js "^1.6.11"
+fetch-blob@^3.1.2, fetch-blob@^3.1.4:
+ version "3.2.0"
+ resolved "https://registry.yarnpkg.com/fetch-blob/-/fetch-blob-3.2.0.tgz#f09b8d4bbd45adc6f0c20b7e787e793e309dcce9"
+ integrity sha512-7yAQpD2UMJzLi1Dqv7qFYnPbaPx7ZfFK6PiIxQ4PfkGPyNyl2Ugx+a/umUonmKqjhM4DnfbMvdX6otXq83soQQ==
+ dependencies:
+ node-domexception "^1.0.0"
+ web-streams-polyfill "^3.0.3"
+
file-entry-cache@^6.0.1:
version "6.0.1"
resolved "https://registry.yarnpkg.com/file-entry-cache/-/file-entry-cache-6.0.1.tgz#211b2dd9659cb0394b073e7323ac3c933d522027"
@@ -6790,6 +6803,13 @@ format@^0.2.0:
resolved "https://registry.yarnpkg.com/format/-/format-0.2.2.tgz#d6170107e9efdc4ed30c9dc39016df942b5cb58b"
integrity sha512-wzsgA6WOq+09wrU1tsJ09udeR/YZRaeArL9e1wPbFg3GG2yDnC2ldKpxs4xunpFF9DgqCqOIra3bc1HWrJ37Ww==
+formdata-polyfill@^4.0.10:
+ version "4.0.10"
+ resolved "https://registry.yarnpkg.com/formdata-polyfill/-/formdata-polyfill-4.0.10.tgz#24807c31c9d402e002ab3d8c720144ceb8848423"
+ integrity sha512-buewHzMvYL29jdeQTVILecSaZKnt/RJWjoZCF5OW60Z67/GmSLBkOFM7qh1PI3zFNtJbaZL5eQu1vLfazOwj4g==
+ dependencies:
+ fetch-blob "^3.1.2"
+
forwarded@0.2.0:
version "0.2.0"
resolved "https://registry.yarnpkg.com/forwarded/-/forwarded-0.2.0.tgz#2269936428aad4c15c7ebe9779a84bf0b2a81811"
@@ -9415,6 +9435,11 @@ no-case@^3.0.4:
lower-case "^2.0.2"
tslib "^2.0.3"
+node-domexception@^1.0.0:
+ version "1.0.0"
+ resolved "https://registry.yarnpkg.com/node-domexception/-/node-domexception-1.0.0.tgz#6888db46a1f71c0b76b3f7555016b63fe64766e5"
+ integrity sha512-/jKZoMpw0F8GRwl4/eLROPA3cfcXtLApP0QzLmUT/HuPCZWyB7IY9ZrMeKw2O/nFIqPQB3PVM9aYm0F312AXDQ==
+
node-downloader-helper@^2.1.6:
version "2.1.9"
resolved "https://registry.yarnpkg.com/node-downloader-helper/-/node-downloader-helper-2.1.9.tgz#a59ee7276b2bf708bbac2cc5872ad28fc7cd1b0e"
@@ -9444,6 +9469,15 @@ node-fetch@^2.0.0, node-fetch@^2.6.1:
dependencies:
whatwg-url "^5.0.0"
+node-fetch@^3.3.2:
+ version "3.3.2"
+ resolved "https://registry.yarnpkg.com/node-fetch/-/node-fetch-3.3.2.tgz#d1e889bacdf733b4ff3b2b243eb7a12866a0b78b"
+ integrity sha512-dRB78srN/l6gqWulah9SrxeYnxeddIG30+GOqK/9OlLVyLg3HPnr6SqOWTWOXKRwC2eGYCkZ59NNuSgvSrpgOA==
+ dependencies:
+ data-uri-to-buffer "^4.0.0"
+ fetch-blob "^3.1.4"
+ formdata-polyfill "^4.0.10"
+
node-forge@^1:
version "1.3.1"
resolved "https://registry.yarnpkg.com/node-forge/-/node-forge-1.3.1.tgz#be8da2af243b2417d5f646a770663a92b7e9ded3"
@@ -13055,6 +13089,11 @@ web-namespaces@^2.0.0:
resolved "https://registry.yarnpkg.com/web-namespaces/-/web-namespaces-2.0.1.tgz#1010ff7c650eccb2592cebeeaf9a1b253fd40692"
integrity sha512-bKr1DkiNa2krS7qxNtdrtHAmzuYGFQLiQ13TsorsdT6ULTkPLKuu5+GsFpDlg6JFjUTwX2DyhMPG2be8uPrqsQ==
+web-streams-polyfill@^3.0.3:
+ version "3.3.3"
+ resolved "https://registry.yarnpkg.com/web-streams-polyfill/-/web-streams-polyfill-3.3.3.tgz#2073b91a2fdb1fbfbd401e7de0ac9f8214cecb4b"
+ integrity sha512-d2JWLCivmZYTSIoge9MsgFCZrt571BikcWGYkjC1khllbTeDlGqZ2D8vD8E/lJa8WGWbb7Plm8/XJYV7IJHZZw==
+
webidl-conversions@^3.0.0:
version "3.0.1"
resolved "https://registry.yarnpkg.com/webidl-conversions/-/webidl-conversions-3.0.1.tgz#24534275e2a7bc6be7bc86611cc16ae0a5654871"