power-vending / laravel-api-query-builder
Laravel API query builder - forked and customized from nealarec/laravel-json-query-builder
Package info
github.com/power-vending/laravel-api-query-builder
pkg:composer/power-vending/laravel-api-query-builder
Requires
- php: ^8.0
- doctrine/dbal: ^3.0|^4.0
- illuminate/support: ^8.0|^9.0|^10.0|^11.0|^12.0
Requires (Dev)
- fakerphp/faker: ^1.9.1
- mockery/mockery: ^1.4.4
- orchestra/testbench: ^8.5|^9.0|^10.0|^11.0|^12.0
- phpunit/phpunit: ^10.0
This package is auto-updated.
Last update: 2026-05-13 20:55:27 UTC
README
Índice
- O que é este pacote
- Como funciona
- Requisitos do sistema
- Instalação
- Configuração inicial
- Rota de schema
- Operadores de busca
- Uso básico
- Parâmetros disponíveis
- Trabalhando com relacionamentos
- Exemplos práticos completos
- Customizações avançadas
- Erros retornados pela API
- Solução de problemas
- Testes
- Créditos e licença
O que é este pacote
Este pacote Laravel permite que você construa consultas (queries) dinâmicas no banco de dados usando parâmetros JSON através de requisições HTTP. Em vez de criar manualmente cada filtro, ordenação e paginação em seus controllers, este pacote processa automaticamente os parâmetros enviados pelo frontend.
Para que serve
Imagine que você tem uma API REST que retorna uma lista de produtos. Sem este pacote, você precisaria criar código para cada tipo de filtro possível:
// Sem o pacote - você precisa tratar cada caso manualmente if ($request->has('name')) { $query->where('name', 'like', '%' . $request->name . '%'); } if ($request->has('min_price')) { $query->where('price', '>=', $request->min_price); } if ($request->has('category')) { $query->where('category_id', $request->category); }
Com este pacote, o frontend envia um JSON estruturado e tudo é processado automaticamente:
// Com o pacote - uma linha resolve tudo return Product::query()->requestPaginate();
Origem
Este pacote foi originalmente desenvolvido por Neal Arec (nealarec/laravel-api-query-builder) e foi internalizado pela Power Vending para permitir customizações específicas e manutenção independente nos projetos internos da empresa.
Como funciona
O pacote funciona interpretando parâmetros JSON enviados via query string (GET) ou body (POST) de requisições HTTP e transformando-os em queries Eloquent do Laravel.
Fluxo de funcionamento
- O frontend faz uma requisição HTTP com parâmetros JSON
- O pacote intercepta esses parâmetros
- Valida e processa cada parâmetro (filtros, ordenação, paginação, etc)
- Constrói a query Eloquent correspondente
- Retorna os dados de acordo com as especificações
Exemplo visual
Frontend envia:
GET /api/products?search={"name":"LIKE:Notebook","price":"GT:1000"}&order_by={"price":"asc"}
Pacote transforma em:
SELECT * FROM products
WHERE name LIKE '%Notebook%'
AND price > 1000
ORDER BY price ASC
Retorna:
{
"data": [...],
"current_page": 1,
"total": 42
}
Requisitos do sistema
Para usar este pacote, você precisa ter as seguintes versões instaladas:
PHP
- Versão 8.0 ou superior
- Extensões necessárias: PDO, Mbstring
Laravel
O pacote é compatível com múltiplas versões do Laravel:
- Laravel 8.x
- Laravel 9.x
- Laravel 10.x
- Laravel 11.x
- Laravel 12.x
Doctrine DBAL
- Versão 3.0 ou superior
- Versão 4.0 também suportada
O Doctrine DBAL é necessário para algumas operações de análise de schema do banco de dados.
Composer
Necessário para gerenciar as dependências do PHP.
Instalação
Instalação via Composer
Instale o pacote usando o Composer:
composer require power-vending/api-query-builder
O pacote será instalado automaticamente e o Laravel irá registrar o service provider.
Configuração inicial
Passo 1: Publicar arquivos de configuração
Publique o arquivo de configuração do pacote para o seu projeto:
php artisan vendor:publish --tag=api-query-builder-config
Este comando irá criar o arquivo config/api-query-builder.php no diretório de configurações do seu projeto.
Passo 2: Entender o arquivo de configuração
Abra o arquivo config/api-query-builder.php. Você verá algo parecido com:
return [ // Operadores de busca disponíveis 'operators' => [ // Lista de classes de operadores ], // Colunas que nunca podem ser acessadas via query 'global_forbidden_columns' => [ 'password', 'remember_token', ], // Outras configurações... ];
Passo 3: Adicionar a trait ao Model
Para que um Model aceite os parâmetros do pacote, adicione a trait ApiQueryBuilder:
<?php namespace App\Models; use Illuminate\Database\Eloquent\Model; use PowerVending\LaravelApiQueryBuilder\Traits\ApiQueryBuilder; class Product extends Model { use ApiQueryBuilder; }
A trait disponibiliza os métodos requestQuery() e requestPaginate() no Model. Sem ela, o pacote não tem efeito sobre o Model.
Passo 4: Configurar colunas proibidas (Segurança)
É muito importante configurar quais colunas não devem ser acessíveis via query para proteger dados sensíveis:
'global_forbidden_columns' => [ 'password', // Senhas nunca devem ser retornadas 'remember_token', // Tokens de sessão 'api_token', // Tokens de API 'secret_key', // Chaves secretas 'credit_card', // Dados bancários 'cpf', // Dados pessoais sensíveis ],
Qualquer tentativa de acessar essas colunas será bloqueada automaticamente.
Também é possível definir colunas proibidas diretamente nos Models, o que é útil para proteger campos específicos de cada tabela:
<?php namespace App\Models; use Illuminate\Database\Eloquent\Model; use PowerVending\LaravelApiQueryBuilder\Traits\ApiQueryBuilder; class User extends Model { use ApiQueryBuilder; /** * Colunas que não podem ser acessadas via API query */ protected $forbiddenColumns = [ 'password', 'remember_token', 'two_factor_secret', 'api_token', ]; }
Ordem de precedência das colunas proibidas:
global_forbidden_columns(config) - aplica a todos os models$forbiddenColumns(model) - específico da modelmodel_options[Model::class]['forbidden_columns'](config) - sobrescreve tudo
IMPORTANTE: As três fontes são mescladas (união). Se quiser usar apenas config, não defina $forbiddenColumns na model.
Passo 5: Configurar estrutura de rotas do pacote
O pacote expõe suas rotas por meio da chave routes na configuração.
No projeto consumidor, configure diretamente no arquivo config/api-query-builder.php:
<?php use PowerVending\LaravelApiQueryBuilder\Http\Controllers\SchemaController; return [ 'resource_models' => [ 'products' => \App\Models\Product::class, 'product-categories' => \App\Models\Category::class, ], 'routes' => [ 'engineering.api.v1.api_query_builder.schema.show' => [ 'method' => 'get', 'uri' => 'eng-api/v1/api-query-builder/{resource}/schema', 'action' => [SchemaController::class, 'show'], 'middlewares' => ['api', 'auth:sanctum', 'acl'], ], ], ];
Se você não quiser customizar o path, pode manter a rota padrão:
'routes' => [ 'api.query_builder.schema.show' => [ 'method' => 'get', 'uri' => 'api-query-builder/{resource}/schema', 'action' => [\PowerVending\LaravelApiQueryBuilder\Http\Controllers\SchemaController::class, 'show'], 'middlewares' => ['api'], ], ],
Estrutura esperada:
'routes' => [ 'nome.completo.da.rota' => [ 'method' => 'get', 'uri' => 'api-query-builder/{resource}/schema', 'action' => [\PowerVending\LaravelApiQueryBuilder\Http\Controllers\SchemaController::class, 'show'], 'middlewares' => ['api'], ], ],
Campos da rota:
method: verbo HTTP (get,post,put,patch,delete,options)uri: caminho completo da rotaaction: array com[Controller::class, 'method']middlewares: lista de middlewares por rota
Exemplo de override no projeto consumidor:
'routes' => [ 'engineering.api.v1.api_query_builder.schema.show' => [ 'method' => 'get', 'uri' => 'eng-api/v1/api-query-builder/{resource}/schema', 'action' => [\PowerVending\LaravelApiQueryBuilder\Http\Controllers\SchemaController::class, 'show'], 'middlewares' => ['api', 'auth:sanctum', 'acl'], ], ],
Observação importante:
- A rota só é registrada quando a
actionaponta para controller/método válidos do namespace do pacote - Configurações inválidas de
actionsão ignoradas para evitar exposição indevida
Rota de schema
A rota de schema retorna metadados para construção de filtros no frontend, incluindo:
- model e table
- searchable_columns (tipo, operadores e nullable)
- sortable_columns
- relations (árvore aninhada)
Rota padrão:
GET /api-query-builder/{resource}/schema
Parâmetros aceitos:
resource(path): chave definida emapi-query-builder.resource_modelsrelations[](query, opcional): lista de relações para limitar/override do retorno
Exemplos de chamada:
GET /api-query-builder/products/schema GET /api-query-builder/products/schema?relations[]=category GET /api-query-builder/products/schema?relations[]=category.parent
Comportamento de relations:
- Sem
relations: usa relações configuradas emmodel_options[Model::class]['relations'] relations[]informado: sobrescreve as relações padrão para aquela requisição- Relação inexistente: lança erro de validação de relação (tratável pelo Handler da aplicação)
Exemplo resumido de resposta:
{
"model": "App\\Models\\Product",
"table": "products",
"searchable_columns": {
"name": {
"type": "varchar(191)",
"operators": ["STARTS_WITH", "ENDS_WITH", "LIKE", "NE", "EQ"],
"nullable": true
}
},
"sortable_columns": ["id", "name"],
"relations": {
"category": {
"model": "App\\Models\\Category",
"table": "categories",
"relations": {
"parent": {
"model": "App\\Models\\Category",
"table": "categories",
"relations": []
}
}
}
}
}
Relacionamentos polimórficos
Relacionamentos polimórficos criados com morphTo() não definem um modelo de destino fixo, o que impede o schema de resolver corretamente os campos e relacionamentos do lado polimórfico.
// Exemplo típico — o schema NÃO consegue introspeccionar o modelo de destino public function requester(): MorphTo { return $this->morphTo(); }
Para que o schema consiga carregar os metadados corretamente, é recomendado (opcional) criar relacionamentos auxiliares tipados com belongsTo, um por cada modelo de destino possível. Cada relacionamento deve filtrar pelo campo *_type correspondente:
// Relacionamentos auxiliares tipados — o schema resolve cada modelo corretamente public function requesterUser(): BelongsTo { return $this->belongsTo(User::class, 'requester_id') ->where('requester_type', User::class); } public function requesterOperator(): BelongsTo { return $this->belongsTo(Operator::class, 'requester_id') ->where('requester_type', Operator::class); }
Com isso, ao solicitar o schema, você pode referenciar cada variação diretamente:
GET /api-query-builder/tickets/schema?relations[]=requester_user&relations[]=requester_operator
E o retorno trará os campos de cada modelo destino de forma independente:
{
"relations": {
"requester_user": {
"model": "App\\Models\\User",
"table": "users",
"relations": {}
},
"requester_operator": {
"model": "App\\Models\\Operator",
"table": "operators",
"relations": {}
}
}
}
Nota: O relacionamento
morphTo()original pode ser mantido no model normalmente — os relacionamentos auxiliares são apenas para uso com o schema e não interferem no comportamento padrão do Eloquent.
Operadores de busca
Antes de começar a usar o pacote, é fundamental entender os operadores disponíveis. Os operadores definem como a comparação será feita nos filtros (igual a, maior que, contém, etc).
Sintaxe geral
"coluna": "OPERADOR:valor"
Para múltiplos valores, use ; (ponto e vírgula) como separador:
"coluna": "OPERADOR:valor1;valor2;valor3"
Comportamento de EQ e NE em colunas string
Os operadores EQ: e NE: têm comportamento condicional que depende do tipo da coluna resolvido via introspecção de schema e da quantidade de valores:
| Situação | SQL gerado |
|---|---|
Tipo resolvido como string + valor único (sem wildcards) |
LIKE / NOT LIKE |
Qualquer outro caso (múltiplos valores, tipo generic, tipo numérico, relação aninhada) |
IN / NOT IN |
O tipo da coluna é determinado em tempo de execução consultando o schema do banco. Se a coluna não for encontrada no schema do model (por exemplo, em buscas dentro de relações aninhadas ou quando a introspecção falha), o tipo cai para generic e o comportamento passa a ser sempre IN.
Exemplo com tipo resolvido como texto (string, varchar, text, etc.):
{"name": "EQ:Notebook Dell"} -> WHERE name LIKE 'Notebook Dell' (1 valor, tipo texto - match exato)
{"name": "EQ:%Dell%"} -> WHERE name LIKE '%Dell%' (1 valor com wildcard - contém)
{"name": "EQ:Dell;HP"} -> WHERE name IN ('Dell', 'HP') (múltiplos valores)
Exemplo com tipo numérico ou genérico:
{"category_id": "EQ:1"} -> WHERE category_id IN (1)
Comportamento do
EQ:com um único valor de texto: Agora detecta corretamente colunas do tipovarchar,text,char, etc. (além destring) e usaLIKEem vez deIN. Para busca parcial (contém), use o micro-operador%nos valores ou prefira os operadoresLIKE:,STARTS_WITH:ouENDS_WITH:.
PostgreSQL: Usa
ILIKEem vez deLIKEnas situações em que LIKE é gerado.
Lista completa de operadores
EQ - Equals (IN / LIKE condicional)
Comportamento:
- Colunas de texto (varchar, text, char, string, etc.) com um único valor: usa
LIKE - Colunas de texto com múltiplos valores: usa
IN - Colunas numéricas ou outros tipos: sempre usa
IN
{"category_id": "EQ:1"} -> WHERE category_id IN (1)
{"category_id": "EQ:1;2;3"} -> WHERE category_id IN (1, 2, 3)
{"name": "EQ:Notebook;Mouse"} -> WHERE name IN ('Notebook', 'Mouse')
{"name": "EQ:Notebook Dell"} -> WHERE name LIKE 'Notebook Dell' (match exato)
{"name": "EQ:%Dell%"} -> WHERE name LIKE '%Dell%' (contém)
{"name": "EQ:Notebook%"} -> WHERE name LIKE 'Notebook%' (começa com)
{"name": "EQ:%Gamer"} -> WHERE name LIKE '%Gamer' (termina com)
Dica: Para buscas textuais com wildcards automáticos, use
LIKE:(adiciona%em ambos os lados),STARTS_WITH:ouENDS_WITH:.
NE - Not Equals (NOT IN / NOT LIKE condicional)
Inverso do EQ:. Segue as mesmas regras de detecção de tipo.
Comportamento:
- Colunas de texto (varchar, text, char, string, etc.) com um único valor: usa
NOT LIKE - Colunas de texto com múltiplos valores: usa
NOT IN - Colunas numéricas ou outros tipos: sempre usa
NOT IN
{"category_id": "NE:3"} -> WHERE category_id NOT IN (3)
{"category_id": "NE:3;4"} -> WHERE category_id NOT IN (3, 4)
{"name": "NE:Usado;Defeito"} -> WHERE name NOT IN ('Usado', 'Defeito')
{"name": "NE:Notebook Dell"} -> WHERE name NOT LIKE 'Notebook Dell'
{"name": "NE:%teste%"} -> WHERE name NOT LIKE '%teste%'
GT - Greater Than (Maior que)
Aceita exatamente um valor.
{"price": "GT:1000"} -> WHERE price > 1000
{"stock": "GT:0"} -> WHERE stock > 0
{"discount": "GT:10"} -> WHERE discount > 10
GE - Greater than or Equal (Maior ou igual a)
Aceita exatamente um valor.
{"price": "GE:1000"} -> WHERE price >= 1000
{"stock": "GE:5"} -> WHERE stock >= 5
LT - Less Than (Menor que)
Aceita exatamente um valor.
{"price": "LT:100"} -> WHERE price < 100
{"discount": "LT:50"} -> WHERE discount < 50
LE - Less than or Equal (Menor ou igual a)
Aceita exatamente um valor.
{"price": "LE:100"} -> WHERE price <= 100
{"discount": "LE:20"} -> WHERE discount <= 20
BT - Between (Entre dois valores)
Aceita exatamente dois valores separados por ;.
{"price": "BT:100;500"} -> WHERE price BETWEEN 100 AND 500
{"stock": "BT:10;100"} -> WHERE stock BETWEEN 10 AND 100
{"created_at": "BT:2024-01-01;2024-12-31"} -> WHERE created_at BETWEEN '2024-01-01' AND '2024-12-31'
NB - Not Between (Não está entre)
Inverso do BT:. Aceita exatamente dois valores separados por ;.
{"price": "NB:100;500"} -> WHERE price NOT BETWEEN 100 AND 500
{"discount": "NB:0;5"} -> WHERE discount NOT BETWEEN 0 AND 5
LIKE - Like (Contém)
Busca o valor em qualquer posição do texto. O % é adicionado automaticamente em ambos os lados.
{"name": "LIKE:Dell"} -> WHERE name LIKE '%Dell%'
{"description": "LIKE:gamer"} -> WHERE description LIKE '%gamer%'
Nota: Diferente do EQ:, o operador LIKE: sempre usa LIKE, independente da quantidade de valores ou tipo da coluna.
STARTS_WITH - Starts With (Começa com)
{"name": "STARTS_WITH:Notebook"} -> WHERE name LIKE 'Notebook%'
{"sku": "STARTS_WITH:PRD"} -> WHERE sku LIKE 'PRD%'
{"code": "STARTS_WITH:DEV"} -> WHERE code LIKE 'DEV%'
ENDS_WITH - Ends With (Termina com)
{"name": "ENDS_WITH:Gamer"} -> WHERE name LIKE '%Gamer'
{"filename": "ENDS_WITH:.pdf"} -> WHERE filename LIKE '%.pdf'
Micro-operadores
Micro-operadores são prefixos aplicados diretamente em cada valor (após o operador principal) para modificar o comportamento de busca individualmente. Funcionam com EQ: e NE:.
! — Negação de valor individual
Nega somente aquele valor dentro de uma lista. Combina IN e NOT IN na mesma query.
{"status": "EQ:active;!deleted;!archived"}
WHERE status IN ('active') AND status NOT IN ('deleted', 'archived')
Diferença entre
NE:e!:
NE:val1;val2— nega toda a lista:NOT IN ('val1', 'val2')EQ:val1;!val2— val1 vai para IN, val2 vai para NOT IN
% — Wildcard (LIKE parcial)
Adiciona busca LIKE quando colocado no início, no fim, ou em ambas as extremidades do valor.
{"name": "EQ:Notebook%;%Mouse;%Gamer%"}
WHERE name LIKE 'Notebook%' AND name LIKE '%Mouse' AND name LIKE '%Gamer%'
Funciona com negação também:
{"name": "EQ:!%usado%"} -> WHERE name NOT LIKE '%usado%'
null e !null — Verificação de nulo
{"discount": "EQ:null"} -> WHERE discount IS NULL
{"discount": "EQ:!null"} -> WHERE discount IS NOT NULL
Operadores lógicos por coluna (&& e ||)
Permitem combinar múltiplas condições de operadores para uma mesma coluna usando AND ou OR. A sintaxe é colocar o operador lógico entre dois pares OPERADOR:valor.
{"id": "EQ:1||EQ:2"}
WHERE id IN (1) OR id IN (2)
{"name": "EQ:Notebook%&&EQ:%Dell"}
WHERE name LIKE 'Notebook%' AND name LIKE '%Dell'
Atenção: A precedência segue a lógica booleana padrão: AND tem prioridade sobre OR. Portanto
x&&y||z&&qequivale a(x AND y) OR (z AND q).
Operadores lógicos como chave (&& e || top-level)
Além de usar && e || dentro do valor de uma coluna, você também pode usá-los como chaves dentro do objeto search para agrupar condições de colunas diferentes com AND ou OR.
&& como chave — AND entre grupos de colunas
{
"search": {
"&&": [
{"status": "EQ:active"},
{"price": "GT:100"}
]
}
}
WHERE (status IN ('active') AND price > 100)
|| como chave — OR entre grupos de colunas
{
"search": {
"||": [
{"status": "EQ:active"},
{"status": "EQ:pending"}
]
}
}
WHERE (status IN ('active') OR status IN ('pending'))
Combinando || e && com condições normais
{
"search": {
"category_id": "EQ:5",
"||": [
{"status": "EQ:active"},
{"status": "EQ:pending"}
]
}
}
WHERE category_id IN (5) AND (status IN ('active') OR status IN ('pending'))
Exemplo avançado: agrupamento aninhado
{
"search": {
"&&": [
{"price": "GT:100"},
{
"||": [
{"status": "EQ:active"},
{"status": "EQ:featured"}
]
}
]
}
}
WHERE ( price > 100 AND (status IN ('active') OR status IN ('featured')) )
Diferença entre os dois usos:
- Dentro do valor (
"name": "EQ:A||EQ:B") — aplica OR/AND sobre a mesma coluna- Como chave (
"||": [...]) — aplica OR/AND sobre grupos de colunas distintas
IMPORTANTE - Ordem dos operadores na configuração
A ordem dos operadores no arquivo config/api-query-builder.php é crítica. Operadores com mais caracteres devem vir antes dos com menos caracteres, pois o parser busca o primeiro que encontrar na string.
'operators' => [ StartsWith::class, // STARTS_WITH: (13 chars) EndsWith::class, // ENDS_WITH: (10 chars) Like::class, // LIKE: (5 chars) NotEquals::class, // NE: (3 chars) NotBetween::class, // NB: (3 chars) LessThanOrEqual::class, // LE: (3 chars) GreaterThanOrEqual::class, // GE: (3 chars) Between::class, // BT: (3 chars) Equals::class, // EQ: (3 chars) LessThan::class, // LT: (3 chars) GreaterThan::class, // GT: (3 chars) ],
Não altere essa ordem a menos que saiba exatamente o que está fazendo.
Uso básico
Passo 1: Adicionar o Trait ao Model
Para usar o pacote em um Model, adicione o trait ApiQueryBuilder:
<?php namespace App\Models; use Illuminate\Database\Eloquent\Model; use PowerVending\LaravelApiQueryBuilder\Traits\ApiQueryBuilder; class Product extends Model { use ApiQueryBuilder; protected $fillable = [ 'name', 'description', 'price', 'category_id', 'stock', ]; }
O trait ApiQueryBuilder adiciona os métodos requestQuery() e requestPaginate() ao seu model, que processam automaticamente os parâmetros JSON das requisições.
Passo 2: Usar no Controller
No seu controller, use o método requestPaginate() para processar automaticamente os parâmetros da requisição:
<?php namespace App\Http\Controllers\Api; use App\Models\Product; use Illuminate\Http\Request; class ProductController extends Controller { public function index(Request $request) { // Método requestPaginate() - retorna resultados paginados return Product::query()->requestPaginate(); } public function search(Request $request) { // Método requestQuery() - retorna o Builder para customizações $query = Product::query()->requestQuery(); // Você pode adicionar condições extras ao Builder $query->where('company_id', auth()->user()->company_id); return $query->get(); } }
Diferença entre os métodos:
requestPaginate(): Processa os parâmetros e retorna resultados paginados automaticamenterequestQuery(): Processa os parâmetros e retorna o Builder para você adicionar mais condições
Pronto! Agora seu endpoint já aceita todos os parâmetros JSON do pacote.
Passo 3: Fazer uma requisição do frontend
Do frontend, você pode fazer requisições como:
// JavaScript - Fetch API const params = { search: { name: "LIKE:Notebook", price: "GT:1000" }, order_by: { price: "asc" }, _per_page: 15, page: 1 }; const queryString = new URLSearchParams(params).toString(); const response = await fetch(`/api/products?${queryString}`); const data = await response.json();
Parâmetros disponíveis
Agora que você já conhece os operadores de busca, vamos explorar todos os parâmetros que você pode usar nas requisições JSON para construir queries dinâmicas.
1. search (Filtros)
Permite filtrar os dados usando os operadores que você acabou de aprender na seção anterior.
Sintaxe:
{
"search": {
"coluna": "OPERADOR:valor"
}
}
Exemplo:
{
"search": {
"name": "LIKE:Notebook",
"price": "GT:1000",
"status": "EQ:active"
}
}
Query string:
?search={"name":"LIKE:Notebook","price":"GT:1000","status":"EQ:active"}
2. relations (Carregar relacionamentos)
Carrega relacionamentos Eloquent junto com a entidade principal (eager loading).
Sintaxe básica (relacionamentos simples)
{
"relations": ["relacionamento1", "relacionamento2"]
}
Exemplo:
{
"relations": ["category", "manufacturer", "reviews"]
}
Query string:
?relations=["category","manufacturer","reviews"]
Isso irá executar algo como:
Product::with(['category', 'manufacturer', 'reviews'])->get();
Sintaxe avançada (relacionamentos com configurações)
Você pode passar configurações específicas para cada relacionamento, permitindo filtrar, ordenar e selecionar campos dentro do relacionamento:
{
"relations": [
"category",
{
"reviews": {
"search": {
"rating": "GE:4"
},
"order_by": {
"created_at": "desc"
},
"returns": ["id", "rating", "comment", "created_at"]
}
}
]
}
O que acontece:
category: carrega normalmente (simples)reviews: carrega apenas reviews com rating >= 4, ordenadas por data decrescente, retornando apenas as colunas especificadas
Possibilidades dentro das configurações de relacionamento:
search- Filtrar registros do relacionamentoorder_by- Ordenar registros do relacionamentoreturns- Selecionar colunas específicas do relacionamentolimit- Limitar quantidade de registros do relacionamento
Exemplo completo: Produtos com reviews filtradas
{
"search": {
"price": "BT:100;1000",
"status": "EQ:active"
},
"relations": [
"category",
{
"reviews": {
"search": {
"rating": "GE:4",
"verified": "EQ:1"
},
"order_by": {
"helpful_votes": "desc",
"created_at": "desc"
}
}
},
{
"images": {
"search": {
"is_primary": "EQ:0"
},
"order_by": {
"order": "asc"
}
}
}
],
"_per_page": 20
}
Resultado:
- Busca produtos entre 100 e 1000, status ativo
- Carrega categoria (todas)
- Carrega apenas reviews verificadas com rating >= 4, ordenadas por votos úteis
- Carrega apenas imagens que não são primárias, ordenadas por ordem
Casos de uso práticos
1. Carregar últimos 5 pedidos de cada cliente:
{
"relations": [
{
"orders": {
"order_by": {"created_at": "desc"},
"limit": 5
}
}
]
}
2. Carregar comentários ativos, mais recentes primeiro:
{
"relations": [
{
"comments": {
"search": {"status": "EQ:approved"},
"order_by": {"created_at": "desc"}
}
}
]
}
3. Misturar relacionamentos simples e configurados:
{
"relations": [
"user",
"category",
{
"tags": {
"order_by": {"name": "asc"}
}
}
]
}
3. order_by (Ordenação)
Define a ordem dos resultados.
Sintaxe:
{
"order_by": {
"coluna1": "asc",
"coluna2": "desc"
}
}
Exemplo:
{
"order_by": {
"price": "asc",
"created_at": "desc"
}
}
Query string:
?order_by={"price":"asc","created_at":"desc"}
Valores aceitos:
asc- ordem crescente (A-Z, 0-9, mais antigo para mais novo)desc- ordem decrescente (Z-A, 9-0, mais novo para mais antigo)
4. limit e offset (Paginação manual)
Controla quantos registros retornar e a partir de qual posição.
Sintaxe:
?limit=10&offset=20
Explicação:
limit: quantos registros retornar (máximo)offset: quantos registros pular antes de começar a retornar
Exemplo:
?limit=10&offset=0 -> Registros 1-10 (primeira página)
?limit=10&offset=10 -> Registros 11-20 (segunda página)
?limit=10&offset=20 -> Registros 21-30 (terceira página)
5. page e _per_page (Paginação automática)
Paginação estilo Laravel Paginator.
Sintaxe:
?page=1&_per_page=15
Explicação:
page: número da página (começa em 1)_per_page: quantos registros por página
Exemplo:
?page=1&_per_page=20 -> Primeira página, 20 itens
?page=2&_per_page=20 -> Segunda página, 20 itens
Resposta:
{
"current_page": 1,
"data": [...],
"first_page_url": "http://api.com/products?page=1",
"from": 1,
"last_page": 5,
"last_page_url": "http://api.com/products?page=5",
"next_page_url": "http://api.com/products?page=2",
"path": "http://api.com/products",
"per_page": 20,
"prev_page_url": null,
"to": 20,
"total": 100
}
6. returns (Selecionar colunas específicas)
Retorna apenas as colunas especificadas (SELECT específico).
Sintaxe:
{
"returns": ["coluna1", "coluna2", "coluna3"]
}
Exemplo:
{
"returns": ["id", "name", "price"]
}
Query string:
?returns=["id","name","price"]
Isso gera:
SELECT id, name, price FROM products
Em vez de:
SELECT * FROM products
Benefícios:
- Reduz o tráfego de rede
- Melhora a performance
- Protege dados sensíveis
7. excepts (Excluir colunas específicas)
Retorna todas as colunas EXCETO as especificadas.
Sintaxe:
{
"excepts": ["coluna1", "coluna2"]
}
Exemplo:
{
"excepts": ["password", "remember_token", "api_token"]
}
Query string:
?excepts=["password","remember_token","api_token"]
Nota: Não use returns e excepts ao mesmo tempo. Escolha um ou outro.
8. count (Adicionar contagem)
Adiciona count(*) as count ao SELECT. Útil para queries de agregação.
Sintaxe:
?count=true
Exemplo:
{
"search": {
"status": "EQ:active"
},
"count": true
}
?search={"status":"EQ:active"}&count=true
9. group_by (Agrupamento)
Agrupa os resultados por uma ou mais colunas.
Sintaxe:
{
"group_by": ["coluna1", "coluna2"]
}
Exemplo:
{
"group_by": ["category_id", "status"]
}
Query string:
?group_by=["category_id","status"]
Isso gera:
GROUP BY category_id, status
10. soft_deleted (Incluir registros deletados)
Inclui registros com soft delete na query. Por padrão, o Eloquent exclui registros com deleted_at preenchido. Este parâmetro remove esse escopo.
Sintaxe:
?soft_deleted=true
Requer que o Model use a trait SoftDeletes do Laravel:
use Illuminate\Database\Eloquent\SoftDeletes; class Product extends Model { use SoftDeletes; use ApiQueryBuilder; }
Exemplo:
?search={"name":"LIKE:Produto"}&soft_deleted=true
-- SEM soft_deleted=true SELECT * FROM products WHERE deleted_at IS NULL AND name LIKE '%Produto%' -- COM soft_deleted=true SELECT * FROM products WHERE name LIKE '%Produto%'
11. doesnt_have_relations (Excluir registros com relacionamento)
Filtra apenas registros que não possuem os relacionamentos especificados (equivale ao doesntHave() do Eloquent).
Sintaxe:
{
"doesnt_have_relations": ["relacionamento1", "relacionamento2"]
}
Exemplo:
{
"doesnt_have_relations": ["orders", "reviews"]
}
Query string:
?doesnt_have_relations=["orders","reviews"]
Isso gera:
$query->doesntHave('orders')->doesntHave('reviews');
Útil para encontrar registros "órfãos":
{
"doesnt_have_relations": ["category"]
}
→ Retorna produtos que não estão associados a nenhuma categoria.
Trabalhando com relacionamentos
O pacote suporta busca, ordenação e carregamento através de relacionamentos Eloquent usando notação de ponto (dot notation).
Tipos de relacionamentos suportados
BelongsTo(Pertence a)HasOne(Tem um)HasMany(Tem muitos)
Busca em campos de relacionamentos
Você pode buscar por colunas de tabelas relacionadas usando a notação de ponto no parâmetro search.
Exemplo: Buscar produtos por nome da categoria
Model:
class Product extends Model { use ApiQueryBuilder; public function category() { return $this->belongsTo(Category::class); } }
JSON:
{
"search": {
"category.name": "LIKE:Electronics",
"price": "GT:100"
},
"relations": ["category"]
}
SQL gerado:
SELECT products.* FROM products LEFT JOIN categories ON products.category_id = categories.id WHERE categories.name LIKE '%Electronics%' AND products.price > 100
Busca em múltiplos níveis de relacionamento
Você pode buscar em relacionamentos aninhados:
{
"search": {
"category.parent.name": "EQ:Technology",
"store.city.name": "LIKE:São Paulo"
},
"relations": ["category.parent", "store.city"]
}
Ordenação por relacionamento
Você pode ordenar os resultados pela coluna de uma tabela relacionada usando notação de ponto.
Exemplo de Model com relacionamento
<?php namespace App\Models; use Illuminate\Database\Eloquent\Model; use PowerVending\LaravelApiQueryBuilder\Traits\ApiQueryBuilder; class Product extends Model { use ApiQueryBuilder; public function brand() { return $this->belongsTo(Brand::class); } }
Ordenando por coluna do relacionamento
JSON:
{
"order_by": {
"brand.name": "desc",
"created_at": "asc"
}
}
O que acontece internamente:
- O pacote detecta que
brand.nameé um relacionamento - Verifica se o método
brand()existe no model - Faz um
LEFT JOINcom a tabelabrands - Ordena pelo campo
nameda tabela relacionada
SQL gerado:
SELECT products.* FROM products LEFT JOIN brands ON products.brand_id = brands.id ORDER BY brands.name DESC, products.created_at ASC
Qualificação automática de colunas
Quando você usa ordenação por relacionamento, pode ocorrer ambiguidade se ambas as tabelas tiverem colunas com o mesmo nome (ex: id, name, created_at).
O pacote resolve isso automaticamente qualificando as colunas do returns com o nome da tabela.
Exemplo:
{
"returns": ["id", "name", "created_at"],
"order_by": {
"brand.name": "desc"
}
}
SQL gerado:
SELECT products.id, products.name, products.created_at FROM products LEFT JOIN brands ON products.brand_id = brands.id ORDER BY brands.name DESC
Sem a qualificação automática, o SQL seria:
SELECT id, name -- ERRO: Column 'name' in SELECT is ambiguous
Exemplo completo com relacionamento
JSON completo:
{
"_per_page": 10,
"page": 1,
"search": {
"status": "EQ:active"
},
"relations": [
"brand"
],
"returns": [
"id",
"brand_id",
"name",
"created_at"
],
"order_by": {
"brand.name": "desc",
"created_at": "asc"
}
}
Explicação:
- Busca produtos com status "active"
- Carrega o relacionamento
brand(eager loading) - Retorna apenas as colunas especificadas
- Ordena primeiro pelo nome da marca (decrescente), depois pela data de criação (crescente)
- Pagina com 10 itens por página
Convenção de nomes
O nome do relacionamento no JSON deve corresponder ao nome do método no Model:
// No Model public function brand() // método em camelCase { return $this->belongsTo(Brand::class); }
// No JSON { "order_by": { "brand.name": "desc" // usa o nome do método } }
Combinando busca e ordenação em relacionamentos
Você pode combinar busca, ordenação e eager loading em relacionamentos para queries complexas.
Exemplo completo: Produtos com busca e ordenação por categoria
Models:
class Product extends Model { use ApiQueryBuilder; public function category() { return $this->belongsTo(Category::class); } public function brand() { return $this->belongsTo(Brand::class); } }
JSON:
{
"search": {
"category.name": "LIKE:Electronics",
"brand.country": "EQ:Brazil",
"price": "BT:500;5000",
"stock": "GT:0"
},
"order_by": {
"category.name": "asc",
"brand.name": "asc",
"price": "desc"
},
"relations": ["category", "brand"],
"returns": [
"id",
"name",
"price",
"stock",
"category_id",
"brand_id"
],
"_per_page": 15
}
SQL gerado:
SELECT products.id, products.name, products.price, products.stock, products.category_id, products.brand_id FROM products LEFT JOIN categories ON products.category_id = categories.id LEFT JOIN brands ON products.brand_id = brands.id WHERE categories.name LIKE '%Electronics%' AND brands.country IN ('Brazil') AND products.price BETWEEN 500 AND 5000 AND products.stock > 0 ORDER BY categories.name ASC, brands.name ASC, products.price DESC LIMIT 15
Configurações avançadas de relacionamentos
Além de buscar e ordenar usando JOINs (notação de ponto), você pode configurar como os relacionamentos são carregados passando objetos no parâmetro relations.
Sintaxe: Relacionamentos com configurações inline
{
"relations": [
"simpleRelation",
{
"relationName": {
"search": {...},
"order_by": {...},
"returns": [...],
"limit": 10
}
}
]
}
O que isso faz:
- Aplica filtros, ordenação e limites dentro do relacionamento
- Útil para relacionamentos
HasManyonde você quer os N mais recentes/relevantes - Não usa JOIN - usa subquery no eager loading
Exemplo 1: Últimos 5 pedidos de cada cliente
Model:
class Customer extends Model { use ApiQueryBuilder; public function orders() { return $this->hasMany(Order::class); } }
JSON:
{
"search": {
"status": "EQ:active"
},
"relations": [
{
"orders": {
"order_by": {
"created_at": "desc"
},
"limit": 5
}
}
]
}
Resultado:
- Cada cliente ativo terá seus 5 pedidos mais recentes carregados
Exemplo 2: Produtos com reviews aprovadas e bem avaliadas
Model:
class Product extends Model { use ApiQueryBuilder; public function reviews() { return $this->hasMany(Review::class); } }
JSON:
{
"relations": [
"category",
{
"reviews": {
"search": {
"status": "EQ:approved",
"rating": "GE:4"
},
"order_by": {
"helpful_votes": "desc",
"created_at": "desc"
},
"returns": ["id", "rating", "comment", "helpful_votes", "created_at"]
}
}
],
"_per_page": 20
}
Resultado:
- Produtos com categoria carregada normalmente
- Apenas reviews aprovadas com rating >= 4
- Ordenadas por votos úteis e data
- Retorna apenas os campos especificados
Exemplo 3: Misturando configurações simples e avançadas
{
"search": {
"status": "EQ:published"
},
"relations": [
"author",
"category",
{
"comments": {
"search": {
"status": "EQ:approved"
},
"order_by": {
"created_at": "desc"
}
}
},
{
"tags": {
"order_by": {
"name": "asc"
}
}
}
]
}
Resultado:
authorecategory: carregados normalmente (sem filtros)comments: apenas aprovados, mais recentes primeirotags: ordenadas alfabeticamente
Diferença: JOIN vs Eager Loading Configurado
JOIN (notação de ponto em search ou order_by):
{
"search": {
"category.name": "LIKE:Electronics"
}
}
→ Usa LEFT JOIN, afeta a query principal, filtra produtos
Eager Loading Configurado (objeto em relations):
{
"relations": [
{
"reviews": {
"search": {"rating": "GE:4"}
}
}
]
}
→ Usa subquery separada, não afeta produtos, apenas filtra reviews carregadas
Casos de uso práticos
1. Last 10 activities de cada usuário:
{
"relations": [
{
"activities": {
"order_by": {"created_at": "desc"},
"limit": 10
}
}
]
}
2. Apenas notificações não lidas:
{
"relations": [
{
"notifications": {
"search": {"read_at": "EQ:null"}
}
}
]
}
3. Top 3 produtos de cada categoria:
{
"relations": [
{
"products": {
"order_by": {"sales_count": "desc"},
"limit": 3
}
}
]
}
Limitações e boas práticas
✅ Suportado:
- Busca em campos de relacionamentos
BelongsTo,HasOne - Ordenação em campos de relacionamentos
BelongsTo,HasOne - Múltiplos relacionamentos na mesma query
- Relacionamentos aninhados (ex:
category.parent.name)
⚠️ Atenção:
- Relacionamentos
HasManypodem gerar resultados duplicados - Use
returnspara evitar conflitos de coluna com mesmo nome - Sempre inclua o relacionamento no
relationsquando buscar/ordenar por ele
💡 Dica de performance:
- Evite muitos JOINs desnecessários
- Use índices nas colunas de foreign keys
- Considere usar
returnspara limitar as colunas selecionadas
Exemplos práticos completos
Exemplo 1: Lista de produtos com filtros básicos
Cenário: Listar produtos ativos, com preço entre 100 e 1000, ordenados por preço crescente.
JSON:
{
"search": {
"status": "EQ:active",
"price": "BT:100;1000"
},
"order_by": {
"price": "asc"
},
"_per_page": 20,
"page": 1
}
Query string:
GET /api/products?search={"status":"EQ:active","price":"BT:100;1000"}&order_by={"price":"asc"}&_per_page=20&page=1
Controller:
public function index() { return Product::query()->requestPaginate(); }
Exemplo 2: Busca de usuários por nome e email
Cenário: Buscar usuários cujo nome ou email contenha "silva".
JSON:
{
"search": {
"name": "LIKE:silva"
},
"returns": ["id", "name", "email", "created_at"],
"_per_page": 50
}
Nota: Para buscar em múltiplos campos com OR, você precisará customizar a query.
Exemplo 3: Relatório de vendas por período
Cenário: Vendas realizadas entre duas datas, com informações completas do cliente.
JSON:
{
"search": {
"created_at": "BT:2024-01-01;2024-12-31",
"status": "EQ:completed"
},
"relations": ["customer", "items", "items.product"],
"order_by": {
"created_at": "desc"
},
"returns": [
"id",
"customer_id",
"total",
"status",
"created_at"
]
}
Exemplo 4: Listagem com relacionamento ordenado
Cenário: Produtos ordenados pelo nome da marca.
JSON:
{
"search": {
"status": "EQ:active"
},
"relations": ["brand"],
"order_by": {
"brand.name": "asc",
"id": "asc"
},
"_per_page": 25
}
Exemplo 5: Clientes com últimos pedidos filtrados e ordenados
Cenário: Listar clientes ativos com seus 5 últimos pedidos completados.
Models:
class Customer extends Model { use ApiQueryBuilder; public function orders() { return $this->hasMany(Order::class); } public function address() { return $this->belongsTo(Address::class); } }
JSON:
{
"search": {
"status": "EQ:active"
},
"relations": [
"address",
{
"orders": {
"search": {
"status": "EQ:completed"
},
"order_by": {
"completed_at": "desc"
},
"returns": ["id", "total", "status", "completed_at"],
"limit": 5
}
}
],
"order_by": {
"created_at": "desc"
},
"_per_page": 20
}
Resultado:
- Lista clientes ativos ordenados por data de cadastro (mais recentes primeiro)
- Carrega o endereço de cada cliente
- Carrega apenas os 5 pedidos mais recentes que estão completados
- Retorna apenas os campos especificados dos pedidos
- Pagina 20 clientes por vez
Controller:
public function index() { return Customer::requestPaginate(); }
Exemplo 6: Blog posts com comments aprovados e tags ordenadas
Cenário: Posts publicados com comentários aprovados e tags ordenadas alfabeticamente.
JSON:
{
"search": {
"status": "EQ:published",
"published_at": "LE:2026-04-14"
},
"relations": [
"author",
"category",
{
"comments": {
"search": {
"status": "EQ:approved",
"parent_id": "EQ:null"
},
"order_by": {
"created_at": "desc"
}
}
},
{
"tags": {
"order_by": {
"name": "asc"
}
}
}
],
"order_by": {
"published_at": "desc"
},
"_per_page": 15
}
Resultado:
- Posts publicados até hoje
- Autor e categoria carregados normalmente
- Apenas comentários aprovados de primeiro nível (sem parent), mais recentes primeiro
- Tags ordenadas alfabeticamente
- Posts ordenados por data de publicação (mais recentes primeiro)
Exemplo 7: Exportação com todos os campos exceto sensíveis
Cenário: Exportar todos os dados de usuários exceto campos sensíveis.
JSON:
{
"excepts": ["password", "remember_token", "api_token", "two_factor_secret"],
"limit": 10000
}
Exemplo 8: Dashboard com múltiplos filtros
Cenário: Dashboard de produtos com múltiplos filtros aplicados.
JSON:
{
"search": {
"category_id": "EQ:5",
"stock": "GT:0",
"price": "LT:5000",
"name": "LIKE:Notebook",
"is_featured": "EQ:1"
},
"relations": ["category", "manufacturer", "reviews"],
"order_by": {
"featured_order": "asc",
"price": "asc"
},
"returns": [
"id",
"name",
"price",
"stock",
"category_id",
"manufacturer_id"
],
"_per_page": 12,
"page": 1
}
Customizações avançadas
Criar um operador customizado
Se os operadores padrão não atendem sua necessidade, você pode criar operadores personalizados.
Passo 1: Criar a classe do operador
Crie um arquivo em app/SearchCallbacks/CustomOperator.php:
<?php namespace App\SearchCallbacks; use PowerVending\LaravelApiQueryBuilder\SearchCallbacks\AbstractCallback; class CustomOperator extends AbstractCallback { // Define o prefixo do operador public const OPERATOR = 'CUSTOM:'; /** * Executa a lógica do operador * * @param \Illuminate\Database\Eloquent\Builder $builder * @param string $column * @param string $value * @return void */ public function __invoke($builder, string $column, string $value) { // Sua lógica customizada aqui // Exemplo: busca case-sensitive $builder->whereRaw("BINARY {$column} = ?", [$value]); } }
Passo 2: Registrar o operador na configuração
Edite config/api-query-builder.php:
use App\SearchCallbacks\CustomOperator; return [ 'operators' => [ CustomOperator::class, // Adicione no topo (lembre da ordem!) // ... outros operadores ], ];
Passo 3: Usar o operador
{
"search": {
"name": "CUSTOM:SuaEmpresa"
}
}
Exemplos de operadores customizados úteis
Operador IN (valores múltiplos)
class InOperator extends AbstractCallback { public const OPERATOR = 'IN:'; public function __invoke($builder, string $column, string $value) { // Espera valores separados por vírgula: "IN:1,2,3,4" $values = explode(',', $value); $builder->whereIn($column, $values); } }
Uso:
{"category_id": "IN:1,5,8,12"}
Operador NULL
class IsNullOperator extends AbstractCallback { public const OPERATOR = 'NULL:'; public function __invoke($builder, string $column, string $value) { if ($value === 'true' || $value === '1') { $builder->whereNull($column); } else { $builder->whereNotNull($column); } } }
Uso:
{"deleted_at": "NULL:true"} // Registros com deleted_at NULL
{"deleted_at": "NULL:false"} // Registros com deleted_at NOT NULL
Modificar query antes de executar
Você pode adicionar condições adicionais à query antes dela ser executada:
public function index(Request $request) { // Adiciona condições adicionais $query = Product::query() ->where('company_id', auth()->user()->company_id) ->where('is_deleted', false); // Processa os parâmetros JSON na query existente return $query->requestPaginate(); }
Validar parâmetros antes de processar
public function index(Request $request) { // Valida os parâmetros $request->validate([ 'search' => 'sometimes|json', 'order_by' => 'sometimes|json', '_per_page' => 'sometimes|integer|min:1|max:100', ]); return Product::requestPaginate(); }
Erros retornados pela API
Quando uma requisição contém parâmetros inválidos, o pacote lança exceções que podem ser mapeadas para respostas HTTP 400. Esta seção descreve cada mensagem de erro, sua causa e como corrigi-la.
Relation '<RELATION>' does not exist on model '<MODEL_CLASS>'.
Exceção: InvalidRelationException
Causa: Foi informado um relacionamento que não existe no model (ou em algum nível de relacionamento aninhado).
Situações em que pode ocorrer:
- No parâmetro
relations. - No parâmetro
searchquando a chave representa relacionamento (ex.: sub-search em objeto ou notação com ponto comorelation.column). - No parâmetro
doesnt_have_relations.
Exemplos que causam o erro:
{
"relations": ["unknown_relation"]
}
{
"search": {
"unknown_relation": {
"search": {
"id": "EQ:1"
}
}
}
}
{
"search": {
"unknown_relation.description": "EQ:abc"
}
}
{
"doesnt_have_relations": ["unknown_relation"]
}
Solução:
- Verifique o nome da relação no model Eloquent e use o nome correto.
- Em relações aninhadas, valide cada segmento do caminho (ex.:
orders.items.product). - Se necessário, padronize o nome enviado pelo frontend para o método real da relação no backend.
Os erros abaixo são produzidos pela classe InvalidOperatorUsageException e a mensagem é segura para ser exibida diretamente ao consumidor da API.
The '<OP>' operator is not supported for text-type fields. Only comparable field types (numeric, date, etc.) are allowed.
Operadores afetados: GT, GE, LT, LE
Causa: O operador de comparação foi usado em uma coluna cujo tipo resolvido pelo pacote é textual (string, varchar, char, text, longtext, etc.). Esses operadores exigem tipos comparáveis (numérico, data, etc.).
Exemplo que causa o erro:
{"name": "GE:Notebook"}
Soluções possíveis:
- Use
EQ:ouLIKE:para colunas de texto. - Se a coluna armazena datas ou números como string (ex:
VARCHARcontendo"2024-01-15"), aplique um cast personalizado no model para que o pacote não trate o campo como texto puro:
// App/Models/Product.php protected $casts = [ 'expires_at' => \App\Casts\Iso8601DateTimeString::class, ];
Isso faz com que o tipo resolvido pelo pacote deixe de ser string e os operadores GT/GE/LT/LE passem a funcionar.
The '<OP>' operator expects exactly one value, but multiple were provided.
Operadores afetados: GT, GE, LT, LE
Causa: Foi passado mais de um valor separado por ; para um operador que aceita apenas um valor.
Exemplo que causa o erro:
{"price": "GE:100;200"}
Solução: Passe apenas um valor:
{"price": "GE:100"}
Se o objetivo é um intervalo, use BT: (between):
{"price": "BT:100;200"}
No value was provided for the '<OP>' operator.
Operadores afetados: GT, GE, LT, LE
Causa: O operador foi enviado sem nenhum valor após os dois pontos.
Exemplo que causa o erro:
{"price": "GE:"}
Solução: Sempre forneça um valor após o operador:
{"price": "GE:100"}
The '<OP>' operator expects exactly 2 values, but N were provided.
Operadores afetados: BT, NB
Causa: O operador BT: (between) ou NB: (not between) exige exatamente dois valores separados por ;, mas a quantidade recebida foi diferente.
Exemplos que causam o erro:
{"price": "BT:100"} // apenas 1 valor
{"price": "BT:100;200;300"} // 3 valores
Solução: Forneça exatamente dois valores:
{"price": "BT:100;500"}
Solução de problemas
Erro: "Column not found"
Sintoma: Erro SQL dizendo que uma coluna não existe.
Causas possíveis:
- Nome da coluna digitado errado no JSON
- Coluna realmente não existe na tabela
- Tentando acessar coluna de relacionamento sem JOIN
Solução:
Verifique se a coluna existe:
php artisan tinker >>> Schema::hasColumn('products', 'nome_da_coluna')
Se for coluna de relacionamento, use notação de ponto:
{"order_by": {"category.name": "asc"}}
Erro: "Ambiguous column"
Sintoma: Column 'id' in SELECT is ambiguous
Causa: Você está usando order_by com relacionamento mas não especificou returns, ou especificou colunas sem qualificação.
Solução:
Sempre especifique returns quando usar ordenação por relacionamento:
{
"returns": ["id", "name"],
"order_by": {"category.name": "asc"}
}
O pacote irá qualificar automaticamente as colunas.
Erro: "Forbidden column"
Sintoma: Tentativa de acessar uma coluna proibida.
Causa: A coluna está na lista de colunas proibidas (via model ou config).
Solução:
- Verifique a propriedade
$forbiddenColumnsna model - Verifique
config/api-query-builder.php→global_forbidden_columns - Verifique
config/api-query-builder.php→model_options[YourModel::class]['forbidden_columns'] - Remova a coluna da lista ou use outra coluna para buscar
Isso é uma proteção de segurança. Se você realmente precisa acessar essa coluna:
- Remova da lista de colunas proibidas (não recomendado para campos sensíveis)
- Crie um endpoint separado específico para esse caso
Operador não funciona
Sintoma: Operador sendo interpretado como valor literal.
Causas possíveis:
- Operador não está registrado na configuração
- Ordem dos operadores está incorreta
- Sintaxe incorreta
Solução:
Verifique a ordem no config/api-query-builder.php:
'operators' => [ // MAIORES PRIMEIRO \PowerVending\LaravelApiQueryBuilder\SearchCallbacks\StartsWith::class, \PowerVending\LaravelApiQueryBuilder\SearchCallbacks\EndsWith::class, // ... depois os menores ],
Limpe o cache de configuração:
php artisan config:clear php artisan config:cache
Performance lenta
Sintoma: Queries demorando muito tempo.
Causas possíveis:
- Falta de índices no banco de dados
- Muitos relacionamentos sendo carregados
- Paginação com número muito alto de registros
Soluções:
Adicione índices nas colunas mais filtradas:
// Migration Schema::table('products', function (Blueprint $table) { $table->index('status'); $table->index('category_id'); $table->index('price'); $table->index(['status', 'category_id']); // índice composto });
Use limit ou _per_page para limitar resultados:
{"_per_page": 25}
Evite carregar relacionamentos desnecessários:
{
"relations": ["category"], // apenas o necessário
// evite: "relations": ["category", "reviews", "images", "manufacturer"]
}
JSON inválido na query string
Sintoma: Erro de parsing JSON.
Causa: JSON mal formatado na URL.
Solução:
Sempre encode o JSON na URL:
// JavaScript const params = { search: JSON.stringify({ name: "LIKE:Produto", status: "EQ:active" }) }; const url = '/api/products?' + new URLSearchParams(params);
Ou use ferramentas que fazem isso automaticamente (Axios, etc).
Relacionamento não encontrado
Sintoma: Erro dizendo que o relacionamento não existe.
Causa: Nome do relacionamento incorreto ou método não existe no model.
Solução:
Verifique se o método existe no model:
class Product extends Model { // O método deve existir e ser público public function category() { return $this->belongsTo(Category::class); } }
Use o nome exato do método no JSON:
{
"order_by": {"category.name": "asc"} // 'category' = nome do método
}
Testes
O pacote inclui testes automatizados usando PHPUnit.
Executar todos os testes
composer test
Ou com PHPUnit diretamente:
vendor/bin/phpunit
Executar testes específicos
vendor/bin/phpunit --filter NomeDoTeste
Estrutura de testes
Os testes estão localizados em packages/api-query-builder/tests/:
tests/
├── Feature/ # Testes de integração
│ ├── SearchTest.php
│ ├── OrderByTest.php
│ └── RelationTest.php
├── Unit/ # Testes unitários
│ ├── OperatorTest.php
│ └── ConfigTest.php
└── TestCase.php # Classe base de testes
Criar novos testes
Para adicionar novos testes, crie uma classe que estenda TestCase:
<?php namespace Tests\Feature; use Tests\TestCase; use App\Models\Product; class CustomFeatureTest extends TestCase { public function test_custom_operator_works() { // Arrange Product::factory()->create(['name' => 'Test Product']); // Act $result = Product::jsonSearch(['name' => 'CUSTOM:Test']); // Assert $this->assertCount(1, $result); } }