Skip to content

Latest commit

 

History

History
177 lines (132 loc) · 8.23 KB

README.md

File metadata and controls

177 lines (132 loc) · 8.23 KB

cm-service

Ruff uv

This is the Rubin Observatory data processing campaign management ReST service. cm-service is developed with FastAPI and Safir. Learn more at https://cm-service.lsst.io.

Developer Quick Start

You can build and run cm-service on any system which has Python 3.11 or greater, uv, make, and Docker w/ the Docker Compose V2 CLI plugin (this includes, in particular, recent MacOS with Docker Desktop). Proceed as follows:

  • Ensure uv is installed (via Homebrew (macOS), pipx, or a preferred alternative method).

    • You may run make uv to bootstrap uv in your environment.
    • Suggested: set UV_PYTHON_PREFERENCE=only-managed to prevent non-uv Pythons from being used.
    • USDF: set UV_NATIVE_TLS=true for compatibility with the Squid proxy.
  • Install Python virtual environment, dependencies and activate pre-commit hooks with make init.

    • uv will automatically install any needed Python version.
    • Note: the ctrl-bps-htcondor plugin is not available on macOS and will not be installed on this platform.
  • Ensure the virtual environment is activated before interacting with other make targets:

    • source .venv/bin/activate
  • Spin up a debug instance of the service running in the foreground of the current shell with make run. This will launch a subsidiary Postgres instance locally in a Docker container via Docker Compose. The foreground debug service instance will log to stdout and will dynamically reload as you edit and save source files.

    • You may also choose to run both the service/worker and the database in Docker by running docker compose --profile full up; in particular this will exercise the Docker build process.

    • You may choose to (re)build the service container with docker compose build [--no-cache] cmservice to build, but not start, the service container (with --no-cache invalidating the build cache if needed).

  • Access the monitoring web application at http://localhost:8080/web_app/campaigns/

  • Exit your debug instance with ^C. The subsidiary Postgres container launched under Docker Compose will remain active, and will be re-used on any subsequent make run.

  • Shut down the subsidiary Postgres container if/when desired with docker compose down. Database state will be maintained in local Docker volumes and re-used on the next run. If you wish to clear the database state volumes as well to start completely fresh, do docker compose down -v.

Additional developer conveniences:

  • Browse the integrated online ReST API documentation while the service is running at http://localhost:8080/cm-service/v1/docs.

  • Run the pytest test suite at any point with make test. This will launch subsidiary containers if necessary via Docker Compose, or will make use of any that may already be running. Tests are performed in their own Postgres schema, so as not interfere with debugging state in a running debug service instance.

  • To run the playwright tests, add --run-playwright to the pytest arguments

  • Run the pre-commit hooks to lint and reformat your code via make lint.

  • Run the mypy static type hint checker with make typing.

  • Connect the psql command line client directly to the subsidiary Postgres for direct database inspection or manipulation via make psql.

Project Management

  • uv add <dependency> will add a dependency to the project; uv add --group dev ... will mark it as a development dependency.

  • make update will cause uv to upgrade all available dependencies within the constraints set for the dependency in the pyproject.toml file.

  • uv run <cmd> will execute <cmd> within the project environment, which is useful for cases where the virtual environment is not activated.

  • make clean will remove the virtual environment and allow make init to create it from scratch. The lock file is not removed by this target.

  • The lock file can be regenerated by removing (or renaming) the uv.lock file and running make uv.lock or the direct command uv lock.

  • uv uses a global package cache for speed and efficiency; you manage this cache with uv cache prune or uv cache clean for the nuclear option.

Release Management

This project is developed using a Trunk-Based Development pattern, where the trunk branch is named main. Developers should work in short-lived feature branches and commit their work to the trunk in accordance with the LSST Development Workflow.

Releases are performed at an unspecified cadence, to be no shorter than 1 week and subject to these rules:

  • Releases are named according to their semantic version (major.minor.patch).
  • Releases are made by adding a named tag to the trunk branch.
  • Each release will increment the minor version and set the patch level to 0, e.g., 1.0.12 -> 1.1.0
  • If a bugfix commit needs to be added to a release, then a retroactive branch will be created from the release tag; the commit is cherry-picked into the release branch and a new tag is written with an incremented patch level, e.g., 1.23.0 -> 1.23.1.
  • The major version is incremented only in the presence of user-facing breaking changes.

This project uses python-semantic-release to manage releases. A release should be triggered by any ticket branch being merged into main. A release must increment the application version according to semantic versioning (i.e., the use of major-minor-patch version tokens); and must create a matching git tag.

The make release target is designed to shepherd the manual release management process:

  1. The version number in src/lsst/cmservice/__init__.py:__version__ is written according to the new version.

  2. The updated version file is committed to git.

  3. If on main, a git tag is created. Git tags are not created for prerelease versions (releases made on user or ticket branches.)

Debugging

VSCode

When using VSCode, you can use the debugger built into the Python extension to start an instance of the service in a debug context by creating a .vscode/launch.json file in the Workspace. This configuration file tells VSCode how to start an application for debugging, after which VSCode's breakpoints and other debug tools are available.

This example launch.json file illustrates the launch of a debuggable CM Service and Daemon instances using usdf-cm-dev resources:

{
  "version": "0.2.0",
  "configurations": [
    {
      "name": "Debug: CM-Service [usdf-cm-dev]",
      "type": "debugpy",
      "request": "launch",
      "module": "uvicorn",
      "args": [
        "--port", "8080", "--reload",
        "lsst.cmservice.main:app"
      ],
      "envFile": "${workspaceFolder}/.env.usdf-cm-dev"
    },
    {
      "name": "Debug: User Daemon [usdf-cm-dev]",
      "type": "debugpy",
      "request": "launch",
      "module": "lsst.cmservice.cli.client",
      "args": ["queue", "daemon", "--row_id", "${input:rowID}"],
      "envFile": "${workspaceFolder}/.env.usdf-cm-dev"
    },
    {
      "name": "Debug: System Daemon [usdf-cm-dev]",
      "type": "debugpy",
      "request": "launch",
      "module": "lsst.cmservice.daemon",
      "envFile": "${workspaceFolder}/.env.usdf-cm-dev"
    }
  ],
  "inputs": [
    {
      "id": "rowID",
      "description": "A Campaign Row ID for the debug client queue",
      "type": "promptString",
      "default": "1"
    }
  ]
}

Note that this configuration references an envFile; there is a make target for generating .env files that can be consumed by VSCode debug configurations: make get-env-<k8s cluster> (e.g., make get-env-usdf-cm-dev will produce a file .env.usdf-cm-dev with environment variables set to values appropriate for services running in the named Kubernetes cluster). Otherwise, you may create a custom .env file and reference it from Launch Configuration instead.

The debug Daemon is launched using a prompted value for the --row_id parameter, which will cause VSCode to open a prompt for this input.