Skip to main content

Pythonic wrapper for macOS launchd: plist generation + launchctl lifecycle, as a library and a CLI.

Project description

launchy

CI PyPI Python License: MIT

Schedule things on macOS without writing plists or memorizing launchctl verbs. Python library and CLI for launchd.

from pathlib import Path
from launchy import Job

Job(
    label="com.launchy.backup.daily",
    program=["/usr/bin/python3", str(Path.home() / "backup.py")],
    calendar={"hour": 2, "minute": 0},
    keep_alive={"successful_exit": False},
    log_dir=Path.home() / "Library/Logs/backup",
).install()
launchy install com.launchy.backup.daily \
    --program /usr/bin/python3 --program ~/backup.py \
    --at 02:00 --keep-alive
launchy info com.launchy.backup.daily      # config + state
launchy doctor                              # health-check everything
launchy ls --failed                         # only jobs with non-zero last exit

Install

# library + CLI on PATH
uv tool install 'launchy[cli]'

# library only
uv add launchy

Requires macOS and Python 3.14+.


Each feature is documented as Library then CLI. Skip ahead to Reference for the trigger/scope tables.

Creating jobs

A Job is the unit of work. Pass triggers, env, log paths; call install(). The install is idempotent — re-running with the same label bootouts the old version and bootstraps the new one.

Library

from datetime import timedelta
from pathlib import Path
from launchy import Job, Scope

job = Job(
    label="com.launchy.sync.photos",
    program=["/usr/local/bin/photo-sync"],
    interval=timedelta(minutes=15),
    env={"PHOTO_DIR": "/Volumes/Photos"},
    log_dir=Path.home() / "Library/Logs/photo-sync",
    scope=Scope.USER,
)
job.install()

CLI

launchy install com.launchy.sync.photos \
    --program /usr/local/bin/photo-sync \
    --every 15m \
    --env PHOTO_DIR=/Volumes/Photos \
    --log-dir ~/Library/Logs/photo-sync

The install message reports loaded status, plist path, and inlines doctor warnings if the new job looks broken (missing program, unwritable log dir, no trigger set).

Listing & finding jobs

Library

from launchy import Job

# all jobs across user, all-users, and system scopes; pass scope= to filter
all_jobs = Job.list_with_status()                          # list[tuple[Job, JobStatus]]
mine = Job.list_with_status(prefix="com.launchy.")         # filter
one = Job.find("com.launchy.backup.daily")                 # auto-detects scope

Job.list_with_status() parallelizes the per-job launchctl print calls in a thread pool.

CLI

launchy list                        # all scopes
launchy ls --scope user             # filter (ls is an alias)
launchy ls -s                       # add a Schedule column
launchy ls -g zoom                  # case-insensitive substring on label
launchy ls --running                # loaded + has pid
launchy ls --stopped                # loaded + no pid
launchy ls --failed                 # last exit non-zero
launchy ls -1 -g zoom               # one label per line, pipeable
launchy ls --json                   # structured output for jq

Inspecting one job

Library

job = Job.find("com.launchy.backup.daily")
job.status()       # JobStatus(label, loaded, pid, last_exit_code)
job.render()       # plist XML as a string
job.diagnose()     # list[Diagnostic] — see Health checks below

CLI

launchy info <label>     # config + state on one screen
launchy status <label>   # just loaded/pid/exit (--json for structured)
launchy show <label>     # rendered plist XML

Health checks

doctor catches the silent failures launchd swallows: missing program paths, unwritable log dirs, intervals so short they pin the CPU, working dirs that don't exist, jobs with no trigger that will never run, plists on disk that aren't loaded. Exit code is non-zero on any failure — wire it into CI if you ship plists from a repo.

Library

from launchy import Job, Diagnostic

for d in Job.find("com.launchy.backup.daily").diagnose():
    print(d.severity, d.check, d.detail)

Diagnostic.severity is "ok", "warn", or "fail".

CLI

launchy doctor <label>      # check one job
launchy doctor              # sweep everything
launchy doctor -v           # include passing checks too

Lifecycle

Library

job.start()     # kickstart -k
job.stop()      # SIGTERM
job.reload()    # bootout + bootstrap (after editing the plist on disk)
job.disable()   # mark disabled in launchd, then unload. Survives reboots.
job.enable()    # remove disabled flag and bootstrap.

