TypeScript passed it clean. The code reviewer approved it. It shipped to production. Three months later, a penetration tester sent a report.
The vulnerable line:
const result = await pool.query(
"SELECT * FROM orders WHERE user_id = " + req.query.userId
);
SQL injection has been a known problem for decades. OWASP A03:2021. Parameterized queries are widely understood. And it still ships โ not because developers don't know, but because the three structural forms that actually appear in node-postgres codebases look harmless in code review, one line at a time. (CWE-89)
And now there's a second author on the team that reaches for those exact three forms by default: the coding assistant. Trained on the same corpus that produced this bug for twenty years, it regenerates it on demand โ cleaner-looking, which makes it harder to catch.
Here are the three patterns, why each survives review, why AI assistants reproduce all three, and how a pg-specific ESLint rule catches them statically โ no matter who (or what) wrote the line.
Why a pg-specific rule โ not a generic SQL injection linter
Most SQL injection detectors work on one signal: string concatenation near a SQL keyword. If they see "SELECT" + variable, they flag it. This produces false positives on non-query string building, and misses injection via template literals โ which is syntactically distinct from + but equally dangerous.
A pg-specific rule knows three things a generic tool doesn't:
The API surface. Only fires on
.query()calls โpool.query(),client.query(). Not on other string operations that happen to mention SQL keywords.The parameterization contract. pg uses
$1, $2positional placeholders, with values passed as the second argument array. If the second argument is a non-empty array, the rule treats the first argument as parameterized and stays silent. Note:client.query("SELECT..." + x, [])with an empty array would still be a vulnerability โ the rule checks for the presence of a values argument, not that every dynamic part is covered by a placeholder.Cross-line assignment taint. When a SQL string is built via concatenation and stored in a variable before
.query(), the variable is marked tainted. The rule fires at the assignment โ not just at the call site.
This is why the rule correctly classifies all six cases in its test suite: three vulnerable patterns flagged, three parameterized patterns silent. There is one known false-positive class โ covered in the trade-offs section below โ but the core patterns have no FPs on legitimate parameterized code. The rule is intraprocedural โ taint tracking doesn't cross function boundaries โ but the direct-access patterns below are the ones that actually appear in production code.
Pattern 1: Direct string concatenation
// โ Flagged โ string + user input in a .query() call
const result = await client.query(
"SELECT * FROM users WHERE email = '" + email + "'"
);
Why it survives code review: The concatenation looks harmless in isolation. The reviewer sees string building. Their mental model doesn't ask "where does email come from?" โ that context lives in the route handler, several stack frames up. Nobody holds the full data-flow in mind while reviewing a database layer.
// โ
Parameterized โ rule stays silent
const result = await client.query(
"SELECT * FROM users WHERE email = $1",
[email]
);
The $1 placeholder + second-argument values array is pg's escaping contract. The database driver handles quoting and type coercion. This pattern cannot be accidentally broken.
Pattern 2: Template literal interpolation
// โ Flagged โ same vulnerability, different syntax
const result = await pool.query(
`SELECT * FROM orders WHERE user_id = ${userId} AND status = '${status}'`
);
Why this is especially dangerous: Template literals feel like interpolation โ "variables in a string." Developers who know concatenation is unsafe sometimes don't connect template expressions to the same risk. The syntax is cleaner, so the code feels safer. It isn't.
The detection here is unambiguous: any ${...} expression inside the first argument to .query() โ without a corresponding values array as the second argument โ is a SQL injection surface.
// โ
Parameterized โ stays silent
const result = await pool.query(
"SELECT * FROM orders WHERE user_id = $1 AND status = $2",
[userId, status]
);
Note: a concatenation with a sanitization wrapper โ client.query("WHERE id = " + sanitize(userId)) โ is still flagged. The rule cannot verify that sanitize() is pg-safe. Parameterization is always the fix.
Pattern 3: Cross-line variable assignment
This is the pattern that gets through code review most often.
// โ Flagged at the assignment โ variable is marked tainted
const sql = "SELECT * FROM products WHERE category = '" + category + "'";
const result = await client.query(sql);
At the .query(sql) call, sql looks like a named variable. Nothing at that call site suggests injection. The reviewer's eye is on the call โ not on where sql was built two lines earlier.
The rule tracks this: when a SQL string is assigned via concatenation or template interpolation, the variable is tainted. If that variable is subsequently passed to .query(), the rule fires at the assignment โ where injection was introduced.
// โ
Safe โ stays silent
const sql = "SELECT * FROM products WHERE category = $1";
const result = await client.query(sql, [category]);
The pentester's report? Pattern 3. The sql variable nobody traced back to req.query.
Your AI assistant ships all three by default
These three patterns predate AI. They got harder the moment a coding assistant joined the team โ because the assistant was trained on the same corpus that produced them.
Ask Claude, Gemini, or Copilot to "write a function that fetches orders for a user id from Postgres," and watch which form it reaches for. In my runs it lands on Pattern 1 or Pattern 2 more often than parameterized $1 โ not because the model doesn't know parameterization, but because string-built SQL is the statistically dominant shape in its training data, and the prompt asked for a query, not for a safe query. Parameterization is a constraint. The prompt described behavior, so the model fulfilled behavior. (Try it yourself โ the output is non-deterministic, so re-run a few times and watch the failure class stay constant even as the exact line changes.)
This is the same negative-space failure I measured at scale. When I let Claude write 80 functions, 65โ75% carried at least one security defect. And when I broke a 700-function benchmark down by security domain across five Claude and Gemini models, database operations were a weak spot for every model โ and, tellingly, the model that "won" generation did so by writing simple, parameterized queries, while the ones that generated more elaborate, senior-looking database code triggered more pg rules. It's the database-layer cousin of the NestJS service Claude shipped with six holes: correct, compiling, and quietly unsafe.
The uncomfortable part for review: AI-generated SQL looks more trustworthy than the human kind. It's clean, consistently formatted, and uses a tidy template literal. Pattern 2 โ the template-literal form โ is exactly what a reviewer skims past as "modern, readable code." The linter doesn't skim. It sees ${userId} inside the first argument to .query() and fires, whether a human or a model typed it.
Run it on your assistant's output before you run it on your colleague's. Same rule, same install, no model-specific tuning:
npm install eslint-plugin-pg --save-dev
// eslint.config.mjs
import pg from "eslint-plugin-pg";
export default [
{
plugins: { pg },
rules: { "pg/no-unsafe-query": "error" },
},
];
Because the rule is structural โ not model-aware โ the methodology transfers to any assistant. Paste the same prompt into Gemini via the Gemini CLI, scan the output with pg/no-unsafe-query, and compare the count to Claude's. The model changes; the three patterns don't.
What about ORM escape hatches?
Most production Node.js teams use Prisma, Drizzle, Knex, or TypeORM. Those ORMs parameterize by default โ but they all have raw query escape hatches ($queryRaw, knex.raw, sequelize.literal) where Pattern 1 and 2 reappear. A pg-specific rule won't catch those; the relevant rules are in the ORM's own lint ecosystem.
For teams using pg directly โ internal APIs, data pipelines, microservices โ the three patterns above cover the injection surface. Prisma shops have different lint priorities.
The trade-offs (and the one false positive)
The install and config are above โ pg/no-unsafe-query set to error is the whole setup. Two things worth knowing before you turn it on in CI:
vs. Semgrep/CodeQL: Interprocedural SAST tools can trace taint across function boundaries. ESLint can't โ it's intraprocedural. The trade-off: ESLint runs in your editor on every keystroke and in pre-commit hooks with no CI pipeline required. For a pg team that wants SQL injection feedback where they see TypeScript errors โ including on the SQL an AI assistant just generated โ that speed matters more than the wider taint scope.
Known false positive: client.query("SELECT * FROM " + SCHEMA_NAME) where SCHEMA_NAME is a hardcoded constant. The rule fires because it can't distinguish constants from dynamic inputs. Workaround: use pg-format for identifier quoting, or restructure to a parameterized form.
Full rule docs and configuration: eslint.interlace.tools/docs/security/plugin-pg/rules/no-unsafe-query
Has a parameterized query ever been "refactored" back into concatenation in your codebase โ by a teammate who thought they were cleaning it up, or by an AI assistant that "simplified" the $1 away? Which pattern was it, and how far did it get before someone caught it?
โ Related: Your node-postgres Data Layer Fails 4 Ways in Production โ SQL injection is only the first ยท node-postgres will happily build a CVSS 9.8 SQL injection for you โ 13 ESLint rules say no ยท 30 minutes of ESLint found 26 critical bugs in an inherited codebase
ยท Rule docs ยท โญ GitHub













