Merge branch 'master' into addloops

pull/2857/head
bunsenstraat 2 years ago committed by GitHub
commit 57aa828ae2
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
  1. 20
      .github/workflows/run-sut.yml
  2. 9
      apps/remix-ide-e2e/src/tests/editorAutoComplete.test.ts
  3. 8
      apps/remix-ide-e2e/src/tests/editorHoverContext.test.ts
  4. 7
      apps/remix-ide-e2e/src/tests/editorReferences.test.ts
  5. 7
      apps/remix-ide-e2e/src/tests/editor_error_marker.test.ts
  6. 7
      apps/remix-ide-e2e/src/tests/editor_line_text.test.ts
  7. 15
      apps/remix-ide-e2e/src/tests/erc721.test.ts
  8. 4
      apps/remix-ide-e2e/src/tests/url.test.ts
  9. 8
      apps/remix-ide-e2e/src/tests/workspace.test.ts
  10. 151
      apps/remix-ide/contracts/ballot.sol
  11. 28
      apps/remix-ide/contracts/tests/Ballot_test.sol
  12. 4
      apps/remix-ide/src/app.js
  13. 52
      apps/remix-ide/src/app/editor/editor.js
  14. 14
      apps/remix-ide/src/app/files/fileManager.ts
  15. 30
      apps/remix-ide/src/app/plugins/parser/code-parser.tsx
  16. 2
      apps/remix-ide/src/assets/js/loader.js
  17. 3
      libs/remix-ui/editor/src/lib/remix-ui-editor.tsx
  18. 4
      libs/remix-ui/editor/src/lib/web-types.ts
  19. 19
      libs/remix-ui/settings/src/lib/remix-ui-settings.tsx
  20. 5
      libs/remix-ui/workspace/src/lib/actions/index.ts
  21. 3
      libs/remix-ws-templates/package.json
  22. 14
      libs/remix-ws-templates/src/templates/ozerc20/contracts/SampleERC20.sol
  23. 7
      libs/remix-ws-templates/src/templates/ozerc20/index.ts
  24. 2
      libs/remix-ws-templates/src/templates/ozerc20/scripts/deploy_with_ethers.ts
  25. 2
      libs/remix-ws-templates/src/templates/ozerc20/scripts/deploy_with_web3.ts
  26. 18
      libs/remix-ws-templates/src/templates/ozerc20/tests/MyToken_test.sol
  27. 18
      libs/remix-ws-templates/src/templates/ozerc20/tests/SampleERC20_test.sol
  28. 14
      libs/remix-ws-templates/src/templates/ozerc721/contracts/SampleERC721.sol
  29. 7
      libs/remix-ws-templates/src/templates/ozerc721/index.ts
  30. 2
      libs/remix-ws-templates/src/templates/ozerc721/scripts/deploy_with_ethers.ts
  31. 2
      libs/remix-ws-templates/src/templates/ozerc721/scripts/deploy_with_web3.ts
  32. 18
      libs/remix-ws-templates/src/templates/ozerc721/tests/MyToken_test.sol
  33. 18
      libs/remix-ws-templates/src/templates/ozerc721/tests/SampleERC721_test.sol
  34. 2
      package.json
  35. 12
      yarn.lock

@ -0,0 +1,20 @@
name: Running Solidity Unit Tests
on: [push]
jobs:
run_sol_contracts_job:
runs-on: ubuntu-latest
name: A job to run solidity unit tests on github actions CI
steps:
- name: Checkout
uses: actions/checkout@v2
- name: Environment Setup
uses: actions/setup-node@v3
with:
node-version: 14.17.6
- name: Run SUT Action
uses: EthereumRemix/sol-test@v1
with:
test-path: 'apps/remix-ide/contracts/tests'
compiler-version: '0.8.15'

