Ethereum has the hash function keccak256
built in, which is a version of SHA3. A hash function basically maps an input into a random 256-bit hexadecimal number. A slight change in the input will cause a large change in the hash.
It's useful for many purposes in Ethereum, but for right now we're just going to use it for pseudo-random number generation.
Also important, keccak256
expects a single parameter of type bytes
. This means that we have to "pack" any parameters before calling keccak256
:
Example:
keccak256(abi.encodePacked("aaaab"));
keccak256(abi.encodePacked("aaaac"));
As you can see, the returned values are totally different despite only a 1 character change in the input.
Typecasting
Sometimes you need to convert between data types. Take the following example:
uint8 a = 5;
uint b = 6;
uint8 c = a * b;
uint8 c = a * uint8(b);
In the above, a * b
returns a uint
, but we were trying to store it as a uint8
, which could cause potential problems. By casting it as a uint8
, it works and the compiler won't throw an error.
Our contract is almost finished! Now let's add an event.
Events are a way for your contract to communicate that something happened on the blockchain to your app front-end, which can be 'listening' for certain events and take action when they happen.
Example:
event IntegersAdded(uint x, uint y, uint result);
function add(uint _x, uint _y) public returns (uint) {
uint result = _x + _y;
emit IntegersAdded(_x, _y, result);
return result;
}
Your app front-end could then listen for the event. A JavaScript implementation would look something like:
YourContract.IntegersAdded(function(error, result) {
})
Solidity contract is complete! Now we need to write a JavaScript frontend that interacts with the contract.
Ethereum has a JavaScript library called Web3.js.
Addresses
The Ethereum blockchain is made up of accounts, which you can think of like bank accounts. An account has a balance of Ether (the currency used on the Ethereum blockchain), and you can send and receive Ether payments to other accounts, just like your bank account can wire transfer money to other bank accounts.
Each account has an address, which you can think of like a bank account number. It's a unique identifier that points to that account, and it looks like this:
0x0cE446255506E92DF41614C46F1d6df9Cc969183
Understand that an address is owned by a specific user (or a smart contract).
So we can use it as a unique ID for ownership . When a user creates new entity/object by interacting with an app, will set ownership of those entity/object to the Ethereum address that called the function.
Mappings
Mappings are another way of storing organized data in Solidity.
Defining a mapping looks like this:
// For a financial app, storing a uint that holds the user's account balance:
mapping (address => uint) public accountBalance;
// Or could be used to store / lookup usernames based on userId
mapping (uint => string) userIdToName;
A mapping is essentially a key-value store for storing and looking up data. In the first example, the key is an address and the value is a uint, and in the second example the key is a uint and the value a string.
msg.sender
In Solidity, there are certain global variables that are available to all functions. One of these is msg.sender
, which refers to the address
of the person (or smart contract) who called the current function.
Note: In Solidity, function execution always needs to start with an external caller. A contract will just sit on the blockchain doing nothing until someone calls one of its functions. So there will always be a msg.sender
.
Here's an example of using msg.sender
and updating a mapping
:
mapping (address => uint) favoriteNumber;
function setMyNumber(uint _myNumber) public {
favoriteNumber[msg.sender] = _myNumber;
}
function whatIsMyNumber() public view returns (uint) {
return favoriteNumber[msg.sender];
}
In this trivial example, anyone could call setMyNumber
and store a uint
in our contract, which would be tied to their address. Then when they called whatIsMyNumber
, they would be returned the uint
that they stored.
Using msg.sender
gives you the security of the Ethereum blockchain — the only way someone can modify someone else's data would be to steal the private key associated with their Ethereum address.
Require
How can we make function can only be called once per player?
For that we use require
. require
makes it so that the function will throw an error and stop executing if some condition is not true:
function sayHiToVitalik(string memory _name) public returns (string memory) {
require(keccak256(abi.encodePacked(_name)) == keccak256(abi.encodePacked("Vitalik")));
return "Hi!";
}
If you call this function with sayHiToVitalik("Vitalik")
, it will return "Hi!". If you call it with any other input, it will throw an error and not execute.
Thus require
is quite useful for verifying certain conditions that must be true before running a function.
One feature of Solidity that makes this more manageable is contract inheritance:
contract Doge {
function catchphrase() public returns (string memory) {
return "So Wow CryptoDoge";
}
}
contract BabyDoge is Doge {
function anotherCatchphrase() public returns (string memory) {
return "Such Moon BabyDoge";
}
}
BabyDoge
inherits from Doge
. That means if you compile and deploy BabyDoge
, it will have access to both catchphrase()
and anotherCatchphrase()
(and any other public functions we may define on Doge
).
This can be used for logical inheritance (such as with a subclass, a Cat
is an Animal
). But it can also be used simply for organizing your code by grouping similar logic together into different contracts.
Import
When you have multiple files and you want to import one file into another, Solidity uses the import
keyword:
import "./someothercontract.sol";
contract newContract is SomeOtherContract {
}
So if we had a file named someothercontract.sol
in the same directory as this contract (that's what the ./
means), it would get imported by the compiler.
Storage vs Memory (Data location)
In Solidity, there are two locations you can store variables — in storage
and in memory
.
Storage refers to variables stored permanently on the blockchain. Memory variables are temporary, and are erased between external function calls to your contract. Think of it like your computer's hard disk vs RAM.
Most of the time you don't need to use these keywords because Solidity handles them by default. State variables (variables declared outside of functions) are by default storage
and written permanently to the blockchain, while variables declared inside functions are memory
and will disappear when the function call ends.
However, there are times when you do need to use these keywords, namely when dealing with structs and arrays within functions:
contract SandwichFactory {
struct Sandwich {
string name;
string status;
}
Sandwich[] sandwiches;
function eatSandwich(uint _index) public {
Sandwich storage mySandwich = sandwiches[_index];
mySandwich.status = "Eaten!";
Sandwich memory anotherSandwich = sandwiches[_index + 1];
anotherSandwich.status = "Eaten!";
sandwiches[_index + 1] = anotherSandwich;
}
}