diff --git a/apps/contract-verification/src/profile.json b/apps/contract-verification/src/profile.json index f56f684c71..1cc07d5b72 100644 --- a/apps/contract-verification/src/profile.json +++ b/apps/contract-verification/src/profile.json @@ -6,7 +6,7 @@ "events": [], "methods": [], "kind": "none", - "icon": "", + "icon": "", "location": "sidePanel", "url": "", "repo": "https://github.com/ethereum/remix-project/tree/master/apps/contract-verification", diff --git a/apps/remix-ide-e2e/src/commands/addAtAddressInstance.ts b/apps/remix-ide-e2e/src/commands/addAtAddressInstance.ts index 48270c6b9b..2ea64cd43b 100644 --- a/apps/remix-ide-e2e/src/commands/addAtAddressInstance.ts +++ b/apps/remix-ide-e2e/src/commands/addAtAddressInstance.ts @@ -19,11 +19,11 @@ function addInstance (browser: NightwatchBrowser, address: string, isValidFormat .waitForElementVisible('.ataddressinput') .click('.ataddressinput') .setValue('.ataddressinput', address, function () { - if (!isValidFormat || !isValidChecksum) browser.assert.elementPresent('button[id^="runAndDeployAtAdressButton"]:disabled') + if (!isValidFormat || !isValidChecksum) browser.assert.elementPresent('button[id^="runAndDeployAtAddressButton"]:disabled') else if (isAbi) { browser .click({ - selector: '//*[@id="runAndDeployAtAdressButtonContainer"]', + selector: '//*[@id="runAndDeployAtAddressButtonContainer"]', locateStrategy: 'xpath' }) .waitForElementPresent('[data-id="udappNotify-modal-footer-ok-react"]', 5000) @@ -34,7 +34,7 @@ function addInstance (browser: NightwatchBrowser, address: string, isValidFormat }) } else { browser.click({ - selector: '//*[@id="runAndDeployAtAdressButtonContainer"]', + selector: '//*[@id="runAndDeployAtAddressButtonContainer"]', locateStrategy: 'xpath' }) } diff --git a/apps/remix-ide/src/app/tabs/locales/en/home.json b/apps/remix-ide/src/app/tabs/locales/en/home.json index a638fd2b02..e4e5554eb4 100644 --- a/apps/remix-ide/src/app/tabs/locales/en/home.json +++ b/apps/remix-ide/src/app/tabs/locales/en/home.json @@ -26,6 +26,7 @@ "home.solhintPluginDesc": "Solhint is an open source project for linting Solidity code.", "home.sourcifyPluginDesc": "Solidity contract and metadata verification service.", "home.unitTestPluginDesc": "Write and run unit tests for your contracts in Solidity.", + "home.contractVerificationDesc": "Verify contracts on multiple services at the same time.", "home.dgitPluginDesc": "Add source control to your projects.", "home.oneClickDappDesc": "Quickly generate smart contract interfaces", "home.getStarted": "Get Started", diff --git a/apps/remix-ide/src/assets/img/contractVerification.webp b/apps/remix-ide/src/assets/img/contractVerification.webp new file mode 100644 index 0000000000..6d7dd5ae6d Binary files /dev/null and b/apps/remix-ide/src/assets/img/contractVerification.webp differ diff --git a/apps/remix-ide/team-best-practices.md b/apps/remix-ide/team-best-practices.md index 2088cd5bf9..477f1304a0 100644 --- a/apps/remix-ide/team-best-practices.md +++ b/apps/remix-ide/team-best-practices.md @@ -158,7 +158,7 @@ Before starting to coding, we should ensure all devs / contributors are aware of ### 1) Bugs: - A critical bug should get the label `Blocker`, and every effort should be put to fix it. - Addressing a non critical and non planned bug can be done: - - After having notified in the `remix-dev` channel if the bug does not involves UX or public API changes. + - After having notified in the `remix-dev` channel if the bug does not involve UX or public API changes. - After a dev meeting (e.g the regular standup) if the bug involves any UX or public API changes. ### 2) Support: diff --git a/libs/remix-lib/src/execution/txRunnerVM.ts b/libs/remix-lib/src/execution/txRunnerVM.ts index 6c4c40b27b..12bfd3636e 100644 --- a/libs/remix-lib/src/execution/txRunnerVM.ts +++ b/libs/remix-lib/src/execution/txRunnerVM.ts @@ -75,6 +75,33 @@ export class TxRunnerVM { } } + runEmptyBlock (callback: VMExecutionCallBack) { + const EIP1559 = this.commonContext.hardfork() !== 'berlin' // berlin is the only pre eip1559 fork that we handle. + const coinbases = ['0x0e9281e9c6a0808672eaba6bd1220e144c9bb07a', '0x8945a1288dc78a6d8952a92c77aee6730b414778', '0x94d76e24f818426ae84aa404140e8d5f60e10e7e'] + const difficulties = [69762765929000, 70762765929000, 71762765929000] + const difficulty = this.commonContext.consensusType() === ConsensusType.ProofOfStake ? 0 : difficulties[this.blocks.length % difficulties.length] + const block = Block.fromBlockData({ + header: { + timestamp: new Date().getTime() / 1000 | 0, + number: this.blocks.length, + coinbase: coinbases[this.blocks.length % coinbases.length], + difficulty, + gasLimit: 0, + baseFeePerGas: EIP1559 ? '0x1' : undefined, + parentHash: this.blockParentHash + } + }, { common: this.commonContext }) + + this.blockParentHash = block.hash() + this.runBlockInVm(null, block, async (err, result) => { + if (!err) { + this.getVMObject().vm.blockchain.putBlock(block) + this.blocks.push(block.serialize()) + } + callback(err, result) + }) + } + async runInVm (tx: InternalTransaction, callback: VMExecutionCallBack) { const { to, data, value, gasLimit, useCall, signed } = tx let { from } = tx @@ -183,7 +210,7 @@ export class TxRunnerVM { const result: RunTxResult = results.results[0] callback(null, { result, - transactionHash: bytesToHex(Buffer.from(tx.hash())), + transactionHash: tx ? bytesToHex(Buffer.from(tx.hash())) : null, block, tx }) diff --git a/libs/remix-simulator/README.md b/libs/remix-simulator/README.md index a7921c7c9e..4c3717fefc 100644 --- a/libs/remix-simulator/README.md +++ b/libs/remix-simulator/README.md @@ -39,7 +39,7 @@ * [X] eth_getCode * [~] eth_sign * [X] eth_sendTransaction -* [_] eth_sendRawTransaction +* [x] eth_sendRawTransaction * [X] eth_call * [~] eth_estimateGas * [X] eth_getBlockByHash diff --git a/libs/remix-simulator/src/methods/accounts.ts b/libs/remix-simulator/src/methods/accounts.ts index c28818c8e6..c35e7f46a6 100644 --- a/libs/remix-simulator/src/methods/accounts.ts +++ b/libs/remix-simulator/src/methods/accounts.ts @@ -1,7 +1,7 @@ import { signTypedData, SignTypedDataVersion, TypedMessage, MessageTypes } from '@metamask/eth-sig-util' import { privateToAddress, toChecksumAddress, isValidPrivate, Address, toBytes, bytesToHex, Account } from '@ethereumjs/util' import { privateKeyToAccount } from 'web3-eth-accounts' -import { toBigInt } from 'web3-utils' +import { toBigInt, toHex } from 'web3-utils' import * as crypto from 'crypto' type AccountType = { @@ -13,9 +13,11 @@ export class Web3Accounts { accounts: Record accountsKeys: Record vmContext + options - constructor (vmContext) { + constructor (vmContext, options) { this.vmContext = vmContext + this.options = options // TODO: make it random and/or use remix-libs this.accounts = {} @@ -97,6 +99,8 @@ export class Web3Accounts { eth_getBalance (payload, cb) { const address = payload.params[0] this.vmContext.vm().stateManager.getAccount(Address.fromString(address)).then((account) => { + if (!account) return cb(null, toBigInt(0).toString(10)) + if (!account.balance) return cb(null, toBigInt(0).toString(10)) cb(null, toBigInt(account.balance).toString(10)) }).catch((error) => { cb(error) @@ -119,7 +123,9 @@ export class Web3Accounts { } eth_chainId (_payload, cb) { - return cb(null, '0x539') // 0x539 is hex of 1337 + if (!this.options.chainId) return cb(null, '0x539') // 0x539 is hex of 1337 + const id = (typeof this.options.chainId === 'number') ? toHex(this.options.chainId) : this.options.chainId + return cb(null, id) } eth_signTypedData_v4 (payload, cb) { diff --git a/libs/remix-simulator/src/methods/evm.ts b/libs/remix-simulator/src/methods/evm.ts new file mode 100644 index 0000000000..16fde3c180 --- /dev/null +++ b/libs/remix-simulator/src/methods/evm.ts @@ -0,0 +1,79 @@ +import { Block } from '@ethereumjs/block' +import { ConsensusType } from '@ethereumjs/common' +import type { VMContext } from '../vm-context' +import type { Transactions } from '../methods/transactions' + +export class EVM { + vmContext: VMContext + transactions: Transactions + + constructor (vmContext: VMContext, transactions: Transactions) { + this.vmContext = vmContext + this.transactions = transactions + } + + methods () { + return { + evm_setAutomine: this.evm_setAutomine.bind(this), + evm_setIntervalMining: this.evm_setIntervalMining.bind(this), + evm_snapshot: this.evm_snapshot.bind(this), + evm_revert: this.evm_revert.bind(this), + evm_increaseTime: this.evm_increaseTime.bind(this), + evm_setNextBlockTimestamp: this.evm_setNextBlockTimestamp.bind(this), + evm_setBlockGasLimit: this.evm_setBlockGasLimit.bind(this), + evm_mine: this.evm_mine.bind(this) + } + } + + evm_setAutomine (payload, cb) { + // always on + cb() + } + + evm_setIntervalMining (payload, cb) { + cb() + } + + evm_snapshot (payload, cb) { + cb() + } + + evm_revert (payload, cb) { + cb() + } + + evm_increaseTime (payload, cb) { + cb() + } + + evm_setNextBlockTimestamp (payload, cb) { + cb() + } + + evm_setBlockGasLimit (payload, cb) { + cb() + } + + async evm_mine (payload, cb) { + const runEmptyBlock = () => { + return new Promise((resolve, reject) => { + this.transactions.txRunnerVMInstance.runEmptyBlock((error, result) => { + if (error) { + reject(error) + return + } + this.vmContext.addBlock(result.block, false, true) + resolve(result) + }) + }) + } + + const blocks = payload.params[0].blocks + + for (let b = 0; b < Number(blocks); b++) { + await runEmptyBlock() + console.log('mining...', b, this.vmContext.latestBlockNumber) + } + cb() + } +} diff --git a/libs/remix-simulator/src/methods/miner.ts b/libs/remix-simulator/src/methods/miner.ts new file mode 100644 index 0000000000..b44058f5a7 --- /dev/null +++ b/libs/remix-simulator/src/methods/miner.ts @@ -0,0 +1,18 @@ +export class Miner { + vmContext + + constructor (vmContext) { + this.vmContext = vmContext + } + + methods () { + return { + miner_start: this.miner_start.bind(this), + miner_stop: this.miner_stop.bind(this) + } + } + + miner_start (payload, cb) { cb() } + + miner_stop (payload, cb) { cb() } +} diff --git a/libs/remix-simulator/src/methods/net.ts b/libs/remix-simulator/src/methods/net.ts index 1a064719f4..3c9c7a612c 100644 --- a/libs/remix-simulator/src/methods/net.ts +++ b/libs/remix-simulator/src/methods/net.ts @@ -1,20 +1,24 @@ -export function methods (): Record { - return { - net_version: net_version, - net_listening: net_listening, - net_peerCount: net_peerCount +export class Net { + vmContext + options + + constructor (vmContext, options) { + this.vmContext = vmContext + this.options = options } -} -export function net_version (payload, cb): void { - // should be configured networkId - cb(null, 1337) -} + methods () { + return { + net_version: this.net_version.bind(this), + net_listening: this.net_listening.bind(this), + net_peerCount: this.net_peerCount.bind(this) + } + } -export function net_listening (payload, cb): void { - cb(null, true) -} + net_version (payload, cb) { cb(null, 1337) } + + net_listening (payload, cb) { cb(null, true)} -export function net_peerCount (payload, cb): void { - cb(null, 0) + net_peerCount (payload, cb) { cb(null, 0)} } + diff --git a/libs/remix-simulator/src/provider.ts b/libs/remix-simulator/src/provider.ts index b6c52345fe..8fae322c6b 100644 --- a/libs/remix-simulator/src/provider.ts +++ b/libs/remix-simulator/src/provider.ts @@ -6,9 +6,11 @@ import merge from 'merge' import { Web3Accounts } from './methods/accounts' import { Filters } from './methods/filters' import { methods as miscMethods } from './methods/misc' -import { methods as netMethods } from './methods/net' +import { Net } from './methods/net' import { Transactions } from './methods/transactions' +import { Miner } from './methods/miner' import { Debug } from './methods/debug' +import { EVM } from './methods/evm' import { VMContext } from './vm-context' import { Web3PluginBase } from 'web3' @@ -30,6 +32,7 @@ export type JSONRPCResponseCallback = (err: Error, result?: JSONRPCResponsePaylo export type State = Record export type ProviderOptions = { + chainId?: number fork?: string, nodeUrl?: string, blockNumber?: number | 'latest', @@ -47,14 +50,16 @@ export class Provider { methods connected: boolean initialized: boolean + initializing: boolean pendingRequests: Array constructor (options: ProviderOptions = {} as ProviderOptions) { + console.log(options) this.options = options this.connected = true this.vmContext = new VMContext(options['fork'], options['nodeUrl'], options['blockNumber'], options['stateDb'], options['blocks']) - this.Accounts = new Web3Accounts(this.vmContext) + this.Accounts = new Web3Accounts(this.vmContext, options) this.Transactions = new Transactions(this.vmContext) this.methods = {} @@ -62,12 +67,15 @@ export class Provider { this.methods = merge(this.methods, (new Blocks(this.vmContext, options)).methods()) this.methods = merge(this.methods, miscMethods()) this.methods = merge(this.methods, (new Filters(this.vmContext)).methods()) - this.methods = merge(this.methods, netMethods()) + this.methods = merge(this.methods, (new Net(this.vmContext, options)).methods()) this.methods = merge(this.methods, this.Transactions.methods()) this.methods = merge(this.methods, (new Debug(this.vmContext)).methods()) + this.methods = merge(this.methods, (new EVM(this.vmContext, this.Transactions)).methods()) + this.methods = merge(this.methods, (new Miner(this.vmContext)).methods()) } async init () { + this.initializing = true this.initialized = false this.pendingRequests = [] await this.vmContext.init() @@ -80,6 +88,7 @@ export class Provider { }) this.pendingRequests = [] } + this.initializing = false } _send(payload: JSONRPCRequestPayload, callback: (err: Error, result?: JSONRPCResponsePayload) => void) { diff --git a/libs/remix-simulator/test/blocks.ts b/libs/remix-simulator/test/blocks.ts index e0eb39d6c9..0b4b2ad52c 100644 --- a/libs/remix-simulator/test/blocks.ts +++ b/libs/remix-simulator/test/blocks.ts @@ -63,6 +63,17 @@ describe('blocks', () => { }) }) + describe('evm_mine', () => { + it('should mine empty block using evm_mine', async function () { + await web3.provider.request({ + method: 'evm_mine', + params: [{ blocks: 3 }], + }) + const number = await web3.eth.getBlockNumber() + assert.equal(number, 3) + }) + }) + describe('eth_getBlockByHash', () => { it('should get block given its hash', async () => { const correctBlock = await web3.eth.getBlock(0) diff --git a/libs/remix-ui/home-tab/src/lib/components/homeTabFeaturedPlugins.tsx b/libs/remix-ui/home-tab/src/lib/components/homeTabFeaturedPlugins.tsx index 199a141ee7..eb01655998 100644 --- a/libs/remix-ui/home-tab/src/lib/components/homeTabFeaturedPlugins.tsx +++ b/libs/remix-ui/home-tab/src/lib/components/homeTabFeaturedPlugins.tsx @@ -82,6 +82,11 @@ function HomeTabFeaturedPlugins({ plugin }: HomeTabFeaturedPluginsProps) { plugin.verticalIcons.select('solidityUnitTesting') _paq.push(['trackEvent', 'hometabActivate', 'userActivate', 'solidityUnitTesting']) } + const startContractVerification = async () => { + await plugin.appManager.activatePlugin(['contract-verification']) + plugin.verticalIcons.select('contract-verification') + _paq.push(['trackEvent', 'hometabActivate', 'userActivate', 'contract-verification']) + } return (
@@ -116,14 +121,14 @@ function HomeTabFeaturedPlugins({ plugin }: HomeTabFeaturedPluginsProps) { itemClass="w-100" > startCodeAnalyzer()} + maintainedBy="Remix" + callback={() => startContractVerification()} /> startLearnEth()} /> + startCodeAnalyzer()} + /> { + const [is_streaming, setIS_streaming] = useState(false) + + const HandleCopyToClipboard = () => { + const markdown = document.getElementsByClassName('nlux-chatSegments-container') + if (markdown.length < 1) return + + const codeBlocks = markdown[0].getElementsByClassName('code-block') + Array.from(codeBlocks).forEach((block) => { + const copyButtons = block.getElementsByClassName('nlux-comp-copyButton') + Array.from(copyButtons).forEach((cp_btn) => { + const hdlr = async () => { + copy(block.textContent) + } + cp_btn.removeEventListener('click', async() => { hdlr() }) + cp_btn.addEventListener('click', async () => { hdlr() }) + }) + }) + } + + useEffect(() => { + HandleCopyToClipboard(); + }, [is_streaming]); + const send: StreamSend = async ( prompt: string, observer: StreamingAdapterObserver, ) => { GenerationParams.stream_result = true + setIS_streaming(true) GenerationParams.return_stream_response = GenerationParams.stream_result let response = null @@ -32,13 +56,15 @@ export const Default = (props) => { observer.next(' ') // Add a space to flush the last message ChatHistory.pushHistory(prompt, result) observer.complete() + setTimeout(() => { setIS_streaming(false) }, 1000) } ) else { observer.next(response) observer.complete() - } + setTimeout(() => { setIS_streaming(false) }, 1000) + } }; ChatApi = useAiChatApi(); const conversationStarters: ConversationStarter[] = [ @@ -73,9 +99,9 @@ export const Default = (props) => { submitShortcut: 'Enter', hideStopButton: false, }} - messageOptions={{ showCodeBlockCopyButton: false, + messageOptions={{ showCodeBlockCopyButton: true, editableUserMessages: true, - streamingAnimationSpeed: 2, + streamingAnimationSpeed: 1, waitTimeBeforeStreamCompletion: 1000, syntaxHighlighter: highlighter }} diff --git a/libs/remix-ui/run-tab/src/lib/actions/index.ts b/libs/remix-ui/run-tab/src/lib/actions/index.ts index a8c914b9c7..134b88c007 100644 --- a/libs/remix-ui/run-tab/src/lib/actions/index.ts +++ b/libs/remix-ui/run-tab/src/lib/actions/index.ts @@ -66,4 +66,4 @@ export const setNetworkName = (networkName: string) => setNetworkNameFromProvide export const updateSelectedContract = (contractName) => setSelectedContract(dispatch, contractName) export const syncContracts = () => syncContractsInternal(plugin) export const isValidProxyAddress = (address: string) => isValidContractAddress(plugin, address) -export const isValidProxyUpgrade = (proxyAddress: string, contractName: string, solcInput: SolcInput, solcOutput: SolcOutput, solcVersion: string) => isValidContractUpgrade(plugin, proxyAddress, contractName, solcInput, solcOutput, solcVersion) +export const isValidProxyUpgrade = (proxyAddress: string, contractName: string, solcInput: SolcInput, solcOutput: SolcOutput, solcVersion: string) => isValidContractUpgrade(plugin, proxyAddress, contractName, solcInput, solcOutput, solcVersion) \ No newline at end of file diff --git a/libs/remix-ui/run-tab/src/lib/components/contractDropdownUI.tsx b/libs/remix-ui/run-tab/src/lib/components/contractDropdownUI.tsx index b2a6573874..14ecd27699 100644 --- a/libs/remix-ui/run-tab/src/lib/components/contractDropdownUI.tsx +++ b/libs/remix-ui/run-tab/src/lib/components/contractDropdownUI.tsx @@ -517,10 +517,10 @@ export function ContractDropdownUI(props: ContractDropdownProps) {
-
+