diff --git a/plugin/src/atoms/deployment.ts b/plugin/src/atoms/deployment.ts index da5c3e74..9d4d3f53 100644 --- a/plugin/src/atoms/deployment.ts +++ b/plugin/src/atoms/deployment.ts @@ -1,13 +1,14 @@ import { atom } from 'jotai' import { type CallDataObject, type Input } from '../utils/types/contracts' +export type Status = 'IDLE' | 'IN_PROGRESS' | 'ERROR' | 'DONE' const isDeployingAtom = atom(false) -const deployStatusAtom = atom('') +const deployStatusAtom = atom('IDLE') const isDelcaringAtom = atom(false) -const declStatusAtom = atom('') +const declStatusAtom = atom('IDLE') const constructorCalldataAtom = atom({}) @@ -19,38 +20,78 @@ const declTxHashAtom = atom('') const deployTxHashAtom = atom('') -type Key = 'isDeploying' | 'deployStatus' | 'isDeclaring' | 'declStatus' | 'constructorCalldata' | 'constructorInputs' | 'notEnoughInputs' | 'declTxHash' | 'deployTxHash' +type Key = + | 'isDeploying' + | 'deployStatus' + | 'isDeclaring' + | 'declStatus' + | 'constructorCalldata' + | 'constructorInputs' + | 'notEnoughInputs' + | 'declTxHash' + | 'deployTxHash' interface SetDeploymentAtom { key: Key value: string | boolean | CallDataObject | Input[] } -const deploymentAtom = atom((get) => { - return { - isDeploying: get(isDeployingAtom), - deployStatus: get(deployStatusAtom), - isDeclaring: get(isDelcaringAtom), - declStatus: get(declStatusAtom), - constructorCalldata: get(constructorCalldataAtom), - constructorInputs: get(constructorInputsAtom), - notEnoughInputs: get(notEnoughInputsAtom), - declTxHash: get(declTxHashAtom), - deployTxHash: get(deployTxHashAtom) +const deploymentAtom = atom( + (get) => { + return { + isDeploying: get(isDeployingAtom), + deployStatus: get(deployStatusAtom), + isDeclaring: get(isDelcaringAtom), + declStatus: get(declStatusAtom), + constructorCalldata: get(constructorCalldataAtom), + constructorInputs: get(constructorInputsAtom), + notEnoughInputs: get(notEnoughInputsAtom), + declTxHash: get(declTxHashAtom), + deployTxHash: get(deployTxHashAtom) + } + }, + (_get, set, newValue: SetDeploymentAtom) => { + switch (newValue?.key) { + case 'isDeploying': + typeof newValue?.value === 'boolean' && + set(isDeployingAtom, newValue?.value) + break + case 'deployStatus': + typeof newValue?.value === 'string' && + set(deployStatusAtom, newValue?.value as Status) + break + case 'isDeclaring': + typeof newValue?.value === 'boolean' && + set(isDelcaringAtom, newValue?.value) + break + case 'declStatus': + typeof newValue?.value === 'string' && + set(declStatusAtom, newValue?.value as Status) + break + case 'constructorCalldata': + typeof newValue?.value === 'object' && + !Array.isArray(newValue?.value) && + set(constructorCalldataAtom, newValue?.value) + break + case 'constructorInputs': + Array.isArray(newValue?.value) && + set(constructorInputsAtom, newValue?.value) + break + case 'notEnoughInputs': + typeof newValue?.value === 'boolean' && + set(notEnoughInputsAtom, newValue?.value) + break + case 'declTxHash': + typeof newValue?.value === 'string' && + set(declTxHashAtom, newValue?.value) + break + case 'deployTxHash': + typeof newValue?.value === 'string' && + set(deployTxHashAtom, newValue?.value) + break + } } -}, (_get, set, newValue: SetDeploymentAtom) => { - switch (newValue?.key) { - case 'isDeploying': typeof newValue?.value === 'boolean' && set(isDeployingAtom, newValue?.value); break - case 'deployStatus': typeof newValue?.value === 'string' && set(deployStatusAtom, newValue?.value); break - case 'isDeclaring': typeof newValue?.value === 'boolean' && set(isDelcaringAtom, newValue?.value); break - case 'declStatus': typeof newValue?.value === 'string' && set(declStatusAtom, newValue?.value); break - case 'constructorCalldata': typeof newValue?.value === 'object' && !Array.isArray(newValue?.value) && set(constructorCalldataAtom, newValue?.value); break - case 'constructorInputs': Array.isArray(newValue?.value) && set(constructorInputsAtom, newValue?.value); break - case 'notEnoughInputs': typeof newValue?.value === 'boolean' && set(notEnoughInputsAtom, newValue?.value); break - case 'declTxHash': typeof newValue?.value === 'string' && set(declTxHashAtom, newValue?.value); break - case 'deployTxHash': typeof newValue?.value === 'string' && set(deployTxHashAtom, newValue?.value); break - } -}) +) export { isDeployingAtom, diff --git a/plugin/src/features/Deployment/index.tsx b/plugin/src/features/Deployment/index.tsx index f24f83d1..79575763 100644 --- a/plugin/src/features/Deployment/index.tsx +++ b/plugin/src/features/Deployment/index.tsx @@ -1,16 +1,13 @@ import React, { useEffect, useState } from 'react' - import { type BigNumberish } from 'ethers' +import { constants } from 'starknet' +import { useAtom, useAtomValue, useSetAtom } from 'jotai' import CompiledContracts from '../../components/CompiledContracts' -import { - type Contract -} from '../../utils/types/contracts' +import { type Contract } from '../../utils/types/contracts' import { getConstructor, getShortenedHash } from '../../utils/utils' import Container from '../../components/ui_components/Container' import { type AccordianTabs } from '../Plugin' -import { constants } from 'starknet' -import { useAtom, useAtomValue, useSetAtom } from 'jotai' import transactionsAtom from '../../atoms/transactions' import './styles.css' @@ -36,6 +33,7 @@ import { import { useWaitForTransaction } from '@starknet-react/core' import { type CallbackReturnType, ConstructorForm } from 'starknet-abi-forms' import { useIcon } from '../../hooks/useIcons' +import { DeclareStatusLabels } from '../../utils/constants' interface DeploymentProps { setActiveTab: (tab: AccordianTabs) => void } @@ -78,7 +76,10 @@ const Deployment: React.FC = ({ setActiveTab }) => { const env = useAtomValue(envAtom) const declTxStatus = useWaitForTransaction({ hash: declTxHash, watch: true }) - const deployTxStatus = useWaitForTransaction({ hash: deployTxHash, watch: true }) + const deployTxStatus = useWaitForTransaction({ + hash: deployTxHash, + watch: true + }) const [chainId, setChainId] = useState( constants.StarknetChainId.SN_GOERLI @@ -108,52 +109,68 @@ const Deployment: React.FC = ({ setActiveTab }) => { console.log('declTxHash', declTxHash, declTxStatus.status, env) if (declTxHash === '') { setIsDeclaring(false) - setDeclStatus('') + setDeclStatus('IDLE') return } if (env !== 'wallet') return if (declTxStatus.status === 'success') { - setDeclStatus('done') + setDeclStatus('DONE') setIsDeclaring(false) remixClient.emit('statusChanged', { key: 'succeed', type: 'success', title: `Contract ${selectedContract?.name ?? ''} declared!` }) - remixClient.call('terminal', 'log', { - value: JSON.stringify(declTxStatus.data, null, 2), - type: 'info' - }).catch(() => {}) + remixClient + .call('terminal', 'log', { + value: JSON.stringify(declTxStatus.data, null, 2), + type: 'info' + }) + .catch(() => {}) - remixClient.call('terminal', 'log', { - value: `--------------------- End getting declare contract: ${selectedContract?.name ?? ''} tx receipt ------------------`, - type: 'info' - }).catch(() => {}) + remixClient + .call('terminal', 'log', { + value: `--------------------- End getting declare contract: ${ + selectedContract?.name ?? '' + } tx receipt ------------------`, + type: 'info' + }) + .catch(() => {}) } if (declTxStatus.status === 'error') { - setDeclStatus('error') + setDeclStatus('ERROR') remixClient.emit('statusChanged', { key: 'failed', type: 'error', title: 'Declaration failed' }) - remixClient.call('terminal', 'log', { - value: JSON.stringify(declTxStatus, null, 2), - type: 'info' - }).catch(() => {}) + remixClient + .call('terminal', 'log', { + value: JSON.stringify(declTxStatus, null, 2), + type: 'info' + }) + .catch(() => {}) - remixClient.call('terminal', 'log', { - value: `--------------------- End getting declare contract: ${selectedContract?.name ?? ''} tx receipt ------------------`, - type: 'info' - }).catch(() => {}) + remixClient + .call('terminal', 'log', { + value: `--------------------- End getting declare contract: ${ + selectedContract?.name ?? '' + } tx receipt ------------------`, + type: 'info' + }) + .catch(() => {}) setIsDeclaring(false) } if (declTxStatus.status === 'pending') { if (isDeclaring) { - remixClient.call('terminal', 'log', { - value: `--------------------- Getting declare contract: ${selectedContract?.name ?? ''} tx receipt --------------------`, - type: 'info' - }).catch(() => {}) + remixClient + .call('terminal', 'log', { + value: `--------------------- Getting declare contract: ${ + selectedContract?.name ?? '' + } tx receipt --------------------`, + type: 'info' + }) + .catch(() => {}) } } }, [declTxHash, declTxStatus.status]) @@ -162,67 +179,76 @@ const Deployment: React.FC = ({ setActiveTab }) => { console.log('deployTxHash', deployTxHash, deployTxStatus.status, env) if (deployTxHash === '') { setIsDeploying(false) - setDeployStatus('') + setDeployStatus('IDLE') return } if (env !== 'wallet') return if (deployTxStatus.status === 'success') { - setDeployStatus('done') + setDeployStatus('DONE') setIsDeploying(false) remixClient.emit('statusChanged', { key: 'succeed', type: 'success', title: `Contract ${selectedContract?.name ?? ''} deployed!` }) - remixClient.call('terminal', 'log', { - value: JSON.stringify(deployTxStatus.data, null, 2), - type: 'info' - }).catch(() => {}) + remixClient + .call('terminal', 'log', { + value: JSON.stringify(deployTxStatus.data, null, 2), + type: 'info' + }) + .catch(() => {}) - remixClient.call('terminal', 'log', { - value: `--------------------- End getting deploy contract: ${selectedContract?.name ?? ''} tx receipt ------------------`, - type: 'info' - }).catch(() => {}) - setActiveTab('interaction') + remixClient + .call('terminal', 'log', { + value: `--------------------- End getting deploy contract: ${ + selectedContract?.name ?? '' + } tx receipt ------------------`, + type: 'info' + }) + .catch(() => {}) } if (deployTxStatus.status === 'error') { - setDeployStatus('error') + setDeployStatus('ERROR') remixClient.emit('statusChanged', { key: 'failed', type: 'error', title: 'Deployment failed' }) - remixClient.call('terminal', 'log', { - value: JSON.stringify(deployTxStatus, null, 2), - type: 'info' - }).catch(() => {}) + remixClient + .call('terminal', 'log', { + value: JSON.stringify(deployTxStatus, null, 2), + type: 'info' + }) + .catch(() => {}) - remixClient.call('terminal', 'log', { - value: `--------------------- End getting deploy contract: ${selectedContract?.name ?? ''} tx receipt ------------------`, - type: 'info' - }).catch(() => {}) + remixClient + .call('terminal', 'log', { + value: `--------------------- End getting deploy contract: ${ + selectedContract?.name ?? '' + } tx receipt ------------------`, + type: 'info' + }) + .catch(() => {}) setIsDeploying(false) } if (deployTxStatus.status === 'pending') { if (isDeploying) { - remixClient.call('terminal', 'log', { - value: `--------------------- Getting deploy contract: ${selectedContract?.name ?? ''} tx receipt --------------------`, - type: 'info' - }).catch(() => {}) + remixClient + .call('terminal', 'log', { + value: `--------------------- Getting deploy contract: ${ + selectedContract?.name ?? '' + } tx receipt --------------------`, + type: 'info' + }) + .catch(() => {}) } } }, [deployTxHash, deployTxStatus.status]) const declareContract = async (): Promise => { + setDeclStatus('IDLE') setIsDeclaring(true) - remixClient.emit('statusChanged', { - key: 'loading', - type: 'info', - title: `Declaring ${selectedContract?.name ?? ''} ...` - }) - let updatedTransactions = transactions - try { if (account === null || provider === null) { throw new Error('No account or provider selected!') @@ -231,7 +257,12 @@ const Deployment: React.FC = ({ setActiveTab }) => { if (selectedContract === null) { throw new Error('No contract selected for deployment!') } - setDeclStatus('Declaring...') + remixClient.emit('statusChanged', { + key: 'loading', + type: 'info', + title: `Declaring ${selectedContract?.name ?? ''} ...` + }) + setDeclStatus('IN_PROGRESS') try { try { await account.getClassByHash(selectedContract.classHash) @@ -245,7 +276,7 @@ const Deployment: React.FC = ({ setActiveTab }) => { )} already has been declared, you can proceed to deployment` ) setIsDeclaring(false) - setDeclStatus('done') + setDeclStatus('DONE') remixClient.emit('statusChanged', { key: 'succeed', type: 'success', @@ -256,11 +287,14 @@ const Deployment: React.FC = ({ setActiveTab }) => { value: `------------------------ Declaring contract: ${selectedContract.name} ------------------------`, type: 'info' }) - const declareResponse = await account.declare({ - contract: selectedContract.sierra, - classHash: selectedContract.classHash, - compiledClassHash: selectedContract.compiledClassHash - }, { maxFee: 1e18 }) + const declareResponse = await account.declare( + { + contract: selectedContract.sierra, + classHash: selectedContract.classHash, + compiledClassHash: selectedContract.compiledClassHash + }, + { maxFee: 1e18 } + ) await remixClient.call('terminal', 'log', { value: JSON.stringify(declareResponse, null, 2), type: 'info' @@ -287,7 +321,9 @@ const Deployment: React.FC = ({ setActiveTab }) => { value: `--------------------- Getting declare contract: ${selectedContract.name} tx receipt --------------------`, type: 'info' }) - const txReceipt = await account.waitForTransaction(declareResponse.transaction_hash) + const txReceipt = await account.waitForTransaction( + declareResponse.transaction_hash + ) await remixClient.call('terminal', 'log', { value: JSON.stringify(txReceipt, null, 2), type: 'info' @@ -296,13 +332,13 @@ const Deployment: React.FC = ({ setActiveTab }) => { value: `--------------------- End getting declare contract: ${selectedContract.name} tx receipt ------------------`, type: 'info' }) - setDeclStatus('done') + setDeclStatus('DONE') setIsDeclaring(false) } } setContractDeclaration(selectedContract) } catch (error) { - setDeclStatus('error') + setDeclStatus('ERROR') setIsDeclaring(false) if (error instanceof Error) { await remixClient.call('terminal', 'log', { @@ -317,7 +353,7 @@ const Deployment: React.FC = ({ setActiveTab }) => { } } } catch (error) { - setDeclStatus('error') + setDeclStatus('ERROR') setIsDeclaring(false) if (error instanceof Error) { await remixClient.call('terminal', 'log', { @@ -334,12 +370,8 @@ const Deployment: React.FC = ({ setActiveTab }) => { } const deploy = async (calldata: BigNumberish[]): Promise => { + setDeployStatus('IDLE') setIsDeploying(true) - remixClient.emit('statusChanged', { - key: 'loading', - type: 'info', - title: `Deploying ${selectedContract?.name ?? ''} ...` - }) const classHash = selectedContract?.classHash const updatedTransactions = transactions try { @@ -350,18 +382,25 @@ const Deployment: React.FC = ({ setActiveTab }) => { if (selectedContract === null) { throw new Error('No contract selected for deployment!') } + remixClient.emit('statusChanged', { + key: 'loading', + type: 'info', + title: `Deploying ${selectedContract?.name ?? ''} ...` + }) - setDeployStatus('Deploying...') - + setDeployStatus('IN_PROGRESS') await remixClient.call('terminal', 'log', { value: `------------------------ Deploying contract: ${selectedContract.name} ------------------------`, type: 'info' }) - const deployResponse = await account.deploy({ - classHash: classHash ?? selectedContract.classHash, - constructorCalldata: calldata - }, { maxFee: 1e18 }) + const deployResponse = await account.deploy( + { + classHash: classHash ?? selectedContract.classHash, + constructorCalldata: calldata + }, + { maxFee: 1e18 } + ) await remixClient.call('terminal', 'log', { value: JSON.stringify(deployResponse, null, 2), @@ -385,14 +424,16 @@ const Deployment: React.FC = ({ setActiveTab }) => { ]) if (env === 'wallet') setDeployTxHash(deployResponse.transaction_hash) else { - setDeployStatus('done') + setDeployStatus('DONE') setIsDeploying(false) await remixClient.call('terminal', 'log', { value: `--------------------- Getting deploy contract: ${selectedContract.name} tx receipt --------------------`, type: 'info' }) - const txReceipt = await account.waitForTransaction(deployResponse.transaction_hash) + const txReceipt = await account.waitForTransaction( + deployResponse.transaction_hash + ) await remixClient.call('terminal', 'log', { value: JSON.stringify(txReceipt, null, 2), type: 'info' @@ -404,9 +445,12 @@ const Deployment: React.FC = ({ setActiveTab }) => { }) setActiveTab('interaction') } - setContractDeployment(selectedContract, deployResponse.contract_address[0]) + setContractDeployment( + selectedContract, + deployResponse.contract_address[0] + ) } catch (error) { - setDeployStatus('error') + setDeployStatus('ERROR') setIsDeploying(false) if (error instanceof Error) { await remixClient.call('terminal', 'log', { @@ -443,10 +487,7 @@ const Deployment: React.FC = ({ setActiveTab }) => { if (account == null) return const declaredContract = { ...currentContract, - declaredInfo: [ - ...currentContract.declaredInfo, - { chainId, env } - ] + declaredInfo: [...currentContract.declaredInfo, { chainId, env }] } const updatedContracts = contracts.map((contract) => { if (contract.classHash === declaredContract.classHash) { @@ -470,7 +511,7 @@ const Deployment: React.FC = ({ setActiveTab }) => { deployedInfo: [ ...currentContract.deployedInfo, { address: account.address, chainId } - ].flatMap(deployment => { + ].flatMap((deployment) => { const key = `${deployment.address}-${deployment.chainId}` if (deployedInfoMap.has(key)) { return [] @@ -496,89 +537,81 @@ const Deployment: React.FC = ({ setActiveTab }) => { {contracts.length > 0 && selectedContract != null ? (
-
+
- {'deploy-icon'}/ + {'deploy-icon'}
Deploy your selected contract
- +
+ + {account != null && selectedContract.deployedInfo.some( (info) => @@ -607,16 +640,19 @@ const Deployment: React.FC = ({ setActiveTab }) => {
) : ( -
-
- {'deploy-icon'}/ -
-

No contracts ready for deployment yet, compile a cairo contract

-
+ )} ) } +const NoContractReadyView = (): JSX.Element => ( +
+
+ {'deploy-icon'} +
+

No contracts ready for deployment yet, compile a cairo contract

+
+) export default Deployment diff --git a/plugin/src/features/Plugin/index.tsx b/plugin/src/features/Plugin/index.tsx index 1f76bd01..7d5dabe3 100644 --- a/plugin/src/features/Plugin/index.tsx +++ b/plugin/src/features/Plugin/index.tsx @@ -15,9 +15,7 @@ import TransactionHistory from '../TransactionHistory' import Footer from '../Footer' import StateAction from '../../components/StateAction' import BackgroundNotices from '../../components/BackgroundNotices' -import { - useCurrentExplorer -} from '../../components/ExplorerSelector' +import { useCurrentExplorer } from '../../components/ExplorerSelector' import { useAtomValue, useSetAtom, useAtom } from 'jotai' import { isCompilingAtom, statusAtom } from '../../atoms/compilation' import { deploymentAtom, isDelcaringAtom } from '../../atoms/deployment' @@ -40,17 +38,16 @@ const Plugin: React.FC = () => { const isCompiling = useAtomValue(isCompilingAtom) const status = useAtomValue(statusAtom) - const { - isDeploying, - deployStatus - } = useAtomValue(deploymentAtom) + const { isDeploying, deployStatus } = useAtomValue(deploymentAtom) const isDeclaring = useAtomValue(isDelcaringAtom) const isDeclaringOrDeploying = isDeploying || isDeclaring // Interaction state variables - const [interactionStatus, setInteractionStatus] = useState<'loading' | 'success' | 'error' | ''>('') + const [interactionStatus, setInteractionStatus] = useState< + 'loading' | 'success' | 'error' | '' + >('') const [currentAccordian, setCurrentAccordian] = useState('compile') @@ -69,30 +66,36 @@ const Plugin: React.FC = () => { const { remixClient } = useRemixClient() const envViteVersion: string | undefined = import.meta.env.VITE_VERSION - const pluginVersion = envViteVersion !== undefined ? `v${envViteVersion}` : 'v0.2.5' + const pluginVersion = + envViteVersion !== undefined ? `v${envViteVersion}` : 'v0.2.5' useEffect(() => { const fetchCairoVersions = async (): Promise => { try { if (apiUrl !== undefined) { - const response = await fetch( - `${apiUrl}/cairo_versions`, - { - method: 'GET', - headers: { - 'Content-Type': 'application/octet-stream' - }, - redirect: 'follow' - } - ) + const response = await fetch(`${apiUrl}/cairo_versions`, { + method: 'GET', + headers: { + 'Content-Type': 'application/octet-stream' + }, + redirect: 'follow' + }) const versions = JSON.parse(await response.text()) versions.sort() setVersions(versions) } } catch (e) { - await remixClient.call('notification' as any, 'toast', '🔴 Failed to fetch cairo versions from the compilation server') + await remixClient.call( + 'notification' as any, + 'toast', + '🔴 Failed to fetch cairo versions from the compilation server' + ) console.error(e) - await remixClient.terminal.log(`🔴 Failed to fetch cairo versions from the compilation server ${e as string}` as any) + await remixClient.terminal.log( + `🔴 Failed to fetch cairo versions from the compilation server ${ + e as string + }` as any + ) } } @@ -104,7 +107,9 @@ const Plugin: React.FC = () => { setCairoVersion(getVersions[getVersions.length - 1]) } } - fetchCairo().catch(e => { console.error(e) }) + fetchCairo().catch((e) => { + console.error(e) + }) }, 10000) }, [remixClient]) @@ -194,15 +199,11 @@ const Plugin: React.FC = () => { return ( -
-
+
+
-
- ALPHA -
-
- Using {pluginVersion} -
+
ALPHA
+
Using {pluginVersion}
@@ -218,115 +219,123 @@ const Plugin: React.FC = () => { {/* apply styles */}
- Home - Transactions - Info - Settings + + Home + + + Transactions + + + Info + + + Settings +
- + - + { - handleTabView('compile') - }} + onClick={() => { + handleTabView('compile') + }} > - - 1 -

Compile

- + 1 +

Compile

+ -
+ /> +
- + { - handleTabView('deploy') - }} + onClick={() => { + handleTabView('deploy') + }} > - - 2 -

Deploy

- + 2 +

Deploy

+ -
+ /> +
- + { - handleTabView('interaction') - }} - > + onClick={() => { + handleTabView('interaction') + }} + > 3

Interact

- +
-
- - - -
-
-
- - - - - - - - - - - - - -
-
-
+ + + + + + + + + + + + + + + + + + + + +
+
+
) diff --git a/plugin/src/utils/constants.ts b/plugin/src/utils/constants.ts index e0ae9003..842b9bed 100644 --- a/plugin/src/utils/constants.ts +++ b/plugin/src/utils/constants.ts @@ -1,11 +1,9 @@ import { constants } from 'starknet' +import { type Status } from '../atoms/deployment' const devnetUrl = 'http://127.0.0.1:5050' -type Network = - | 'goerli' - | 'sepolia' - | 'mainnet' +type Network = 'goerli' | 'sepolia' | 'mainnet' const networks = [ { name: 'Testnet', value: 'goerli' }, @@ -82,3 +80,17 @@ export { } export type { Network } + +export const DeclareStatusLabels: Record = { + IDLE: 'Idle', + IN_PROGRESS: 'Declaring...', + ERROR: 'Error', + DONE: 'Done' +} + +export const DeployStatusLabels: Record = { + IDLE: 'Idle', + IN_PROGRESS: 'Deploying...', + ERROR: 'Error', + DONE: 'Done' +}