To install all of our protobuf, rust, and go toolchains install mise, then run:
mise trust
mise install
Recommended: Add mise shell integration so that the mise environment will automatically activate when you are this repo, giving you access to all executables and environment variables. Otherwise you will need will need to either manually mise activate [SHELL] or run all commands with the mise exec prefix.
We use pkg-config in the build process.
One way to install it is through brew:
brew install pkgconf
lefthook (optional)
Lefthook gives us an easy way to declare pre-commit hooks (as well as other git hooks) so you're not pushing up PRs that immediately fail in CI.
You can either install it through mise (above, recommended) or brew:
brew install lefthook
Once it's installed, run lefthook install to install the git hooks, which will automatically run
when you did git commit. You can also run the hooks manually with lefthook run pre-commit.
After modifying the proto files, you can generate the Go files with the following command:
make
Our SO implementation uses ZMQ to listen for block updates from bitcoind. Install it with:
brew install zeromq
Note: whatever bitcoind you are running will also need to have been compiled with ZMQ. The default installation via brew has ZMQ, but binaries downloaded from the bitcoin core website do not.
brew install bitcoin
We use atlas to manage our database migrations. Install via mise install.
To make a migration, follow these steps:
- Make your change to the schema, run
make ent - Generate migration files by running
./scripts/gen-migration.sh <name>: - With
run-everything.sh, the migration will be automatically applied to each operator's database. But if you want to apply a migration manually, you can run (e.g. DB name issparkoperator_0):
atlas migrate apply --dir "file://so/ent/migrate/migrations" --url "postgresql://127.0.0.1:5432/sparkoperator_0?sslmode=disable"
- Commit the migration files, and submit a PR.
If you are adding atlas migrations for the first time to an existing DB, you will need to run the migration command with the --baseline flag.
atlas migrate apply --dir "file://so/ent/migrate/migrations" --url "postgresql://127.0.0.1:5432/sparkoperator_0?sslmode=disable" --baseline 20250228224813
If spark_frost.udl file has issue with VSCode, you can add the following to your settings.json file:
"files.associations": {
"spark_frost.udl": "plaintext"
}
Golang linting uses golangci-lint, installed with mise install.
To run the linters, use either of
mise lint
golangci-lint run
We use Zap for logging in the SO. Here are some best practices:
- Zap includes APIs both for structured logging (with key-value pairs) and unstructured logging
(with
fmt-style formatting), called "Sugared" and "Unsugared" respectively. Since we index structured attribute keys in our logging backend, they should be reserved for common attributes that we want searchable. For example:error: we use this for logging all errors (throughzap.Error).identity_public_key: we use this for logging the identity of the public key that is making a particular request. It should not be used for logging one-off attributes that don't have a clear purpose or common meaning across the codebase, i.e.count,value, etc. Use unstructured logging for those. If you're unsure, err towards unstructured logging.
- Converting between
zap.Loggerandzap.SugaredLoggeris easy and cheap. UseLogger.Sugar()to go fromzap.Loggertozap.SugaredLogger, andSugaredLogger.Desugar()to go the other way. - Most places in the codebase have a
zap.Loggerinjected into theContextwith common fields already present. This includes all requests, tasks, and most other major components (i.e. chainwatcher). Uselogging.GetLoggerFromContextto get the logger instance from the context. - Structured and unstructured logging can be mixed for a given log message. For example, the following example is perfectly acceptable:
logger := logging.GetLoggerFromContext(ctx)
...
logger.With(zap.Error(err)).Sugar().Infof("Failed to broadcast node tx for node %s", node.ID)- Be mindful with what you log. Be extremely selective on logs that are logged on every request (i.e. logs in middleware). Logs that describe regular execution are discouraged, instead logs should be reserved for important events (abnormalities or errors). This is especially true on high-traffic endpoints (i.e. read endpoints).
mise test-go # works from any directory
mise test # works from the spark folder
or
In spark folder, run:
go test $(go list ./... | grep -v -E "so/grpc_test|so/tree")
The E2E test environment can be run locally via ./run-everything.sh.
brew install tmux
brew install sqlx-cli # required for LRC20 Node
brew install cargo # required for LRC20 Node
See bitcoin section above.
You also need to enable TCP/IP connections to the database.
You might need to edit the following files found in your postgres data directory. If you installed postgres via homebrew, it is probably in /usr/local/var/postgres. If you can connect to the database via psql, you can find the data directory by running psql -U postgres -c "SHOW data_directory;".
A sample postgresql.conf:
hba_file = './pg_hba.conf'
ident_file = './pg_ident.conf'
listen_addresses = '*'
log_destination = 'stderr'
log_line_prefix = '[%p] '
port = 5432
A sample pg_hba.conf:
#type database user address method
local all all trust
host all all 127.0.0.1/32 trust
host all all ::1/128 trust
Golang integration tests are in the spark/so/grpc_test folder.
JS SDK integration tests are across the different JS packages, but can be run together in the js/sdks directory, or in each package's own directory, via yarn test:integration.
In the root folder, run:
# Local environment
./run-everything.sh
then run your tests
mise test-grpc # from anywhere in the repo
# OR
go test -failfast=false -p=2 ./so/grpc_test/... # in the spark folder
# OR
gotestsum --format testname --rerun-fails ./so/grpc_test/... # if you want prettier results and retries
In the sdks/js folder, you can run:
yarn install
yarn build
yarn test:integration
- For local testing, operator (go) and signer (rust) logs are found in
_data/run_X/logs. - If you don't want to deal with
tmuxcommands yourself, you can easily interact with tmux using theiterm2GUI and tmux control mode. From withiniterm2, you can run:
tmux -CC attach -t operator
-
The first time you run
run-everything.shit will take a while to start up. You might actually need to run it a couple of times for everything to work properly. Attach to theoperatorsession and check out the logs. -
Having trouble with mise? You can always run
mise implodeand it will remove mise entirely so you can start over.
To run the server with local changes go to your sparkcore folder in the lightspark webdev project:
export QUART_APP=sparkcore.server
export QUART_ENV=development
export GRPC_DNS_RESOLVER=native
export QUART_CONFIG=core-dev.py
# run this line if accessed AWS PROD earlier, since PROD credentials supersede dev credentials.
unset AWS_SESSION_TOKEN AWS_ACCESS_KEY_ID AWS_SECRET_ACCESS_KEY AWS_TOKEN_EXPIRATION
aws sso login
QUART_CONFIG=core-dev.py uv run quart -A sparkcore.server -E development run
Follow commented direction here: https://github.com/lightsparkdev/webdev/blob/150dd1ceecf85e122ec3ebd608d3c9f7c44f1969/sparkcore/sparkcore/spark/handlers/__tests__/test_transfer_v2.py#L35
Update the sspClientOptions for dev-regtest-config.json
"sspClientOptions": {
"baseUrl": "http://127.0.0.1:5000",
"schemaEndpoint": "graphql/spark/rc",
"identityPublicKey": "028c094a432d46a0ac95349d792c2e3730bd60c29188db716f56a99e39b95338b4"
}
Running yarn cli:dev will now point to you local SSP server and will be connected to dev SOs and dev dbs.
Releases are triggered by creating a release, which kicks off the Release workflow.
Every hour we build a Release Candidate. The Release Candidate does the following:
- Chooses the most recent SO build that has successfully gone through CI.
- Runs all hermetic tests against the build.
- Restarts the SOs and LRC20 node in Loadtest with the build.
- (Soon) runs Artillery tests.
In regular circumstances (i.e. unless fixing a production issue), releasing from the release candidate is greatly preferred. You can create a release from the release candidate here.
Each release has a name, which should follow prod-spark-YYYY.MM.DD.N, where N starts at 1 and
increments for each release on a given day.
