Connect to an Indexer¶
Before your app can upload, download, or share data with Sia, it must first connect to an indexer. An indexer acts as your application’s gateway to the Sia network. It handles:
- Verifying your app’s identity.
- One-time approval flow.
- Tracking your pinned objects and their metadata.
- Coordination with storage providers on the network.
Prerequisites¶
In order for your app to establish a connection to an indexer, you will need:
Authentication Requirements¶
Each new instance of your app will require a unique App Key, which is deterministically derived from:
- A BIP-39 recovery phrase
- Your 32-byte App ID
The resulting App Key is a public/private key pair. The public key is registered with the indexer during onboarding, while the private key should be stored securely by the app.
The BIP-39 recovery phrase should be treated as the user's master key.
- The recovery phrase must never be stored by your application, but instead stored securely by the user.
- It should be used only once during onboarding to derive the App Key.
- Your application should export and store the App Key securely for future sessions.
Example¶
import asyncio
from indexd_ffi import (
generate_recovery_phrase,
uniffi_set_event_loop,
Builder,
AppMeta,
)
async def main():
# IMPORTANT: required for UniFFI async trait callbacks (Reader/Writer/etc.)
uniffi_set_event_loop(asyncio.get_running_loop())
# Create a builder to manage the connection flow
builder = Builder("https://app.sia.storage")
# Configure your app identity details
meta = AppMeta(
id=b"your-32-byte-app-id.............",
name="My App",
description="Demo application",
service_url="https://example.com",
logo_url=None,
callback_url=None
)
# Request app connection and get the approval URL
builder = await builder.request_connection(meta)
print("Open this URL to approve the app:", builder.response_url())
# Wait for the user to approve the request
try:
builder = await builder.wait_for_approval()
except Exception as e:
raise Exception("\nApp was not approved (rejected or request expired)") from e
# Ask the user for their recovery phrase
recovery_phrase = input("\nEnter your recovery phrase (type `seed` to generate a new one): ").strip()
if recovery_phrase == "seed":
recovery_phrase = generate_recovery_phrase()
print("\nRecovery phrase:", recovery_phrase)
# Register an SDK instance with your recovery phrase.
sdk = await builder.register(recovery_phrase)
# The App Key should be exported and stored securely for future launches, but we don't demonstrate storage here.
app_key = sdk.app_key()
print("\nApp Key export (persist however your app prefers):", app_key.export())
print("\nApp Connected!")
asyncio.run(main())
🚧 Coming soon
🚧 Coming soon
🚧 Coming soon
🚧 Coming soon
🚧 Coming soon
🚧 Coming soon
Deep Dive¶
Why approval is required¶
The indexer enforces a one-time authorization step, so the user must explicitly grant your app access to their account.
After approval, the SDK can connect without user interaction using the stored app key.
App Metadata¶
During request_connection, you supply metadata that will be displayed during app approval:
id— Your 32-byte App IDname— Name of your applicationdescription— Explains the purpose of your appservice_url— The URL representing your applogo_url(optional) — An icon shown to the usercallback_url(optional) — Used if your approval flow involves redirects
Approval failures¶
Approval can fail if:
- The request expires before the user approves it
- The user declines the request (the indexer will not approve it)
- There is a network or connectivity issue while polling