Skip to main content

CLI tools for setting up and retiring GitHub-backed course repositories

Project description

course-setup

CLI tools for setting up and retiring GitHub-backed course repositories.

Installation

Install as a uv tool (recommended):

uv tool install course-setup

This makes setup-course, retire-course, unretire-course, archive-course, and setup-course-config available on your PATH. All five commands support --version and --help, which display the version number, PyPI URL, author name (Reuven Lerner), and email. To upgrade:

uv tool upgrade course-setup

You can also install with pip (pip install course-setup).

Configuration

Generate a starter config file:

setup-course-config

This creates a config.toml file in your platform's config directory (e.g., ~/Library/Application Support/course-setup/ on macOS, ~/.config/course-setup/ on Linux, %APPDATA%\course-setup\ on Windows). Open it and fill in your settings:

[github]
token = "ghp_YOUR_TOKEN_HERE"

[paths]
archive = "/path/to/your/archive"
# readme_source = "/path/to/custom/README.md"   # or a URL

[defaults]
notebook_type = "jupyter"   # or "marimo"
Setting Required Description
[github] token Yes GitHub personal access token. Alternatively, set the GITHUB_TOKEN environment variable.
[paths] archive Yes Directory where retired courses are archived.
[paths] readme_source No Local path or URL to a custom README for new courses. Omit to use the bundled default.
[paths] additional_files No List of file/directory paths to copy into every new course (e.g. data files, exercise notebooks).
[defaults] notebook_type No "jupyter" (default) or "marimo".
[defaults] verbose No true or false (default). Sets the default verbosity for setup-course.
[defaults] private No true or false (default). When true, setup-course creates private GitHub repos by default.
[defaults] extras_group No Default dependency group when --extras is not passed (e.g. "python").
[defaults] weekend No "standard" (skip Sat/Sun) or "israeli" (skip Fri/Sat). Default for --skip-weekends/--skip-israeli-weekends.

To regenerate the config file, use setup-course-config --force.

Usage

setup-course — Create a new course repo

setup-course -c Acme -t python-intro
Flag Description
-c, --client Client name (required)
-t, --topic Course topic (required)
-d, --date YYYY-MM override (defaults to current month). Validated: must be a real month, not more than 2 years ahead.
-n, --num-sessions Number of sessions (creates one notebook per session)
--freq Session frequency: daily or weekly (requires -n, defaults to daily)
--first-notebook-date Start date for notebook files (YYYY-MM-DD); defaults to today
--skip-weekends Skip Saturdays and Sundays when scheduling notebooks
--skip-israeli-weekends Skip Fridays and Saturdays when scheduling notebooks
--notebook-type jupyter or marimo (overrides config default)
--extras Dependency groups to add to the course pyproject.toml (see below)
--add-imports Pre-populate notebooks with import statements from --extras groups
-v, --verbose Show detailed output (paths, filenames, dependencies)
--private Create the GitHub repo as private instead of public (overrides config default)
--dry-run Preview what would be created without making any changes

Dependency groups

Group Packages
python ipython
data numpy, pandas, xlrd, openpyxl, pyarrow
viz matplotlib, seaborn, plotly
geo geopandas, folium, shapely
db duckdb, sqlalchemy
ml scikit-learn

You can also define custom groups in your config.toml under [extras]. Entries can be package names or references to other groups (built-in or custom):

[extras]
finance = ["yfinance", "pandas-datareader"]
reuven = ["python", "data", "plotly"]   # expands python & data groups + plotly

Example — a Pandas course with Python extras and data/viz packages:

setup-course -c Acme -t pandas --extras python data

This will:

  1. Create a directory and GitHub repo named {client}-{topic}-{YYYY-MM} (public by default; use --private for private)
  2. Create a notebook per session, named {client}-{topic}-{YYYY-MM-DD} (.ipynb for Jupyter, .py for Marimo)
  3. Generate a pyproject.toml with the notebook dependency and gitautopush
  4. Include a .gitignore for Python, virtual environments, and IDE files
  5. Configure the local .git/config with the GitHub SSH remote
  6. Make an initial commit and push to GitHub
  7. Run uv sync to install all dependencies

