Skip to main content

A lightweight async Python CMS for crafting modern websites

Project description

Skrift

Python 3.12+ License: MIT Built with Litestar

A modern Litestar-powered content management framework with multi-provider OAuth authentication, role-based access control, and WordPress-like template resolution.

Features

  • Multi-Provider OAuth: Authenticate with Google, GitHub, Microsoft, Discord, Facebook, or Twitter/X
  • Role-Based Access Control: Flexible permission system with Admin, Editor, Author, and Moderator roles
  • Setup Wizard: Guided first-time configuration without manual file editing
  • Admin Interface: Web-based management for users, pages, and site settings
  • WordPress-like Templates: Hierarchical template resolution for content pages
  • Dynamic Controllers: Load controllers from app.yaml configuration
  • SQLAlchemy Integration: Async database support with SQLite/PostgreSQL
  • Client-Side Sessions: Encrypted cookie sessions for horizontal scalability
  • Hook/Filter System: WordPress-like extensibility with async support
  • SEO Metadata: Built-in meta descriptions, OpenGraph tags, and robots directives
  • Content Scheduling: Schedule pages to publish at a future date
  • Page Revisions: Automatic content history with restore capability
  • Sitemap & Robots.txt: Auto-generated with filter extensibility

Quick Start

Prerequisites

  • Python 3.13+

Installation

# Install Skrift
pip install skrift

# Or install from git
pip install git+https://github.com/ZechCodes/skrift.git

Getting Started

Create a project directory and set up your environment:

mkdir mysite && cd mysite

# Create minimal environment file
echo "SECRET_KEY=$(python -c 'import secrets; print(secrets.token_urlsafe(32))')" > .env

# Start Skrift
skrift

Open http://localhost:8080 to launch the setup wizard.

Setup Wizard

The setup wizard guides you through initial configuration:

  1. Database Configuration: Choose SQLite (dev) or PostgreSQL (production)
  2. Authentication Providers: Configure OAuth credentials
  3. Site Settings: Set site name, tagline, and copyright info
  4. Admin Account: Create your first admin user via OAuth login

After completing the wizard, an app.yaml configuration file is created in your project directory.

Manual Configuration

Alternatively, create app.yaml manually:

controllers:
  - skrift.controllers.auth:AuthController
  - skrift.admin.controller:AdminController
  - skrift.controllers.web:WebController

db:
  url: sqlite+aiosqlite:///./app.db

auth:
  redirect_base_url: http://localhost:8080
  providers:
    google:
      client_id: $GOOGLE_CLIENT_ID
      client_secret: $GOOGLE_CLIENT_SECRET
      scopes: [openid, email, profile]

Then run migrations and start the server:

skrift-db upgrade head
skrift

Documentation

Project Structure

skrift/
├── skrift/              # Main Python package
│   ├── asgi.py          # Application factory
│   ├── config.py        # Settings management
│   ├── controllers/     # Route handlers (auth, web, sitemap)
│   ├── admin/           # Admin panel
│   ├── auth/            # RBAC and guards
│   ├── db/              # Models and services
│   │   ├── models/      # Page, User, Role, PageRevision
│   │   └── services/    # page_service, revision_service
│   ├── lib/             # Core utilities
│   │   ├── hooks.py     # Hook/filter system
│   │   ├── seo.py       # SEO metadata utilities
│   │   ├── flash.py     # Enhanced flash messages
│   │   └── template.py  # Template resolver
│   └── setup/           # Setup wizard
├── templates/           # Jinja2 templates
├── static/              # Static assets
├── alembic/             # Database migrations
├── docs/                # Documentation
├── app.yaml             # Application config (generated)
└── main.py              # Development entry point

Configuration

Environment Variables

Variable Required Description
SECRET_KEY Yes Session encryption key
DEBUG No Enable debug mode (default: false)
DATABASE_URL No Database connection string
OAUTH_REDIRECT_BASE_URL No OAuth callback base URL

OAuth credentials are configured per-provider (e.g., GOOGLE_CLIENT_ID, GOOGLE_CLIENT_SECRET).

app.yaml

Application configuration is stored in app.yaml (generated by setup wizard):

controllers:
  - skrift.controllers.auth:AuthController
  - skrift.admin.controller:AdminController
  - skrift.controllers.web:WebController

db:
  url: $DATABASE_URL
  pool_size: 5

auth:
  redirect_base_url: $OAUTH_REDIRECT_BASE_URL
  providers:
    google:
      client_id: $GOOGLE_CLIENT_ID
      client_secret: $GOOGLE_CLIENT_SECRET

Environment variables (prefixed with $) are interpolated at runtime.

Deployment

Minimal VPS Deployment

# Install Skrift
pip install skrift

# Create project directory
mkdir -p /opt/skrift && cd /opt/skrift

# Configure environment
cat > .env << EOF
SECRET_KEY=$(python -c "import secrets; print(secrets.token_urlsafe(32))")
DATABASE_URL=sqlite+aiosqlite:///./app.db
OAUTH_REDIRECT_BASE_URL=https://yourdomain.com
EOF

# Start server (use setup wizard or create app.yaml manually)
skrift

Production with Hypercorn

hypercorn skrift.asgi:app --workers 4 --bind 0.0.0.0:8080

See the Deployment Guide for detailed instructions including Docker, Docker Compose, and Kubernetes deployments.

Database Migrations

# Apply migrations
skrift-db upgrade head

# Create new migration
skrift-db revision --autogenerate -m "description"

# Rollback
skrift-db downgrade -1

Template Resolution

Templates follow WordPress-like hierarchical resolution:

URL Path Templates Tried
/about page-about.html -> page.html
/services/web page-services-web.html -> page-services.html -> page.html

Contributing

  1. Fork the repository
  2. Create a feature branch
  3. Make your changes
  4. Submit a pull request

License

MIT

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

skrift-0.1.0a54.tar.gz (153.0 kB view details)

Uploaded Source

Built Distribution

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

skrift-0.1.0a54-py3-none-any.whl (217.2 kB view details)

Uploaded Python 3

File details

Details for the file skrift-0.1.0a54.tar.gz.

File metadata

  • Download URL: skrift-0.1.0a54.tar.gz
  • Upload date:
  • Size: 153.0 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: uv/0.10.7 {"installer":{"name":"uv","version":"0.10.7","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"24.04","id":"noble","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":true}

File hashes

Hashes for skrift-0.1.0a54.tar.gz
Algorithm Hash digest
SHA256 7a58e4be378078778c1ced296767cc14bfc91d706eabf5cc5d035756f213ae59
MD5 fc43c83fbf867ebe57207a04c1e0e705
BLAKE2b-256 2587f1cce8a744b2d4c39ac10e20a297e48e203c0fec64843f3dafd374dcdaba

See more details on using hashes here.

File details

Details for the file skrift-0.1.0a54-py3-none-any.whl.

File metadata

  • Download URL: skrift-0.1.0a54-py3-none-any.whl
  • Upload date:
  • Size: 217.2 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: uv/0.10.7 {"installer":{"name":"uv","version":"0.10.7","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"24.04","id":"noble","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":true}

File hashes

Hashes for skrift-0.1.0a54-py3-none-any.whl
Algorithm Hash digest
SHA256 299a5ae3fb20ad1d30baa52053693ac77dd293c6f6719a567d15b796d0c49997
MD5 5d632f84db1c23854b28c5f7d52a3a1d
BLAKE2b-256 80d8ae1215d7cc1b9abd9aa484e899efb65b792fcbb3b45f9e1d2eedc4da86e1

See more details on using hashes here.

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