matatirosoln / sql-to-odata
Convert SQL queries to OData query syntax.
Requires
- php: ^8.4
- phpmyadmin/sql-parser: ^5.0
Requires (Dev)
- phpunit/phpunit: ^11.0
README
A PHP library for converting SQL queries to OData (Open Data Protocol) query syntax. Targets OData 4.01 at the intermediate conformance level, primarily designed for use with the Claris FileMaker OData API (FileMaker Server 2025 / v22+).
We would be very interested in hearing from potential users who have access to alternative OData implementations and may be interested in working to extend this library to ensure that it is broadly useful as possible.
Requirements
- PHP ^8.4
- Composer
Installation
composer require matatirosoln/sql-to-odata
Usage
Call parse() on any supported SQL statement. It returns a typed query object you can inspect to build your OData request.
use Matatirosoln\SqlToOdata\SqlToOdata; $converter = new SqlToOdata(); $query = $converter->parse($sql);
SELECT
Returns a SelectQuery with entitySet and queryString properties.
use Matatirosoln\SqlToOdata\Query\SelectQuery; $query = $converter->parse("SELECT Id, Name FROM Users WHERE Status = 'Active' ORDER BY Name ASC LIMIT 10"); // $query->entitySet => 'Users' // $query->queryString => '?$select=Id,Name&$filter=Status eq \'Active\'&$orderby=Name asc&$top=10' assert($query instanceof SelectQuery);
INSERT
Returns an InsertQuery with entitySet and body properties. body is an associative array of column-value pairs with PHP-native types, suitable for JSON-encoding into a POST request body.
use Matatirosoln\SqlToOdata\Query\InsertQuery; $query = $converter->parse("INSERT INTO Users (Name, Age, Active) VALUES ('John', 30, true)"); // $query->entitySet => 'Users' // $query->body => ['Name' => 'John', 'Age' => 30, 'Active' => true] assert($query instanceof InsertQuery);
UPDATE
Returns an UpdateQuery with entitySet, body, and filter properties. filter is an OData filter expression derived from the WHERE clause. UPDATE without a WHERE clause throws a ConversionException.
use Matatirosoln\SqlToOdata\Query\UpdateQuery; $query = $converter->parse("UPDATE Users SET Name = 'Jane', Age = 31 WHERE Id = 1"); // $query->entitySet => 'Users' // $query->body => ['Name' => 'Jane', 'Age' => 31] // $query->filter => 'Id eq 1' assert($query instanceof UpdateQuery);
DELETE
Returns a DeleteQuery with entitySet and filter properties. DELETE without a WHERE clause throws a ConversionException.
use Matatirosoln\SqlToOdata\Query\DeleteQuery; $query = $converter->parse('DELETE FROM Users WHERE Id = 1'); // $query->entitySet => 'Users' // $query->filter => 'Id eq 1' assert($query instanceof DeleteQuery);
Dispatching on query type
Use match or instanceof checks to handle each statement type:
use Matatirosoln\SqlToOdata\Query\DeleteQuery; use Matatirosoln\SqlToOdata\Query\InsertQuery; use Matatirosoln\SqlToOdata\Query\SelectQuery; use Matatirosoln\SqlToOdata\Query\UpdateQuery; $query = $converter->parse($sql); match (true) { $query instanceof SelectQuery => handleSelect($query), $query instanceof InsertQuery => handleInsert($query), $query instanceof UpdateQuery => handleUpdate($query), $query instanceof DeleteQuery => handleDelete($query), };
Supported SQL features
Targets OData 4.01 intermediate conformance level (spec).
SELECT clauses
| SQL | OData |
|---|---|
SELECT col1, col2 |
$select=col1,col2 |
SELECT * |
(omitted — returns all fields) |
JOIN / LEFT JOIN |
$expand |
WHERE |
$filter |
ORDER BY |
$orderby |
LIMIT n |
$top=n |
LIMIT n OFFSET m |
$top=n&$skip=m |
JOIN → $expand
SQL JOINs are translated to OData $expand, which follows navigation properties defined in the OData service metadata. Columns selected from a joined table are promoted into a nested $select within the expand:
$query = $converter->parse( 'SELECT Users.Id, Users.Name, Orders.OrderDate FROM Users JOIN Orders ON Users.Id = Orders.UserId' ); // $query->entitySet => 'Users' // $query->queryString => '?$select=Id,Name&$expand=Orders($select=OrderDate)'
Multiple JOINs produce a comma-separated $expand:
$query = $converter->parse( 'SELECT * FROM Users JOIN Orders ON Users.Id = Orders.UserId JOIN Addresses ON Users.Id = Addresses.UserId' ); // $query->queryString => '?$expand=Orders,Addresses'
Note: Unlike SQL, OData
$expandis constrained to navigation properties defined in the service metadata — arbitrary cross-entity joins are not supported by the OData protocol itself.
WHERE operators
| SQL | OData |
|---|---|
= |
eq |
!= / <> |
ne |
> |
gt |
>= |
ge |
< |
lt |
<= |
le |
AND / OR |
and / or |
IS NULL |
eq null |
IS NOT NULL |
ne null |
LIKE '%val%' |
contains(col, 'val') |
LIKE 'val%' |
startswith(col, 'val') |
LIKE '%val' |
endswith(col, 'val') |
IN (a, b, c) |
(col eq a or col eq b or col eq c) |
Exceptions
All errors throw Matatirosoln\SqlToOdata\Exception\ConversionException. Common cases:
- Invalid or unparseable SQL
- UPDATE or DELETE without a WHERE clause
- Subqueries (not supported)
- Unsupported statement types (e.g.
CREATE,DROP)
Running tests
composer install ./vendor/bin/phpunit
License
MIT — see LICENSE.
Contact
Steve Winter — Matatiro Solutions Ltd — steve@msdev.nz