@ -12,6 +12,15 @@ module.exports = {
before: function (browser: NightwatchBrowser, done: VoidFunction) { before: function (browser: NightwatchBrowser, done: VoidFunction) {
init(browser, done, 'http://127.0.0.1:8080', false) init(browser, done, 'http://127.0.0.1:8080', false)
}, },
'Should enable settings': function (browser: NightwatchBrowser) {
browser
.clickLaunchIcon('settings')
.click('[data-id="settingsAutoCompleteLabel"]')
.click('[data-id="settingsShowGasLabel"]')
.click('[data-id="displayErrorsLabel"]')
},
'Should add test and base files #group1': function (browser: NightwatchBrowser) { 'Should add test and base files #group1': function (browser: NightwatchBrowser) {
browser.addFile(examples.testContract.name, examples.testContract) browser.addFile(examples.testContract.name, examples.testContract)
.addFile(examples.baseContract.name, examples.baseContract) .addFile(examples.baseContract.name, examples.baseContract)

@ -18,6 +18,14 @@ module.exports = {
init(browser, done, 'http://127.0.0.1:8080', false) init(browser, done, 'http://127.0.0.1:8080', false)
}, },
'Should enable settings': function (browser: NightwatchBrowser) {
browser
.clickLaunchIcon('settings')
.click('[data-id="settingsAutoCompleteLabel"]')
.click('[data-id="settingsShowGasLabel"]')
.click('[data-id="displayErrorsLabel"]')
},
'Should load the test file': function (browser: NightwatchBrowser) { 'Should load the test file': function (browser: NightwatchBrowser) {
browser.openFile('contracts') browser.openFile('contracts')
.openFile('contracts/3_Ballot.sol') .openFile('contracts/3_Ballot.sol')

@ -20,6 +20,13 @@ module.exports = {
init(browser, done, 'http://127.0.0.1:8080', false) init(browser, done, 'http://127.0.0.1:8080', false)
}, },
'Should enable settings': function (browser: NightwatchBrowser) {
browser
.clickLaunchIcon('settings')
.click('[data-id="settingsAutoCompleteLabel"]')
.click('[data-id="settingsShowGasLabel"]')
.click('[data-id="displayErrorsLabel"]')
},
'Should load the test file': function (browser: NightwatchBrowser) { 'Should load the test file': function (browser: NightwatchBrowser) {
browser.openFile('contracts') browser.openFile('contracts')
.openFile('contracts/3_Ballot.sol') .openFile('contracts/3_Ballot.sol')

@ -8,6 +8,13 @@ module.exports = {
before: function (browser: NightwatchBrowser, done: VoidFunction) { before: function (browser: NightwatchBrowser, done: VoidFunction) {
init(browser, done, 'http://127.0.0.1:8080', true) init(browser, done, 'http://127.0.0.1:8080', true)
}, },
'Should enable settings': function (browser: NightwatchBrowser) {
browser
.clickLaunchIcon('settings')
.click('[data-id="settingsAutoCompleteLabel"]')
.click('[data-id="settingsShowGasLabel"]')
.click('[data-id="displayErrorsLabel"]')
},
'Should add error marker': function (browser: NightwatchBrowser) { 'Should add error marker': function (browser: NightwatchBrowser) {
browser browser
.openFile('contracts') .openFile('contracts')

@ -8,6 +8,13 @@ module.exports = {
before: function (browser: NightwatchBrowser, done: VoidFunction) { before: function (browser: NightwatchBrowser, done: VoidFunction) {
init(browser, done, 'http://127.0.0.1:8080', true) init(browser, done, 'http://127.0.0.1:8080', true)
}, },
'Should enable settings': function (browser: NightwatchBrowser) {
browser
.clickLaunchIcon('settings')
.click('[data-id="settingsAutoCompleteLabel"]')
.click('[data-id="settingsShowGasLabel"]')
.click('[data-id="displayErrorsLabel"]')
},
'Should add line texts': function (browser: NightwatchBrowser) { 'Should add line texts': function (browser: NightwatchBrowser) {
browser browser
.openFile('contracts') .openFile('contracts')

@ -26,20 +26,17 @@ module.exports = {
.execute(function () { (document.querySelector('[data-id="fileSystemModalDialogModalFooter-react"] .modal-ok') as HTMLElement).click() }) .execute(function () { (document.querySelector('[data-id="fileSystemModalDialogModalFooter-react"] .modal-ok') as HTMLElement).click() })
.pause(100) .pause(100)
.waitForElementVisible('*[data-id="treeViewLitreeViewItemcontracts"]') .waitForElementVisible('*[data-id="treeViewLitreeViewItemcontracts"]')
.waitForElementVisible('*[data-id="treeViewLitreeViewItemcontracts/SampleERC721.sol"]') .waitForElementVisible('*[data-id="treeViewLitreeViewItemcontracts/MyToken.sol"]')
.openFile('contracts/SampleERC721.sol') .openFile('contracts/MyToken.sol')
.verifyContracts(['SampleERC721']) .verifyContracts(['MyToken'])
// deploy contract // deploy contract
.clickLaunchIcon('udapp') .clickLaunchIcon('udapp')
.selectContract('SampleERC721') .selectContract('MyToken')
.createContract('E,E') .createContract('')
.testFunction('last', .testFunction('last',
{ {
status: 'true Transaction mined and execution succeed', status: 'true Transaction mined and execution succeed',
'decoded input': { 'decoded input': {}
'string tokenName': 'E',
'string tokenSymbol': 'E'
}
}).end() }).end()
} }
} }

@ -85,7 +85,7 @@ module.exports = {
}) })
}, },
'Should load Etherscan verified contractss from URL "address" param) #group2': function (browser: NightwatchBrowser) { 'Should load Etherscan verified contracts from URL "address" param)': !function (browser: NightwatchBrowser) {
browser browser
.pause(5000) .pause(5000)
.url('http://127.0.0.1:8080/#address=0x56db08fb78bc6689a1ef66efd079083fed0e4915') .url('http://127.0.0.1:8080/#address=0x56db08fb78bc6689a1ef66efd079083fed0e4915')
@ -248,7 +248,7 @@ module.exports = {
.openFile('contracts/governance/UnionGovernor.sol') .openFile('contracts/governance/UnionGovernor.sol')
}, },
'Should execute function call from URL parameters #group2': function (browser: NightwatchBrowser) { 'Should execute function call from URL parameters #group1': function (browser: NightwatchBrowser) {
browser browser
.switchWorkspace('default_workspace') .switchWorkspace('default_workspace')
.url('http://127.0.0.1:8080?calls=fileManager//open//contracts/3_Ballot.sol///terminal//log//log') .url('http://127.0.0.1:8080?calls=fileManager//open//contracts/3_Ballot.sol///terminal//log//log')

@ -124,7 +124,7 @@ module.exports = {
.execute(function () { (document.querySelector('[data-id="fileSystemModalDialogModalFooter-react"] .modal-ok') as HTMLElement).click() }) .execute(function () { (document.querySelector('[data-id="fileSystemModalDialogModalFooter-react"] .modal-ok') as HTMLElement).click() })
.pause(100) .pause(100)
.waitForElementVisible('*[data-id="treeViewLitreeViewItemcontracts"]') .waitForElementVisible('*[data-id="treeViewLitreeViewItemcontracts"]')
.waitForElementVisible('*[data-id="treeViewLitreeViewItemcontracts/SampleERC20.sol"]') .waitForElementVisible('*[data-id="treeViewLitreeViewItemcontracts/MyToken.sol"]')
.waitForElementVisible('*[data-id="treeViewLitreeViewItemscripts"]') .waitForElementVisible('*[data-id="treeViewLitreeViewItemscripts"]')
.waitForElementVisible('*[data-id="treeViewLitreeViewItemscripts/deploy_with_web3.ts"]') .waitForElementVisible('*[data-id="treeViewLitreeViewItemscripts/deploy_with_web3.ts"]')
// check js and ts files are not transformed // check js and ts files are not transformed
@ -156,7 +156,7 @@ module.exports = {
'Incorrect content') 'Incorrect content')
}) })
.waitForElementVisible('*[data-id="treeViewLitreeViewItemtests"]') .waitForElementVisible('*[data-id="treeViewLitreeViewItemtests"]')
.waitForElementVisible('*[data-id="treeViewLitreeViewItemtests/SampleERC20_test.sol"]') .waitForElementVisible('*[data-id="treeViewLitreeViewItemtests/MyToken_test.sol"]')
}, },
'Should create ERC721 workspace with files #group1': function (browser: NightwatchBrowser) { 'Should create ERC721 workspace with files #group1': function (browser: NightwatchBrowser) {
@ -172,7 +172,7 @@ module.exports = {
.execute(function () { (document.querySelector('[data-id="fileSystemModalDialogModalFooter-react"] .modal-ok') as HTMLElement).click() }) .execute(function () { (document.querySelector('[data-id="fileSystemModalDialogModalFooter-react"] .modal-ok') as HTMLElement).click() })
.pause(100) .pause(100)
.waitForElementVisible('*[data-id="treeViewLitreeViewItemcontracts"]') .waitForElementVisible('*[data-id="treeViewLitreeViewItemcontracts"]')
.waitForElementVisible('*[data-id="treeViewLitreeViewItemcontracts/SampleERC721.sol"]') .waitForElementVisible('*[data-id="treeViewLitreeViewItemcontracts/MyToken.sol"]')
.waitForElementVisible('*[data-id="treeViewLitreeViewItemscripts"]') .waitForElementVisible('*[data-id="treeViewLitreeViewItemscripts"]')
.waitForElementVisible('*[data-id="treeViewLitreeViewItemscripts/deploy_with_web3.ts"]') .waitForElementVisible('*[data-id="treeViewLitreeViewItemscripts/deploy_with_web3.ts"]')
// check js and ts files are not transformed // check js and ts files are not transformed
@ -204,7 +204,7 @@ module.exports = {
'Incorrect content') 'Incorrect content')
}) })
.waitForElementVisible('*[data-id="treeViewLitreeViewItemtests"]') .waitForElementVisible('*[data-id="treeViewLitreeViewItemtests"]')
.waitForElementVisible('*[data-id="treeViewLitreeViewItemtests/SampleERC721_test.sol"]') .waitForElementVisible('*[data-id="treeViewLitreeViewItemtests/MyToken_test.sol"]')
}, },
// WORKSPACE TEMPLATES E2E END // WORKSPACE TEMPLATES E2E END

