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.1.tar.gz (16.4 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.1-py3-none-any.whl (19.1 kB view details)

Uploaded Python 3

File details

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

File metadata

  • Download URL: shrtnr-1.0.1.tar.gz
  • Upload date:
  • Size: 16.4 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.1.tar.gz
Algorithm Hash digest
SHA256 8a020a723e2efe708def9e629a1e86e7b020da768aafced3c996c2e1707711d4
MD5 c68a5d3b22d1ea85604c96db4170ae22
BLAKE2b-256 e3e153fe363f4cf32386e5adb9f6f15e905ff3a018c3275cb5cc8e99227c851d

See more details on using hashes here.

Provenance

The following attestation bundles were made for shrtnr-1.0.1.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.1-py3-none-any.whl.

File metadata

  • Download URL: shrtnr-1.0.1-py3-none-any.whl
  • Upload date:
  • Size: 19.1 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.1-py3-none-any.whl
Algorithm Hash digest
SHA256 da21b382f7bd4bd1c713eff2f1cca904c7a4996222994164c83ef6bad5acb7bf
MD5 49c3936ed43332e920828e053763fcbf
BLAKE2b-256 ae8e8b79fdd8a2c8d7613025f4d6b320a78ca3df9a78f09ec3e52e5ba71a04d1

See more details on using hashes here.

Provenance

The following attestation bundles were made for shrtnr-1.0.1-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