Skip to main content

Linter for PostgreSQL migrations

Project description

squawk npm

Linter for Postgres migrations & SQL

Quick Start | Playground | Rules Documentation | GitHub Action | DIY GitHub Integration

Why?

Prevent unexpected downtime caused by database migrations and encourage best practices around Postgres schemas and SQL.

Install

npm install -g squawk-cli

# or via PYPI
pip install squawk-cli

# or install binaries directly via the releases page
https://github.com/sbdchd/squawk/releases

Or via Docker

You can also run Squawk using Docker. The official image is available on GitHub Container Registry.

# Assuming you want to check sql files in the current directory
docker run --rm -v $(pwd):/data ghcr.io/sbdchd/squawk:latest *.sql

Or via the Playground

Use the WASM powered playground to check your SQL locally in the browser!

https://play.squawkhq.com

Or via VSCode

https://marketplace.visualstudio.com/items?itemName=sbdchd.squawk

Usage

 squawk example.sql
warning[prefer-bigint-over-int]: Using 32-bit integer fields can result in hitting the max `int` limit.
  ╭▸ example.sql:6:10
  6      "id" serial NOT NULL PRIMARY KEY,
            ━━━━━━
     help: Use 64-bit integer values instead to prevent hitting this limit.
  ╭╴
6      "id" bigserial NOT NULL PRIMARY KEY,
  ╰╴         +++
warning[prefer-identity]: Serial types make schema, dependency, and permission management difficult.
  ╭▸ example.sql:6:10
  6      "id" serial NOT NULL PRIMARY KEY,
            ━━━━━━
     help: Use an `IDENTITY` column instead.
  ╭╴
6 -     "id" serial NOT NULL PRIMARY KEY,
6 +     "id" integer generated by default as identity NOT NULL PRIMARY KEY,
  ╰╴
warning[prefer-text-field]: Changing the size of a `varchar` field requires an `ACCESS EXCLUSIVE` lock, that will prevent all reads and writes to the table.
  ╭▸ example.sql:7:13
  7      "alpha" varchar(100) NOT NULL
               ━━━━━━━━━━━━
     help: Use a `TEXT` field with a `CHECK` constraint.
  ╭╴
7 -     "alpha" varchar(100) NOT NULL
7 +     "alpha" text NOT NULL
  ╰╴
warning[require-concurrent-index-creation]: During normal index creation, table updates are blocked, but reads are still allowed.
   ╭▸ example.sql:10:1
   10  CREATE INDEX "field_name_idx" ON "table_name" ("field_name");
    ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
       help: Use `concurrently` to avoid blocking writes.
   ╭╴
10  CREATE INDEX concurrently "field_name_idx" ON "table_name" ("field_name");
   ╰╴             ++++++++++++
warning[constraint-missing-not-valid]: By default new constraints require a table scan and block writes to the table while that scan occurs.
   ╭▸ example.sql:12:24
   12  ALTER TABLE table_name ADD CONSTRAINT field_name_constraint UNIQUE (field_name);
                           ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
       help: Use `NOT VALID` with a later `VALIDATE CONSTRAINT` call.
warning[disallowed-unique-constraint]: Adding a `UNIQUE` constraint requires an `ACCESS EXCLUSIVE` lock which blocks reads and writes to the table while the index is built.
   ╭▸ example.sql:12:28
   12  ALTER TABLE table_name ADD CONSTRAINT field_name_constraint UNIQUE (field_name);
                               ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
       help: Create an index `CONCURRENTLY` and create the constraint using the index.

Find detailed examples and solutions for each rule at https://squawkhq.com/docs/rules
Found 6 issues in 1 file (checked 1 source file)

squawk --help

Find problems in your SQL

Usage: squawk [OPTIONS] [path]... [COMMAND]

Commands:
  server            Run the language server
  upload-to-github  Comment on a PR with Squawk's results
  help              Print this message or the help of the given subcommand(s)

Arguments:
  [path]...
          Paths or patterns to search

Options:
      --exclude-path <EXCLUDED_PATH>
          Paths to exclude

          For example:

          `--exclude-path=005_user_ids.sql --exclude-path=009_account_emails.sql`

          `--exclude-path='*user_ids.sql'`

  -e, --exclude <rule>
          Exclude specific warnings

          For example: --exclude=require-concurrent-index-creation,ban-drop-database

  -i, --include <rule>
          Include opt-in rules that are disabled by default

          Rules listed in --exclude take precedence over --include.

          For example: --include=require-table-schema

      --pg-version <PG_VERSION>
          Specify postgres version

          For example: --pg-version=13.0

      --debug <format>
          Output debug format

          [possible values: lex, parse, ast]

      --reporter <REPORTER>
          Style of error reporting

          [possible values: tty, gcc, json, gitlab]

      --stdin-filepath <filepath>
          Path to use in reporting for stdin

      --verbose
          Enable debug logging output

  -c, --config <CONFIG_PATH>
          Path to the squawk config file (.squawk.toml)

      --assume-in-transaction
          Assume that a transaction will wrap each SQL file when run by a migration tool

          Use --no-assume-in-transaction to override any config file that sets this

      --no-error-on-unmatched-pattern
          Do not exit with an error when provided path patterns do not match any files

  -h, --help
          Print help (see a summary with '-h')

  -V, --version
          Print version

