TypeDAL 3.17.0__tar.gz → 3.17.1__tar.gz
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Potentially problematic release.
This version of TypeDAL might be problematic. Click here for more details.
- {typedal-3.17.0 → typedal-3.17.1}/CHANGELOG.md +6 -0
- {typedal-3.17.0 → typedal-3.17.1}/PKG-INFO +1 -1
- {typedal-3.17.0 → typedal-3.17.1}/src/typedal/__about__.py +1 -1
- {typedal-3.17.0 → typedal-3.17.1}/src/typedal/core.py +3 -3
- {typedal-3.17.0 → typedal-3.17.1}/src/typedal/helpers.py +24 -4
- {typedal-3.17.0 → typedal-3.17.1}/tests/test_helpers.py +5 -1
- {typedal-3.17.0 → typedal-3.17.1}/.github/workflows/su6.yml +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/.gitignore +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/.readthedocs.yml +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/README.md +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/coverage.svg +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/docs/1_getting_started.md +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/docs/2_defining_tables.md +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/docs/3_building_queries.md +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/docs/4_relationships.md +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/docs/5_py4web.md +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/docs/6_migrations.md +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/docs/7_mixins.md +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/docs/css/code_blocks.css +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/docs/index.md +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/docs/requirements.txt +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/example_new.py +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/example_old.py +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/mkdocs.yml +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/pyproject.toml +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/src/typedal/__init__.py +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/src/typedal/caching.py +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/src/typedal/cli.py +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/src/typedal/config.py +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/src/typedal/fields.py +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/src/typedal/for_py4web.py +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/src/typedal/for_web2py.py +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/src/typedal/mixins.py +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/src/typedal/py.typed +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/src/typedal/serializers/as_json.py +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/src/typedal/types.py +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/src/typedal/web2py_py4web_shared.py +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/tests/__init__.py +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/tests/configs/simple.toml +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/tests/configs/valid.env +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/tests/configs/valid.toml +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/tests/test_cli.py +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/tests/test_config.py +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/tests/test_docs_examples.py +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/tests/test_json.py +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/tests/test_main.py +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/tests/test_mixins.py +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/tests/test_mypy.py +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/tests/test_orm.py +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/tests/test_py4web.py +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/tests/test_query_builder.py +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/tests/test_relationships.py +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/tests/test_row.py +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/tests/test_stats.py +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/tests/test_table.py +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/tests/test_web2py.py +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/tests/test_xx_others.py +0 -0
- {typedal-3.17.0 → typedal-3.17.1}/tests/timings.py +0 -0
|
@@ -2,6 +2,12 @@
|
|
|
2
2
|
|
|
3
3
|
<!--next-version-placeholder-->
|
|
4
4
|
|
|
5
|
+
## v3.17.1 (2025-09-20)
|
|
6
|
+
|
|
7
|
+
### Fix
|
|
8
|
+
|
|
9
|
+
* Smarter adapt so Fields and Tables can also be safely inserted in `sql_expression` ([`d5cad6a`](https://github.com/trialandsuccess/TypeDAL/commit/d5cad6a13ddf50ec6e5762075fbeae1e44067da6))
|
|
10
|
+
|
|
5
11
|
## v3.17.0 (2025-09-20)
|
|
6
12
|
|
|
7
13
|
### Feature
|
|
@@ -835,10 +835,10 @@ class TypeDAL(pydal.DAL): # type: ignore
|
|
|
835
835
|
def sql_expression(
|
|
836
836
|
self,
|
|
837
837
|
sql_fragment: str,
|
|
838
|
-
*raw_args:
|
|
838
|
+
*raw_args: Any,
|
|
839
839
|
output_type: str | None = None,
|
|
840
|
-
**raw_kwargs:
|
|
841
|
-
) ->
|
|
840
|
+
**raw_kwargs: Any,
|
|
841
|
+
) -> Expression:
|
|
842
842
|
"""
|
|
843
843
|
Creates a pydal Expression object representing a raw SQL fragment.
|
|
844
844
|
|
|
@@ -337,6 +337,26 @@ class classproperty:
|
|
|
337
337
|
return self.fget(owner)
|
|
338
338
|
|
|
339
339
|
|
|
340
|
+
def smarter_adapt(db: TypeDAL, placeholder: Any) -> str:
|
|
341
|
+
"""
|
|
342
|
+
Smarter adaptation of placeholder to quote if needed.
|
|
343
|
+
|
|
344
|
+
Args:
|
|
345
|
+
db: Database object.
|
|
346
|
+
placeholder: Placeholder object.
|
|
347
|
+
|
|
348
|
+
Returns:
|
|
349
|
+
Quoted placeholder if needed, except for numbers (smart_adapt logic)
|
|
350
|
+
or fields/tables (use already quoted rname).
|
|
351
|
+
"""
|
|
352
|
+
return typing.cast(
|
|
353
|
+
str,
|
|
354
|
+
getattr(placeholder, "sql_shortref", None) # for tables
|
|
355
|
+
or getattr(placeholder, "sqlsafe", None) # for fields
|
|
356
|
+
or db._adapter.smart_adapt(placeholder), # for others
|
|
357
|
+
)
|
|
358
|
+
|
|
359
|
+
|
|
340
360
|
def sql_escape(db: TypeDAL, sql_fragment: str, *raw_args: Any, **raw_kwargs: Any) -> str:
|
|
341
361
|
"""
|
|
342
362
|
Generates escaped SQL fragments with placeholders.
|
|
@@ -358,18 +378,18 @@ def sql_escape(db: TypeDAL, sql_fragment: str, *raw_args: Any, **raw_kwargs: Any
|
|
|
358
378
|
|
|
359
379
|
elif raw_args:
|
|
360
380
|
# list
|
|
361
|
-
return sql_fragment % tuple(db
|
|
381
|
+
return sql_fragment % tuple(smarter_adapt(db, placeholder) for placeholder in raw_args)
|
|
362
382
|
else:
|
|
363
383
|
# dict
|
|
364
|
-
return sql_fragment % {key: db
|
|
384
|
+
return sql_fragment % {key: smarter_adapt(db, placeholder) for key, placeholder in raw_kwargs.items()}
|
|
365
385
|
|
|
366
386
|
|
|
367
387
|
def sql_expression(
|
|
368
388
|
db: TypeDAL,
|
|
369
389
|
sql_fragment: str,
|
|
370
|
-
*raw_args:
|
|
390
|
+
*raw_args: Any,
|
|
371
391
|
output_type: str | None = None,
|
|
372
|
-
**raw_kwargs:
|
|
392
|
+
**raw_kwargs: Any,
|
|
373
393
|
) -> Expression:
|
|
374
394
|
"""
|
|
375
395
|
Creates a pydal Expression object representing a raw SQL fragment.
|
|
@@ -223,7 +223,7 @@ def test_sql_expression():
|
|
|
223
223
|
expr1 = sql_expression(database, "date('now') > %s", "2025-01-01")
|
|
224
224
|
expr2 = database.sql_expression("date('now') > %(value)s", value="2025-01-01")
|
|
225
225
|
|
|
226
|
-
assert expr1 == expr2
|
|
226
|
+
assert str(expr1) == str(expr2)
|
|
227
227
|
assert str(expr1) == "date('now') > '2025-01-01'"
|
|
228
228
|
# past -> should yield result
|
|
229
229
|
result = database(expr1).select(TestSqlExpression.value, expr2)[0]
|
|
@@ -240,3 +240,7 @@ def test_sql_expression():
|
|
|
240
240
|
# far future -> should not yield result
|
|
241
241
|
result3 = database(expr3).select(TestSqlExpression.value, expr3).as_list()
|
|
242
242
|
assert not result3
|
|
243
|
+
|
|
244
|
+
# test quoting fields and tables:
|
|
245
|
+
assert str(database.sql_expression("LOWER(%s)", TestSqlExpression.value)) == 'LOWER("test_sql_expression"."value")'
|
|
246
|
+
assert str(database.sql_expression("LOWER(%s.value)", TestSqlExpression)) == 'LOWER("test_sql_expression".value)'
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|