Thanks for your interest in improving the Farcaster Hub!
No contribution is too small and we welcome your help. There's always something to work on, no matter how experienced you are. If you're looking for ideas, start with the good first issue or help wanted sections in the issues. You can help make Farcaster better by:
- Opening issues or adding details to existing issues
- Fixing bugs in the code
- Making tests or the ci builds faster
- Improving the documentation
- Keeping packages up-to-date
- Proposing and implementing new features
Before you get down to coding, take a minute to consider this:
- If your proposal modifies the farcaster protocol, open an issue there first.
- If your proposal is a non-trivial change, consider opening an issue first to get buy-in.
- If your issue is a small bugfix or improvement, you can simply make the changes and open the PR.
First, ensure that the following are installed globally on your machine:
Then, from the root folder run:
yarn install
to install dependenciesyarn build
to build Hubble and its dependenciesyarn test
to ensure that the test suite runs correctly
All commits need to be signed with a GPG key. This adds a second factor of authentication that proves that it came from you, and not someone who managed to compromise your GitHub account. You can enable signing by following these steps:
-
Generate GPG Keys and upload them to your Github account, GPG Suite is recommended for OSX
-
Use
gpg-agent
to remember your password locally
vi ~/.gnupg/gpg-agent.conf
default-cache-ttl 100000000
max-cache-ttl 100000000
-
Configure Git to use your keys when signing.
-
Configure Git to always sign commits by running
git config --global commit.gpgsign true
-
Commit all changes with your usual git commands and you should see a
Verified
badge near your commits
The repository is a monorepo with a primary application in the /apps/
folder that imports several packages /packages/
. It is composed of yarn workspaces and uses TurboRepo as its build system.
You can run commands like yarn test
and yarn build
which TurboRepo will automatically parallelize and execute across all workspaces. To execute the application, you'll need to navigate into the app folder and follow the instructions there. The TurboRepo documentation covers other important topics like:
TurboRepo uses a local cache which can be disabled by adding the --force
option to yarn commands. Remote caching is not enabled since the performance gains at our scale are not worth the cost of introducing subtle caching bugs.
When proposing a change, make sure that you've followed all of these steps before you ask for a review.
All changes that involve features or bugfixes should be accompanied by tests, and remember that:
- Unit tests should live side-by-side with code as
foo.test.ts
- Tests that span multiple files should live in
src/test/
under the appropriate subfolder - Tests should use factories instead of stubs wherever possible.
- Critical code paths should have 100% test coverage, which you can check in the Coveralls CI.
If your PR has changes to gRPC or protobuf files, you must update the public documentation website. See the Protobuf README for instructions on how to auto-gen the documentation.
All PR's should have supporting documentation that makes reviewing and understanding the code easy. You should:
- Update high-level changes in the contract docs.
- Always use TSDoc style comments for functions, variables, constants, events and params.
- Prefer single-line comments
/** The comment */
when the TSDoc comment fits on a single line. - Always use regular comments
//
for inline commentary on code. - Comments explaining the 'why' when code is not obvious.
- Do not comment obvious changes (e.g.
starts the db
before the linedb.start()
) - Add a
Safety: ..
comment explaining every use ofas
. - Prefer active, present-tense doing form (
Gets the connection
) over other forms (Connection is obtained
,Get the connection
,We get the connection
,will get a connection
)
Errors are not handled using throw
and try / catch
as is common with Javascript programs. This pattern makes it hard for people to reason about whether methods are safe which leads to incomplete test coverage, unexpected errors and less safety. Instead we use a more functional approach to dealing with errors. See this issue for the rationale behind this approach.
All errors must be constructed using the HubError
class which extends Error. It is stricter than error and requires a Hub Error Code (e.g. unavailable.database_error
) and some additional context. Codes are used a replacement for error subclassing since they can be easily serialized over network calls. Codes also have multiple levels (e.g. database_error
is a type of unavailable
) which help with making decisions about error handling.
Functions that can fail should always return HubResult
which is a type that can either be the desired value or an error. Callers of the function should inspect the value and handle the success and failure case explicitly. The HubResult is an alias over neverthrow's Result. If you have never used a language where this is common (like Rust) you may want to start with the API docs. This pattern ensures that:
- Readers can immediately tell whether a function is safe or unsafe
- Readers know the type of error that may get thrown
- Authors can never accidentally ignore an error.
We also enforce the following rules during code reviews:
Always return HubResult<T>
instead of throwing if the function can fail
// incorrect usage
const parseMessage = (message: string): Uint8Array => {
if (message == '') throw new HubError(...);
return message;
};
// correct usage
const parseMessage = (message: string): HubResult<Uint8Array> => {
if (message == '') return new HubError(...)
return ok(message);
};
Always wrap external calls with Result.fromThrowable
or ResultAsync.fromPromise
and wrap external an Error
into a HubError
.
// incorrect usage
const parseMessage = (message: string): string => {
try {
return JSON.parse(message);
} catch (err) {
return err as Error;
}
};
// correct usage: wrap the external call for safety
const parseMessage = (message: string): HubResult<string> => {
return Result.fromThrowable(
() => JSON.parse(message),
(err) => new HubError('bad_request.parse_failure', err as Error)
)();
};
// correct usage: build a convenience method so you can call it easily
const safeJsonStringify = Result.fromThrowable(
JSON.stringify,
() => new HubError('bad_request', 'json stringify failure')
);
const result = safeJsonStringify(json);
Prefer result.match
to handle HubResult since it is explicit about how all branches are handled
const result = computationThatMightFail().match(
(str) => str.toUpperCase(),
(error) => err(error)
);
Only use isErr()
in cases where you want to short-circuit early on failure and refactoring is unwieldy or not performant
public something(): HubResult<number> {
const result = computationThatMightFail();
if (result.isErr()) return err(new HubError('unavailable', 'down'));
// do a lot of things that would be unwieldy to put in a match
// ...
// ...
return ok(200);
}
Use _unsafeUnwrap()
and _unsafeUnwrapErr()
in tests to assert results
// when expecting an error
const error = foo()._unsafeUnwrapErr();
expect(error.errCode).toEqual('bad_request');
expect(error.message).toMatch('invalid AddressInfo family');
Prefer combine
and combineWithAllErrors
when operating on multiple results
const results = await Promise.all(things.map((thing) => foo(thing)));
// 1. Only fail if all failed
const combinedResults = Result.combineWithAllErrors(results) as Result<void[], HubError[]>;
if (combinedResults.isErr() && combinedResults.error.length == things.length) {
return err(new HubError('unavailable', 'could not connect to any bootstrap nodes'));
}
// 2. Fail if at least one failed
const combinedResults = Result.combine(results);
if (combinedResults.isErr()) {
return err(new HubError('unavailable', 'could not connect to any bootstrap nodes'));
}
All submissions must be opened as a Pull Request and reviewed and approved by a project member. The CI build process will ensure that all tests pass and that all linters have been run correctly. In addition, you should ensure that:
- The PR titles must follow the Conventional Commits specification
- Commit titles should follow the Conventional Commits specification
As an example, a good PR title would look like this:
fix(signers): validate signatures correctly
While a good commit message might look like this:
fix(signers): validate signatures correctly
Called Signer.verify with the correct parameter to ensure that older signature
types would not pass verification in our Signer Sets
All PRs with meaningful changes should have a changeset which is a short description of the modifications being made to each package. Changesets are automatically converted into a changelog when the repo manager runs a release process.
- Run
yarn changeset
to start the process - Select the packages being modified with the space key
- Select minor version if breaking change or patch otherwise, since we haven't release 1.0 yet
- Commit the generates files into your branch.
Permissions to publish to the @farcaster organization in NPM is necessary. This is a non-reversible process so if you are at all unsure about how to proceed, please reach out to Varun (Github | Warpcast)
- Checkout a new branch and run
yarn changeset version
- Review CHANGELOG.md and confirm that it is accurate
- Check that
package.json
was bumped correctly - If protocol version change, bump
FARCASTER_VERSIONS_SCHEDULE
andFARCASTER_VERSION
- Create commit, merge to main, check out commit on main and run
yarn build
- Publish changes by running
yarn changeset publish
- Fetch and update tags with
git fetch origin --tags && yarn changeset tag && git tag -f @latest
- Delete the biome tags
git tag -d [email protected]
- Push tags with
git push upstream HEAD --tags -f
- If docker build does not start
git push upstream --delete @farcaster/hubble@<version> && git push upstream --tags @farcaster/hubble@<version>
to re-trigger it. - Create a GitHub Release for Hubble, copying over the changelog and marking it as the latest.
- If this is a non-patch change, create an NFT for the release.
- Make sure that the Docker image for the latest release gets built and published to Docker hub.
Some of the CPU intensive code is written in Rust for speed. We import the Rust modules via Neon that are built as a part of the @farcaster/core
package.
To add new code to Rust,
- Add it to
packages/core/src/addon/
- Add a bridge implementation and types into
packages/core/src/addon/addon.js
andpackages/core/src/addon/addon.d.ts
- Export the callable typescript function in
packages/core/src/rustfunctions.ts
. This function can then be used throught the project to transparently call into Rust from Typescript
One-time changes to RocksDB and the data stored can be made as a part of migrations. Migrations are scripts that are run once, at startup. They are run blocking, so you can safely make big changes before the Hub starts running.
- Create a new function in
apps/hubble/src/db/migrations/
that executes the change and its associated tests - Increment
LATEST_DB_SCHEMA_VERSION
inmigrations.ts
- Add a new entry to the
migrations
constant with the migration number and the function to execute the migration inmigrations.ts
When users upgrade their hub and restart, the migration is executed at startup.
- Pick a libp2p release and navigate to its package.json file
- Copy the required versions of
libp2p
,@libp2p/*
,@chainsafe/*
@multiformats/*
packages to our package.json - For unspecified packages read their changelog and make a best guess about versions (e.g.
@chainsafe/libp2p-gossipsub
and@libp2p/pubsub-peer-discovery
) - Follow the migration guide for the versions you are upgrading to
If you run into any unexpected issues open a discussion in the libp2p forum. @achingbrain on the Filecoin slack maintains this project and can be helpful with major issues.
- Use
npm adduser
to log into the account that can publish to @farcaster on npm - Make a branch, run
yarn changeset version
and merge the changes into main - Pull latest main, run
yarn changeset publish