Rules

Individual rules can be disabled via the --exclude flag

squawk --exclude=adding-field-with-default,disallowed-unique-constraint example.sql

Disabling rules via comments

Rule violations can be ignored via the squawk-ignore comment:

-- squawk-ignore ban-drop-column
alter table t drop column c cascade;

You can also ignore multiple rules by making a comma seperated list:

-- squawk-ignore ban-drop-column, renaming-column,ban-drop-database
alter table t drop column c cascade;

To ignore a rule for the entire file, use squawk-ignore-file:

-- squawk-ignore-file ban-drop-column
alter table t drop column c cascade;
-- also ignored!
alter table t drop column d cascade;

Or leave off the rule names to ignore all rules for the file

-- squawk-ignore-file
alter table t drop column c cascade;
create table t (a int);

Configuration file

Rules can also be disabled with a configuration file.

By default, Squawk will traverse up from the current directory to find a .squawk.toml configuration file. You may specify a custom path with the -c or --config flag.

squawk --config=~/.squawk.toml example.sql

The --exclude flag will always be prioritized over the configuration file.

Example .squawk.toml

excluded_rules = [
    "require-concurrent-index-creation",
    "require-concurrent-index-deletion",
]

See the Squawk website for documentation on each rule with examples and reasoning.

Bot Setup

Squawk works as a CLI tool but can also create comments on GitHub Pull Requests using the upload-to-github subcommand.

Here's an example comment created by squawk using the example.sql in the repo:

https://github.com/sbdchd/squawk/pull/14#issuecomment-647009446

See the "GitHub Integration" docs for more information.

pre-commit hook

Integrate Squawk into Git workflow with pre-commit. Add the following to your project's .pre-commit-config.yaml:

repos:
  - repo: https://github.com/sbdchd/squawk
    rev: 2.52.0
    hooks:
      - id: squawk
        files: path/to/postgres/migrations/written/in/sql

Note the files parameter as it specifies the location of the files to be linted.

Prior Art / Related

Related Blog Posts / SE Posts / PG Docs

Dev

cargo install
cargo run
./s/test
./s/lint
./s/fmt

... or with nix:

$ nix develop
[nix-shell]$ cargo run
[nix-shell]$ cargo insta review
[nix-shell]$ ./s/test
[nix-shell]$ ./s/lint
[nix-shell]$ ./s/fmt

Adding a New Rule

When adding a new rule, running cargo xtask new-rule will create stubs for your rule in the Rust crate and in Documentation site.

cargo xtask new-rule 'prefer big serial'

Releasing a New Version

  1. Run s/update-version

    # update version in squawk/Cargo.toml, package.json, flake.nix to 4.5.3
    s/update-version 4.5.3
    
  2. Update the CHANGELOG.md

    Include a description of any fixes / additions. Make sure to include the PR numbers and credit the authors.

  3. Create a new release on GitHub

    Use the text and version from the CHANGELOG.md

Algolia

The squawkhq.com Algolia index can be found on the crawler website. Algolia reindexes the site every day at 5:30 (UTC).

How it Works

