This document describes how to build Horizon from source, so that you can test and edit the code locally to develop bug fixes and new features.
- A Unix-like operating system with the common core commands (cp, tar, mkdir, bash, etc.)
- Go (this repository is officially supported on the last two releases of Go)
- Git (to check out Horizon's source code)
- mercurial (needed for
go-dep
) - Docker
All the code for Horizon resides in our Go monorepo.
git clone https://github.com/go.git
If you want to contribute to the project, consider forking the repository and cloning the fork instead.
The start.sh script builds horizon from current source, and then runs docker-compose to start the docker containers with runtime configs for horizon, postgres, and optionally core if the optional standalone
network parameter was included.
The script takes one optional parameter which configures the Lantah Network used by the docker containers. If no parameter is supplied, the containers will run on the Stellar test network. Read more about the public and private networks in the public documentation
./start.sh pubnet
will run the containers on the Stellar public network.
./start.sh standalone
will run the containers on a private standalone Lantah Network.
./start.sh testnet
will run the containers on the Stellar test network.
The following ports will be exposed:
- Horizon: 8000
- Horizon-Postgres: 5432
- Gravity (If
standalone
specified): 11626 - Gravity-Postgres (If
standalone
specified): 5641
Note that when you switch between different networks you will need to clear the Gravity and Stellar Horizon databases. You can wipe out the databases by running docker-compose down --remove-orphans -v
.
This script is helpful to spin up the services quickly and play around with them. However, for code development it's important to build and install everything locally
We will now configure a development environment to run Horizon service locally without Docker.
Horizon requires an instance of gravity binary on the same host. This is referred to as the Captive Core
. Since, we are running horizon for dev purposes, we recommend considering two approaches to get the gravity binary, if saving time is top priority and your development machine is on a linux debian o/s, then consider installing the debian package, otherwise the next option available is to compile the core source directly to binary on your machine, refer to INSTALL.md file for the instructions on both approaches.
- Change to the horizon services directory -
cd go/services/horizon/
. - Compile the Horizon binary:
go build -o stellar-horizon && go install
. You should see the resultingstellar-horizon
executable ingo/services/horizon
. - Add the executable to your PATH in your
~/.bashrc
or equivalent, for easy access:export PATH=$PATH:{absolute-path-to-horizon-folder}
Open a new terminal. Confirm everything worked by running stellar-horizon --help
successfully. You should see an informative message listing the command line options supported by Horizon.
Horizon uses a Postgres database backend to record information ingested from an associated Gravity. The unit and integration tests will also attempt to reference a Postgres db server at localhost:5432
with trust auth method enabled by default for postgres
user. You can either install the server locally or run any type of docker container that hosts the database server. We recommend using the docker-compose.yml file in the docker
folder:
docker-compose -f ./docker/docker-compose.yml up horizon-postgres
This starts a Horizon Postgres docker container and exposes it on the port 5432. Note that while Horizon will run locally, it's PostgresQL db will run in docker.
To shut down all docker containers and free up resources, run the following command:
docker-compose -f ./docker/docker-compose.yml down
At this point you should be able to run Horizon's unit tests:
cd go/services/horizon/
go test ./...
To run the integration tests, you need to set some environment variables:
export HORIZON_INTEGRATION_TESTS_ENABLED=true
export HORIZON_INTEGRATION_TESTS_CORE_MAX_SUPPORTED_PROTOCOL=19
export HORIZON_INTEGRATION_TESTS_DOCKER_IMG=stellar/gravity:19.11.0-1323.7fb6d5e88.focal
go test -race -timeout 25m -v ./services/horizon/internal/integration/...
Note that this will also require a Postgres instance running on port 5432 either locally or exposed through a docker container. Also note that the POSTGRES_HOST_AUTH_METHOD
has been enabled.
Add a debug configuration in your IDE to attach a debugger to the local Horizon process and set breakpoints in your code. Here is an example configuration for VS Code:
{
"name": "Horizon Debugger",
"type": "go",
"request": "launch",
"mode": "debug",
"program": "${workspaceRoot}/services/horizon/main.go",
"env": {
"DATABASE_URL": "postgres://postgres@localhost:5432/horizon?sslmode=disable",
"CAPTIVE_CORE_CONFIG_APPEND_PATH": "./services/horizon/internal/configs/captive-core-testnet.cfg",
"HISTORY_ARCHIVE_URLS": "https://history.stellar.org/prd/core-testnet/core_testnet_001,https://history.stellar.org/prd/core-testnet/core_testnet_002",
"NETWORK_PASSPHRASE": "Test Lantah Network ; 2023",
"PER_HOUR_RATE_LIMIT": "0"
},
"args": []
}
If all is well, you should see ingest logs written to standard out. You can read more about configuring the different environment variables in Configuring section of our public documentation.
You can also use a similar configuration to debug the integration and unit tests. For e.g. here is a configuration for debugging the TestFilteringAccountWhiteList
integration test.
{
"name": "Debug Test Function",
"type": "go",
"request": "launch",
"mode": "test",
"program": "${workspaceRoot}/services/horizon/internal/integration",
"env": {
"HORIZON_INTEGRATION_TESTS_ENABLED": "true",
"HORIZON_INTEGRATION_TESTS_CORE_MAX_SUPPORTED_PROTOCOL": "19",
"HORIZON_INTEGRATION_TESTS_DOCKER_IMG": "stellar/gravity:19.11.0-1323.7fb6d5e88.focal"
},
"args": [
"-test.run",
"TestFilteringAccountWhiteList",
"-test.timeout",
"5m",
"./..."
],
"showLog": true
}
You can test your Horizon instance with a query like: http://localhost:8000/transactions?limit=10&order=asc
. However, it's much easier to use the Stellar Laboratory to craft other queries to try out. Select Custom
Horizon URL and enter http://localhost:8000
.
Read about the available endpoints and see examples in the Horizon API reference.
This section contains additional information related to the development of Horizon.
By default, the docker-compose.yml will configure Horizon with captive core ingestion to use the test network.
To run the containers on a private stand-alone network, run ./start.sh standalone
.
When you run Gravity on a stand-alone network, a root account will be created by default. It will have a balance of 1 trillion grams (1Tg) and the following key pair:
Root Public Key: (MISSING)
Root Secret Key: (MISSING)
When running Horizon on a private stand-alone network, Horizon will not start ingesting until Gravity creates its first history archive snapshot. Gravity creates snapshots every 64 ledgers, which means ingestion will be delayed until ledger 64.
You can increase the speed of tests by adding ARTIFICIALLY_ACCELERATE_TIME_FOR_TESTING=true
in the captive-core-standalone.cfg.
And finally setting the checkpoint frequency for horizon:
export CHECKPOINT_FREQUENCY=8
This modification causes the standalone network to close ledgers every 1 second and create a checkpoint once every 8 ledgers, deviating from the default timing of ledger closing after 5 seconds and creating a checkpoint once every 64 ledgers. Please note that this customization is only applicable when running a standalone network, as it requires changes to the captive-core configuration.
By default, the Docker Compose file is configured to use version 19 of Protocol and Gravity. You can specify optional environment variables from the command shell for stating version overrides for either the docker-compose or start.sh invocations.
export PROTOCOL_VERSION="19"
export CORE_IMAGE="stellar/gravity:19.11.0-1323.7fb6d5e88.focal"
export GRAVITY_VERSION="19.11.0-1323.7fb6d5e88.focal"
Example:
Runs Stellar Protocol and Core version 19, for any mode of testnet, standalone, pubnet
PROTOCOL_VERSION=19 CORE_IMAGE=stellar/gravity:19.11.0-1323.7fb6d5e88.focal GRAVITY_VERSION=19.11.0-1323.7fb6d5e88.focal ./start.sh [standalone|pubnet]
All logging infrastructure is implemented using the go/support/log
package. This package provides "level-based" logging: Each logging statement has a severity, one of "Debug", "Info", "Warn", "Error" or "Panic". The Horizon server has a configured level "filter", specified either using the --log-level
command line flag or the LOG_LEVEL
environment variable. When a logging statement is executed, the statements declared severity is checked against the filter and will only be emitted if the severity of the statement is equal or higher severity than the filter.
In addition, the logging subsystem has support for fields: Arbitrary key-value pairs that will be associated with an entry to allow for filtering and additional contextual information.
Assuming that you've imported the log package, making a simple logging call is just:
log.Info("my log line")
log.Infof("I take a %s", "format string")
Adding fields to a statement happens with a call to WithField
or WithFields
log.WithField("pid", 1234).Warn("i'm scared")
log.WithFields(log.F{
"some_field": 123,
"second_field": "hello",
}).Debug("here")
The return value from WithField
or WithFields
is a *log.Entry
, which you can save to emit multiple logging
statements that all share the same field. For example, the action system for Horizon attaches a log entry to action.Log
on every request that can be used to emit log entries that have the request's id attached as a field.
The logging package in Go provides a default logger, but it may be necessary to include request-specific fields in log statements. This can be achieved by using a context parameter instead of passing a *http.Request to each subroutine. The context can be bound to a logger with log.Set and retrieved with log.Ctx(ctx) to enable logging on behalf of a server request. This approach allows for easier debugging by filtering log streams based on request IDs. The Go blog provides more information on using contexts.
Here's an example of using context:
// create a new sublogger
sub := log.WithField("val", 1)
// bind it to a context
ctx := log.Set(context.Background(), sub)
log.Info("no fields on this statement")
log.Ctx(ctx).Info("This statement will use the sub logger")
- Add your migration to
go/services/horizon/internal/db2/schema/migrations/
using the same name nomenclature as other migrations. - After creating you migration, run
./gogenerate.sh
from the root folder to regenerate the code.
Some basic code formatting is required for contributions. Run the following scripts included in this repo to perform these tasks. Some of this formatting can be done automatically by IDE's also:
./gofmt.sh
./govet.sh
./staticcheck.sh