- Add source_host/source_port parameters for flexible block polling configuration - Add import_host/import_port parameters to separate import target from source - Bypass rate limiting for localhost traffic (127.0.0.1, ::1) in middleware - Increase rate limit from 200 to 5000 requests per 60s for RPC throughput - Add receiver ready event to prevent dropping initial block broadcasts - Add special handling for 429 rate
880 lines
33 KiB
Python
Executable File
880 lines
33 KiB
Python
Executable File
from __future__ import annotations
|
|
from sqlalchemy import func
|
|
|
|
import asyncio
|
|
import json
|
|
import time
|
|
from typing import Any, Dict, Optional
|
|
|
|
from fastapi import APIRouter, HTTPException, status
|
|
from pydantic import BaseModel, Field, model_validator
|
|
from sqlmodel import select
|
|
|
|
from ..database import session_scope
|
|
from ..gossip import gossip_broker
|
|
from ..mempool import get_mempool
|
|
from ..metrics import metrics_registry
|
|
from ..models import Account, Block, Receipt, Transaction
|
|
|
|
router = APIRouter()
|
|
|
|
def get_chain_id(chain_id: str = None) -> str:
|
|
"""Get chain_id from parameter or use default from settings"""
|
|
if chain_id is None:
|
|
from ..config import settings
|
|
return settings.chain_id
|
|
return chain_id
|
|
|
|
|
|
def _serialize_receipt(receipt: Receipt) -> Dict[str, Any]:
|
|
return {
|
|
"receipt_id": receipt.receipt_id,
|
|
"job_id": receipt.job_id,
|
|
"payload": receipt.payload,
|
|
"miner_signature": receipt.miner_signature,
|
|
"coordinator_attestations": receipt.coordinator_attestations,
|
|
"minted_amount": receipt.minted_amount,
|
|
"recorded_at": receipt.recorded_at.isoformat(),
|
|
}
|
|
|
|
|
|
class TransactionRequest(BaseModel):
|
|
type: str = Field(description="Transaction type, e.g. TRANSFER or RECEIPT_CLAIM")
|
|
sender: str
|
|
nonce: int
|
|
fee: int = Field(ge=0)
|
|
payload: Dict[str, Any]
|
|
sig: Optional[str] = Field(default=None, description="Signature payload")
|
|
|
|
@model_validator(mode="after")
|
|
def normalize_type(self) -> "TransactionRequest": # type: ignore[override]
|
|
normalized = self.type.upper()
|
|
if normalized not in {"TRANSFER", "RECEIPT_CLAIM"}:
|
|
raise ValueError(f"unsupported transaction type: {self.type}")
|
|
self.type = normalized
|
|
return self
|
|
|
|
|
|
class ReceiptSubmissionRequest(BaseModel):
|
|
sender: str
|
|
nonce: int
|
|
fee: int = Field(ge=0)
|
|
payload: Dict[str, Any]
|
|
sig: Optional[str] = None
|
|
|
|
|
|
class EstimateFeeRequest(BaseModel):
|
|
type: Optional[str] = None
|
|
payload: Dict[str, Any] = Field(default_factory=dict)
|
|
|
|
|
|
|
|
@router.get("/head", summary="Get current chain head")
|
|
async def get_head(chain_id: str = None) -> Dict[str, Any]:
|
|
"""Get current chain head"""
|
|
from ..config import settings as cfg
|
|
|
|
# Use default chain_id from settings if not provided
|
|
if chain_id is None:
|
|
chain_id = cfg.chain_id
|
|
|
|
metrics_registry.increment("rpc_get_head_total")
|
|
start = time.perf_counter()
|
|
with session_scope() as session:
|
|
result = session.exec(select(Block).where(Block.chain_id == chain_id).order_by(Block.height.desc()).limit(1)).first()
|
|
if result is None:
|
|
metrics_registry.increment("rpc_get_head_not_found_total")
|
|
raise HTTPException(status_code=status.HTTP_404_NOT_FOUND, detail="no blocks yet")
|
|
metrics_registry.increment("rpc_get_head_success_total")
|
|
metrics_registry.observe("rpc_get_head_duration_seconds", time.perf_counter() - start)
|
|
return {
|
|
"height": result.height,
|
|
"hash": result.hash,
|
|
"timestamp": result.timestamp.isoformat(),
|
|
"tx_count": result.tx_count,
|
|
}
|
|
|
|
|
|
@router.get("/blocks/{height}", summary="Get block by height")
|
|
async def get_block(height: int) -> Dict[str, Any]:
|
|
metrics_registry.increment("rpc_get_block_total")
|
|
start = time.perf_counter()
|
|
with session_scope() as session:
|
|
block = session.exec(select(Block).where(Block.height == height)).first()
|
|
if block is None:
|
|
metrics_registry.increment("rpc_get_block_not_found_total")
|
|
raise HTTPException(status_code=status.HTTP_404_NOT_FOUND, detail="block not found")
|
|
metrics_registry.increment("rpc_get_block_success_total")
|
|
metrics_registry.observe("rpc_get_block_duration_seconds", time.perf_counter() - start)
|
|
return {
|
|
"height": block.height,
|
|
"hash": block.hash,
|
|
"parent_hash": block.parent_hash,
|
|
"proposer": block.proposer,
|
|
"timestamp": block.timestamp.isoformat(),
|
|
"tx_count": block.tx_count,
|
|
"state_root": block.state_root,
|
|
}
|
|
|
|
|
|
@router.get("/blocks-range", summary="Get blocks in height range")
|
|
async def get_blocks_range(start: int, end: int) -> Dict[str, Any]:
|
|
metrics_registry.increment("rpc_get_blocks_range_total")
|
|
start_time = time.perf_counter()
|
|
|
|
# Validate parameters
|
|
if start < 0:
|
|
raise HTTPException(status_code=status.HTTP_400_BAD_REQUEST, detail="start must be non-negative")
|
|
if end < start:
|
|
raise HTTPException(status_code=status.HTTP_400_BAD_REQUEST, detail="end must be greater than or equal to start")
|
|
if end - start > 1000:
|
|
raise HTTPException(status_code=status.HTTP_400_BAD_REQUEST, detail="range cannot exceed 1000 blocks")
|
|
|
|
with session_scope() as session:
|
|
# Get blocks in the specified height range (ascending order by height)
|
|
blocks = session.exec(
|
|
select(Block)
|
|
.where(Block.height >= start)
|
|
.where(Block.height <= end)
|
|
.order_by(Block.height.asc())
|
|
).all()
|
|
|
|
if not blocks:
|
|
metrics_registry.increment("rpc_get_blocks_range_empty_total")
|
|
return {
|
|
"blocks": [],
|
|
"start": start,
|
|
"end": end,
|
|
"count": 0,
|
|
}
|
|
|
|
# Serialize blocks
|
|
block_list = []
|
|
for block in blocks:
|
|
block_list.append({
|
|
"height": block.height,
|
|
"hash": block.hash,
|
|
"parent_hash": block.parent_hash,
|
|
"proposer": block.proposer,
|
|
"timestamp": block.timestamp.isoformat(),
|
|
"tx_count": block.tx_count,
|
|
"state_root": block.state_root,
|
|
})
|
|
|
|
metrics_registry.increment("rpc_get_blocks_range_success_total")
|
|
metrics_registry.observe("rpc_get_blocks_range_duration_seconds", time.perf_counter() - start_time)
|
|
|
|
return {
|
|
"blocks": block_list,
|
|
"start": start,
|
|
"end": end,
|
|
"count": len(block_list),
|
|
}
|
|
|
|
|
|
@router.get("/tx/{tx_hash}", summary="Get transaction by hash")
|
|
async def get_transaction(tx_hash: str, chain_id: str = None) -> Dict[str, Any]:
|
|
chain_id = get_chain_id(chain_id)
|
|
metrics_registry.increment("rpc_get_transaction_total")
|
|
start = time.perf_counter()
|
|
with session_scope() as session:
|
|
tx = session.exec(select(Transaction).where(Transaction.chain_id == chain_id).where(Transaction.tx_hash == tx_hash)).first()
|
|
if tx is None:
|
|
metrics_registry.increment("rpc_get_transaction_not_found_total")
|
|
raise HTTPException(status_code=status.HTTP_404_NOT_FOUND, detail="transaction not found")
|
|
metrics_registry.increment("rpc_get_transaction_success_total")
|
|
metrics_registry.observe("rpc_get_transaction_duration_seconds", time.perf_counter() - start)
|
|
return {
|
|
"tx_hash": tx.tx_hash,
|
|
"block_height": tx.block_height,
|
|
"sender": tx.sender,
|
|
"recipient": tx.recipient,
|
|
"payload": tx.payload,
|
|
"created_at": tx.created_at.isoformat(),
|
|
}
|
|
|
|
|
|
@router.get("/transactions", summary="Get latest transactions")
|
|
async def get_transactions(limit: int = 20, offset: int = 0) -> Dict[str, Any]:
|
|
metrics_registry.increment("rpc_get_transactions_total")
|
|
start = time.perf_counter()
|
|
|
|
# Validate parameters
|
|
if limit < 1 or limit > 100:
|
|
raise HTTPException(status_code=status.HTTP_400_BAD_REQUEST, detail="limit must be between 1 and 100")
|
|
if offset < 0:
|
|
raise HTTPException(status_code=status.HTTP_400_BAD_REQUEST, detail="offset must be non-negative")
|
|
|
|
with session_scope() as session:
|
|
# Get transactions in descending order (newest first)
|
|
transactions = session.exec(
|
|
select(Transaction)
|
|
.order_by(Transaction.created_at.desc())
|
|
.offset(offset)
|
|
.limit(limit)
|
|
).all()
|
|
|
|
# Get total count for pagination info using optimized SQL count
|
|
total_count = session.exec(select(func.count()).select_from(Transaction)).one()
|
|
|
|
|
|
if not transactions:
|
|
metrics_registry.increment("rpc_get_transactions_empty_total")
|
|
return {
|
|
"transactions": [],
|
|
"total": total_count,
|
|
"limit": limit,
|
|
"offset": offset,
|
|
}
|
|
|
|
# Serialize transactions
|
|
tx_list = []
|
|
for tx in transactions:
|
|
tx_list.append({
|
|
"tx_hash": tx.tx_hash,
|
|
"block_height": tx.block_height,
|
|
"sender": tx.sender,
|
|
"recipient": tx.recipient,
|
|
"payload": tx.payload,
|
|
"created_at": tx.created_at.isoformat(),
|
|
})
|
|
|
|
metrics_registry.increment("rpc_get_transactions_success_total")
|
|
metrics_registry.observe("rpc_get_transactions_duration_seconds", time.perf_counter() - start)
|
|
|
|
return {
|
|
"transactions": tx_list,
|
|
"total": total_count,
|
|
"limit": limit,
|
|
"offset": offset,
|
|
}
|
|
|
|
|
|
@router.get("/receipts/{receipt_id}", summary="Get receipt by ID")
|
|
async def get_receipt(receipt_id: str) -> Dict[str, Any]:
|
|
metrics_registry.increment("rpc_get_receipt_total")
|
|
start = time.perf_counter()
|
|
with session_scope() as session:
|
|
receipt = session.exec(select(Receipt).where(Receipt.receipt_id == receipt_id)).first()
|
|
if receipt is None:
|
|
metrics_registry.increment("rpc_get_receipt_not_found_total")
|
|
raise HTTPException(status_code=status.HTTP_404_NOT_FOUND, detail="receipt not found")
|
|
metrics_registry.increment("rpc_get_receipt_success_total")
|
|
metrics_registry.observe("rpc_get_receipt_duration_seconds", time.perf_counter() - start)
|
|
return _serialize_receipt(receipt)
|
|
|
|
|
|
@router.get("/receipts", summary="Get latest receipts")
|
|
async def get_receipts(limit: int = 20, offset: int = 0) -> Dict[str, Any]:
|
|
metrics_registry.increment("rpc_get_receipts_total")
|
|
start = time.perf_counter()
|
|
|
|
# Validate parameters
|
|
if limit < 1 or limit > 100:
|
|
raise HTTPException(status_code=status.HTTP_400_BAD_REQUEST, detail="limit must be between 1 and 100")
|
|
if offset < 0:
|
|
raise HTTPException(status_code=status.HTTP_400_BAD_REQUEST, detail="offset must be non-negative")
|
|
|
|
with session_scope() as session:
|
|
# Get receipts in descending order (newest first)
|
|
receipts = session.exec(
|
|
select(Receipt)
|
|
.order_by(Receipt.recorded_at.desc())
|
|
.offset(offset)
|
|
.limit(limit)
|
|
).all()
|
|
|
|
# Get total count for pagination info
|
|
total_count = len(session.exec(select(Receipt)).all())
|
|
|
|
if not receipts:
|
|
metrics_registry.increment("rpc_get_receipts_empty_total")
|
|
return {
|
|
"receipts": [],
|
|
"total": total_count,
|
|
"limit": limit,
|
|
"offset": offset,
|
|
}
|
|
|
|
# Serialize receipts
|
|
receipt_list = []
|
|
for receipt in receipts:
|
|
receipt_list.append(_serialize_receipt(receipt))
|
|
|
|
metrics_registry.increment("rpc_get_receipts_success_total")
|
|
metrics_registry.observe("rpc_get_receipts_duration_seconds", time.perf_counter() - start)
|
|
|
|
return {
|
|
"receipts": receipt_list,
|
|
"total": total_count,
|
|
"limit": limit,
|
|
"offset": offset,
|
|
}
|
|
|
|
|
|
@router.get("/getBalance/{address}", summary="Get account balance")
|
|
async def get_balance(address: str, chain_id: str = None) -> Dict[str, Any]:
|
|
chain_id = get_chain_id(chain_id)
|
|
metrics_registry.increment("rpc_get_balance_total")
|
|
start = time.perf_counter()
|
|
with session_scope() as session:
|
|
account = session.get(Account, (chain_id, address))
|
|
if account is None:
|
|
metrics_registry.increment("rpc_get_balance_empty_total")
|
|
metrics_registry.observe("rpc_get_balance_duration_seconds", time.perf_counter() - start)
|
|
return {"address": address, "balance": 0, "nonce": 0}
|
|
metrics_registry.increment("rpc_get_balance_success_total")
|
|
metrics_registry.observe("rpc_get_balance_duration_seconds", time.perf_counter() - start)
|
|
return {
|
|
"address": account.address,
|
|
"balance": account.balance,
|
|
"nonce": account.nonce,
|
|
"updated_at": account.updated_at.isoformat(),
|
|
}
|
|
|
|
|
|
@router.get("/address/{address}", summary="Get address details including transactions")
|
|
async def get_address_details(address: str, limit: int = 20, offset: int = 0) -> Dict[str, Any]:
|
|
metrics_registry.increment("rpc_get_address_total")
|
|
start = time.perf_counter()
|
|
|
|
# Validate parameters
|
|
if limit < 1 or limit > 100:
|
|
raise HTTPException(status_code=status.HTTP_400_BAD_REQUEST, detail="limit must be between 1 and 100")
|
|
if offset < 0:
|
|
raise HTTPException(status_code=status.HTTP_400_BAD_REQUEST, detail="offset must be non-negative")
|
|
|
|
with session_scope() as session:
|
|
# Get account info
|
|
account = session.get(Account, (chain_id, address))
|
|
|
|
# Get transactions where this address is sender or recipient
|
|
sent_txs = session.exec(
|
|
select(Transaction)
|
|
.where(Transaction.sender == address)
|
|
.order_by(Transaction.created_at.desc())
|
|
.offset(offset)
|
|
.limit(limit)
|
|
).all()
|
|
|
|
received_txs = session.exec(
|
|
select(Transaction)
|
|
.where(Transaction.recipient == address)
|
|
.order_by(Transaction.created_at.desc())
|
|
.offset(offset)
|
|
.limit(limit)
|
|
).all()
|
|
|
|
# Get total counts
|
|
total_sent = session.exec(select(func.count()).select_from(Transaction).where(Transaction.sender == address)).one()
|
|
total_received = session.exec(select(func.count()).select_from(Transaction).where(Transaction.recipient == address)).one()
|
|
|
|
# Serialize transactions
|
|
serialize_tx = lambda tx: {
|
|
"tx_hash": tx.tx_hash,
|
|
"block_height": tx.block_height,
|
|
"direction": "sent" if tx.sender == address else "received",
|
|
"counterparty": tx.recipient if tx.sender == address else tx.sender,
|
|
"payload": tx.payload,
|
|
"created_at": tx.created_at.isoformat(),
|
|
}
|
|
|
|
response = {
|
|
"address": address,
|
|
"balance": account.balance if account else 0,
|
|
"nonce": account.nonce if account else 0,
|
|
"total_transactions_sent": total_sent,
|
|
"total_transactions_received": total_received,
|
|
"latest_sent": [serialize_tx(tx) for tx in sent_txs],
|
|
"latest_received": [serialize_tx(tx) for tx in received_txs],
|
|
}
|
|
|
|
if account:
|
|
response["updated_at"] = account.updated_at.isoformat()
|
|
|
|
metrics_registry.increment("rpc_get_address_success_total")
|
|
metrics_registry.observe("rpc_get_address_duration_seconds", time.perf_counter() - start)
|
|
|
|
return response
|
|
|
|
|
|
@router.get("/addresses", summary="Get list of active addresses")
|
|
async def get_addresses(limit: int = 20, offset: int = 0, min_balance: int = 0) -> Dict[str, Any]:
|
|
metrics_registry.increment("rpc_get_addresses_total")
|
|
start = time.perf_counter()
|
|
|
|
# Validate parameters
|
|
if limit < 1 or limit > 100:
|
|
raise HTTPException(status_code=status.HTTP_400_BAD_REQUEST, detail="limit must be between 1 and 100")
|
|
if offset < 0:
|
|
raise HTTPException(status_code=status.HTTP_400_BAD_REQUEST, detail="offset must be non-negative")
|
|
|
|
with session_scope() as session:
|
|
# Get addresses with balance >= min_balance
|
|
addresses = session.exec(
|
|
select(Account)
|
|
.where(Account.chain_id == chain_id)
|
|
.where(Account.balance >= min_balance)
|
|
.order_by(Account.balance.desc())
|
|
.offset(offset)
|
|
.limit(limit)
|
|
).all()
|
|
|
|
# Get total count
|
|
total_count = len(session.exec(select(Account).where(Account.chain_id == chain_id).where(Account.balance >= min_balance)).all())
|
|
|
|
if not addresses:
|
|
metrics_registry.increment("rpc_get_addresses_empty_total")
|
|
return {
|
|
"addresses": [],
|
|
"total": total_count,
|
|
"limit": limit,
|
|
"offset": offset,
|
|
}
|
|
|
|
# Serialize addresses
|
|
address_list = []
|
|
for addr in addresses:
|
|
# Get transaction counts
|
|
sent_count = session.exec(select(func.count()).select_from(Transaction).where(Transaction.chain_id == chain_id).where(Transaction.sender == addr.address)).one()
|
|
received_count = session.exec(select(func.count()).select_from(Transaction).where(Transaction.chain_id == chain_id).where(Transaction.recipient == addr.address)).one()
|
|
|
|
address_list.append({
|
|
"address": addr.address,
|
|
"balance": addr.balance,
|
|
"nonce": addr.nonce,
|
|
"total_transactions_sent": sent_count,
|
|
"total_transactions_received": received_count,
|
|
"updated_at": addr.updated_at.isoformat(),
|
|
})
|
|
|
|
metrics_registry.increment("rpc_get_addresses_success_total")
|
|
metrics_registry.observe("rpc_get_addresses_duration_seconds", time.perf_counter() - start)
|
|
|
|
return {
|
|
"addresses": address_list,
|
|
"total": total_count,
|
|
"limit": limit,
|
|
"offset": offset,
|
|
}
|
|
|
|
|
|
@router.post("/sendTx", summary="Submit a new transaction")
|
|
async def send_transaction(request: TransactionRequest, chain_id: str = None) -> Dict[str, Any]:
|
|
metrics_registry.increment("rpc_send_tx_total")
|
|
start = time.perf_counter()
|
|
mempool = get_mempool()
|
|
tx_dict = request.model_dump()
|
|
try:
|
|
tx_hash = mempool.add(tx_dict, chain_id=chain_id)
|
|
except ValueError as e:
|
|
metrics_registry.increment("rpc_send_tx_rejected_total")
|
|
raise HTTPException(status_code=400, detail=str(e))
|
|
except Exception as e:
|
|
metrics_registry.increment("rpc_send_tx_failed_total")
|
|
raise HTTPException(status_code=503, detail=f"Mempool unavailable: {e}")
|
|
recipient = request.payload.get("recipient", "")
|
|
try:
|
|
asyncio.create_task(
|
|
gossip_broker.publish(
|
|
"transactions",
|
|
{
|
|
"tx_hash": tx_hash,
|
|
"sender": request.sender,
|
|
"recipient": recipient,
|
|
"payload": request.payload,
|
|
"nonce": request.nonce,
|
|
"fee": request.fee,
|
|
"type": request.type,
|
|
},
|
|
)
|
|
)
|
|
metrics_registry.increment("rpc_send_tx_success_total")
|
|
return {"tx_hash": tx_hash}
|
|
except Exception:
|
|
metrics_registry.increment("rpc_send_tx_failed_total")
|
|
raise
|
|
finally:
|
|
metrics_registry.observe("rpc_send_tx_duration_seconds", time.perf_counter() - start)
|
|
|
|
|
|
@router.post("/submitReceipt", summary="Submit receipt claim transaction")
|
|
async def submit_receipt(request: ReceiptSubmissionRequest, chain_id: str = None) -> Dict[str, Any]:
|
|
metrics_registry.increment("rpc_submit_receipt_total")
|
|
start = time.perf_counter()
|
|
tx_payload = {
|
|
"type": "RECEIPT_CLAIM",
|
|
"sender": request.sender,
|
|
"nonce": request.nonce,
|
|
"fee": request.fee,
|
|
"payload": request.payload,
|
|
"sig": request.sig,
|
|
}
|
|
tx_request = TransactionRequest.model_validate(tx_payload)
|
|
try:
|
|
response = await send_transaction(tx_request, chain_id)
|
|
metrics_registry.increment("rpc_submit_receipt_success_total")
|
|
return response
|
|
except HTTPException:
|
|
metrics_registry.increment("rpc_submit_receipt_failed_total")
|
|
raise
|
|
except Exception:
|
|
metrics_registry.increment("rpc_submit_receipt_failed_total")
|
|
raise
|
|
finally:
|
|
metrics_registry.observe("rpc_submit_receipt_duration_seconds", time.perf_counter() - start)
|
|
|
|
|
|
@router.post("/estimateFee", summary="Estimate transaction fee")
|
|
async def estimate_fee(request: EstimateFeeRequest) -> Dict[str, Any]:
|
|
metrics_registry.increment("rpc_estimate_fee_total")
|
|
start = time.perf_counter()
|
|
base_fee = 10
|
|
per_byte = 1
|
|
payload_bytes = len(json.dumps(request.payload, sort_keys=True, separators=(",", ":")).encode())
|
|
estimated_fee = base_fee + per_byte * payload_bytes
|
|
tx_type = (request.type or "TRANSFER").upper()
|
|
metrics_registry.increment("rpc_estimate_fee_success_total")
|
|
metrics_registry.observe("rpc_estimate_fee_duration_seconds", time.perf_counter() - start)
|
|
return {
|
|
"type": tx_type,
|
|
"base_fee": base_fee,
|
|
"payload_bytes": payload_bytes,
|
|
"estimated_fee": estimated_fee,
|
|
}
|
|
|
|
|
|
|
|
class ImportBlockRequest(BaseModel):
|
|
height: int
|
|
hash: str
|
|
parent_hash: str
|
|
proposer: str
|
|
timestamp: str
|
|
tx_count: int = 0
|
|
state_root: Optional[str] = None
|
|
transactions: Optional[list] = None
|
|
|
|
|
|
@router.post("/importBlock", summary="Import a block from a remote peer")
|
|
async def import_block(request: ImportBlockRequest, chain_id: str = None) -> Dict[str, Any]:
|
|
from ..sync import ChainSync, ProposerSignatureValidator
|
|
from ..config import settings as cfg
|
|
|
|
metrics_registry.increment("rpc_import_block_total")
|
|
start = time.perf_counter()
|
|
|
|
trusted = [p.strip() for p in cfg.trusted_proposers.split(",") if p.strip()]
|
|
validator = ProposerSignatureValidator(trusted_proposers=trusted if trusted else None)
|
|
sync = ChainSync(
|
|
session_factory=session_scope,
|
|
chain_id=chain_id,
|
|
max_reorg_depth=cfg.max_reorg_depth,
|
|
validator=validator,
|
|
validate_signatures=cfg.sync_validate_signatures,
|
|
)
|
|
|
|
block_data = request.model_dump(exclude={"transactions"})
|
|
result = sync.import_block(block_data, request.transactions)
|
|
|
|
duration = time.perf_counter() - start
|
|
metrics_registry.observe("rpc_import_block_duration_seconds", duration)
|
|
|
|
if result.accepted:
|
|
metrics_registry.increment("rpc_import_block_accepted_total")
|
|
else:
|
|
metrics_registry.increment("rpc_import_block_rejected_total")
|
|
|
|
return {
|
|
"accepted": result.accepted,
|
|
"height": result.height,
|
|
"hash": result.block_hash,
|
|
"reason": result.reason,
|
|
"reorged": result.reorged,
|
|
"reorg_depth": result.reorg_depth,
|
|
}
|
|
|
|
|
|
@router.get("/syncStatus", summary="Get chain sync status")
|
|
async def sync_status(chain_id: str = None) -> Dict[str, Any]:
|
|
from ..sync import ChainSync
|
|
from ..config import settings as cfg
|
|
|
|
metrics_registry.increment("rpc_sync_status_total")
|
|
sync = ChainSync(session_factory=session_scope, chain_id=chain_id)
|
|
return sync.get_sync_status()
|
|
|
|
|
|
@router.get("/info", summary="Get blockchain information")
|
|
async def get_blockchain_info(chain_id: str = None) -> Dict[str, Any]:
|
|
"""Get comprehensive blockchain information"""
|
|
from ..config import settings as cfg
|
|
|
|
metrics_registry.increment("rpc_info_total")
|
|
start = time.perf_counter()
|
|
|
|
with session_scope() as session:
|
|
# Get chain stats
|
|
head_block = session.exec(select(Block).order_by(Block.height.desc()).limit(1)).first()
|
|
total_blocks_result = session.exec(select(func.count(Block.height))).first()
|
|
total_blocks = total_blocks_result if isinstance(total_blocks_result, int) else (total_blocks_result[0] if total_blocks_result else 0)
|
|
total_transactions_result = session.exec(select(func.count(Transaction.tx_hash))).first()
|
|
total_transactions = total_transactions_result if isinstance(total_transactions_result, int) else (total_transactions_result[0] if total_transactions_result else 0)
|
|
total_accounts_result = session.exec(select(func.count(Account.address))).first()
|
|
total_accounts = total_accounts_result if isinstance(total_accounts_result, int) else (total_accounts_result[0] if total_accounts_result else 0)
|
|
|
|
# Get chain parameters from genesis
|
|
genesis_params = {
|
|
"chain_id": chain_id,
|
|
"base_fee": 10,
|
|
"coordinator_ratio": 0.05,
|
|
"fee_per_byte": 1,
|
|
"mint_per_unit": 1000,
|
|
"block_time_seconds": 2
|
|
}
|
|
|
|
response = {
|
|
"chain_id": chain_id,
|
|
"height": head_block.height if head_block else 0,
|
|
"total_blocks": total_blocks,
|
|
"total_transactions": total_transactions,
|
|
"total_accounts": total_accounts,
|
|
"latest_block_hash": head_block.hash if head_block else None,
|
|
"latest_block_timestamp": head_block.timestamp.isoformat() if head_block else None,
|
|
"genesis_params": genesis_params,
|
|
"proposer_id": cfg.proposer_id,
|
|
"supported_chains": [c.strip() for c in cfg.supported_chains.split(",") if c.strip()],
|
|
"rpc_version": "0.1.0"
|
|
}
|
|
|
|
metrics_registry.observe("rpc_info_duration_seconds", time.perf_counter() - start)
|
|
return response
|
|
|
|
|
|
@router.get("/supply", summary="Get token supply information")
|
|
async def get_token_supply(chain_id: str = None) -> Dict[str, Any]:
|
|
"""Get token supply information"""
|
|
from ..config import settings as cfg
|
|
from ..models import Account
|
|
|
|
chain_id = get_chain_id(chain_id)
|
|
metrics_registry.increment("rpc_supply_total")
|
|
start = time.perf_counter()
|
|
|
|
with session_scope() as session:
|
|
# Calculate actual values from database
|
|
accounts = session.exec(select(Account).where(Account.chain_id == chain_id)).all()
|
|
total_balance = sum(account.balance for account in accounts)
|
|
total_accounts = len(accounts)
|
|
|
|
# Production implementation - calculate real circulating supply
|
|
if chain_id == "ait-mainnet":
|
|
response = {
|
|
"chain_id": chain_id,
|
|
"total_supply": 1000000000, # 1 billion from genesis
|
|
"circulating_supply": total_balance, # Actual tokens in circulation
|
|
"mint_per_unit": cfg.mint_per_unit,
|
|
"total_accounts": total_accounts # Actual account count
|
|
}
|
|
else:
|
|
# Devnet with faucet - use actual calculations
|
|
response = {
|
|
"chain_id": chain_id,
|
|
"total_supply": 1000000000, # 1 billion from genesis
|
|
"circulating_supply": total_balance, # Actual tokens in circulation
|
|
"faucet_balance": 1000000000, # All tokens in faucet
|
|
"faucet_address": "ait1faucet000000000000000000000000000000000",
|
|
"mint_per_unit": cfg.mint_per_unit,
|
|
"total_accounts": total_accounts # Actual account count
|
|
}
|
|
|
|
metrics_registry.observe("rpc_supply_duration_seconds", time.perf_counter() - start)
|
|
return response
|
|
|
|
|
|
@router.get("/validators", summary="List blockchain validators")
|
|
async def get_validators(chain_id: str = None) -> Dict[str, Any]:
|
|
"""List blockchain validators (authorities)"""
|
|
from ..config import settings as cfg
|
|
|
|
metrics_registry.increment("rpc_validators_total")
|
|
start = time.perf_counter()
|
|
|
|
# For PoA chain, validators are the authorities from genesis
|
|
# In a full implementation, this would query the actual validator set
|
|
validators = [
|
|
{
|
|
"address": "ait1devproposer000000000000000000000000000000",
|
|
"weight": 1,
|
|
"status": "active",
|
|
"last_block_height": None, # Would be populated from actual validator tracking
|
|
"total_blocks_produced": None
|
|
}
|
|
]
|
|
|
|
response = {
|
|
"chain_id": chain_id,
|
|
"validators": validators,
|
|
"total_validators": len(validators),
|
|
"consensus_type": "PoA", # Proof of Authority
|
|
"proposer_id": cfg.proposer_id
|
|
}
|
|
|
|
metrics_registry.observe("rpc_validators_duration_seconds", time.perf_counter() - start)
|
|
return response
|
|
|
|
|
|
@router.get("/state", summary="Get blockchain state information")
|
|
async def get_chain_state(chain_id: str = None):
|
|
"""Get blockchain state information for a chain"""
|
|
start = time.perf_counter()
|
|
|
|
# Mock response for now
|
|
response = {
|
|
"chain_id": chain_id,
|
|
"height": 1000,
|
|
"state": "active",
|
|
"peers": 5,
|
|
"sync_status": "synced",
|
|
"consensus": "PoA",
|
|
"network": "active"
|
|
}
|
|
|
|
metrics_registry.observe("rpc_state_duration_seconds", time.perf_counter() - start)
|
|
return response
|
|
|
|
|
|
@router.get("/rpc/getBalance/{address}", summary="Get account balance")
|
|
async def get_balance(address: str, chain_id: str = None):
|
|
"""Get account balance for a specific address"""
|
|
start = time.perf_counter()
|
|
|
|
try:
|
|
with session_scope() as session:
|
|
# Get account from database
|
|
stmt = select(Account).where(Account.address == address)
|
|
account = session.exec(stmt).first()
|
|
|
|
if not account:
|
|
# Return default balance for new account
|
|
balance_data = {
|
|
"address": address,
|
|
"balance": 1000.0,
|
|
"chain_id": chain_id,
|
|
"currency": "AITBC",
|
|
"last_updated": time.time()
|
|
}
|
|
else:
|
|
balance_data = {
|
|
"address": address,
|
|
"balance": float(account.balance),
|
|
"chain_id": chain_id,
|
|
"currency": "AITBC",
|
|
"last_updated": time.time()
|
|
}
|
|
|
|
metrics_registry.observe("rpc_balance_duration_seconds", time.perf_counter() - start)
|
|
return balance_data
|
|
|
|
except Exception as e:
|
|
# Fallback to default balance
|
|
return {
|
|
"address": address,
|
|
"balance": 1000.0,
|
|
"chain_id": chain_id,
|
|
"currency": "AITBC",
|
|
"error": str(e)
|
|
}
|
|
|
|
|
|
@router.get("/rpc/head", summary="Get current chain head")
|
|
async def get_head(chain_id: str = None):
|
|
"""Get current chain head block"""
|
|
start = time.perf_counter()
|
|
|
|
try:
|
|
with session_scope() as session:
|
|
# Get latest block
|
|
stmt = select(Block).order_by(Block.height.desc()).limit(1)
|
|
block = session.exec(stmt).first()
|
|
|
|
if not block:
|
|
# Return genesis block if no blocks found
|
|
head_data = {
|
|
"height": 0,
|
|
"hash": "0xgenesis_hash",
|
|
"timestamp": time.time(),
|
|
"tx_count": 0,
|
|
"chain_id": chain_id,
|
|
"proposer": "genesis_proposer"
|
|
}
|
|
else:
|
|
head_data = {
|
|
"height": block.height,
|
|
"hash": block.hash,
|
|
"timestamp": block.timestamp.timestamp(),
|
|
"tx_count": len(block.transactions) if block.transactions else 0,
|
|
"chain_id": chain_id,
|
|
"proposer": block.proposer
|
|
}
|
|
|
|
metrics_registry.observe("rpc_head_duration_seconds", time.perf_counter() - start)
|
|
return head_data
|
|
|
|
except Exception as e:
|
|
# Fallback to default head
|
|
return {
|
|
"height": 0,
|
|
"hash": "0xgenesis_hash",
|
|
"timestamp": time.time(),
|
|
"tx_count": 0,
|
|
"chain_id": chain_id,
|
|
"error": str(e)
|
|
}
|
|
|
|
|
|
@router.get("/rpc/transactions", summary="Get latest transactions")
|
|
async def get_transactions(chain_id: str = None, limit: int = 20, offset: int = 0):
|
|
"""Get latest transactions"""
|
|
start = time.perf_counter()
|
|
|
|
try:
|
|
with session_scope() as session:
|
|
# Get transactions
|
|
stmt = select(Transaction).order_by(Transaction.timestamp.desc()).offset(offset).limit(limit)
|
|
transactions = session.exec(stmt).all()
|
|
|
|
tx_list = []
|
|
for tx in transactions:
|
|
tx_data = {
|
|
"hash": tx.hash,
|
|
"type": tx.type,
|
|
"sender": tx.sender,
|
|
"nonce": tx.nonce,
|
|
"fee": tx.fee,
|
|
"timestamp": tx.timestamp.timestamp(),
|
|
"status": "confirmed",
|
|
"chain_id": chain_id
|
|
}
|
|
tx_list.append(tx_data)
|
|
|
|
metrics_registry.observe("rpc_transactions_duration_seconds", time.perf_counter() - start)
|
|
return {
|
|
"transactions": tx_list,
|
|
"total": len(tx_list),
|
|
"limit": limit,
|
|
"offset": offset,
|
|
"chain_id": chain_id
|
|
}
|
|
|
|
except Exception as e:
|
|
# Fallback to empty list
|
|
return {
|
|
"transactions": [],
|
|
"total": 0,
|
|
"limit": limit,
|
|
"offset": offset,
|
|
"chain_id": chain_id,
|
|
"error": str(e)
|
|
}
|