This article isn’t about cryptocurrency or decentralized finance. Instead, we’ll explore public EVM blockchains and how they can be used in your next project, depending on your specific needs and goals. I’ll dive into the pros, cons, and practical examples, using the 0xweb library that I’ve been working on.
It is already up and running. Simply define your data model as a contract and deploy it.
Once your data is uploaded, it remains accessible as long as the blockchain operates. I can assume that it will be far longer than your other hosting subscription.
The separation of reading and writing processes in blockchain ensures 100% uptime for read operations, especially when leveraging multiple RPC providers for redundancy.
Blockchains inherently provide a higher level of security than conventional hosting solutions. Data exploits are possible only if vulnerabilities exist in your data model’s logic.
Unless encrypted, your data remains open, accessible, and verifiable by anyone, promoting transparency.
Domain names are unnecessary for this type of backend. Instead, a list of decentralized node providers can be used, allowing client libraries to select the most efficient option for end-users.
Thanks to the features above, blockchain-based backends inherently build user trust by ensuring data security and 24/7 availability, even if project maintenance and development stops.
You can integrate other data models stored on the blockchain, or other projects can build upon your data model.
Users can leverage numerous third-party projects to monitor or automate actions, significantly expanding the possibilities of your data model.
The data can be accessed from any point in the past.
Load historical custom events or use WebSockets to listen for real-time incoming events, enabling dynamic application responses.
The “wallet” concept enables users to authenticate themselves by signing messages, providing seamless and decentralized user identification.
Users can modify or extend data in your storage based on the permissions you define. Importantly, the costs of these modifications are borne by the users themselves. By selecting a low-cost blockchain, these fees can remain negligible, often amounting to only a few cents per transaction.
Though it follows a true pay-as-you-go model, you pay just for the SLOTs you store in. Each SLOT has 32 bytes, it costs 20000 GAS to write new data or 5000 GAS to update the data. Let's take Polygon as an example, with a 30-gwei GAS price and a $0.60 POL price.
20000GAS × 30gwei = 0.008 POL × $0.60 = $0.00032
This is a lot, so the “Floppy Disk“ emoji represents the storage amounts in the best way, which means it is best suited for smaller datasets if you pay on your own. However, a unique advantage is that users can bear the costs of their own storage and actions, a feature not found in other technologies. While this approach might hinder the mass adoption of your app, it is widely accepted within the blockchain community.
Blockchain data models support functions for interacting with the data, but their computational capabilities have constraints. These limitations depend on the RPC nodes you use for read actions and the strict GAS limits imposed on write actions (transactions). While basic operations, loops, and deeper call stacks are typically manageable, the blockchain is unsuited for heavy computational workloads.
That said, given the relatively small data sizes typically involved, the existing limits are usually sufficient for most use cases.
If you're new to blockchain development, you may have heard that it’s complicated and hard to get started with. However, this is not true. Blockchain development uses familiar concepts, semantics, and syntax, making it easier to learn than it might seem.
https://github.com/0xweb-org/examples-backend
For this article, let’s create an application version manager contract. Imagine you have a desktop application that requires a backend to check for new versions and retrieve the download link whenever a new version is published. Below is the final contract, demonstrating most of the key concepts:
import { Ownable } from "@openzeppelin/contracts/access/Ownable.sol";
struct Package {
uint version;
uint timestamp;
string url;
bytes32 sha256;
}
contract AppVersionManager is Ownable {
// Events that are emitted on data updates
event NewApplicationInfo();
event NewPackage(uint version, uint timestamp);
// Custom error, when title for the application is empty
error TitleIsEmpty();
// Some application information
string public title;
// @TODO: add further application related properties if required
// Latest package
Package public package;
// Track all versions and their packages
mapping (uint => Package) public packages;
// List of all previous versions
uint[] public versions;
constructor () Ownable(msg.sender) {
}
function updateInfo(string calldata newTitle) external onlyOwner {
if (bytes(newTitle).length == 0) {
revert TitleIsEmpty();
}
title = newTitle;
emit NewApplicationInfo();
}
function updatePackage(Package calldata newPackage) external onlyOwner {
require(newPackage.version > package.version, "Newer package already published");
packages[package.version] = package;
package = newPackage;
versions.push(package.version);
emit NewPackage(package.version, block.timestamp);
}
function findPackageAtTimestamp (uint timestamp) external view returns (Package memory) {
if (package.timestamp <= timestamp) {
return package;
}
// the countdown loop to find the latest package for the timestamp
int i = int(versions.length);
while (--i > -1) {
Package memory pkg = packages[versions[uint(i)]];
if (pkg.timestamp <= timestamp) {
return pkg;
}
}
revert("No package found");
}
function getPackage (uint version) external view returns (Package memory) {
if (version == package.version) {
return package;
}
return packages[version];
}
}
Every developer can read and understand this code with minimal effort. If you’re familiar with TypeScript, most of the concepts here will already make sense. To make it even clearer, I’ve created an equivalent TypeScript example: AppVersionManager.ts 🔗.
In simple terms, a contract in Solidity can be thought of as a stateful class instance. The concepts of properties, methods, types, and inheritance are already well-known in object-oriented programming. The main concept to explain here is the onlyOwner
modifier (similar to a decorator in TypeScript).
Every blockchain account is essentially a pair of private and public keys. The account's ID, known as the address, is derived from the public key. When a transaction is executed, the sender’s address is passed as msg.sender
. Using this, we can store your address in the constructor (during contract deployment). Later, the onlyOwner
modifier ensures that only you, as the contract owner, can execute the updateInfo
and updatePackage
functions. If someone else attempts these actions, the transaction will be reverted. The onlyOwner
modifier is provided by the Ownable
contract, which is part of the widely-used OpenZeppelin library. This library includes many other useful contracts to streamline blockchain development.
Another important topic to discuss is the concept of Proxies, which split storage and implementation into two separate contracts. Contract implementations in Solidity are immutable, meaning you cannot add new functions or properties after deployment. To work around this, you can deploy a “Proxy” contract. The Proxy handles storage and contains only one fallback
function, which delegates calls to the implementation contract while maintaining the storage context of the Proxy.
This concept may sound complex, but it’s similar to how this
works in JavaScript. Here’s a quick analogy to help clarify:
const foo = new Proxy({ bar: 'Lorem' }, {
get (obj, prop) {
return fooImplementation[prop].bind(obj)
},
});
const fooImplementation = { logValue () { console.log('Bar value:', this.bar) } }
foo.logValue();
The proxy contract holds a reference to the implementation contract. If you want to add new functions, you simply deploy a new implementation contract and update the proxy to reference this new contract, forwarding function calls to the updated instance. It’s a straightforward process, but there’s an edge case to consider: constructors.
When deploying an implementation contract, its constructor operates within the storage of the implementation contract itself. This means that setters like title = "Hello World"
will not modify the proxy's storage. To address this, we use the initializer function concept:
initialize
function.initialize
method to be called in the context of the proxy contract.
As a result, updating the title
property, for example, will correctly update it in the proxy’s storage.
Here’s an upgraded implementation version of our AppVersionManager: AppVersionManagerUpgradeable.sol.
The proxy contract itself is quite universal and independent of the implementation. Several well-known standards for proxies are available in the OpenZeppelin library.
With the knowledge of these concepts and the examples above, you’re ready to develop smart contracts for your business cases.
First, we need to select the blockchain where we want to deploy our contract. For this example, I’ve chosen Polygon. It offers low transaction costs, has been around for a long time, and has consistently performed well. Its stable and efficient infrastructure, combined with a Total Value Locked (TVL) of $0.9 billion, makes it a reliable choice. Deploying your contracts to public blockchains means coexisting with financial institutions. The TVL metric reflects the trust these institutions place in the blockchain’s reliability.
Moreover, if conditions change, you can always redeploy the contract to another blockchain in the future.
The demo project also serves as the CI test repository, so all the commands can be found here: https://github.com/0xweb-org/examples-backend/blob/master/deploy-cli.sh
# Install 0xweb library from NPM into the prject folder
npm i 0xweb
# Install required dependencies to compile/deploy *.sol files
npx 0xweb init --hardhat --openzeppelin
# Create or import the account. Private key will be encrypted with pin AND machine key.
npx 0xweb accounts new --name foo --pin test --login
# Save the private key securly and ensure the account has some POL tokens
# Deploy. The foo account is selected as default.
npx 0xweb deploy ./contracts/AppVersionManager.sol --chain polygon --pin test
# Set title
npx 0xweb c write AppVersionManager updateInfo --newTitle MySuperApp --pin test
# Set latest package information
npx 0xweb c write AppVersionManager updatePackage --arg0 'load(./data/package.json)' --pin test
With just a few commands, you’ve deployed the contract and updated the data. That’s it for the backend—it's now up and running “forever” without requiring any further actions from your side. The costs for this deployment, at a GAS price of 70 gwei and a POL price of $0.51, would be:
|
GAS |
POL |
$ |
---|---|---|---|
Deploy |
850352 |
0.059 |
0.03 |
Save Title |
47517 |
0.0033 |
0.001 |
Save Package Data |
169549 |
0.0118 |
0.006 |
Total |
|
|
0.037 |
You spend just 4 cents to set up a decentralized, secure, and long-running service with no maintenance required.
To query your contract data, you’ll need RPC node providers. Dozens of free providers are available at https://chainlist.org. You can choose multiple providers, and a good Web3 library can utilize a round-robin strategy at runtime to select the most efficient one for your end users. With 0xweb, the generated TypeScript or JavaScript classes not only select the best endpoints but also abstract away all blockchain communication. The clients contain high-level methods for fetching data, making the process seamless and efficient.
# The deploy command also generates the class, but manual install is also possible
npx 0xweb i 0x<address> --name AppVersionManager --chain polygon
import { AppVersionManager } from './0xc/polygon/AppVersionManager/AppVersionManager'
const manager = new AppVersionManager();
console.log(`Title`, await manager.title());
console.log(`Package`, await manager.package());
For other programming languages, there are numerous libraries available to simplify querying the blockchain. After deployment, you’ll have the contract address and ABI (interface).
Alternatively, you can launch a middleware server to query contract data using 0xweb.
npx 0xweb server start --port 3000
curl http://localhost:3000/api/c/read/AppVersionManager/package?chain=polygon
One advantage is that you don’t need to include any libraries in your application - raw HTTP requests. However, this approach relies on an additional server that you’ll need to manage. It’s often better to query the blockchain directly using 0xweb-generated classes or other blockchain libraries available.
This article showcased how blockchains can be both simple and powerful, offering unique advantages compared to traditional hosting solutions.
In the next article, I plan to explore decentralized BLOB storage networks such as Greenfield and Arweave, highlighting their features and benefits.
If you have any suggestions or ideas for additional features to include in the 0xweb library, feel free to share them in the comments or reach out directly at tnbts@0xweb.org.