Skip to main content

SDK for the shrtnr URL shortener API

Project description

shrtnr

Python SDK for shrtnr, a self-hosted URL shortener on Cloudflare Workers. Create short links, manage slugs, and read click analytics.

PyPI license

Install

pip install shrtnr

Quick start

from shrtnr import Shrtnr

client = Shrtnr(base_url="https://your-shrtnr.example.com", api_key="sk_your_api_key")

link = client.links.create(url="https://example.com/very-long-path")
print(link.slugs[0].slug)  # "a3x9"

Async usage:

import asyncio
from shrtnr import AsyncShrtnr

async def main():
    async with AsyncShrtnr(base_url="https://your-shrtnr.example.com", api_key="sk_...") as client:
        link = await client.links.create(url="https://example.com")
        print(link.id)

asyncio.run(main())

Configuration

Shrtnr(
    base_url="https://your-shrtnr.example.com",  # required
    api_key="sk_...",                             # required; from the admin dashboard
    timeout=30.0,                                 # optional; seconds (default: 30)
    http_client=custom_httpx_client,              # optional; inject a custom httpx.Client
)

AsyncShrtnr accepts the same parameters, but takes an httpx.AsyncClient for http_client.

Both classes work as context managers:

with Shrtnr(base_url="...", api_key="sk_...") as client:
    links = client.links.list()

Resources

Links (client.links)

Method Description
get(id, *, range=None) Get a link with click count
list(*, owner=None, range=None) List all links
create(*, url, label=None, slug_length=None, expires_at=None, allow_duplicate=None) Create a short link
update(id, *, url=None, label=None, expires_at=None) Update URL, label, or expiry
disable(id) Stop redirecting
enable(id) Resume redirecting
delete(id) Permanently delete
analytics(id, *, range=None) Click breakdown by country, device, referrer, etc.
timeline(id, *, range=None) Click counts bucketed over time
qr(id, *, slug=None, size=None) QR code as SVG string
bundles(id) Bundles this link belongs to
# Shorten a URL
link = client.links.create(url="https://example.com", label="Landing page")

# Get a 7-day click count
fresh = client.links.get(link.id, range="7d")

# Full analytics for the last 30 days
stats = client.links.analytics(link.id, range="30d")
print(stats.total_clicks, stats.countries, stats.browsers)

Slugs (client.slugs)

Method Description
lookup(slug) Find a link by slug
add(link_id, slug) Add a custom slug
disable(link_id, slug) Disable a slug
enable(link_id, slug) Re-enable a slug
remove(link_id, slug) Remove a slug
# Add a campaign slug then disable it when the campaign ends
client.slugs.add(link.id, "spring-sale")
client.slugs.disable(link.id, "spring-sale")

# Look up a link by its slug
found = client.slugs.lookup("spring-sale")

Bundles (client.bundles)

Groups of related links with combined analytics.

Method Description
get(id, *, range=None) Get a bundle with click summary
list(*, archived=None, range=None) List bundles
create(*, name, description=None, icon=None, accent=None) Create a bundle
update(id, *, name=None, description=None, icon=None, accent=None) Update metadata
delete(id) Permanently delete
archive(id) Hide from default listing
unarchive(id) Restore an archived bundle
analytics(id, *, range=None) Combined click analytics
links(id) List links in the bundle
add_link(id, link_id) Add a link
remove_link(id, link_id) Remove a link
# Create a bundle and add links to it
bundle = client.bundles.create(name="Spring 2026", accent="green")
client.bundles.add_link(bundle.id, link_a.id)
client.bundles.add_link(bundle.id, link_b.id)

# Combined analytics for the last 7 days
stats = client.bundles.analytics(bundle.id, range="7d")
print(stats.total_clicks)

Models

All model fields use snake_case, matching the wire format. Types are frozen dataclasses.

Key types exported from shrtnr:

  • Link, Slug, Bundle, BundleWithSummary
  • ClickStats, TimelineData, NameCount, TimelineBucket, TimelineSummary
  • DeletedResult, AddedResult, RemovedResult
  • TimelineRange (Literal["24h", "7d", "30d", "90d", "1y", "all"])
  • BundleAccent (Literal["orange", "red", "green", "blue", "purple"])

Errors

Every 4xx/5xx response raises ShrtnrError. Network failures also raise ShrtnrError with status=0.

from shrtnr import ShrtnrError

try:
    client.links.get(99999)
except ShrtnrError as err:
    print(err.status)         # 404
    print(err.server_message) # "not found"
    print(str(err))           # "shrtnr API error (HTTP 404): not found"

See also

Attribution

shrtnr is developed and maintained by Oddbit.

If you publish a fork or derivative work, retain the license and notice files, preserve applicable copyright and attribution notices, and clearly indicate that your version has been modified.

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

shrtnr-1.0.2.tar.gz (17.1 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

shrtnr-1.0.2-py3-none-any.whl (19.6 kB view details)

Uploaded Python 3

File details

Details for the file shrtnr-1.0.2.tar.gz.

File metadata

  • Download URL: shrtnr-1.0.2.tar.gz
  • Upload date:
  • Size: 17.1 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for shrtnr-1.0.2.tar.gz
Algorithm Hash digest
SHA256 c2b5853bbe237fa6db9552d5ef2beac170cfceb02256ef339d6b44530ebc3218
MD5 e0417bd6bf3aafb654572652f1f19c66
BLAKE2b-256 206b43288089d122a9aa31f36168d10aed9819b450119ba459dc3db5768f1aad

See more details on using hashes here.

Provenance

The following attestation bundles were made for shrtnr-1.0.2.tar.gz:

Publisher: release-sdk-python.yml on oddbit/shrtnr

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file shrtnr-1.0.2-py3-none-any.whl.

File metadata

  • Download URL: shrtnr-1.0.2-py3-none-any.whl
  • Upload date:
  • Size: 19.6 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for shrtnr-1.0.2-py3-none-any.whl
Algorithm Hash digest
SHA256 10804a66a4ff60a8d55f15626ceab4d27fc44b69850d365a908657100899cee9
MD5 8ed4804a0eaf47801028ab697edd75b8
BLAKE2b-256 66177ad08f2220849076ed2b24827ce2b20c308cf80d5e958559b28ecaff77b2

See more details on using hashes here.

Provenance

The following attestation bundles were made for shrtnr-1.0.2-py3-none-any.whl:

Publisher: release-sdk-python.yml on oddbit/shrtnr

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page