|
| 1 | +from datasette.utils import await_me_maybe |
1 | 2 | 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 |
0 commit comments