@ -1,10 +1,13 @@
pragma solidity ^0.4.0; // SPDX-License-Identifier: GPL-3.0
/// @title Voting with delegation. pragma solidity >=0.7.0 <0.9.0;
/**
* @title Ballot
* @dev Implements voting process along with vote delegation
*/
contract Ballot { contract Ballot {
// This declares a new complex type which will
// be used for variables later.
// It will represent a single voter.
struct Voter { struct Voter {
uint weight; // weight is accumulated by delegation uint weight; // weight is accumulated by delegation
bool voted; // if true, that person already voted bool voted; // if true, that person already voted
@ -12,33 +15,31 @@ contract Ballot {
uint vote; // index of the voted proposal uint vote; // index of the voted proposal
} }
// This is a type for a single proposal.
struct Proposal { struct Proposal {
// If you can limit the length to a certain number of bytes,
// always use one of bytes1 to bytes32 because they are much cheaper
bytes32 name; // short name (up to 32 bytes) bytes32 name; // short name (up to 32 bytes)
uint voteCount; // number of accumulated votes uint voteCount; // number of accumulated votes
} }
address public chairperson; address public chairperson;
// This declares a state variable that
// stores a \`Voter\` struct for each possible address.
mapping(address => Voter) public voters; mapping(address => Voter) public voters;
// A dynamically-sized array of \`Proposal\` structs.
Proposal[] public proposals; Proposal[] public proposals;
/// Create a new ballot to choose one of \`proposalNames\`. /**
function Ballot(bytes32[] proposalNames) { * @dev Create a new ballot to choose one of 'proposalNames'.
* @param proposalNames names of proposals
*/
constructor(bytes32[] memory proposalNames) {
chairperson = msg.sender; chairperson = msg.sender;
voters[chairperson].weight = 1; voters[chairperson].weight = 1;
// For each of the provided proposal names,
// create a new proposal object and add it
// to the end of the array.
for (uint i = 0; i < proposalNames.length; i++) { for (uint i = 0; i < proposalNames.length; i++) {
// \`Proposal({...})\` creates a temporary // 'Proposal({...})' creates a temporary
// Proposal object and \`proposals.push(...)\` // Proposal object and 'proposals.push(...)'
// appends it to the end of \`proposals\`. // appends it to the end of 'proposals'.
proposals.push(Proposal({ proposals.push(Proposal({
name: proposalNames[i], name: proposalNames[i],
voteCount: 0 voteCount: 0
@ -46,98 +47,92 @@ contract Ballot {
} }
} }
// Give \`voter\` the right to vote on this ballot. /**
// May only be called by \`chairperson\`. * @dev Give 'voter' the right to vote on this ballot. May only be called by 'chairperson'.
function giveRightToVote(address voter) { * @param voter address of voter
if (msg.sender != chairperson || voters[voter].voted) { */
// \`throw\` terminates and reverts all changes to function giveRightToVote(address voter) public {
// the state and to Ether balances. It is often require(
// a good idea to use this if functions are msg.sender == chairperson,
// called incorrectly. But watch out, this "Only chairperson can give right to vote."
// will also consume all provided gas. );
throw; require(
} !voters[voter].voted,
"The voter already voted."
);
require(voters[voter].weight == 0);
voters[voter].weight = 1; voters[voter].weight = 1;
} }
/// Delegate your vote to the voter \`to\`. /**
function delegate(address to) { * @dev Delegate your vote to the voter 'to'.
// assigns reference * @param to address to which vote is delegated
Voter sender = voters[msg.sender]; */
if (sender.voted) function delegate(address to) public {
throw; Voter storage sender = voters[msg.sender];
require(!sender.voted, "You already voted.");
// Forward the delegation as long as require(to != msg.sender, "Self-delegation is disallowed.");
// \`to\` also delegated.
// In general, such loops are very dangerous, while (voters[to].delegate != address(0)) {
// because if they run too long, they might
// need more gas than is available in a block.
// In this case, the delegation will not be executed,
// but in other situations, such loops might
// cause a contract to get "stuck" completely.
while (
voters[to].delegate != address(0) &&
voters[to].delegate != msg.sender
) {
to = voters[to].delegate; to = voters[to].delegate;
}
// We found a loop in the delegation, not allowed. // We found a loop in the delegation, not allowed.
if (to == msg.sender) { require(to != msg.sender, "Found loop in delegation.");
throw;
} }
// Since \`sender\` is a reference, this
// modifies \`voters[msg.sender].voted\`
sender.voted = true; sender.voted = true;
sender.delegate = to; sender.delegate = to;
Voter delegate = voters[to]; Voter storage delegate_ = voters[to];
if (delegate.voted) { if (delegate_.voted) {
// If the delegate already voted, // If the delegate already voted,
// directly add to the number of votes // directly add to the number of votes
proposals[delegate.vote].voteCount += sender.weight; proposals[delegate_.vote].voteCount += sender.weight;
} else { } else {
// If the delegate did not vote yet, // If the delegate did not vote yet,
// add to her weight. // add to her weight.
delegate.weight += sender.weight; delegate_.weight += sender.weight;
} }
} }
/// Give your vote (including votes delegated to you) /**
/// to proposal \`proposals[proposal].name\`. * @dev Give your vote (including votes delegated to you) to proposal 'proposals[proposal].name'.
function vote(uint proposal) { * @param proposal index of proposal in the proposals array
Voter sender = voters[msg.sender]; */
if (sender.voted) function vote(uint proposal) public {
throw; Voter storage sender = voters[msg.sender];
require(sender.weight != 0, "Has no right to vote");
require(!sender.voted, "Already voted.");
sender.voted = true; sender.voted = true;
sender.vote = proposal; sender.vote = proposal;
// If \`proposal\` is out of the range of the array, // If 'proposal' is out of the range of the array,
// this will throw automatically and revert all // this will throw automatically and revert all
// changes. // changes.
proposals[proposal].voteCount += sender.weight; proposals[proposal].voteCount += sender.weight;
} }
/// @dev Computes the winning proposal taking all /**
/// previous votes into account. * @dev Computes the winning proposal taking all previous votes into account.
function winningProposal() constant * @return winningProposal_ index of winning proposal in the proposals array
returns (uint winningProposal) */
function winningProposal() public view
returns (uint winningProposal_)
{ {
uint winningVoteCount = 0; uint winningVoteCount = 0;
for (uint p = 0; p < proposals.length; p++) { for (uint p = 0; p < proposals.length; p++) {
if (proposals[p].voteCount > winningVoteCount) { if (proposals[p].voteCount > winningVoteCount) {
winningVoteCount = proposals[p].voteCount; winningVoteCount = proposals[p].voteCount;
winningProposal = p; winningProposal_ = p;
} }
} }
} }
// Calls winningProposal() function to get the index /**
// of the winner contained in the proposals array and then * @dev Calls winningProposal() function to get the index of the winner contained in the proposals array and then
// returns the name of the winner * @return winnerName_ the name of the winner
function winnerName() constant */
returns (bytes32 winnerName) function winnerName() public view
returns (bytes32 winnerName_)
{ {
winnerName = proposals[winningProposal()].name; winnerName_ = proposals[winningProposal()].name;
} }
} }

@ -0,0 +1,28 @@
// SPDX-License-Identifier: GPL-3.0
pragma solidity >=0.7.0 <0.9.0;
import "remix_tests.sol"; // this import is automatically injected by Remix.
import "hardhat/console.sol";
import "../ballot.sol";
contract BallotTest {
bytes32[] proposalNames;
Ballot ballotToTest;
function beforeAll () public {
proposalNames.push(bytes32("candidate1"));
ballotToTest = new Ballot(proposalNames);
}
function checkWinningProposal () public {
console.log("Running checkWinningProposal");
ballotToTest.vote(0);
Assert.equal(ballotToTest.winningProposal(), uint(0), "proposal at index 0 should be the winning proposal");
Assert.equal(ballotToTest.winnerName(), bytes32("candidate1"), "candidate1 should be the winner name");
}
function checkWinninProposalWithReturnValue () public view returns (bool) {
return ballotToTest.winningProposal() == 0;
}
}

@ -354,10 +354,6 @@ class AppComponent {
const queryParams = new QueryParams() const queryParams = new QueryParams()
const params = queryParams.get() const params = queryParams.get()
if (isElectron()) {
this.appManager.activatePlugin('remixd')
}
try { try {
this.engine.register(await this.appManager.registeredPlugins()) this.engine.register(await this.appManager.registeredPlugins())
} catch (e) { } catch (e) {

@ -213,6 +213,34 @@ class Editor extends Plugin {
return ext && this.modes[ext] ? this.modes[ext] : this.modes.txt return ext && this.modes[ext] ? this.modes[ext] : this.modes.txt
} }
async handleTypeScriptDependenciesOf (path, content, readFile) {
if (path.endsWith('.ts')) {
// extract the import, resolve their content
// and add the imported files to Monaco through the `addModel`
// so Monaco can provide auto completion
const paths = path.split('/')
paths.pop()
const fromPath = paths.join('/') // get current execution context path
for (const match of content.matchAll(/import\s+.*\s+from\s+(?:"(.*?)"|'(.*?)')/g)) {
let pathDep = match[2]
if (pathDep.startsWith('./') || pathDep.startsWith('../')) pathDep = resolve(fromPath, pathDep)
if (pathDep.startsWith('/')) pathDep = pathDep.substring(1)
if (!pathDep.endsWith('.ts')) pathDep = pathDep + '.ts'
try {
// we can't use the fileManager plugin call directly
// because it's itself called in a plugin context, and that causes a timeout in the plugin stack
const contentDep = await readFile(pathDep)
if (contentDep !== null) {
this.emit('addModel', contentDep, 'typescript', pathDep, false)
}
} catch (e) {
console.log(e)
}
}
}
}
/** /**
* Create an editor session * Create an editor session
* @param {string} path path of the file * @param {string} path path of the file
@ -222,30 +250,6 @@ class Editor extends Plugin {
async _createSession (path, content, mode) { async _createSession (path, content, mode) {
if (!this.activated) return if (!this.activated) return
if (path.endsWith('.ts')) {
try {
// extract the import, resolve their content
// and add the imported files to Monaco through the `addModel`
// so Monaco can provide auto completion
let content = await this.call('fileManager', 'readFile', path)
const paths = path.split('/')
paths.pop()
const fromPath = paths.join('/') // get current execution context path
for (const match of content.matchAll(/import\s+.*\s+from\s+(?:"(.*?)"|'(.*?)')/g)) {
let path = match[2]
if (path.startsWith('./') || path.startsWith('../')) path = resolve(fromPath, path)
if (path.startsWith('/')) path = path.substring(1)
if (!path.endsWith('.ts')) path = path + '.ts'
if (await this.call('fileManager', 'exists', path)) {
content = await this.call('fileManager', 'readFile', path)
this.emit('addModel', content, 'typescript', path, false)
}
}
} catch (e) {
console.log('unable to resolve dependency of', path, e)
}
}
this.emit('addModel', content, mode, path, false) this.emit('addModel', content, mode, path, false)
return { return {
path, path,

@ -632,6 +632,13 @@ class FileManager extends Plugin {
console.log(error) console.log(error)
throw error throw error
} }
try {
// This make sure dependencies are loaded in the editor context.
// This ensure monaco is aware of deps artifacts, so it can provide basic features like "go to" symbols.
await this.editor.handleTypeScriptDependenciesOf(file, content, path => this.readFile(path))
} catch (e) {
console.log('unable to handle TypeScript dependencies of', file)
}
if (provider.isReadOnly(file)) { if (provider.isReadOnly(file)) {
await this.editor.openReadOnly(file, content) await this.editor.openReadOnly(file, content)
} else { } else {
@ -838,7 +845,7 @@ class FileManager extends Plugin {
const fileName = helper.extractNameFromKey(src) const fileName = helper.extractNameFromKey(src)
if (await this.exists(dest + '/' + fileName)) { if (await this.exists(dest + '/' + fileName)) {
throw createError({ code: 'ENOENT', message: `Cannot move ${src}. File already exists at destination ${dest}`}) throw createError({ code: 'EEXIST', message: `Cannot move ${src}. File already exists at destination ${dest}`})
} }
await this.copyFile(src, dest, fileName) await this.copyFile(src, dest, fileName)
await this.remove(src) await this.remove(src)
@ -865,9 +872,8 @@ class FileManager extends Plugin {
await this._handleIsDir(src, `Cannot move ${src}. Path is not directory.`) await this._handleIsDir(src, `Cannot move ${src}. Path is not directory.`)
await this._handleIsDir(dest, `Cannot move content into ${dest}. Path is not directory.`) await this._handleIsDir(dest, `Cannot move content into ${dest}. Path is not directory.`)
const dirName = helper.extractNameFromKey(src) const dirName = helper.extractNameFromKey(src)
if (await this.exists(dest + '/' + dirName) || src === dest) {
if (await this.exists(dest + '/' + dirName)) { throw createError({ code: 'EEXIST', message: `Cannot move ${src}. Folder already exists at destination ${dest}`})
throw createError({ code: 'ENOENT', message: `Cannot move ${src}. Folder already exists at destination ${dest}`})
} }
await this.copyDir(src, dest, dirName) await this.copyDir(src, dest, dirName)
await this.remove(src) await this.remove(src)

@ -87,6 +87,18 @@ export class CodeParser extends Plugin {
} }
} }
async handleChangeEvents() {
const completionSettings = await this.call('config', 'getAppParameter', 'auto-completion')
if (completionSettings) {
await this.antlrService.getCurrentFileAST()
}
const showGasSettings = await this.call('config', 'getAppParameter', 'show-gas')
const showErrorSettings = await this.call('config', 'getAppParameter', 'display-errors')
if(showGasSettings || showErrorSettings || completionSettings) {
await this.compilerService.compile()
}
}
async onActivation() { async onActivation() {
this.gasService = new CodeParserGasService(this) this.gasService = new CodeParserGasService(this)
@ -102,8 +114,7 @@ export class CodeParser extends Plugin {
this.on('editor', 'didChangeFile', async (file) => { this.on('editor', 'didChangeFile', async (file) => {
await this.call('editor', 'discardLineTexts') await this.call('editor', 'discardLineTexts')
await this.antlrService.getCurrentFileAST() await this.handleChangeEvents()
await this.compilerService.compile()
}) })
this.on('filePanel', 'setWorkspace', async () => { this.on('filePanel', 'setWorkspace', async () => {
@ -113,8 +124,7 @@ export class CodeParser extends Plugin {
this.on('fileManager', 'currentFileChanged', async () => { this.on('fileManager', 'currentFileChanged', async () => {
await this.call('editor', 'discardLineTexts') await this.call('editor', 'discardLineTexts')
await this.antlrService.getCurrentFileAST() await this.handleChangeEvents()
await this.compilerService.compile()
}) })
this.on('solidity', 'loadingCompiler', async (url) => { this.on('solidity', 'loadingCompiler', async (url) => {
@ -188,10 +198,10 @@ export class CodeParser extends Plugin {
const index = {} const index = {}
const contractName: string = contractNode.name const contractName: string = contractNode.name
const callback = (node) => { const callback = (node) => {
if(inScope && node.scope !== contractNode.id if (inScope && node.scope !== contractNode.id
&& !(node.nodeType === 'EnumDefinition' || node.nodeType === 'EventDefinition' || node.nodeType === 'ModifierDefinition')) && !(node.nodeType === 'EnumDefinition' || node.nodeType === 'EventDefinition' || node.nodeType === 'ModifierDefinition'))
return return
if(inScope) node.isClassNode = true; if (inScope) node.isClassNode = true;
node.gasEstimate = this._getContractGasEstimate(node, contractName, fileName, compilatioResult) node.gasEstimate = this._getContractGasEstimate(node, contractName, fileName, compilatioResult)
node.functionName = node.name + this._getInputParams(node) node.functionName = node.name + this._getInputParams(node)
node.contractName = contractName node.contractName = contractName
@ -227,11 +237,11 @@ export class CodeParser extends Plugin {
if ((node.scope && node.scope === baseContract.id) if ((node.scope && node.scope === baseContract.id)
|| node.nodeType === 'EnumDefinition' || node.nodeType === 'EnumDefinition'
|| node.nodeType === 'EventDefinition' || node.nodeType === 'EventDefinition'
) { ) {
baseNodesWithBaseContractScope[node.id] = node baseNodesWithBaseContractScope[node.id] = node
} }
if(node.members){ if (node.members) {
for(const member of node.members){ for (const member of node.members) {
member.contractName = (baseContract as any).name member.contractName = (baseContract as any).name
member.contractId = (baseContract as any).id member.contractId = (baseContract as any).id
member.isBaseNode = true; member.isBaseNode = true;
@ -249,7 +259,7 @@ export class CodeParser extends Plugin {
if (node.nodeType === 'ImportDirective') { if (node.nodeType === 'ImportDirective') {
const imported = await this.resolveImports(node, {}) const imported = await this.resolveImports(node, {})
for (const importedNode of (Object.values(imported) as any)) { for (const importedNode of (Object.values(imported) as any)) {
if (importedNode.nodes) if (importedNode.nodes)
for (const subNode of importedNode.nodes) { for (const subNode of importedNode.nodes) {

@ -45,7 +45,7 @@ function isElectron() {
return false return false
} }
const versionUrl = isElectron() ? 'https://remix.ethereum.org/assets/version.json' : 'assets/version.json' const versionUrl = 'assets/version.json'
fetch(versionUrl, { cache: "no-store" }).then(response => { fetch(versionUrl, { cache: "no-store" }).then(response => {
response.text().then(function (data) { response.text().then(function (data) {
const version = JSON.parse(data); const version = JSON.parse(data);

@ -262,7 +262,8 @@ export const EditorUI = (props: EditorUIProps) => {
// see https://code.visualstudio.com/api/references/theme-color for more settings // see https://code.visualstudio.com/api/references/theme-color for more settings
'editor.background': textbackground, 'editor.background': textbackground,
'editorSuggestWidget.background': lightColor, 'editorSuggestWidget.background': lightColor,
'editorSuggestWidget.selectedBackground': lightColor, 'editorSuggestWidget.selectedBackground': secondaryColor,
'editorSuggestWidget.selectedForeground': textColor,
'editorSuggestWidget.highlightForeground': infoColor, 'editorSuggestWidget.highlightForeground': infoColor,
'editor.lineHighlightBorder': secondaryColor, 'editor.lineHighlightBorder': secondaryColor,
'editor.lineHighlightBackground': textbackground === darkColor ? lightColor : secondaryColor, 'editor.lineHighlightBackground': textbackground === darkColor ? lightColor : secondaryColor,

@ -227,9 +227,9 @@ export const loadTypes = async (monaco) => {
// @ts-ignore // @ts-ignore
const chaiType = await import('raw-loader!@types/chai/index.d.ts') const chaiType = await import('raw-loader!@types/chai/index.d.ts')
monaco.languages.typescript.typescriptDefaults.addExtraLib(chaiType.default) monaco.languages.typescript.typescriptDefaults.addExtraLib(chaiType.default, `file:///node_modules/@types/chai/index.d.ts`)
// @ts-ignore // @ts-ignore
const mochaType = await import('raw-loader!@types/mocha/index.d.ts') const mochaType = await import('raw-loader!@types/mocha/index.d.ts')
monaco.languages.typescript.typescriptDefaults.addExtraLib(mochaType.default) monaco.languages.typescript.typescriptDefaults.addExtraLib(mochaType.default, `file:///node_modules/@types/mocha/index.d.ts`)
} }

@ -41,13 +41,13 @@ export const RemixUiSettings = (props: RemixUiSettingsProps) => {
if (javascriptVM === null || javascriptVM === undefined) ethereumVM(props.config, true, dispatch) if (javascriptVM === null || javascriptVM === undefined) ethereumVM(props.config, true, dispatch)
const useAutoComplete = props.config.get('settings/auto-completion') const useAutoComplete = props.config.get('settings/auto-completion')
if (useAutoComplete === null || useAutoComplete === undefined) useAutoCompletion(props.config, true, dispatch) if (useAutoComplete === null || useAutoComplete === undefined) useAutoCompletion(props.config, false, dispatch)
const displayErrors = props.config.get('settings/display-errors') const displayErrors = props.config.get('settings/display-errors')
if (displayErrors === null || displayErrors === undefined) useDisplayErrors(props.config, true, dispatch) if (displayErrors === null || displayErrors === undefined) useDisplayErrors(props.config, false, dispatch)
const useShowGas = props.config.get('settings/show-gas') const useShowGas = props.config.get('settings/show-gas')
if (useShowGas === null || useShowGas === undefined) useShowGasInEditor(props.config, true, dispatch) if (useShowGas === null || useShowGas === undefined) useShowGasInEditor(props.config, false, dispatch)
} }
useEffect(() => initValue(), [resetState, props.config]) useEffect(() => initValue(), [resetState, props.config])
useEffect(() => initValue(), []) useEffect(() => initValue(), [])
@ -148,9 +148,10 @@ export const RemixUiSettings = (props: RemixUiSettingsProps) => {
const isEditorWrapChecked = props.config.get('settings/text-wrap') || false const isEditorWrapChecked = props.config.get('settings/text-wrap') || false
const isPersonalChecked = props.config.get('settings/personal-mode') || false const isPersonalChecked = props.config.get('settings/personal-mode') || false
const isMatomoChecked = props.config.get('settings/matomo-analytics') || false const isMatomoChecked = props.config.get('settings/matomo-analytics') || false
const isAutoCompleteChecked = props.config.get('settings/auto-completion') === null ? true:props.config.get('settings/auto-completion')
const isShowGasInEditorChecked = props.config.get('settings/show-gas') === null ? true:props.config.get('settings/show-gas') const isAutoCompleteChecked = props.config.get('settings/auto-completion') || false
const displayErrorsChecked = props.config.get('settings/display-errors') === null ? true:props.config.get('settings/display-errors') const isShowGasInEditorChecked = props.config.get('settings/show-gas') || false
const displayErrorsChecked = props.config.get('settings/display-errors') || false
return ( return (
<div className="$border-top"> <div className="$border-top">
<div title="Reset to Default settings." className='d-flex justify-content-end pr-4'> <div title="Reset to Default settings." className='d-flex justify-content-end pr-4'>
@ -188,19 +189,19 @@ export const RemixUiSettings = (props: RemixUiSettingsProps) => {
</div> </div>
<div className='custom-control custom-checkbox mb-1'> <div className='custom-control custom-checkbox mb-1'>
<input onChange={onchangeUseAutoComplete} id="settingsUseAutoComplete" type="checkbox" className="custom-control-input" checked={isAutoCompleteChecked} /> <input onChange={onchangeUseAutoComplete} id="settingsUseAutoComplete" type="checkbox" className="custom-control-input" checked={isAutoCompleteChecked} />
<label className={`form-check-label custom-control-label align-middle ${getTextClass('settings/auto-completion')}`} htmlFor="settingsUseAutoComplete"> <label className={`form-check-label custom-control-label align-middle ${getTextClass('settings/auto-completion')}`} data-id="settingsAutoCompleteLabel" htmlFor="settingsUseAutoComplete">
<span>{useAutoCompleteText}</span> <span>{useAutoCompleteText}</span>
</label> </label>
</div> </div>
<div className='custom-control custom-checkbox mb-1'> <div className='custom-control custom-checkbox mb-1'>
<input onChange={onchangeShowGasInEditor} id="settingsUseShowGas" type="checkbox" className="custom-control-input" checked={isShowGasInEditorChecked} /> <input onChange={onchangeShowGasInEditor} id="settingsUseShowGas" type="checkbox" className="custom-control-input" checked={isShowGasInEditorChecked} />
<label className={`form-check-label custom-control-label align-middle ${getTextClass('settings/show-gas')}`} htmlFor="settingsUseShowGas"> <label className={`form-check-label custom-control-label align-middle ${getTextClass('settings/show-gas')}`} data-id="settingsShowGasLabel" htmlFor="settingsUseShowGas">
<span>{useShowGasInEditorText}</span> <span>{useShowGasInEditorText}</span>
</label> </label>
</div> </div>
<div className='custom-control custom-checkbox mb-1'> <div className='custom-control custom-checkbox mb-1'>
<input onChange={onchangeDisplayErrors} id="settingsDisplayErrors" type="checkbox" className="custom-control-input" checked={displayErrorsChecked} /> <input onChange={onchangeDisplayErrors} id="settingsDisplayErrors" type="checkbox" className="custom-control-input" checked={displayErrorsChecked} />
<label className={`form-check-label custom-control-label align-middle ${getTextClass('settings/display-errors')}`} htmlFor="settingsDisplayErrors"> <label className={`form-check-label custom-control-label align-middle ${getTextClass('settings/display-errors')}`} data-id="displayErrorsLabel" htmlFor="settingsDisplayErrors">
<span>{displayErrorsText}</span> <span>{displayErrorsText}</span>
</label> </label>
</div> </div>

@ -8,6 +8,7 @@ import { createWorkspaceTemplate, getWorkspaces, loadWorkspacePreset, setPlugin,
import { QueryParams } from '@remix-project/remix-lib' import { QueryParams } from '@remix-project/remix-lib'
import { fetchContractFromEtherscan } from '@remix-project/core-plugin' // eslint-disable-line import { fetchContractFromEtherscan } from '@remix-project/core-plugin' // eslint-disable-line
import JSZip from 'jszip' import JSZip from 'jszip'
import isElectron from 'is-electron'
export * from './events' export * from './events'
export * from './workspace' export * from './workspace'
@ -111,6 +112,10 @@ export const initWorkspace = (filePanelPlugin) => async (reducerDispatch: React.
await basicWorkspaceInit(workspaces, workspaceProvider) await basicWorkspaceInit(workspaces, workspaceProvider)
} }
} else await basicWorkspaceInit(workspaces, workspaceProvider) } else await basicWorkspaceInit(workspaces, workspaceProvider)
} else if (isElectron()) {
plugin.call('notification', 'toast', `connecting to localhost...`)
await basicWorkspaceInit(workspaces, workspaceProvider)
await plugin.call('manager', 'activatePlugin', 'remixd')
} else if (localStorage.getItem("currentWorkspace")) { } else if (localStorage.getItem("currentWorkspace")) {
const index = workspaces.findIndex(element => element.name == localStorage.getItem("currentWorkspace")) const index = workspaces.findIndex(element => element.name == localStorage.getItem("currentWorkspace"))
if (index !== -1) { if (index !== -1) {

@ -22,8 +22,11 @@
"homepage": "https://github.com/ethereum/remix-project/tree/master/libs/remix-ws-templates#readme", "homepage": "https://github.com/ethereum/remix-project/tree/master/libs/remix-ws-templates#readme",
"typings": "./src/index.d.ts", "typings": "./src/index.d.ts",
"dependencies": { "dependencies": {
"@openzeppelin/contracts": "^4.7.3",
"@openzeppelin/wizard": "^0.1.1",
"ethers": "^5.4.2", "ethers": "^5.4.2",
"web3": "^1.5.1" "web3": "^1.5.1"
}, },
"gitHead": "0c1957c9b2f890076a5062309bc81b41f93af57c" "gitHead": "0c1957c9b2f890076a5062309bc81b41f93af57c"
} }

@ -1,14 +0,0 @@
// SPDX-License-Identifier: GPL-3.0
pragma solidity >=0.7.0 <0.9.0;
import "@openzeppelin/contracts/token/ERC20/ERC20.sol";
/**
* @title SampleERC20
* @dev Create a sample ERC20 standard token
*/
contract SampleERC20 is ERC20 {
constructor(string memory tokenName, string memory tokenSymbol) ERC20(tokenName, tokenSymbol) {}
}

@ -1,7 +1,8 @@
import { erc20 } from '@openzeppelin/wizard';
export default async () => { export default async () => {
return { return {
// @ts-ignore 'contracts/MyToken.sol': erc20.print(),
'contracts/SampleERC20.sol': (await import('raw-loader!./contracts/SampleERC20.sol')).default,
// @ts-ignore // @ts-ignore
'scripts/deploy_with_ethers.ts': (await import('!!raw-loader!./scripts/deploy_with_ethers.ts')).default, 'scripts/deploy_with_ethers.ts': (await import('!!raw-loader!./scripts/deploy_with_ethers.ts')).default,
// @ts-ignore // @ts-ignore
@ -11,6 +12,6 @@ export default async () => {
// @ts-ignore // @ts-ignore
'scripts/web3-lib.ts': (await import('!!raw-loader!./scripts/web3-lib.ts')).default, 'scripts/web3-lib.ts': (await import('!!raw-loader!./scripts/web3-lib.ts')).default,
// @ts-ignore // @ts-ignore
'tests/SampleERC20_test.sol': (await import('raw-loader!./tests/SampleERC20_test.sol')).default 'tests/MyToken_test.sol': (await import('raw-loader!./tests/MyToken_test.sol')).default
} }
} }

@ -2,7 +2,7 @@ import { deploy } from './ethers-lib'
(async () => { (async () => {
try { try {
const result = await deploy('SampleERC20', ['testToken', 'TST']) const result = await deploy('MyToken', [])
console.log(`address: ${result.address}`) console.log(`address: ${result.address}`)
} catch (e) { } catch (e) {
console.log(e.message) console.log(e.message)

@ -2,7 +2,7 @@ import { deploy } from './web3-lib'
(async () => { (async () => {
try { try {
const result = await deploy('SampleERC20', ['testToken', 'TST']) const result = await deploy('MyToken', [])
console.log(`address: ${result.address}`) console.log(`address: ${result.address}`)
} catch (e) { } catch (e) {
console.log(e.message) console.log(e.message)

@ -0,0 +1,18 @@
// SPDX-License-Identifier: GPL-3.0
pragma solidity >=0.7.0 <0.9.0;
import "remix_tests.sol";
import "../contracts/MyToken.sol";
contract MyTokenTest {
MyToken s;
function beforeAll () public {
s = new MyToken();
}
function testTokenNameAndSymbol () public {
Assert.equal(s.name(), "MyToken", "token name did not match");
Assert.equal(s.symbol(), "MTK", "token symbol did not match");
}
}

@ -1,18 +0,0 @@
// SPDX-License-Identifier: GPL-3.0
pragma solidity >=0.7.0 <0.9.0;
import "remix_tests.sol";
import "../contracts/SampleERC20.sol";
contract SampleERC20Test {
SampleERC20 s;
function beforeAll () public {
s = new SampleERC20("TestToken", "TST");
}
function testTokenNameAndSymbol () public {
Assert.equal(s.name(), "TestToken", "token name did not match");
Assert.equal(s.symbol(), "TST", "token symbol did not match");
}
}

@ -1,14 +0,0 @@
// SPDX-License-Identifier: GPL-3.0
pragma solidity >=0.7.0 <0.9.0;
import "@openzeppelin/contracts/token/ERC721/ERC721.sol";
/**
* @title SampleERC721
* @dev Create a sample ERC721 standard token
*/
contract SampleERC721 is ERC721 {
constructor(string memory tokenName, string memory tokenSymbol) ERC721(tokenName, tokenSymbol) {}
}

@ -1,7 +1,8 @@
import { erc721 } from '@openzeppelin/wizard';
export default async () => { export default async () => {
return { return {
// @ts-ignore 'contracts/MyToken.sol': erc721.print(),
'contracts/SampleERC721.sol': (await import('raw-loader!./contracts/SampleERC721.sol')).default,
// @ts-ignore // @ts-ignore
'scripts/deploy_with_ethers.ts': (await import('!!raw-loader!./scripts/deploy_with_ethers.ts')).default, 'scripts/deploy_with_ethers.ts': (await import('!!raw-loader!./scripts/deploy_with_ethers.ts')).default,
// @ts-ignore // @ts-ignore
@ -11,6 +12,6 @@ export default async () => {
// @ts-ignore // @ts-ignore
'scripts/web3-lib.ts': (await import('!!raw-loader!./scripts/web3-lib.ts')).default, 'scripts/web3-lib.ts': (await import('!!raw-loader!./scripts/web3-lib.ts')).default,
// @ts-ignore // @ts-ignore
'tests/SampleERC721_test.sol': (await import('raw-loader!./tests/SampleERC721_test.sol')).default 'tests/MyToken_test.sol': (await import('raw-loader!./tests/MyToken_test.sol')).default
} }
} }

@ -2,7 +2,7 @@ import { deploy } from './ethers-lib'
(async () => { (async () => {
try { try {
const result = await deploy('SampleERC721', ['testNFT', 'TNFT']) const result = await deploy('MyToken', [])
console.log(`address: ${result.address}`) console.log(`address: ${result.address}`)
} catch (e) { } catch (e) {
console.log(e.message) console.log(e.message)

@ -2,7 +2,7 @@ import { deploy } from './web3-lib'
(async () => { (async () => {
try { try {
const result = await deploy('SampleERC721', ['testToken', 'TST']) const result = await deploy('MyToken', [])
console.log(`address: ${result.address}`) console.log(`address: ${result.address}`)
} catch (e) { } catch (e) {
console.log(e.message) console.log(e.message)

@ -0,0 +1,18 @@
// SPDX-License-Identifier: GPL-3.0
pragma solidity >=0.7.0 <0.9.0;
import "remix_tests.sol";
import "../contracts/MyToken.sol";
contract MyTokenTest {
MyToken s;
function beforeAll () public {
s = new MyToken();
}
function testTokenNameAndSymbol () public {
Assert.equal(s.name(), "MyToken", "token name did not match");
Assert.equal(s.symbol(), "MTK", "token symbol did not match");
}
}

@ -1,18 +0,0 @@
// SPDX-License-Identifier: GPL-3.0
pragma solidity >=0.7.0 <0.9.0;
import "remix_tests.sol";
import "../contracts/SampleERC721.sol";
contract SampleERC721Test {
SampleERC721 s;
function beforeAll () public {
s = new SampleERC721("TestNFT", "TNFT");
}
function testTokenNameAndSymbol () public {
Assert.equal(s.name(), "TestNFT", "token name did not match");
Assert.equal(s.symbol(), "TNFT", "token symbol did not match");
}
}

@ -154,6 +154,8 @@
"@ethersphere/bee-js": "^3.2.0", "@ethersphere/bee-js": "^3.2.0",
"@isomorphic-git/lightning-fs": "^4.4.1", "@isomorphic-git/lightning-fs": "^4.4.1",
"@monaco-editor/react": "4.4.5", "@monaco-editor/react": "4.4.5",
"@openzeppelin/contracts": "^4.7.3",
"@openzeppelin/wizard": "^0.1.1",
"@remixproject/engine": "^0.3.31", "@remixproject/engine": "^0.3.31",
"@remixproject/engine-web": "^0.3.31", "@remixproject/engine-web": "^0.3.31",
"@remixproject/plugin": "^0.3.31", "@remixproject/plugin": "^0.3.31",

@ -3912,6 +3912,18 @@
dependencies: dependencies:
"@octokit/openapi-types" "^11.2.0" "@octokit/openapi-types" "^11.2.0"
"@openzeppelin/contracts@^4.7.3":
version "4.7.3"
resolved "https://registry.yarnpkg.com/@openzeppelin/contracts/-/contracts-4.7.3.tgz#939534757a81f8d69cc854c7692805684ff3111e"
integrity sha512-dGRS0agJzu8ybo44pCIf3xBaPQN/65AIXNgK8+4gzKd5kbvlqyxryUYVLJv7fK98Seyd2hDZzVEHSWAh0Bt1Yw==
"@openzeppelin/wizard@^0.1.1":
version "0.1.1"
resolved "https://registry.yarnpkg.com/@openzeppelin/wizard/-/wizard-0.1.1.tgz#8c183e2c5748869bc3a5317c0330aa36a9ad44fe"
integrity sha512-AGyvn3PIh1vCgAEoRKAXKhtlk4fkA8AHE7G4PyzLnYcASClYCWpSf43WLJCs6S/LORvTZADX1flvF8x2LciJIg==
dependencies:
array.prototype.flatmap "^1.2.4"
"@pmmmwh/react-refresh-webpack-plugin@^0.4.3": "@pmmmwh/react-refresh-webpack-plugin@^0.4.3":
version "0.4.3" version "0.4.3"
resolved "https://registry.yarnpkg.com/@pmmmwh/react-refresh-webpack-plugin/-/react-refresh-webpack-plugin-0.4.3.tgz#1eec460596d200c0236bf195b078a5d1df89b766" resolved "https://registry.yarnpkg.com/@pmmmwh/react-refresh-webpack-plugin/-/react-refresh-webpack-plugin-0.4.3.tgz#1eec460596d200c0236bf195b078a5d1df89b766"

Loading…
Cancel
Save