tutorial

Hello World dApp

What's in it for me?

By following this tutorial, you'll be able to get more confident about:

  • Preparing a Docker image for your app with the ultimate purpose of deploying it on Golem.
  • Converting the VM image to Golem and uploading it to Golem's repository.
  • Creating a descriptor reflecting your app using YAML syntax similar to that used by docker-compose.
  • Deploying your app to Golem using dapp-runner.

Prerequisites

To follow this tutorial in full, you need to have Docker installed on your machine. If you don't have it installed, please refer to the instructions on Docker's website.

info

This step-by-step tutorial will be easier for you to follow if you previously had a chance to launch the yagna service as a requestor and have any experience building portable web applications on Docker, but you should be able to complete it without any prior experience nevertheless.

Choice of tools

The example that we're going to show you here, uses a very simple Python HTTP server.

On the other hand, if you're more acquainted with any other language or platform that enables you to easily create a simple HTTP server and pack it into a Docker image, go for it.

Just as well - if the setup and construction of such a simple app is obvious to you, feel free to jump to Converting the image to Golem

Environment preparation

For the sake of completeness, we're including the steps that prepare our Python environment. Again, you're free to do it your way and skip to Hello World dApp:

Create and activate the virtual environment

python3 -m venv ~/.envs/hello-dapps
source ~/.envs/hello-dapps/bin/activate
pip install -U pip poetry

Initialize the project

mkdir -p hello_golem/server_app
cd hello_golem/server_app/
poetry init --no-interaction --python="^3.9"

Add the requirements

In our little example here, we'll use Flask, a very robust and minimal Python back-end framework.

poetry add Flask

There, we're ready to start coding our app.

Hello world app

info

If you're lost at any moment, feel free to consult our source of the "Hello World" application available at: https://github.com/golemfactory/dapp-experiments/tree/main/05_hello_world

Similarly, instead of coding along, you may just check out the whole thing from the repository:

git clone https://github.com/golemfactory/dapp-experiments.git
cd dapp-experiments/05_hello_world

Fire up your favourite editor or IDE and navigate to the hello_golem/server_app directory that we have set up above.

If you have configured the app using poetry like we did above, you should already see the pyproject.toml and poetry.lock files.

In the directory, add hello_golem.py and just paste the following few lines which are a slightly modified version of Flask's original, minimal example:

from flask import Flask
app = Flask(__name__)
@app.route('/')
def hello_golem():
    return 'Hello from Golem!'
app.run(host="0.0.0.0")

The above code achieves the following:

  1. Import the Flask engine
  2. Define our Flask app
  3. Set up the root route for the app
  4. Launch the Flask server
info

You may wish to customize the "Hello..." message up there, to make your application (and, more importantly, its VM image) unique. Our repository is set up to reject repeated uploads of images with matching signatures so, if you encounter an issue while uploading your application later in this tutorial, please use the image hash that we'll provide you there.

warning

For local testing, we could have simply used just app.run(). That makes the server listen on the local host address (127.0.0.1) by default. We need it to bind to all addresses so that it can later work correctly in Golem.

Local test

Let's test this app locally, before we start preparing it for Golem. Launch the app:

python hello_golem.py

and then point your browser to http://127.0.0.1:5000.

You should see "Hello from Golem!" which proves that our minimal app is working correctly when launched completely stand-alone. Good!

Preparing the Docker image

We need to start with a Docker image since the only currently-supported way of creating a GVMI (Golem Virtual Machine Image) file is by providing a Docker image for conversion.

If you'd like to know more about GVMI images and about the conversion process, please refer to: Golem Images explained

Let's back-up a little and ensure that we're one directory above server_app.

cd ..

To build a docker image, we'll need a file called Dockerfile with the following contents:

