Blockchain technology today allows developers to create decentralized web applications by deploying smart contracts to the blockchain and interacting with the contracts from a frontend interface. Most web developers are familiar with at least one frontend framework such as React or Vue which are perfect for interfacing with blockchain contracts, but how does one begin to create a decentralized application without the smart contract? Let’s explore a popular tool called Ethereum Remix that allows developers to explore the world of smart contracts inside a web sandbox, similiar to codesandbox.io but for Solidity. Write, compile, and deploy your contract and interface with the contract all from a web interface. No terminal required.
Prerequisites:
- Web Browser
- Familiarity with object-oriented programming
- General understanding of Smart Contracts
Let’s get started
- Navigate to https://remix.ethereum.org/ This should take you to the web IDE interface.
-
As you can see there is a default workspace containing all the files you need to get started. There are even 3 sample contracts, try clicking on 1_Storage.sol:
// SPDX-License-Identifier: GPL-3.0 pragma solidity >=0.7.0 <0.9.0; /** * @title Storage * @dev Store & retrieve value in a variable */ contract Storage { uint256 number; /** * @dev Store value in variable * @param num value to store */ function store(uint256 num) public { number = num; } /** * @dev Return value * @return value of 'number' */ function retrieve() public view returns (uint256){ return number; } }
This is an example of a very simple contract. The name of the contract is
Storage
and it stores data of typeuint256
in a variable callednumber
. There are 2 functions or methods that are publicly available. One is calledstore
and the other is calledretrieve
. Thestore
function must be called with one argument of typeuint256
and all it does is update the value provided to the variablenumber
. Calling onretrieve
does not require any arguments and all it does is return the value ofnumber
If that explanation didn’t make any sense, that’s okay. Let’s compile and deploy the contract so we can get a better understanding how it works.
-
Click on the compile tab on the left sidebar (second tab from the top) and click on the button that says “Compile 1_Storage.sol”
If everything compiles successfully, you will see a green check on the compile tab.
-
Click on the 3rd tab from the top, this is the deploy tab.
Don’t worry too much about all the different options right now, just click on the big “Deploy” button. If successfull, you will see an item appear under “Deployed Contracts” click on the tab to open the contract interface.
-
Let’s interact with the contract
As you can see there are 2 buttons, one representing each of the two publicly available functions. The
store
button is orange because this function mutates data inside the contract. Theretrieve
function is grey because it only reads data.Try clicking on the
retrieve
buttonYou should get back:
0: uint256: 0
This means the function returns one piece of data (index 0) of type
uint256
and the value is0
This is good because we never set the value of
number
. The default value foruint256
is0
so this is exactly what we would expect on a fresh deploy.Now lets set the value of
number
to be3
. Enter3
into thestore
function and click on the “store” button.If successful, you will notice some output on the bottom terminal with a green checkmark. This means that the transaction was successful.
If that was successful, then now when we retrieve the value of
number
it should tell us the value is3
. Try clicking onretrieve
again.If all of that worked, you should now get this response:
0: uint256: 3
Try setting the value of
number
to different values. What are the limitations? What happens if you try passing something other than an integer into thestore
function?
Building the CRUD application
-
Now that we have a basic understanding of Solidity and Ethereum Remix, lets write our own CRUD application. In this example, we’re going to create a decentralized application for users to produce and share their own beats. Start by renaming our
1_Storage.sol
file toBeatBox.sol
. Update the name of the contract and remove everything inside.// SPDX-License-Identifier: GPL-3.0 pragma solidity >=0.7.0 <0.9.0; contract BeatBox { }
Now let’s start by defining our User model. In Solidity we do this with a struct
When defining a struct, all you need to do is define the data type, and then give it a name. We are going to use types address, uint, string, and bytes32[]
struct User {
address wallet;
uint id;
string username;
bytes32[] sequences;
}
Next, we are going to store our User data in a mapping. In some ways mapping is analogous to a Table in a traditional SQL database. We can retrieve data on our User model by using a unique id or key. In this case, we are going to use the User address as the unique id. The name of our mapping will be called userMap
mapping (address => User) userMap;
Finally, we are going to store references to our Users in an array so we can access all Users.
address[] userIndex;
All together, our contract looks like this so far:
// SPDX-License-Identifier: GPL-3.0
pragma solidity >=0.7.0 <0.9.0;
contract BeatBox {
struct User {
address wallet;
uint id;
string username;
bytes32[] sequences;
}
mapping (address => User) userMap;
address[] userIndex;
}
-
Now lets create a function to create a User. This function will accept 1 argument of type string. We will store this argument in memory versus storage and call with the argument
_username
. This function is public meaning anyone can call on this function and create a User in this contract.function createUser(string memory _username) public { }
Right now this function does nothing, so let’s write the code to create a User. Inside the function, add the following code to create a User, save it to memory and call it
newUser
. We simply have to pass the correct arguments to our User struct and the User is created.User memory newUser = User({ wallet: msg.sender, id: userIndex.length, username: _username, sequences: new bytes32[](0) });
Inside a Solidity function,
msg.sender
is the address of the user who is making the function call, so we can be certain this is the wallet address of the User.id
will be assigned a value based on the number of users in ouruserIndex
.username
will be assigned the value passed into the function, andsequences
for now will be a bytes32 array of length 0.Now that we have our User saved to memory, let’s add this user to our
userMap
. We can access an entry in our mapping by providing a key (in this case the address of the user) inside square brackets [] and update the entry by setting it equal to a new value (our new User)userMap[msg.sender] = newUser;
Finally let’s add a reference to this new User to our
userIndex
userIndex.push(msg.sender);
in this way, our user
id
’s will be auto-incrementing because each user that’s created will add an index to ouruserIndex
array. -
Now let’s create a function to get all our Users. This function is also public, designated as
view
because it does not mutate data in the contract, and returns a User array stored in memory.function getUsers() public view returns (User[] memory) { }
Again, this function doesn’t currently do anything, so let’s start by saving an empty User array to memory and store it in a variable called
_users
. When creating a new array in Solidity, you must define the length of an array, which in this case we want to be the number of users in ouruserIndex
array, so we can useuserIndex.length
:User[] memory _users = new User[](userIndex.length);
Now that we have our User array saved to memory, we can use a for loop to replace each item with our actual users. For each index of the array, we are going to replace that index with the actual User data from our
userMap
and access it with the reference in ouruserIndex
:for (uint i = 0; i < userIndex.length; i++) { _users[i] = userMap[userIndex[i]]; }
Finally, now that our User array has been filled with our actual User data, we can simply return our
_user
array:return _users;
Our completed function looks like this:
function getUsers() public view returns (User[] memory) { User[] memory _users = new User[](userIndex.length); for (uint i = 0; i < userIndex.length; i++) { _users[i] = userMap[userIndex[i]]; } return _users; }
- Now that we have our
createUser
andgetUsers
functions, let’s see if they work? Compile and deploy and create a user. Make sure you pass in data of type string, for example'test'
Now call on getUsers
and you should get a response like this:
0: tuple(address,uint256,string,bytes32[])[]: 0x5B38Da6a701c568545dCfcB03FcB875f56beddC4,0,'test',
This may be a bit difficult to read in this format, but you can see our User data is all there. The first argument will be the address of the wallet you are using in Remix so it will not match the example. When calling this function from a frontend, the data will be structured as an array of objects so it will be easy to work with:
[
{
wallet: 0x5b38da6a701c568545dcfcb03fcb875f56beddc4,
id: 0,
username: 'test',
sequences: [],
},
];
- As you can see, the user id is 0 because they are the first user in our userIndex (index 0). If we want to start with index 1, we can create an initial user for the developer using the
constructor
:
First, let’s save the value of the developer address so we can use this in the future:
address payable _devAddress;
Next, lets write the code for the constructor:
constructor() {
_devAddress = payable(msg.sender);
userMap[msg.sender] = User({
wallet: msg.sender,
id: userIndex.length,
username: 'developer',
sequences: new bytes32[](0)
});
userIndex.push(msg.sender);
}
This code will run only once, on deployment and the value of msg.sender
will be the address of the deployer. We can save this address to our _devAddress
so we can transfer funds to this address in the future. Also we are creating a user for the developer that will take index 0. This way our users will start with index 1. This can be helpful in validating if a wallet is an existing user.
- Now let’s create a function to update our user. This will be similiar to
createUser
but instead of creating a new User struct, we can simply identify the correct User, and update the neccessary data.
function updateUser(string memory _username) public {
}
Access the correct User and save it to storage as _user
. Since we will be mutating the data we must use storage here:
User storage _user = userMap[msg.sender];
Finally, we can simply update the username with the argument provided:
_user.username = _username;
The completed function looks like this:
function updateUser(string memory _username) public {
User storage _user = userMap[msg.sender];
_user.username = _username;
}
Compile and deploy to make sure all our functions are working as intended.
- Now that we have our User model setup with create, read, and update functionality, lets create our Sequence Model. Sequence will contain all our instrument tracks. We can follow the same pattern for our Sequence Model.
struct Sequence {
bytes32 id;
address owner;
string name;
bytes32[] modules;
uint createdAt;
uint updatedAt;
}
mapping (bytes32 => Sequence) sequenceMap;
bytes32[] sequenceIndex;
Only difference here is we are using bytes32
for our Sequence id so our mapping will map bytes32 to Sequence. We will use a hash function, keccak256 to generate our ids.
- Function for
createSequence
function createSequence(string memory _name) public {
User storage _user = userMap[msg.sender];
bytes32 sequenceId = keccak256(abi.encodePacked(msg.sender, block.timestamp, _name));
sequenceMap[sequenceId] = Sequence({
id: sequenceId,
name: _name,
createdAt: block.timestamp,
updatedAt: block.timestamp,
modules: new bytes32[](0)
});
sequenceIndex.push(sequenceId);
_user.sequences.push(sequenceId);
}
Here, we are pushing the sequenceId
to the sequenceIndex
as well as the User’s sequences
array so the User will keep a reference to the sequence.
- Function for
getAllSequences
- following the same pattern asgetUsers
function getAllSequences() public view returns (Sequence[] memory) {
Sequence[] memory _sequences = new Sequence[](sequenceIndex.length);
for (uint i = 0; i < sequenceIndex.length; i++) {
_sequences[i] = sequenceMap[sequenceIndex[i]];
}
return _sequences;
}
- Function for
getUserSequences
- same asgetAllSequences
but only for User sequences:
function getUserSequences() public view returns (Sequence[] memory) {
User memory _user = userMap[msg.sender];
Sequence[] memory _sequences = new Sequence[](_user.sequences.length);
for (uint i =0; i < _user.sequences.length; i++) {
_sequences[i] = sequenceMap[_user.sequences[i]];
}
return _sequences;
}
- Finally, lets create our Module Model this will be our instrument track:
struct Module {
bytes32 id;
string instrument;
bool beat_1;
bool beat_2;
bool beat_3;
bool beat_4;
uint createdAt;
uint updatedAt;
}
mapping (bytes32 => Module) moduleMap;
bytes32[] moduleIndex;
- Function for
createModule
function createModule(bytes32 _sequenceId, string memory _instrument, bool _beat_1, bool _beat_2, bool _beat_3, bool _beat_4) public {
bytes32 moduleId = keccak256(abi.encodePacked(msg.sender, block.timestamp, _instrument));
moduleMap[moduleId] = Module({
id: moduleId,
instrument: _instrument,
beat_1: _beat_1,
beat_2: _beat_2,
beat_3: _beat_3,
beat_4: _beat_4,
createdAt: block.timestamp,
updatedAt: block.timestamp
});
moduleIndex.push(moduleId);
sequenceMap[_sequenceId].modules.push(moduleId);
}
- Function for
getSequenceModules
function getSequenceModules(bytes32 _sequenceId) public view returns (Module[] memory) {
Sequence memory _sequence = sequenceMap[_sequenceId];
Module[] memory _modules = new Module[](_sequence.modules.length);
for(uint i = 0; i < _sequence.modules.length; i++) {
_modules[i] = moduleMap[_sequence.modules[i]];
}
return _modules;
}
- Finally, just for fun let’s create a function for users to “buy me a coffee” if they want to support the developer.
function buyMeACoffee() public payable {
uint contribution = msg.value;
_devAddress.transfer(contribution);
}
This function takes the value of the funds sent (msg.value) and transfers it to the _devAddress
anyone can call on this function to send funds to the developer so the function is designated public
and payable
Next steps
- Example repository can be seen here
- Now we have the beginning of an application capable of creating, reading and updating data on the blockchain. Give it a try, experiment, modify for your own applications. This example hopefully is a good introduction to smart contract development but really only scratches the surface. Some key concepts that were not included is validating function calls with
assert
andrequire
you can read about that here, and emitting events which you can read about here