disable/enable are distinct from stop/start: they write to launchd's persistent store and survive reboots. Use them to pause a job for a week without uninstalling.

CLI

launchy start <label>
launchy stop <label>
launchy reload <label>
launchy disable <label>
launchy enable <label>

Bulk operations

uninstall, start, stop, reload, disable, enable all accept --prefix or --grep instead of a single label. A confirmation prompt shows the count and a sample; --force/-f skips it.

Library — no separate API; just iterate:

for job in Job.list(prefix="com.launchy.test."):
    job.uninstall()

CLI

launchy rm --grep test --force                  # bulk remove (rm is alias for uninstall)
launchy stop --prefix com.launchy.batch.        # bulk stop
launchy disable --grep zoom                     # interactive confirm

Logs

Library — paths exposed as properties:

job.resolved_stdout_path    # Path or None — explicit override, else log_dir/<label>.out.log
job.resolved_stderr_path

CLI

launchy logs <label>          # tail (last 10 lines)
launchy logs <label> -f       # follow (kqueue-backed via `tail -f`)
launchy logs <label> --err    # stderr instead of stdout

Removing jobs

Library

Job.find("com.launchy.x").uninstall()

CLI

launchy uninstall <label>           # confirmation prompt
launchy rm <label> --force          # skip prompt
launchy rm --grep test --force      # bulk

Shell completion

launchy --install-completion writes a completion script for your shell. Tab-completes installed labels across all scopes using fzf-style fuzzy matching — characters in order, not contiguous, ranked by word-boundary hits and position. So launchy info ferry<TAB> finds dev.ascention.mcp-ferry, and launchy rm espa<TAB> finds com.federicoterzi.espanso.

Shell behaviour varies: fish shows all returned candidates; zsh and bash may filter to prefix-only depending on your config (fzf-tab in zsh restores the full fuzzy experience).

Reference

Triggers

launchd key Python type CLI shorthand
RunAtLoad bool --run-at-load
KeepAlive bool | KeepAliveConditions --keep-alive
StartInterval datetime.timedelta --every 5m
StartCalendarInterval CalendarSpec | list[CalendarSpec] --at "02:00"
WatchPaths list[Path] --watch PATH
QueueDirectories list[Path] --queue-dir P
StartOnMount bool --start-on-mount

CalendarSpec and KeepAliveConditions are TypedDicts with snake_case keys (hour, minute, weekday, successful_exit, etc). launchy maps them to launchd's PascalCase at render time.

For the authoritative spec, see man 5 launchd.plist (online mirror) and Apple's Creating Launch Daemons and Agents.

When to use which

RunAtLoad — fires once when launchd loads the plist: at login for user agents, at boot for daemons. Best for "start this service on login" patterns; combine with KeepAlive for long-running services that should also survive crashes.