FROM python:3.9-slim
RUN pip install -U pip poetry
RUN mkdir /app
COPY server_app/* /app/
WORKDIR /app
RUN poetry install --no-root
ENTRYPOINT poetry run python hello_golem.py

Let's now go through what happens there.

1). Take a stock Python Docker image (the slim version, we won't need anything more here):

FROM python:3.9-slim

2). Install the needed tools:

RUN pip install -U pip poetry

3). Copy the app contents to the image:

RUN mkdir /app
COPY server_app/* /app/

4). Install the app's requirements:

WORKDIR /app
RUN poetry install --no-root

5). Finally, set up the Docker's entrypoint:

ENTRYPOINT poetry run python hello_golem.py

Testing the Docker image

With the Dockerfile ready, we can test whether the app works inside the container:

docker build -t hello-dapps .
docker run -it -p 5000:5000 hello-dapps

Once again, after the image launches, you may connect to http://127.0.0.1:5000/ with your browser, and you should be able to see the "Hello..." message, which confirms that the app has been correctly packaged and is working as it should.

Converting the image to Golem

To be able to use our newly-created image on Golem, we need to convert it to Golem's custom format and then make the image available to providers. The easiest way to achieve the latter is to upload the image to our image repository.

Obtain the gvmkit-build tool

pip install gvmkit-build

Upload the image to the repository

gvmkit-build golem-example --push --nologin

Once the command completes, you should get a line containing the image hash, e.g.:

image already generated
on 102777844: success. hash link 3032b6e97914eb5ee87d71188180d271f04eb9472b6da0d308943b2f

The hash there: 3032b6e97914eb5ee87d71188180d271f04eb9472b6da0d308943b2f is what we're interested in.

If the command fails because the image has already been uploaded, feel free to use the hash above and follow through with the tutorial.

Prepare the Golem dApp descriptor

Now that we have the image ready, uploaded into the Golem repo, and its hash in hand, we can prepare the descriptor of the application that we're going to launch on Golem.

For more in-depth information about the dApp descriptors, please consult the appropriate section of the "Creating Golem dApps" article. Here, we're just covering the bare minimum.

Here's what it looks like:

payloads:
  hello:
    runtime: "vm"
    params:
      image_hash: "3032b6e97914eb5ee87d71188180d271f04eb9472b6da0d308943b2f"
nodes:
  hello:
    payload: "hello"
    init:
      - run:
          args: ["/bin/sh", "-c", "poetry run python hello_golem.py > /dev/null &"]
    http_proxy:
      ports:
        - "5000"

Let's add it as hello_golem.yaml in the hello_golem directory.

There are two obligatory elements in it, the payload and nodes.

The payload

The payload is the definition of what kind of activity you'd like the providers to run on your behalf.

In this case, it's just the reference to the VM image we just uploaded.

payloads:
  hello:
    runtime: "vm"
    params:
      image_hash: "3032b6e97914eb5ee87d71188180d271f04eb9472b6da0d308943b2f"

The node definition

A node entry defines the parameters of the specific instances of services that we want launched on Golem.

nodes:
  hello:
    payload: "hello"
    init:
      - run:
          args: ["/bin/sh", "-c", "poetry run python hello_golem.py > /dev/null &"]
    http_proxy:
      ports:
        - "5000"

A couple of important details here.

Firstly, given the fact that Docker's ENTRYPOINT is not yet supported by the dapp-runner, our service definition must contain any and all commands that will start our service in the container.

Moreover, as the commands included in the init must finish before the service can be treated as started, we need to put the service in the background. For this reason, we're launching our service using the shell and adding the ampersand (&) to the end of the command.

One more caveat is that we need to redirect the output stream of the launched app so that it stays running after we exit the shell.

Lastly, we're adding the http_proxy element because our service is an HTTP app which we wish to be able to talk to. There's currently no way for the service to be exposed directly on the provider's own address but we can use the local HTTP proxy functionality to expose a port on our own machine that will forward traffic to the app through the Golem Network.

The config file

There's one more piece of data that we'll need to run our application. It's the configuration of our Golem requestor that we need to supply for the dapp-runner.

Unless you want to customize your set-up, it'll be easiest to just use the default that comes with the dapp-runner, which you can get with:

curl https://raw.githubusercontent.com/golemfactory/dapp-runner/main/configs/default.yaml > golem_config.yaml

Running your app

Ensure your yagna service is started

First, let's make sure that you have your yagna service up and running and if not, execute this in another console session:

yagna service run
info

If you haven't set-up your yagna service before, please refer to our Yagna installation instruction.

Obtain the application key

Please, run:

yagna app-key list

and copy the value listed in the key column.

If the above command doesn't give you any 32-char keys, just create your app key:

shell yagna app-key create requestor

and copy the value output by this command.

Export your application key to the environment

export YAGNA_APPKEY=<your key>

Install the dapp-runner

python3 -m venv --clear ~/.envs/dapp-runner
source ~/.envs/dapp-runner/bin/activate
pip install dapp-runner

Run the app

Now you're ready to start the app on Golem.

dapp-runner start --config golem_config.yaml hello_golem.yaml

Once the app launches, you should see some status messages describing various stages of the deployment. And finally, you should see:

{"hello": {"local_proxy_address": "http://localhost:8081"}}

The port may be different on your machine, since it's assigned automatically. Copy the address that you get there and paste that into your browser.

Assuming everything went well, you've just managed to create and deploy your own decentralized application on Golem.