Skip to content

Commit 9a61588

Browse files
committed
just horrific, fixes #1
1 parent fa86723 commit 9a61588

File tree

5 files changed

+118
-1
lines changed

5 files changed

+118
-1
lines changed

README.md

+12-1
Original file line numberDiff line numberDiff line change
@@ -15,7 +15,18 @@ Install this plugin in the same environment as Datasette.
1515

1616
## Usage
1717

18-
Usage instructions go here.
18+
Write a hook like:
19+
20+
```python
21+
from datasette import hookimpl
22+
23+
@hookimpl
24+
def rewrite_sql(sql):
25+
if sql == 'select 123':
26+
return 'select 234'
27+
28+
return sql
29+
```
1930

2031
## Development
2132

datasette_rewrite_sql/__init__.py

+88
Original file line numberDiff line numberDiff line change
@@ -1 +1,89 @@
1+
from datasette.utils import await_me_maybe
12
from datasette import hookimpl
3+
from datasette.plugins import pm
4+
from datasette.database import Database
5+
from . import hookspecs
6+
7+
original_execute = Database.execute
8+
original_execute_write = Database.execute_write
9+
original_execute_write_script = Database.execute_write_script
10+
original_execute_write_many = Database.execute_write_many
11+
12+
async def patched_execute(
13+
self,
14+
sql,
15+
params=None,
16+
truncate=False,
17+
custom_time_limit=None,
18+
page_size=None,
19+
log_sql_errors=True,
20+
):
21+
rewritten = await rewrite(self.ds, self, 'execute', sql, params)
22+
return await original_execute(
23+
self,
24+
rewritten,
25+
params,
26+
truncate,
27+
custom_time_limit,
28+
page_size,
29+
log_sql_errors
30+
)
31+
32+
async def patched_execute_write(self, sql, params=None, block=True):
33+
rewritten = await rewrite(self.ds, self, 'execute_write', sql, params)
34+
35+
return await original_execute_write(self, rewritten, params, block)
36+
37+
async def patched_execute_write_script(self, sql, block=True):
38+
rewritten = await rewrite(self.ds, self, 'execute_write_script', sql, None)
39+
40+
return await original_execute_write_script(self, rewritten, block)
41+
42+
async def patched_execute_write_many(self, sql, params_seq, block=True):
43+
rewritten = await rewrite(self.ds, self, 'execute_write_many', sql, params_seq)
44+
return await original_execute_write_many(self, rewritten, params_seq, block)
45+
46+
Database.execute = patched_execute
47+
Database.execute_write = patched_execute_write
48+
Database.execute_write_many = patched_execute_write_many
49+
Database.execute_write_script = patched_execute_write_script
50+
51+
52+
@hookimpl
53+
def startup():
54+
pm.add_hookspecs(hookspecs)
55+
56+
async def rewrite(datasette, database, fn, sql, params):
57+
# A challenge: if two plugins are participating in rewriting, who wins?
58+
#
59+
# Option 1: the first one registered wins.
60+
# Option 2: the output of the first is fed into the second.
61+
#
62+
# With option 2, if two plugins rewrite the output, ought the first one
63+
# get another kick at the can to rewrite the output of the second? That is,
64+
# should rewrites continue until no plugin wants to rewrite again?
65+
#
66+
# That feels like the "right" solution, but is probably hard for programmers
67+
# to reason about -- I can imagine easily getting into an infinite loop.
68+
#
69+
# Instead, let's let each hook rewrite exactly once.
70+
#
71+
# This seems to not be a supported case for pluggy. So, let's do
72+
# some vaguely evil things to do it ourselves.
73+
caller_kwargs = {
74+
'datasette': datasette,
75+
'database': database,
76+
'fn': fn,
77+
'sql': sql,
78+
'params': params
79+
}
80+
81+
for hook_impl in pm.hook.rewrite_sql.get_hookimpls():
82+
args = [caller_kwargs[argname] for argname in hook_impl.argnames]
83+
res = await await_me_maybe(hook_impl.function(*args))
84+
85+
if res is not None:
86+
sql = res
87+
caller_kwargs['sql'] = sql
88+
89+
return sql

datasette_rewrite_sql/hookspecs.py

+8
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
from pluggy import HookspecMarker
2+
3+
hookspec = HookspecMarker("datasette")
4+
5+
6+
@hookspec
7+
def rewrite_sql(datasette, database, fn, sql, params):
8+
"Inspect or rewrite the SQL request."

go

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
#!/bin/bash
2+
set -euo pipefail
3+
datasette --reload --plugins-dir plugins/

plugins/test.py

+7
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
from datasette import hookimpl
2+
3+
@hookimpl
4+
def rewrite_sql(sql):
5+
if sql == 'SELECT 123':
6+
return 'SELECT 234'
7+
return sql

0 commit comments

Comments
 (0)