Production-ready Python library & CLI for Open Badges 3.0 (W3C Verifiable Credentials): issue, verify, and revoke JWT-VC credentials with Bitstring Status Lists and did:web — plus strict OB 2.0 and legacy OB 1.0.
Project description
OpenBadgesLib
A production-ready Python library & CLI for the full Open Badges 3.0 issuer
lifecycle — issue W3C Verifiable Credentials
as JWT-VC, bake them into SVG/PNG, verify them, and revoke or suspend them
with W3C Bitstring Status Lists and did:web. It also ships strict
OpenBadges 2.0 (JWS / hosted assertions) and a frozen OpenBadges 1.0
legacy format, selected with -V {1,2,3} (default 3).
Features
- Sign badge images (SVG and PNG) as strict OB 2.0 JWS / hosted assertions (with a frozen OB 1.0 legacy format)
- Issue and verify OpenBadges 3.0 JWT-VC credentials
- Bake OB 3.0 JWT tokens into SVG and PNG badge images
- RSA 2048-bit (RS256), ECC NIST P-256 (ES256), and Ed25519 (EdDSA) key support
- SHA-256 hashed recipient identity with salt (OB 2.0)
- Expiration and revocation checking
- Issuer-side OB 3.0 revocation and suspension: W3C Bitstring Status List
publication and
--revoke/--suspend/--unsuspendmanagement - did:web issuer identity:
did.jsongeneration and DID resolution - Five command-line tools included
Why openbadgeslib
- The complete OB 3.0 issuer lifecycle in Python — not just issuing, but
publishing trust artefacts (
did:web) and revoking/suspending credentials via Bitstring Status Lists, driven from the CLI or the library. - Native VC-JWT signing with RSA (RS256), ECC P-256 (ES256) and Ed25519 (EdDSA), and offline verification.
- Lean and typed —
mypy --strict, CI on Python 3.10–3.13, a small dependency set, dataclasses + explicit validation (no Pydantic). - Dual-licensed LGPLv3 (library) / BSD-2-Clause (CLI tools).
How it compares
Best-effort comparison from each project's public documentation as of July 2026;
"not documented" is shown as —. Corrections welcome via issue/PR.
| Capability | openbadgeslib | PyOpenBadges |
|---|---|---|
| Open Badges versions | 1.0, 2.0, 3.0 | 3.0 (v2→v3 planned) |
| OB 3.0 proof: VC-JWT (JOSE) | ✅ RS/ES/EdDSA | roadmap |
| OB 3.0 proof: Data Integrity / LDP | — (verify-first, planned) | JSON-LD signing (format unspecified) |
| Revocation / suspension (Bitstring Status List) | ✅ issue + publish + manage | — |
did:web (generate + resolve) |
✅ | — |
| Image baking (SVG + PNG) | ✅ | roadmap |
| Command-line tools | ✅ 5 tools | — |
| Typing / CI | mypy --strict, CI 3.10–3.13 |
— |
| Validation | dataclasses + explicit checks | Pydantic |
For OB 2.0 validation the reference implementation is 1EdTech's Node
openbadges-validator; this
table focuses on the OB 3.0 Python landscape.
Requirements
- Python >= 3.10 (tested on 3.10–3.13)
- pycryptodome >= 3.20
- ecdsa >= 0.19
- pypng >= 0.20220715.0
- PyJWT[crypto] >= 2.8
- cryptography >= 42
- defusedxml >= 0.7
Installation
pip install openbadgeslib
All dependencies are installed automatically. For a development checkout with the test suite and linters:
pip install -e ".[dev]"
Quick Start
# 1. Initialize a configuration directory
openbadges-init ./config/
# 2. Generate a key pair for a badge
openbadges-keygenerator -c ./config/config.ini -g 1
# 3a. Sign a badge — OpenBadges 3.0 (default)
openbadges-signer -c ./config/config.ini -b 1 -r recipient@example.com -o /tmp/ -E
# 3b. Sign a badge — strict OpenBadges 2.0
openbadges-signer -c ./config/config.ini -b 1 -r recipient@example.com -o /tmp/ -E -V 2
# 4a. Verify — OpenBadges 3.0
openbadges-verifier -i /tmp/badge_1_recipient@example.com.svg \
-r recipient@example.com -V 3 -k ./config/keys/verify_rsa_key_1.pem
# 4b. Verify — strict OpenBadges 2.0 (pin a trusted key with -l/--local or -k/--pubkey)
openbadges-verifier -i /tmp/badge_1_recipient@example.com.svg \
-r recipient@example.com -V 2 -l 1
# 5. OpenBadges 3.0 revocation (opt-in: set 'status_lists = revocation, suspension'
# in the badge section before signing). Publish the issuer's did.json and the
# signed Bitstring Status Lists, then revoke and re-publish.
openbadges-publish -c ./config/config.ini -o ./public -V 3
openbadges-publish -c ./config/config.ini -o ./public -V 3 \
--revoke recipient@example.com --reason "issued in error"
See the Quick Start and CLI Reference wiki pages for the full walkthrough and every flag.
Using the library — OpenBadges 2.0 (strict)
from datetime import datetime, timezone
from openbadgeslib.ob2 import OB2Signer, Assertion, IdentityObject, Verification
with open('sign.pem', 'rb') as f:
priv_pem = f.read()
with open('badge.svg', 'rb') as f:
image = f.read()
assertion = Assertion(
recipient=IdentityObject.create('recipient@example.com', salt='s4lt3d'),
badge='https://example.com/badge_1/badge.json',
verification=Verification(type='SignedBadge',
creator='https://example.com/badge_1/key.json'),
issued_on=datetime(2026, 1, 1, tzinfo=timezone.utc),
)
signer = OB2Signer(privkey_pem=priv_pem, algorithm='RS256')
baked_svg = signer.sign_into_svg(assertion, image)
with open('/tmp/signed_badge.svg', 'wb') as f:
f.write(baked_svg)
For the frozen OpenBadges 1.0 legacy API (Badge / Signer / Verifier), import from openbadgeslib.ob1 instead.
Using the library — OpenBadges 3.0 (JWT-VC)
from openbadgeslib.ob3 import (
Issuer, Achievement, OpenBadgeCredential, OB3Signer, OB3Verifier,
)
issuer = Issuer(id='https://example.com/issuer', name='Example Org')
achievement = Achievement(
id='https://example.com/achievements/python',
name='Python Developer',
description='Awarded for Python proficiency',
criteria_narrative='Must pass the Python assessment',
)
credential = OpenBadgeCredential(
issuer=issuer,
recipient_id='mailto:recipient@example.com',
achievement=achievement,
)
with open('sign.pem', 'rb') as f:
priv_pem = f.read()
signer = OB3Signer(privkey_pem=priv_pem, algorithm='RS256')
# Bake the signed JWT-VC into a badge image
with open('badge.svg', 'rb') as f:
baked_svg = signer.sign_into_svg(credential, f.read())
# Verify
with open('verify.pem', 'rb') as f:
verifier = OB3Verifier(pubkey_pem=f.read())
token = OB3Verifier.extract_token_from_svg(baked_svg)
restored = verifier.verify(token, expected_recipient='recipient@example.com')
print('Recipient:', restored.recipient_id)
Documentation
- User & developer guide — the project Wiki: installation, configuration, concepts, the security model, CLI reference and how-to guides.
- API reference — generated from the docstrings and published at luisgf.github.io/openbadgeslib.
Running the test suite
pytest
pytest --cov=openbadgeslib # with coverage report
flake8 openbadgeslib tests # lint
mypy # type check (config in pyproject.toml)
Changelog
See Changelog.txt for the full history, and the
GitHub Releases page for
release notes.
License
The library (openbadgeslib/ package) is licensed under the
GNU Lesser General Public License v3
(LGPLv3). The command-line wrapper tools are licensed under the
BSD 2-Clause license.
Authors
- Luis González Fernández luisgf@luisgf.es
- Jesús Cea Avión jcea@jcea.es
Project details
Release history Release notifications | RSS feed
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 openbadgeslib-3.1.1.tar.gz.
File metadata
- Download URL: openbadgeslib-3.1.1.tar.gz
- Upload date:
- Size: 171.8 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
780720e37631ec87f94c9a108cb2eb7109ce3946abe980b63fc1f1bb42c562b6
|
|
| MD5 |
6a4a4995a8c1c0a1e2d5e26b0a84fa44
|
|
| BLAKE2b-256 |
11064eb701df238ae7116532f4dac006317d0113bcaf6661ee951758840f0a0c
|
File details
Details for the file openbadgeslib-3.1.1-py3-none-any.whl.
File metadata
- Download URL: openbadgeslib-3.1.1-py3-none-any.whl
- Upload date:
- Size: 103.8 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
0ddddf2d50df8049efe27545a165ed00f2483e39aea942fe45e3c0a056438682
|
|
| MD5 |
a9055fa12f63ab4e41f851523221d319
|
|
| BLAKE2b-256 |
edae0de66a512cc7392f61ac7edf5a8f8b485b4a0683b5081069ca8cf77b1dec
|