Civic engagement library: address → reps → delivery
Project description
GovPal Reach
Civic engagement library: discover representatives from an address and deliver messages via email or web forms. Resistbot-like functionality with modular, pluggable architecture.
Project overview
GovPal Reach maps address → representatives (federal, state, local) and supports message delivery via email (Postmark) or web forms (Playwright). Users onboard via SMS/OTP (Twilio Verify, or Plivo).
- Discovery: Geocode address (Census Geocoder), then look up reps (Google Civic, Open States, LA local).
- Delivery: Send messages via email or submit web forms; PGQueuer for durable queue.
- Identity: User profiles, PII encryption, SMS OTP (Twilio primary, or Plivo).
- Orchestrator: High-level API composing discovery and delivery.
Quick start
1. Install
Requires uv (install: curl -LsSf https://astral.sh/uv/install.sh | sh).
uv sync
2. Configure environment
Copy .env.example to .env and fill in values. See SETUP.md for how to obtain credentials.
cp .env.example .env
# Edit .env with your credentials
Email delivery mode (optional): GP_DELIVERY_MODE defaults to capture (no real sends) so new installs don't send email until you explicitly enable it. Set to live to send via Postmark.
capture(default) – No real sends; emails are recorded in memory for inspection (e.g. tests, CI).redirect– All recipients are rewritten to the comma-separated addresses inGP_TEST_RECIPIENTS; subject is tagged with[TO: original].live– Production behavior via Postmark (set explicitly to send real email).
3. Run a simple example
The library is intended to be imported by a host app. Example usage:
from govpal.orchestrator.api import discover_reps_for_address, send_message_to_representatives
# Discover reps for an address
reps = discover_reps_for_address("123 Main St, Los Angeles, CA 90012")
# Send a message to one or more representatives (library adds preamble + closing)
# body = middle content only; subject, profile, phone, and list of reps
results = send_message_to_representatives(
"Your message body here.",
"Subject line",
profile, # constituent profile (first_name, last_name, email, full_address_for_display)
phone,
reps, # list of rep dicts with contact_email
)
# results = [(rep, success), ...]
4. Database schema (Postgres + Alembic required)
GovPal Reach requires PostgreSQL and Alembic for Identity (OTP), queue path, and production use. There is no non-Alembic schema path.
-
Set
DATABASE_URLin your environment (e.g. in.env). -
In your host app's Alembic setup, add the library's versions path to
version_locationsso onealembic upgrade headapplies both your app's and GovPal Reach's schema. Example inenv.py:from govpal.alembic import get_versions_path reach_versions = str(get_versions_path()) # Prepend or append to version_locations (e.g. [reach_versions, "alembic/versions"]) config.set_main_option("version_locations", ...)
-
Run migrations:
alembic upgrade head(or your app's equivalent, e.g.python run_migrations.py).
Migrations are idempotent: after updating the govpal-reach library, running alembic upgrade head again applies any new schema changes and is safe when nothing has changed. You do not need to check whether the library had schema updates.
Modules
| Module | Responsibility |
|---|---|
| Discovery | Address → reps (Google Civic, Open States, Census) |
| Delivery | Email (Postmark), web forms (Playwright), queue |
| Identity | Users, PII, SMS OTP (Twilio or Plivo) |
| Orchestrator | High-level API: discovery, send_message_to_representatives (preamble + closing) |
How to run the system
- As a library: Import
govpalin your app; configure env vars; use orchestrator API. - Production queue path: For durable delivery, use the queue path: create_message → optional update_message → enqueue_messages_for_address. Set
GOVPAL_USE_REAL_QUEUE=1and run the GovPal delivery worker so jobs are processed:uv run pgq run govpal.delivery.worker:create_pgqueuer --max-concurrent-tasks 2. See SETUP.md (PGQueuer section) for create_message/update_message usage and worker run command. - Railway: Deploy with Postgres; run the queue worker as a separate process or worker dyno; env vars set in Railway dashboard.
Using in another project
Install GovPal Reach as a dependency from PyPI (recommended for released versions):
# With uv (recommended)
uv add "govpal-reach>=0.2.4"
# Or with pip
pip install govpal-reach
To pin to a specific GitHub tag (e.g. before a version is on PyPI or to try a release tag):
uv add "govpal-reach @ git+https://github.com/YOUR_ORG/govpal-reach@v0.2.3"
Replace YOUR_ORG and the tag (v0.2.3) as needed. The import name is govpal:
from govpal.orchestrator.api import discover_reps_for_address, create_message, update_message, enqueue_messages_for_address
How to test
# Unit tests (runs in project .venv)
uv run pytest
# With coverage
uv run pytest --cov=govpal --cov-report=term-missing
# Integration tests (multi-module flows; no DB/API needed for composed-flow tests)
uv run pytest -m integration
# With DATABASE_URL and schema applied, the worker integration test also runs; without it that test is skipped.
# Linter
uv run ruff check src/ tests/
uv run ruff format --check src/ tests/
Full test suite (lint + pytest + coverage): see TEST_PLAN_SUMMARY.md.
Documentation
- RESEARCH_SUMMARY.md – APIs, data sources, architecture research
- PLAN_SUMMARY.md – Implementation plan, beads, schema, modules
- TEST_PLAN_SUMMARY.md – Testing strategy per phase
- SETUP.md – External services, credentials, permissions; Releasing: push a
v*tag and CI publishes to PyPI (Trusted Publishing), see section 13
License
MIT
Project details
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file govpal_reach-0.2.5.tar.gz.
File metadata
- Download URL: govpal_reach-0.2.5.tar.gz
- Upload date:
- Size: 56.9 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
b68ed1bcc31a10047a4b312c0d0eab07fe528a02bae5b7a40bbcbe8261d5837e
|
|
| MD5 |
212b5d1185b65f2317e5f794b0222021
|
|
| BLAKE2b-256 |
c4d76c61668883d846668b88a5408b1f214b0d8378f1d3a6b270f618ded2c5ab
|
Provenance
The following attestation bundles were made for govpal_reach-0.2.5.tar.gz:
Publisher:
release.yml on BruceAndTyler/govpal-reach
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
govpal_reach-0.2.5.tar.gz -
Subject digest:
b68ed1bcc31a10047a4b312c0d0eab07fe528a02bae5b7a40bbcbe8261d5837e - Sigstore transparency entry: 976102671
- Sigstore integration time:
-
Permalink:
BruceAndTyler/govpal-reach@babb2b76d8087c8c44dda97763066b57d1fdf02f -
Branch / Tag:
refs/tags/v0.2.5 - Owner: https://github.com/BruceAndTyler
-
Access:
private
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@babb2b76d8087c8c44dda97763066b57d1fdf02f -
Trigger Event:
push
-
Statement type:
File details
Details for the file govpal_reach-0.2.5-py3-none-any.whl.
File metadata
- Download URL: govpal_reach-0.2.5-py3-none-any.whl
- Upload date:
- Size: 72.2 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
5e60ff90f7a94236c6c05f53ddbd7bb3d27597bb8fea3f93d6265bb50776e351
|
|
| MD5 |
371f2ffd761b82ec8e6cdf072332aded
|
|
| BLAKE2b-256 |
f5124804ec535a59b5fece882b66c7f65c467226563e1f3fbafe681c87744919
|
Provenance
The following attestation bundles were made for govpal_reach-0.2.5-py3-none-any.whl:
Publisher:
release.yml on BruceAndTyler/govpal-reach
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
govpal_reach-0.2.5-py3-none-any.whl -
Subject digest:
5e60ff90f7a94236c6c05f53ddbd7bb3d27597bb8fea3f93d6265bb50776e351 - Sigstore transparency entry: 976102673
- Sigstore integration time:
-
Permalink:
BruceAndTyler/govpal-reach@babb2b76d8087c8c44dda97763066b57d1fdf02f -
Branch / Tag:
refs/tags/v0.2.5 - Owner: https://github.com/BruceAndTyler
-
Access:
private
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@babb2b76d8087c8c44dda97763066b57d1fdf02f -
Trigger Event:
push
-
Statement type: