solophp / query-builder
A lightweight, fluent SQL query builder for PHP, providing secure and intuitive database interactions.
v1.4.0
2025-04-02 11:35 UTC
Requires
- php: ^8.2
- solophp/database: ^2.8
README
A lightweight, fluent SQL query builder for PHP, providing secure and intuitive database interactions.
✨ Features
- Fluent Interface: Chainable methods for building queries.
- CRUD Support: Quickly create, read, update, and delete.
- Secure Binding: Automatic placeholder handling to prevent SQL injection.
- Alias Parsing: Easy table aliasing (e.g.,
answers_services|a
). - Condition Groups: Complex
WHERE
clauses with closures. - Join Clauses:
INNER JOIN
,LEFT JOIN
,RIGHT JOIN
support inSELECT
andUPDATE
. - Raw SQL: Safely insert raw SQL snippets when needed.
- RawExpression Support: Use RawExpression objects in select fields for complex expressions.
- Select Bindings: Add parameters directly to the select statement for secure value binding.
- HAVING Support: Add
HAVING
clauses the same way you useWHERE
. - Field Mapping: Consistent field mapping for both search and orderBy operations across joins.
- Search Functionality: Easily implement search with keywords across fields and optional fields mapping when using joins.
📥 Installation
Install via Composer:
composer require solophp/query-builder
🚀 Usage
Initialization
use App\Core\QueryBuilder\QueryBuilder; use Solo\Database; $db = new Database(/* your config */); $qb = new QueryBuilder($db);
📘 API Reference
Core Methods
Method | Description |
---|---|
select(array $fields, array $bindings = []) |
Initiate a SELECT query with optional bindings. |
insert(string $table) |
Initiate an INSERT query. |
update(string $table) |
Initiate an UPDATE query. |
delete(string $table) |
Initiate a DELETE query. |
Query Methods (SELECT)
Method | Description |
---|---|
from(string $table) |
Specify the table and optional alias. |
distinct(bool $distinct = true) |
Use SELECT DISTINCT . |
join(string $table, string $condition, string $type) |
Add a JOIN clause (INNER / LEFT / RIGHT). |
leftJoin(string $table, string $condition) |
Add a LEFT JOIN clause. |
rightJoin(string $table, string $condition) |
Add a RIGHT JOIN clause. |
innerJoin(string $table, string $condition) |
Add an INNER JOIN clause. |
where(string $field, string $operator, mixed $value) |
Basic WHERE condition. |
andWhere(string $field, string $operator, mixed $value) |
AND condition (chained). |
orWhere(string $field, string $operator, mixed $value) |
OR condition. |
whereBetween(string $field, mixed $start, mixed $end) |
WHERE BETWEEN condition. |
whereRaw(string $sql, array $bindings = []) |
Insert a raw SQL snippet in WHERE. |
whereGroup(Closure $callback) |
Group multiple conditions via a closure. |
groupBy(string $field) |
GROUP BY a specified field. |
having(string $field, string $operator, mixed $value) |
HAVING condition (works like where ). |
andHaving(string $field, string $operator, mixed $value) |
AND condition for HAVING. |
orHaving(string $field, string $operator, mixed $value) |
OR condition for HAVING. |
havingBetween(string $field, mixed $start, mixed $end) |
HAVING BETWEEN condition. |
havingRaw(string $sql, array $bindings = []) |
Insert a raw SQL snippet in HAVING. |
havingGroup(Closure $callback) |
Group multiple HAVING conditions via a closure. |
orderBy(?string $field, ?string $direction, array $fieldMap = []) |
ORDER BY clause with optional field mapping support. |
addOrderBy(string $field, string $direction, array $fieldMap = []) |
Add additional order criteria with mapping support. |
limit(int $limit, int $offset = 0) |
Limit and offset for pagination. |
paginate(int $page, int $limit) |
Paginate by page number. |
smartSearch(?string $search, array $searchableFields, array $fieldMap = []) |
Add search conditions with field mapping support. |
Execution & Results
Method | Description |
---|---|
get(?int $fetchMode = null) |
Execute SELECT and return all rows. |
getFirst(?int $fetchMode = null) |
Execute SELECT with LIMIT 1 and return a single row. |
getFieldValue(string $field) |
Fetch a single field value from the first result row. |
getFieldValues(string $field) |
Fetch an array of all values for a specific field. |
getIndexedBy(string $field) |
Return an associative array indexed by a specific field. |
count() |
Execute a SELECT COUNT(*) using the current conditions. |
execute() |
Execute INSERT , UPDATE , or DELETE . |
compile() |
Return the generated SQL string without executing. |
📚 Examples
SELECT
$results = $qb ->select(['id', 'title']) ->from('posts|p') ->where('p.created_at', '>', '2023-01-01') ->leftJoin('users|u', 'u.id = p.author_id') ->orderBy('p.id', 'DESC') ->limit(10) ->get();
SELECT With Bindings
$results = $qb ->select([ 'id', 'title', 'DATE_FORMAT(created_at, ?s) AS formatted_date' ], ['%Y-%m-%d']) ->from('posts') ->get();
Using RawExpressions
use Solo\Database\Expressions\RawExpression; $results = $qb ->select([ 'u.id', 'u.email', new RawExpression('CONCAT(u.first_name, " ", u.last_name) AS full_name'), new RawExpression('DATEDIFF(NOW(), u.created_at) AS days_registered') ]) ->from('users|u') ->where('u.status', '=', 'active') ->get();
SELECT WITH JOIN
$results = $qb ->select(['u.id', 'u.name', 'p.phone']) ->from('users|u') ->leftJoin('profiles|p', 'p.user_id = u.id') ->where('u.status', '=', 'active') ->get();
INSERT
$qb->insert('posts') ->values([ 'title' => 'Hello World', 'content' => 'Welcome to my blog!', 'author_id' => 1 ]) ->execute();
UPDATE
$qb->update('posts') ->set(['title' => 'Updated Title']) ->where('id', '=', 5) ->execute();
UPDATE with JOIN
$qb->update('users AS u') ->innerJoin('profiles p', 'p.user_id = u.id') ->set(['u.status' => 'active', 'p.updated_at' => 'NOW()']) ->where('u.id', '=', 42) ->execute();
DELETE
$qb->delete('posts') ->where('id', 'IN', [4, 5, 6]) ->execute();
Raw SQL
$qb->select() ->from('users') ->whereRaw('LENGTH(username) > ?i', [10]) ->get();
Using WHERE
$qb->select() ->from('orders|o') ->whereGroup(function ($builder) { $builder->where('o.status', '=', 'pending') ->orWhere('o.payment_status', '=', 'failed'); }) ->get();
Using HAVING
$results = $qb ->select(['p.category', 'COUNT(p.id) AS total']) ->from('products|p') ->groupBy('p.category') ->having('total', '>', 10) ->get();
You can also chain multiple conditions, use andHavingGroup()
, orHaving()
, etc., just like with WHERE
.
Complex Grouping with WHERE
$results = $qb->select() ->from('orders') ->whereGroup(function ($builder) { $builder->where('status', '=', 'pending') ->orWhere('priority', '>', 5); }) ->get();
Get a specific field value from the first row
$email = $qb->select(['email']) ->from('users') ->where('id', '=', 42) ->getFieldValue('email');
Get an array of values for a specific field
$emails = $qb->select(['email']) ->from('users') ->where('status', '=', 'active') ->getFieldValues('email'); // Example result: ['user1@example.com', 'user2@example.com', ...]
Indexed Results
$users = $qb->select() ->from('users') ->getIndexedBy('id'); // Returns array keyed by user ID
Counting Records
$total = $qb->select() ->from('posts') ->where('status', '=', 'published') ->count();
Using Field Mapping
Search with field mapping:
$fieldMap = [ 'name' => 'p.name', 'category' => 'c.name' ]; $results = $qb ->select(['p.*', 'c.name AS category_name']) ->from('products|p') ->join('categories|c', 'c.id = p.category_id') ->smartSearch('category:Electronics', ['name', 'category'], $fieldMap) ->get();
OrderBy with field mapping:
$fieldMap = [ 'name' => 'p.name', 'price' => 'p.price', 'category' => 'c.name' ]; $results = $qb ->select(['p.*', 'c.name AS category_name']) ->from('products|p') ->join('categories|c', 'c.id = p.category_id') ->smartSearch('Electronics', ['name', 'category'], $fieldMap) ->orderBy('price', 'DESC', $fieldMap) ->get();
Using smartSearch
Basic search across multiple fields:
$results = $qb ->select() ->from('products|p') ->smartSearch('laptop', ['p.name', 'p.description']) ->get();
Search in a specific field using colon syntax:
$results = $qb ->select() ->from('products|p') ->smartSearch('name:gaming laptop', ['p.name', 'p.description', 'p.category']) ->get();
Combined field mapping for search and sorting:
$fieldMap = [ 'category_name' => 'c.name', 'product_name' => 'p.name', 'price' => 'p.price' ]; $results = $qb ->select(['p.*', 'c.name AS category_name']) ->from('products|p') ->join('categories|c', 'c.id = p.category_id') ->smartSearch('category_name:Electronics', ['category_name', 'product_name'], $fieldMap) ->orderBy('price', 'DESC', $fieldMap) ->get();
The search functionality supports multiple keywords (space-separated) and will apply them with an AND condition.
✅ Requirements
- PHP 8.2+
- solophp/database for database connections
📄 License
MIT License. See LICENSE.