KeepAlive — restart the program when it exits. The bool form True always restarts. The dict form gates restart on conditions:

  • successful_exit: False — restart on crash but not on clean exit. The standard pattern for "service that runs until I explicitly stop it."
  • crashed: True — restart only after abnormal exit (signal/crash).
  • network_state: True — restart while network is reachable. Deprecated on modern macOS; treat as a no-op.
  • path_state: {"/path": True} — restart while a path exists (or doesn't).
  • other_job_enabled: {"com.x.label": True} — restart depending on another job's enabled state.

Conditions aren't mutually exclusive. If you set more than one, launchd ORs them — the job restarts if any condition says it should. In practice you almost always pick one, usually {"successful_exit": False}.

StartInterval (--every 5m) — run every N seconds. Counts from load time, not wall clock — "every hour" doesn't necessarily land on :00. Best for polling and periodic syncs where exact timing doesn't matter. Avoid intervals under 10 seconds (launchy doctor warns).

The --every shorthand accepts a positive integer followed by s (seconds), m (minutes), h (hours), or d (days): 30s, 5m, 1h, 2d. Compound expressions like 1h30m aren't supported — use --interval SECONDS for arbitrary values.

StartCalendarInterval (--at "02:00") — cron-like fire times. Each CalendarSpec is a dict with any of: minute (0–59), hour (0–23), day (1–31, day of month), weekday (0–7, where 0 and 7 are both Sunday), month (1–12). Omitted keys are wildcards. Pass a list of dicts for multiple fire times in one job. Missed runs while the Mac was asleep fire once on wake, not retroactively.

# every day at 2:00 AM
calendar={"hour": 2, "minute": 0}

# every Monday at 9:00 AM
calendar={"weekday": 1, "hour": 9, "minute": 0}

# 1st of every month at noon
calendar={"day": 1, "hour": 12}

# every July 4th at midnight
calendar={"month": 7, "day": 4, "hour": 0, "minute": 0}

# twice a day: 9 AM and 5 PM
calendar=[{"hour": 9, "minute": 0}, {"hour": 17, "minute": 0}]

# every weekday morning (one entry per day; weekday 1=Mon, 5=Fri)
calendar=[{"weekday": d, "hour": 9} for d in range(1, 6)]

The CLI shorthand --at covers daily and weekly cases (--at "02:00", --at "Mon 09:00"); for day-of-month or month constraints, fall back to --calendar "day=1,hour=12".

WatchPaths (--watch PATH) — fire when any of the listed paths is modified. Triggers on metadata changes too (atime/mtime), so expect occasional spurious fires; debounce in your program if needed. Best for reacting to config edits or file drops.

QueueDirectories (--queue-dir PATH) — like WatchPaths but only fires when the directory is non-empty, and keeps firing until your program drains it. Best for ingest queues, mail spools, or any "process incoming files" pattern.

StartOnMount (--start-on-mount) — fire whenever any filesystem mounts. Best for automated actions on external drives (back up to USB on plug-in, sync after a network share comes online). Rare in practice.

Scopes

Scope Plist location Notes
Scope.USER ~/Library/LaunchAgents Default. No privileges required.
Scope.ALL_USERS /Library/LaunchAgents Requires root to install.
Scope.SYSTEM /Library/LaunchDaemons Runs without a user session. Root only.

Job.find(label) and most CLI commands search all three scopes by default; pass scope= or --scope to filter. install defaults to Scope.USER since it has to pick a destination.

Exceptions

from launchy import LaunchyError, JobNotFound, NotInstalled, PermissionDeniedError, LaunchctlError
  • JobNotFoundload/find couldn't locate the label.
  • NotInstalled — lifecycle op on a job whose plist isn't on disk.
  • PermissionDeniedErrorScope.ALL_USERS or Scope.SYSTEM op without root.
  • LaunchctlError — any launchctl non-zero exit; carries returncode, stderr, argv.

Gotchas

  • launchd never catches up on missed runs. If your Mac was asleep when a StartCalendarInterval should have fired, it runs once on wake — not for every missed slot.
  • StartInterval is measured from load time, not wall clock. "Every hour" doesn't necessarily land on :00.
  • After editing a plist on disk you must reload() — launchd caches the loaded version.
  • log_dir must exist before launchd starts the job. launchy creates it on install(). If you point at a network volume that isn't mounted at load time, launchd silently refuses to write logs.

License

MIT.

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

launchy-0.1.1.tar.gz (46.1 kB view details)

Uploaded Source

Built Distribution

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

launchy-0.1.1-py3-none-any.whl (30.4 kB view details)

Uploaded Python 3

File details

Details for the file launchy-0.1.1.tar.gz.

File metadata

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

File hashes

Hashes for launchy-0.1.1.tar.gz
Algorithm Hash digest
SHA256 c5f83bc24ad2150c2ab73bc55cb8a376c2f35e444f71331d53691f643b3bdf73
MD5 4c725b057b4dd3c0878ce1743152e424
BLAKE2b-256 8df4c6d7e2ee82d39271fb78ba615f556ea57b3db1df80a510c3b0e80673ecd3

See more details on using hashes here.

Provenance

The following attestation bundles were made for launchy-0.1.1.tar.gz:

Publisher: release.yml on dalberto/launchy

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

File details

Details for the file launchy-0.1.1-py3-none-any.whl.

File metadata

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

File hashes

Hashes for launchy-0.1.1-py3-none-any.whl
Algorithm Hash digest
SHA256 b07e674fb878a1d315d5f2ae155e3a66fa497cc8e9087eed8ad56d2ab13cc287
MD5 79889fee0886710939e0317f383ad350
BLAKE2b-256 bc638b3d15d0277e54bd456df570df92855c5a3c11f0496f28a37f140673cc62

See more details on using hashes here.

Provenance

The following attestation bundles were made for launchy-0.1.1-py3-none-any.whl:

Publisher: release.yml on dalberto/launchy

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