sad_spirit/pg_builder 问题修复 & 功能扩展

解决BUG、新增功能、兼容多环境部署,快速响应你的开发需求

邮箱:yvsm@zunyunkeji.com | QQ:316430983 | 微信:yvsm316

sad_spirit/pg_builder

最新稳定版本:v3.3.1

Composer 安装命令:

composer require sad_spirit/pg_builder

包简介

Query builder for Postgres backed by SQL parser

README 文档

README

Continuous Integration Static Analysis

This is a query builder for Postgres with a twist: it contains a partial1 reimplementation of PostgreSQL's own query parser. This sets it aside from the usual breed of "write-only" query builders:

  • Query is represented as an Abstract Syntax Tree quite similar to PostgreSQL's internal representation.
  • Query parts can be added to the AST either as objects or as strings (that will be processed by Parser).
  • Nodes can be removed and replaced in AST.
  • AST can be analyzed and transformed, the package takes advantage of this to allow named parameters like :foo instead of standard PostgreSQL's positional parameters $1 and to infer parameters' types from SQL typecasts.
  • Almost all syntax available for SELECT (and VALUES) / INSERT / UPDATE / DELETE / MERGE in PostgreSQL 18 is supported, query being built is automatically checked for correct syntax.

Substantial effort was made to optimise parsing, but not parsing is faster anyway, so there are means to cache parts of AST and the resultant query.

Usage example

use sad_spirit\pg_builder\{ Select, StatementFactory, converters\BuilderSupportDecorator }; use sad_spirit\pg_wrapper\{ Connection, converters\DefaultTypeConverterFactory }; $wantPDO = false; if ($wantPDO) { $pdo = new \PDO('pgsql:host=localhost;user=username;dbname=cms'); // Uses DB connection properties to set up parsing and building of SQL  $factory = StatementFactory::forPDO($pdo); // NB: This still requires sad_spirit/pg_wrapper for type conversion code $converter = new BuilderSupportDecorator(new DefaultTypeConverterFactory(), $factory->getParser()); } else { $connection = new Connection('host=localhost user=username dbname=cms'); // Uses DB connection properties to set up parsing and building of SQL  $factory = StatementFactory::forConnection($connection); // Needed for handling type info extracted from query $connection->setTypeConverterFactory(new BuilderSupportDecorator( $connection->getTypeConverterFactory(), $factory->getParser() )); } // latest 5 news /** @var Select $query */ $query = $factory->createFromString( 'select n.* from news as n order by news_added desc limit 5' ); // we also need pictures for these... $query->list[] = 'p.*'; $query->from[0]->leftJoin('pictures as p')->on = 'n.picture_id = p.picture_id'; // ...and need to limit them to only specific rubrics $query->from[] = 'objects_rubrics as ro'; $query->where->and('ro.rubric_id = any(:rubric::integer[]) and ro.obj_id = n.news_id'); // ...and keep 'em fresh $query->where->and('age(news_added) < :age::interval'); // $generated contains a query, mapping from named parameters to positional ones, types info // it can be easily cached to prevent parsing/building SQL on each request $generated = $factory->createFromAST($query); // Note that we don't have to specify parameter types, these are extracted from query if ($wantPDO) { $result = $pdo->prepare($generated->getSql()); $result->execute($converter->convertParameters( $generated, [ 'rubric' => [19, 20, 21], 'age' => 30 * 24 * 3600 ] )); } else { $result = $generated->executeParams( $connection, [ 'rubric' => [19, 20, 21], 'age' => 30 * 24 * 3600 ] ); } foreach ($result as $row) { print_r($row); } echo $generated->getSql();

the last echo statement will output something like

select n.*, p.* from news as n left join pictures as p on n.picture_id = p.picture_id, objects_rubrics as ro where ro.rubric_id = any($1::pg_catalog.int4[]) and ro.obj_id = n.news_id and age(news_added) < $2::interval order by news_added desc limit 5

if targeting Connection and something like

select n.*, p.* from news as n left join pictures as p on n.picture_id = p.picture_id, objects_rubrics as ro where ro.rubric_id = any(:rubric::pg_catalog.int4[]) and ro.obj_id = n.news_id and age(news_added) < :age::interval order by news_added desc limit 5

if targeting PDO

Installation

Require the package with composer:

composer require "sad_spirit/pg_builder:^3" 

pg_builder requires at least PHP 8.2. Either native pgsql extension with pg_wrapper package or PDO with pgsql support can be used to run the built queries.

Minimum supported PostgreSQL version is 12.

It is highly recommended to use PSR-6 compatible cache in production.

Documentation

For in-depth description of package features, visit pg_builder manual.

1: "Partial" here means the following: PostgreSQL grammar file src/backend/parser/gram.y is about 19K lines long. Of these about 5K lines are used for SELECT / INSERT / UPDATE / DELETE / MERGE queries and are reimplemented here.

统计信息

  • 总下载量: 30.1k
  • 月度下载量: 0
  • 日度下载量: 0
  • 收藏数: 58
  • 点击次数: 3
  • 依赖项目数: 1
  • 推荐数: 0

GitHub 信息

  • Stars: 58
  • Watchers: 2
  • Forks: 6
  • 开发语言: PHP

其他信息

  • 授权协议: BSD-2-Clause
  • 更新时间: 2026-01-04

承接程序开发

PHP开发

VUE

Vue开发

前端开发

小程序开发

公众号开发

系统定制

数据库设计

云部署

网站建设

安全加固