diff --git a/CHANGELOG.md b/CHANGELOG.md index fbe9532f..1dce2696 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,5 +1,10 @@ # CHANGELOG.md +## v0.39.1 (unreleased) + - More precise server timing tracking to debug performance issues + - Fix missing server timing header in some cases + - Implement nice error messages for some header-related errors such as invalid header values. + ## v0.39.0 (2025-10-28) - Ability to execute sql for URL paths with another extension. If you create sitemap.xml.sql, it will be executed for example.com/sitemap.xml - Display source line info in errors even when the database does not return a precise error position. In this case, the entire problematic SQL statement is referenced. diff --git a/examples/official-site/extensions-to-sql.md b/examples/official-site/extensions-to-sql.md index 81ee0902..88fa4067 100644 --- a/examples/official-site/extensions-to-sql.md +++ b/examples/official-site/extensions-to-sql.md @@ -1,257 +1,262 @@ -# Extensions to SQL +## How SQLPage runs your SQL -SQLPage makes some special treatment before executing your SQL queries. +SQLPage reads your SQL file and runs one statement at a time. For each statement, it -When executing your SQL file, SQLPage executes each query one at a time. -It doesn't send the whole file as-is to the database engine. +- decides whether to: + - handle it inside SQLPage, or + - prepare it as a (potentially slightly modified) sql statement on the database. +- extracts values from the request to pass them as prepared statements parameters +- runs [`sqlpage.*` functions](/functions) +- passes the database results to components -## Performance +This page explains every step of the process, +with examples and details about differences between how SQLPage understands SQL and how your database does. -See the [performance page](/performance.sql) for details on the optimizations -made to run your queries as fast as possible. +## What runs where -## Variables +### Handled locally by SQLPage -SQL doesn't have its own mechanism for variables. -SQLPage implements variables in the following way: +- Static simple selects (a tiny, fast subset of SELECT) +- Simple variable assignments that use only literals or variables + - All sqlpage functions + -### POST parameters +### Sent to your database -When sending a POST request, most often by sending a form with the -[form component](/component.sql?component=form), the form data is made -available as variables prefixed by a colon. +Everything else: joins, subqueries, arithmetic, database functions, `SELECT @@VERSION`, `CURRENT_TIMESTAMP`, `SELECT *`, expressions, `FROM`, `WHERE`, `GROUP BY`, `ORDER BY`, `LIMIT`/`FETCH`, `WITH`, `DISTINCT`, etc. -So when this form is sent: +### Mixed statements using `sqlpage.*` functions -`form.sql` -```sql -SELECT - 'form' AS component, - 'POST' AS method, -- form defaults to using the HTTP POST method - 'result.sql' AS action; +[`sqlpage.*` functions](/functions.sql) are executed by SQLPage; your database never sees them. They can run: -SELECT - 'age' AS name, - 'How old are you?' AS label, - 'number' AS type; -``` +- Before the query, when used as values inside conditions or parameters. +- After the query, when used as top-level selected columns (applied per row). + +Examples are shown below. + +## Static simple selects + +A *static simple select* is a very restricted `SELECT` that SQLPage can execute entirely by itself. This avoids back and forths between SQLPage and the database for trivial queries. + +To be static and simple, a statement must satisfy all of the following: + +- No `FROM`, `WHERE`, `GROUP BY`, `HAVING`, `ORDER BY`, `LIMIT`/`FETCH`, `WITH`, `DISTINCT`, `TOP`, windowing, locks, or other clauses. +- Each selected item is of the form `value AS alias`. +- Each `value` is either: + - a literal (single-quoted string, number, boolean, or `NULL`), or + - a variable (like `$name`, `:message`) + +That’s it. If any part is more complex, it is not a static simple select and will be sent to the database. -It will make a request to this page: +#### Examples that ARE static (executed by SQLPage) -`result.sql` ```sql -SELECT - 'text' AS component, - 'You are ' || :age || ' years old!' AS contents; +SELECT 'text' AS component, 'Hello' AS contents; +SELECT 'text' AS component, $name AS contents; ``` -`:age` will be substituted by the actual value of the POST parameter. +#### Examples that are NOT static (sent to the database) -### URL parameters +```sql +-- Has string concatenation +select 'from' as component, 'handle_form.sql?id=' || $id as action; + +-- Has WHERE +select 'text' as component, $alert_message as contents where $should_alert; + +-- Uses database functions or expressions +SELECT 1 + 1 AS two; +SELECT CURRENT_TIMESTAMP AS now; +SELECT @@VERSION AS version; -- SQL Server variables +-- Uses a subquery +SELECT (select 1) AS one; +``` -Likewise, URL parameters are available as variables prefixed by a dollar sign. +## Variables + +SQLPage communicates information about incoming HTTP requests to your SQL code through prepared statement variables. +You can use + - `$var` to reference a GET variable (an URL parameter), + - `:var` to reference a POST variable (a value filled by an user in a form field), + - `set var = ...` to set the value of `$var`. -> URL parameters are often called GET parameters because they can originate -> from a form with 'GET' as the method. +### POST parameters -So the previous example can be reworked to handle URL parameters: +Form fields sent with POST are available as `:name`. -`result.sql` ```sql SELECT - 'text' AS component, - 'You are ' || $age || ' years old!' AS contents; + 'form' AS component, + 'POST' AS method, + 'result.sql' AS action; + +SELECT 'age' AS name, 'How old are you?' AS label, 'number' AS type; ``` -By querying this page with this URL: `/request.sql?age=42` -we would get `You are 42 years old!` as a response. +```sql +-- result.sql +SELECT 'text' AS component, 'You are ' || :age || ' years old!' AS contents; +``` -### The `SET` command +### URL parameters -SQLPage overrides the behavior of `SET` statements in SQL to store variables in SQLPage itself instead of running the statement on the database. +Query-string parameters are available as `$name`. ```sql -SET coalesced_post_id = COALESCE($post_id, 0); +SELECT 'text' AS component, 'You are ' || $age || ' years old!' AS contents; +-- /result.sql?age=42 → You are 42 years old! ``` -`SET` statements are transformed into `SELECT` queries, and their result is stored in a `$`-variable: +When a URL parameter is not set, its value is `NULL`. -```sql -SELECT COALESCE($post_id, 0); -``` +### The SET command -We can override a previous `$`-variable: +`SET` stores a value in SQLPage (not in the database). Only strings and `NULL` are stored. ```sql +-- Give a default value to a variable SET post_id = COALESCE($post_id, 0); ``` -### Limitations +- If the right-hand side is purely literals/variables, SQLPage computes it directly. See the section about *static simple select* above. +- If it needs the database (for example, calls a database function), SQLPage runs an internal `SELECT` to compute it and stores the first column of the first row of results. -`$`-variables and `:`-variables are stored by SQLPage, not in the database. +Only a single textual value (**string or `NULL`**) is stored. +`set id = 1` will store the string `'1'`, not the number `1`. -They can only store a string, or null. +On databases with a strict type system, such as PostgreSQL, if you need a number, you will need to cast your variables: `select * from post where id = $id::int`. -As such, they're not designed to store table-valued results. -They will only store the first value of the first column: +Complex structures can be stored as json strings. -```sql -CREATE TABLE t(a, b); -INSERT INTO t(a, b) VALUES (1, 2), (3, 4); +For larger temporary results, prefer temporary tables on your database; do not send them to SQLPage at all. -SET var = (SELECT * FROM t); +## `sqlpage.*` functions --- now $var contains '1' -``` +Functions under the `sqlpage.` prefix run in SQLPage. See the [functions page](/functions.sql). -Temporary table-valued results can be stored in two ways. +They can run: -## Storing large datasets in the database with temporary tables +### Before sending the query (as input values) + +Used inside conditions or parameters, the function is evaluated first and its result is passed to the database. -This is the most efficient method to store large values. ```sql --- Database connections are reused and temporary tables are stored at the --- connection level, so we make sure the table doesn't exist already -DROP TABLE IF EXISTS my_temp_table; -CREATE TEMPORARY TABLE my_temp_table AS -SELECT a, b -FROM my_stored_table ... - --- Insert data from direct values -INSERT INTO my_temp_table(a, b) -VALUES (1, 2), (3, 4); +SELECT * +FROM blog +WHERE slug = sqlpage.path(); ``` -## Storing rich structured data in memory using JSON - -This can be more convenient, but should only be used for small values, because data -is copied from the database into SQLPage memory, and to the database again at each use. +### After receiving results (as top-level selected columns) -You can use the [JSON functions from your database](/blog.sql?post=JSON+in+SQL%3A+A+Comprehensive+Guide). +Used as top-level selected columns, the query is rewritten to first fetch the raw column, and the function is applied per row in SQLPage. -Here are some examples with SQLite: ```sql --- CREATE TABLE my_table(a, b); --- INSERT INTO my_table(a, b) --- VALUES (1, 2), (3, 4); - -SET my_json = ( - SELECT json_group_array(a) - FROM my_table -); --- [1, 3] - -SET my_json = json_array(1, 2, 3); --- [1, 2, 3] +SELECT sqlpage.read_file_as_text(file_path) AS contents +FROM blog_posts; ``` -## Functions - -Functions starting with `sqlpage.` are executed by SQLPage, not by your database engine. -See the [functions page](/functions.sql) for more details. +## Performance -They're either executed before or after the query is run in the database. +See the [performance page](/performance.sql) for details. In short: -### Executing functions *before* sending a query to the database +- Statements sent to the database are prepared and cached. +- Variables and pre-computed values are bound as parameters. +- This keeps queries fast and repeatable. -When they don't process results coming from the database: +## Working with larger temporary results -```sql -SELECT * FROM blog WHERE slug = sqlpage.path() -``` +### Temporary tables in your database -`sqlpage.path()` will get replaced by the result of the function. +When you reuse the same values multiple times in your page, +store them in a temporary table. -### Executing functions *after* receiving results from the database +```sql +DROP TABLE IF EXISTS filtered_posts; +CREATE TEMPORARY TABLE filtered_posts AS +SELECT * FROM posts where category = $category; -When they process results coming from the database: +select 'alert' as component, count(*) || 'results' as title +from filtered_posts; -```sql -SELECT sqlpage.read_file_as_text(blog_post_file) AS title -FROM blog; +select 'list' as component; +select name from filtered_posts; ``` -The query executed will be: +### Small JSON values in variables + +Useful for small datasets that you want to keep in memory. +See the [guide on JSON in SQL](/blog.sql?post=JSON+in+SQL%3A+A+Comprehensive+Guide). ```sql -SELECT blog_post_file AS title FROM blog; +set product = ( + select json_object('name', name, 'price', price) + from products where id = $product_id +); ``` -Then `sqlpage.read_file_as_text()` will be called on each row. - -## Implementation details of variables and functions +## CSV imports -All queries run by SQLPage in the database are first prepared, then executed. +When you write a compatible `COPY ... FROM 'field'` statement and upload a file with the matching form field name, SQLPage orchestrates the import: -Statements are prepared and cached the first time they're encountered by SQLPage. -Then those cached prepared statements are executed at each run, with parameter substitution. +- PostgreSQL: the file is streamed directly to the database using `COPY FROM STDIN`; the database performs the import. +- Other databases: SQLPage reads the CSV and inserts rows using a prepared `INSERT` statement. Options like delimiter, quote, header, escape, and a custom `NULL` string are supported. With a header row, column names are matched by name; otherwise, the order is used. -All variables and function results are cast as text, to let the -database query optimizer know only strings (or nulls) will be passed. - -Examples: +Example: ```sql --- Source query -SELECT * FROM blog WHERE slug = sqlpage.path(); - --- Prepared statement (SQLite syntax) -SELECT * FROM blog WHERE slug = CAST(?1 AS TEXT) +COPY my_table (col1, col2) +FROM 'my_csv' +(DELIMITER ';', HEADER); ``` -```sql --- Source query -SET post_id = COALESCE($post_id, 0); - --- Prepared statement (SQLite syntax) -SELECT COALESCE(CAST(?1 AS TEXT), 0) -``` +The uploaded file should be provided in a form field with `'file' as type, 'my_csv' as name`. -# Data types +## Data types -Each database has its own rich set of data types. -The data modal in SQLPage itself is simpler, mainly composed of text strings and json objects. +Each database has its own usually large set of data types. +SQLPage itself has a much more rudimentary type system. ### From the user to SQLPage -Form fields and URL parameters may contain arrays. These are converted to JSON strings before processing. +Form fields and URL parameters in HTTP are fundamentally untyped. +They are just sequences of bytes. SQLPage requires them to be valid utf8 strings. -For instance, Loading `users.sql?user[]=Tim&user[]=Tom` will result in a single variable `$user` with the textual value `["Tim", "Tom"]`. +SQLPage follows the convention that when a parameter name ends with `[]`, it represents an array. +Arrays in SQLPage are represented as JSON strings. + +Example: In `users.sql?user[]=Tim&user[]=Tom`, `$user` becomes `'["Tim", "Tom"]'` (a JSON string exploitable with your database's builtin json functions). ### From SQLPage to the database -SQLPage sends only text strings (`VARCHAR`) and `NULL`s to the database, since these are the only possible variable and function return values. +SQLPage sends only strings (`TEXT` or `VARCHAR`) and `NULL`s as parameters. ### From the database to SQLPage -Each row of data returned by a SQL query is converted to a JSON object before being passed to components. +Each row returned by the database becomes a JSON object +before its passed to components: -- Each column becomes a key in the json object. If a row has two columns of the same name, they become an array in the json object. -- Each value is converted to the closest JSON value - - all number types map to json numbers, booleans to booleans, and `NULL` to `null`, - - all text types map to json strings - - date and time types map to json strings containing ISO datetime values - - binary values (BLOBs) map to json strings containing [data URLs](https://developer.mozilla.org/en-US/docs/Web/URI/Reference/Schemes/data) +- Each column is a key. Duplicate column names turn into arrays. +- Numbers, booleans, text, and `NULL` map naturally. +- Dates/times become ISO strings. +- Binary data (BLOBs) becomes a data URL (with mime type auto-detection). #### Example -The following PostgreSQL query: - ```sql -select - 1 as one, - 'x' as my_array, 'y' as my_array, - now() as today, - ''::bytea as my_image; +SELECT + 1 AS one, + 'x' AS my_array, 'y' AS my_array, + now() AS today, + ''::bytea AS my_image; ``` -will result in the following JSON object being passed to components for rendering +Produces something like: ```json { - "one" : 1, - "my_array" : ["x","y"], - "today":"2025-08-30T06:40:13.894918+00:00", - "my_image":"" + "one": 1, + "my_array": ["x", "y"], + "today": "2025-08-30T06:40:13.894918+00:00", + "my_image": "" } ``` \ No newline at end of file diff --git a/src/render.rs b/src/render.rs index 95c968f6..337d08f5 100644 --- a/src/render.rs +++ b/src/render.rs @@ -46,8 +46,10 @@ use crate::webserver::http::RequestContext; use crate::webserver::response_writer::{AsyncResponseWriter, ResponseWriter}; use crate::webserver::ErrorWithStatus; use crate::AppState; +use actix_web::body::MessageBody; use actix_web::cookie::time::format_description::well_known::Rfc3339; use actix_web::cookie::time::OffsetDateTime; +use actix_web::http::header::TryIntoHeaderPair; use actix_web::http::{header, StatusCode}; use actix_web::{HttpResponse, HttpResponseBuilder}; use anyhow::{bail, format_err, Context as AnyhowContext}; @@ -116,7 +118,7 @@ impl HeaderContext { Some(HeaderComponent::HttpHeader) => { self.add_http_header(&data).map(PageContext::Header) } - Some(HeaderComponent::Redirect) => self.redirect(&data).map(PageContext::Close), + Some(HeaderComponent::Redirect) => self.redirect(&data), Some(HeaderComponent::Json) => self.json(&data), Some(HeaderComponent::Csv) => self.csv(&data).await, Some(HeaderComponent::Cookie) => self.add_cookie(&data).map(PageContext::Header), @@ -167,7 +169,9 @@ impl HeaderContext { self.response.status(StatusCode::FOUND); self.has_status = true; } - self.response.insert_header((name.as_str(), value_str)); + let header = TryIntoHeaderPair::try_into_pair((name.as_str(), value_str)) + .map_err(|e| anyhow::anyhow!("Invalid header: {name}:{value_str}: {e:#?}"))?; + self.response.insert_header(header); } Ok(self) } @@ -237,13 +241,13 @@ impl HeaderContext { Ok(self) } - fn redirect(mut self, data: &JsonValue) -> anyhow::Result { + fn redirect(mut self, data: &JsonValue) -> anyhow::Result { self.response.status(StatusCode::FOUND); self.has_status = true; let link = get_object_str(data, "link") .with_context(|| "The redirect component requires a 'link' property")?; self.response.insert_header((header::LOCATION, link)); - Ok(self.into_response_builder().body(())) + self.close_with_body(()) } /// Answers to the HTTP request with a single json object @@ -256,9 +260,7 @@ impl HeaderContext { } else { serde_json::to_vec(contents)? }; - Ok(PageContext::Close( - self.into_response_builder().body(json_response), - )) + self.close_with_body(json_response) } else { let body_type = get_object_str(data, "type"); let json_renderer = match body_type { @@ -320,10 +322,11 @@ impl HeaderContext { .status(StatusCode::FOUND) .insert_header((header::LOCATION, link)); self.has_status = true; - Ok(PageContext::Close(self.into_response_builder().body( + let response = self.into_response( "Sorry, but you are not authorized to access this page. \ Redirecting to the login page...", - ))) + )?; + Ok(PageContext::Close(response)) } else { anyhow::bail!(ErrorWithStatus { status: StatusCode::UNAUTHORIZED @@ -358,9 +361,7 @@ impl HeaderContext { self.response .insert_header((header::CONTENT_TYPE, content_type)); } - Ok(PageContext::Close( - self.into_response_builder().body(body_bytes.into_owned()), - )) + self.close_with_body(body_bytes.into_owned()) } fn log(self, data: &JsonValue) -> anyhow::Result { @@ -374,9 +375,18 @@ impl HeaderContext { } } - fn into_response_builder(mut self) -> HttpResponseBuilder { + fn into_response(mut self, body: B) -> anyhow::Result { self.add_server_timing_header(); - self.response + match self.response.message_body(body) { + Ok(response) => Ok(response.map_into_boxed_body()), + Err(e) => Err(anyhow::anyhow!( + "An error occured while generating the request headers: {e:#}" + )), + } + } + + fn close_with_body(self, body: B) -> anyhow::Result { + Ok(PageContext::Close(self.into_response(body)?)) } async fn start_body(mut self, data: JsonValue) -> anyhow::Result { @@ -394,7 +404,7 @@ impl HeaderContext { } pub fn close(self) -> HttpResponse { - self.into_response_builder().finish() + self.into_response(()).unwrap() } }