Squawk uses its parser (based on rust-analyzer's parser) to create a CST. The linters then use an AST layered on top of the CST to navigate and record warnings, which are then pretty printed!

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

squawk_cli-2.52.0.tar.gz (814.0 kB view details)

Uploaded Source

Built Distributions

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

squawk_cli-2.52.0-py3-none-win_amd64.whl (4.1 MB view details)

Uploaded Python 3Windows x86-64

squawk_cli-2.52.0-py3-none-win32.whl (3.8 MB view details)

Uploaded Python 3Windows x86

squawk_cli-2.52.0-py3-none-manylinux_2_28_x86_64.whl (6.8 MB view details)

Uploaded Python 3manylinux: glibc 2.28+ x86-64

squawk_cli-2.52.0-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl (6.0 MB view details)

Uploaded Python 3manylinux: glibc 2.17+ ARM64

squawk_cli-2.52.0-py3-none-macosx_11_0_arm64.whl (4.5 MB view details)

Uploaded Python 3macOS 11.0+ ARM64

squawk_cli-2.52.0-py3-none-macosx_10_12_x86_64.whl (4.6 MB view details)

Uploaded Python 3macOS 10.12+ x86-64

File details

Details for the file squawk_cli-2.52.0.tar.gz.

File metadata

  • Download URL: squawk_cli-2.52.0.tar.gz
  • Upload date:
  • Size: 814.0 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: maturin/1.13.3

File hashes

Hashes for squawk_cli-2.52.0.tar.gz
Algorithm Hash digest
SHA256 32ebe40eb1cd6edcb913c0438cf4057a24a1b7ad294bb4c6b582f38ad8b0fd1d
MD5 953acff8b8e2bfe7eb480bc3e00f2aca
BLAKE2b-256 a60e79e2d18a1b7b66e12c0468041436b0fa47970669af4ee9ef853f87512be7

See more details on using hashes here.

File details

Details for the file squawk_cli-2.52.0-py3-none-win_amd64.whl.

File metadata

File hashes

Hashes for squawk_cli-2.52.0-py3-none-win_amd64.whl
Algorithm Hash digest
SHA256 b617cb5219275d0e7efdd3c15b53f4ede2221ac0e23e18395bd47a44404842f3
MD5 de1bfc1dcc070939656b2e6e033d0226
BLAKE2b-256 16008550be9ca8f905980f21c52048dc0424eb6002bb8fff06ea353fa3cc998c

See more details on using hashes here.

File details

Details for the file squawk_cli-2.52.0-py3-none-win32.whl.

File metadata

  • Download URL: squawk_cli-2.52.0-py3-none-win32.whl
  • Upload date:
  • Size: 3.8 MB
  • Tags: Python 3, Windows x86
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: maturin/1.13.3

File hashes

Hashes for squawk_cli-2.52.0-py3-none-win32.whl
Algorithm Hash digest
SHA256 673f50fd91712b1b79dbdde1d7f5623cc37dce951b53263bca083adffb19719b
MD5 2b450d0aeb8bb13e998d0ce072e44bd7
BLAKE2b-256 ced5116ece9e3dbc2c8334924c1bf26d3bfe18500eac61abd32bfca1429c44e6

See more details on using hashes here.

File details

Details for the file squawk_cli-2.52.0-py3-none-manylinux_2_28_x86_64.whl.

File metadata

File hashes

Hashes for squawk_cli-2.52.0-py3-none-manylinux_2_28_x86_64.whl
Algorithm Hash digest
SHA256 4a1825c29ab78e0095b2625d3e4ae8a7b40327280ee625b7fe3bebda84a1707b
MD5 231955475100bf5da547510f8fae6d24
BLAKE2b-256 188df103e0eec08d6baeb061564842792497431bcabcf0644262d2f5cf6f34b9

See more details on using hashes here.

File details

Details for the file squawk_cli-2.52.0-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl.

File metadata

File hashes

Hashes for squawk_cli-2.52.0-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl
Algorithm Hash digest
SHA256 07731f4d365b769ace6e49d4ed04bb89e62607b9863a8bb53fc9318eced87832
MD5 e5cb361c394359291c9f888ff341bd0f
BLAKE2b-256 bc7dc520b33c4de4a0ae49cd77426f839b711740716c81129079e39e08eb55f2

See more details on using hashes here.

File details

Details for the file squawk_cli-2.52.0-py3-none-macosx_11_0_arm64.whl.

File metadata

File hashes

Hashes for squawk_cli-2.52.0-py3-none-macosx_11_0_arm64.whl
Algorithm Hash digest
SHA256 b210ead15ca901355f44b0cc4a5f2f23d69a2f48cc7923e06f9c11c4d71748a4
MD5 541ef4c628bf2b50a73713f7888ef9f9
BLAKE2b-256 36e048f3e24bd47a8165d3fa07de2845b756463f020a2cf15550135b67df53f7

See more details on using hashes here.

File details

Details for the file squawk_cli-2.52.0-py3-none-macosx_10_12_x86_64.whl.

File metadata

File hashes

Hashes for squawk_cli-2.52.0-py3-none-macosx_10_12_x86_64.whl
Algorithm Hash digest
SHA256 2d4bb9abc44b22e5fc69d9a611783ebd497062a7a4acd57097f4b4a0b1ac5886
MD5 2ebe9fb21cddbc43027f34c4b7cf13a7
BLAKE2b-256 bb81e15d0ab9b24b27fc4476a70cc10e5f3922fd302b8b22282bf71c0dda66ce

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