Limitations and best practices
R2 SQL is designed for querying partitioned Apache Iceberg tables in your R2 data catalog. This document outlines the supported features, limitations, and best practices of R2 SQL.
| Feature | Supported | Notes | 
|---|---|---|
| Basic SELECT | Yes | Columns, * | 
| Aggregation functions | No | No COUNT, AVG, etc. | 
| Single table FROM | Yes | Note, aliasing not supported | 
| WHERE clause | Yes | Filters, comparisons, equality, etc | 
| JOINs | No | No table joins | 
| Array filtering | No | No array type support | 
| JSON filtering | No | No nested object queries | 
| Simple LIMIT | Yes | 1-10,000 range, no pagination support | 
| ORDER BY | Yes | Any columns of the partition key only | 
| GROUP BY | No | Not supported | 
R2 SQL supports a limited set of SQL clauses: SELECT, FROM, WHERE, ORDER BY, and LIMIT. All other SQL clauses are not supported at the moment. New features will be released in the future, keep an eye on this page for the latest.
- Individual columns: SELECT column1, column2
- All columns: SELECT *
- No JSON field querying: Cannot query individual fields from JSON objects
- No SQL functions: Functions like AVG(),COUNT(),MAX(),MIN(), quantiles are not supported
- No synthetic data: Cannot create synthetic columns like SELECT 1 AS what, "hello" AS greeting
- No field aliasing: SELECT field AS another_name
-- ValidSELECT timestamp, user_id, status FROM my_table;SELECT * FROM my_table;
-- InvalidSELECT user_id AS uid, timestamp AS ts FROM my_table;SELECT COUNT(*) FROM events FROM FROM my_table;SELECT json_field.property FROM my_table;SELECT 1 AS synthetic_column FROM my_table;- Single table queries: SELECT * FROM table_name
- No multiple tables: Cannot specify multiple tables in FROM clause
- No subqueries: SELECT ... FROM (SELECT ...)is not supported
- No JOINs: No INNER, LEFT, RIGHT, or FULL JOINs
- No SQL functions: Cannot use functions like read_parquet()
- No synthetic tables: Cannot create tables from values
- No schema evolution: Schema cannot be altered (no ALTER TABLE, migrations)
- Immutable datasets: No UPDATE or DELETE operations allowed
- Fully defined schema: Dynamic or union-type fields are not supported
- No table aliasing: SELECT * FROM table_name AS alias
--ValidSELECT * FROM http_requests
--InvalidSELECT * FROM table1, table2SELECT * FROM table1 JOIN table2 ON table1.id = table2.idSELECT * FROM (SELECT * FROM events WHERE status = 200)- Simple type filtering: Supports string,boolean,numbertypes, and timestamps expressed as RFC3339
- Boolean logic: Supports AND,OR,NOToperators
- Comparison operators: >,>=,=,<,<=,!=
- Grouped conditions: WHERE col_a="hello" AND (col_b>5 OR col_c != 3)
- Pattern matching: WHERE col_a LIKE ‘hello w%’(prefix matching only)
- NULL Handling : WHERE col_a IS NOT NULL(IS/IS NOT)
- No column-to-column comparisons: Cannot use WHERE col_a = col_b
- No array filtering: Cannot filter on array types (array[number], array[string], array[boolean])
- No JSON/object filtering: Cannot filter on fields inside nested objects or JSON
- No SQL functions: No function calls in WHERE clause
- No arithmetic operators: Cannot use +,-,*,/in conditions
--ValidSELECT * FROM events WHERE timestamp BETWEEN '2024-01-01' AND '2024-01-02'SELECT * FROM logs WHERE status = 200 AND user_type = 'premium'SELECT * FROM requests WHERE (method = 'GET' OR method = 'POST') AND response_time < 1000
--InvalidSELECT * FROM logs WHERE tags[0] = 'error' -- Array filteringSELECT * FROM requests WHERE metadata.user_id = '123' -- JSON field filteringSELECT * FROM events WHERE col_a = col_b -- Column comparisonSELECT * FROM logs WHERE response_time + latency > 5000 -- Arithmetic- ASC: Ascending order
- DESC: Descending order (Default, on full partition key)
- Non-partition keys not supported: ORDER BYon columns other than the partition key is not supported
SELECT * FROM table_name WHERE ... ORDER BY partitionKeySELECT * FROM table_name WHERE ... ORDER BY partitionKey DESC- Simple limits: LIMIT number
- Range: Minimum 1, maximum 10,000
- No pagination: LIMIT offset, countsyntax not supported
- No SQL functions: Cannot use functions to determine limit
- No arithmetic: Cannot use expressions like LIMIT 10 * 50
-- ValidSELECT * FROM events LIMIT 100SELECT * FROM logs WHERE ... LIMIT 10000
-- InvalidSELECT * FROM events LIMIT 100, 50 -- PaginationSELECT * FROM logs LIMIT COUNT(*) / 2 -- FunctionsSELECT * FROM events LIMIT 10 * 10 -- ArithmeticThe following SQL clauses are not supported:
- GROUP BY
- HAVING
- UNION/- INTERSECT/- EXCEPT
- WITH(Common Table Expressions)
- WINDOWfunctions
- INSERT/- UPDATE/- DELETE
- CREATE/- ALTER/- DROP
- Always include time filters in your WHERE clause to ensure efficient queries.
- Use specific column selection instead of SELECT *when possible for better performance.
- Structure your data to avoid nested JSON objects if you need to filter on those fields.
Was this helpful?
- Resources
- API
- New to Cloudflare?
- Directory
- Sponsorships
- Open Source
- Support
- Help Center
- System Status
- Compliance
- GDPR
- Company
- cloudflare.com
- Our team
- Careers
- © 2025 Cloudflare, Inc.
- Privacy Policy
- Terms of Use
- Report Security Issues
- Trademark
-