By default, a single notebook is created for today's date. Use -n to create multiple notebooks for multi-day or multi-week courses:

setup-course -c Acme -t python-intro -n 5              # 5 daily sessions
setup-course -c Acme -t python-intro -n 5 --freq weekly # 5 weekly sessions

retire-course — Archive a course repo

retire-course ./Acme-python-intro-2026-03
Argument / Flag Description
DIRNAME... One or more course directories to retire
--keep-public Archive without making the GitHub repo private

This will (for each directory):

  1. Make the GitHub repo private (unless --keep-public is passed)
  2. Move the local directory to your configured archive path under the current year (prompts for confirmation if the year directory doesn't exist)
  3. Print a retirement summary showing: notebook count, date range, dependencies, archive location, and GitHub URL

You can retire multiple courses at once:

retire-course ./Acme-2026-03 ./Beta-2026-03 ./Gamma-2026-02

If any directory fails, the rest are still processed and errors are reported at the end.

archive-course — Create a zip archive of a course

archive-course ./Acme-python-intro-2026-03
Flag Description
--output, -o Custom output zip path (defaults to {dirname}.zip)
--no-html Skip HTML export of Jupyter notebooks

For Jupyter notebooks, archive-course exports each .ipynb to HTML via nbconvert and includes both the .ipynb and .html files in the zip. Use --no-html to skip the HTML export. The archive excludes .git, .venv, __pycache__, and .ipynb_checkpoints directories. After creating the archive, a summary is printed showing the archive path, file count, size, and which notebooks were included.

unretire-course — Restore a retired course

unretire-course /path/to/archive/2026/Acme-python-intro-2026-03

This will:

  1. Make the GitHub repo public again
  2. Move the directory from the archive back to your current working directory

Live teaching with gitautopush

In a separate terminal, run uv run gitautopush . from inside the course directory. This watches for notebook changes and automatically pushes them to GitHub, so students can follow along in real time by viewing the public repo.

Development

git clone https://github.com/reuven/course-setup.git
cd course-setup
uv sync --dev

Run tests, format, and lint:

uv run pytest
uv run ruff format src/ tests/
uv run ruff check src/ tests/
uv run mypy --strict src/

License

MIT — see LICENSE for details.

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

course_setup-3.0.5.tar.gz (16.1 kB view details)

Uploaded Source

Built Distribution

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

course_setup-3.0.5-py3-none-any.whl (22.2 kB view details)

Uploaded Python 3

File details

Details for the file course_setup-3.0.5.tar.gz.

File metadata

  • Download URL: course_setup-3.0.5.tar.gz
  • Upload date:
  • Size: 16.1 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: uv/0.10.12 {"installer":{"name":"uv","version":"0.10.12","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"macOS","version":null,"id":null,"libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":null}

File hashes

Hashes for course_setup-3.0.5.tar.gz
Algorithm Hash digest
SHA256 f6f773f26766dd19e7012c39eeed85f9cd600f75f0bab656d8536c25b232d36a
MD5 c400de84b96aeafa30640d5dcd164573
BLAKE2b-256 e9f7ebd811eabe84bbed02d07f9b13bdaec6d85acf54cd1c6cc89ffd2f3cc285

See more details on using hashes here.

File details

Details for the file course_setup-3.0.5-py3-none-any.whl.

File metadata

  • Download URL: course_setup-3.0.5-py3-none-any.whl
  • Upload date:
  • Size: 22.2 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: uv/0.10.12 {"installer":{"name":"uv","version":"0.10.12","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"macOS","version":null,"id":null,"libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":null}

File hashes

Hashes for course_setup-3.0.5-py3-none-any.whl
Algorithm Hash digest
SHA256 9cee337746449812e5976a665dc3bab092337f8b90bf6e3239efbc382c9b4635
MD5 2855aa8409c127f8623bce1fe2ecdb4b
BLAKE2b-256 4b53b11aed4e762dc5ac4f8edbf42d6524a80326da7cce09a6ccdd4593300adc

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