Skip to content

feat: implement propose_withdraw on GroupTreasuryContract#246

Open
Kayce10 wants to merge 137 commits into
codebestia:mainfrom
Kayce10:feature/propose-withdraw-group-treasury
Open

feat: implement propose_withdraw on GroupTreasuryContract#246
Kayce10 wants to merge 137 commits into
codebestia:mainfrom
Kayce10:feature/propose-withdraw-group-treasury

Conversation

@Kayce10

@Kayce10 Kayce10 commented Jun 27, 2026

Copy link
Copy Markdown

Summary

Implements propose_withdraw on GroupTreasuryContract as described in issue #122.

Changes

storage.rs

  • Added ProposalCount and Proposal(u32) variants to DataKey
  • Added ProposalStatus enum (Pending, Approved, Rejected, Executed)
  • Added WithdrawProposal struct (id, proposer, to, token, amount, approvals, status, expires_at)
  • Added ProposalCreatedEvent struct

lib.rs

  • Added propose_withdraw(env, proposer, to, token, amount, ttl_ledgers) -> u32
    • Requires proposer auth
    • Panics not a member if proposer is not in the members list
    • Panics amount must be positive if amount <= 0
    • Panics insufficient funds if treasury balance < amount
    • Increments ProposalCount, stores WithdrawProposal with Pending status
    • Auto-adds proposer to approvals list
    • Sets expires_at = ledger.sequence() + ttl_ledgers
    • Emits ProposalCreatedEvent
    • Returns the new proposal ID

test.rs

  • test_propose_withdraw_returns_id_and_stores_proposal — happy path, returned ID is 1
  • test_propose_withdraw_non_member_panics — verifies non-member call panics
  • test_propose_withdraw_insufficient_balance_panics — verifies insufficient funds panic
  • test_propose_withdraw_increments_id — verifies IDs increment sequentially

Acceptance Criteria

  • Non-member call panics
  • Insufficient treasury balance panics with "insufficient funds"
  • Returned ID matches the stored proposal's id

Closes #122

codebestia and others added 30 commits May 12, 2026 14:16
feat(backend): PostgreSQL connection with Drizzle ORM + base schema
feat(backend): SIWS wallet auth with JWT + Socket.IO middleware
feat(backend): real-time messaging via Socket.IO with persistent storage
feat(contracts): TokenTransfer Soroban contract with testnet deploy script
feat(web): landing page for Clicked
ci(backend): format check, lint, and test pipeline
Adds GET /users/:id — requires a valid JWT, returns id/username/avatarUrl
plus wallet address/isPrimary pairs; explicit serialization prevents
leaking internal fields. Returns 404 for unknown or malformed IDs.

Fixes codebestia#10
Adds proposals Soroban contract with cast_vote/execute_proposal:
double-vote prevention, expiry enforcement, yes>no pass threshold,
and cross-contract treasury withdraw on passed proposals. Also adds
group_treasury contract gating withdraw behind the proposals contract.
20 tests cover all acceptance criteria.

Fixes codebestia#39
Adds contracts/scripts/deploy_group_treasury.sh mirroring deploy_token_transfer.sh.
Validates DEPLOYER_SECRET, ADMIN_ADDRESS, TOKEN_CONTRACT_ID, and INITIAL_MEMBERS
(comma-separated) before building, uploading WASM, deploying, and calling initialize.
Outputs contract ID and .env setup instructions. Adds GROUP_TREASURY_CONTRACT_ID
to .env.example.

Fixes codebestia#42
- Add useSocket hook connecting to backend via socket.io-client
- Add /app/conversations/[id]/page.tsx with scrollable message thread
- Messages grouped by date with day separators
- Self messages right-aligned (accent), others left-aligned
- Auto-scroll to bottom only when user is already at the bottom
- Shows avatar, sender name, content, and timestamp per message
- Add reusable validate(schema) Express middleware returning structured 400s
- Add ChallengeSchema and VerifySchema for auth routes
- Replace manual if(!field) guards in auth.ts with validate middleware
- Structured error format: { error, issues: [{ field, message }] }
- Add supertest + 6 tests covering valid input, missing fields, wrong types
feat: group treasury contract, Makefile, and AI agent chat/fraud endpoints
…oint-10

feat(api): implement GET user profile endpoint
- Add last_read_message_id nullable FK column to conversation_members schema
- Generate migration 0002_greedy_hellion.sql (ALTER TABLE + FK constraint)
- Add message_read Socket.IO event handler with membership + message validation
- Persist lastReadMessageId per userId/conversationId in conversation_members
- Broadcast read_receipt { userId, lastReadMessageId } to conversation room
- Prevent spoofed reads: validates message exists in target conversation
- Add 4 tests covering success, non-member error, invalid message, DB persistence
- Extract Express app into src/app.ts for testability (server start stays in index.ts)
- Add supertest dev dependency for HTTP assertion
- Add 10 integration test cases for POST /auth/challenge and POST /auth/verify:
  challenge: valid walletAddress, missing walletAddress, empty body
  verify: new user JWT, existing user JWT, expired nonce, invalid sig,
          missing fields, empty body, malformed wallet address (Keypair throws)
- All mocks (DB, nonce, Stellar SDK) are offline — no real network or DB required
- 21 tests pass across 3 test files
- Add ioredis dependency and src/lib/redis.ts with graceful null fallback
- Cache GET /conversations per userId with 30-second TTL (CONV_CACHE_TTL)
- Cache key format: conversations:<userId>
- Invalidate cache for all conversation members after send_message
- Invalidate cache for all new members after create_conversation
- Redis errors silently degrade: requests fall through to DB
- Add 5 tests: cache hit skips DB, cache miss writes to Redis, null Redis fallback,
  Redis error fallback, per-user key format verified
codebestia and others added 29 commits June 3, 2026 17:36
…rsation-update

feat: add PATCH /conversations/:id endpoint for updating group conver…
feat(web): implement premium responsive app shell and page layouts
[CONTRACT] Add member management to Group Treasury
created apps/web/src/components/ui/Modal.tsx:1
…mponent

add WalletAddress component with copy and explorer link
…ndicator

feat: add online presence indicator to DM avatars and member count to…
feat(ui): implement CopyButton with clipboard API and framer-motion t…
Centralize admin check and update backend configuration
Closes codebestia#122

- Add WithdrawProposal, ProposalStatus, ProposalCreatedEvent types to storage.rs
- Add ProposalCount and Proposal(u32) variants to DataKey
- Implement propose_withdraw(env, proposer, to, token, amount, ttl_ledgers) -> u32
  - Verifies proposer is a member (panics 'not a member' otherwise)
  - Verifies amount > 0
  - Verifies sufficient treasury balance (panics 'insufficient funds' otherwise)
  - Increments ProposalCount, stores WithdrawProposal with Pending status
  - Auto-adds proposer approval, emits ProposalCreatedEvent, returns proposal ID
- Add tests: happy path, non-member panic, insufficient funds panic, ID increment
@codebestia

Copy link
Copy Markdown
Owner

Hello @Kayce10
You are still using the previous repo state. If possible, you can delete the fork and re-fork the repo. Then implement your changes.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Labels

None yet

Projects

None yet

Development

Successfully merging this pull request may close these issues.

Implement propose_withdraw on Group Treasury contract