2009-12-20 10:46:58 +08:00
|
|
|
==========================
|
2008-08-24 06:25:40 +08:00
|
|
|
Performing raw SQL queries
|
|
|
|
==========================
|
|
|
|
|
2009-12-20 10:46:58 +08:00
|
|
|
.. currentmodule:: django.db.models
|
|
|
|
|
2018-11-14 07:15:24 +08:00
|
|
|
Django gives you two ways of performing raw SQL queries: you can use
|
|
|
|
:meth:`Manager.raw()` to `perform raw queries and return model instances`__, or
|
|
|
|
you can avoid the model layer entirely and `execute custom SQL directly`__.
|
2009-12-20 10:46:58 +08:00
|
|
|
|
|
|
|
__ `performing raw queries`_
|
|
|
|
__ `executing custom SQL directly`_
|
|
|
|
|
2018-11-14 07:15:24 +08:00
|
|
|
.. admonition:: Explore the ORM before using raw SQL!
|
|
|
|
|
|
|
|
The Django ORM provides many tools to express queries without writing raw
|
|
|
|
SQL. For example:
|
|
|
|
|
|
|
|
* The :doc:`QuerySet API </ref/models/querysets>` is extensive.
|
|
|
|
* You can :meth:`annotate <.QuerySet.annotate>` and :doc:`aggregate
|
|
|
|
</topics/db/aggregation>` using many built-in :doc:`database functions
|
|
|
|
</ref/models/database-functions>`. Beyond those, you can create
|
|
|
|
:doc:`custom query expressions </ref/models/expressions/>`.
|
|
|
|
|
|
|
|
Before using raw SQL, explore :doc:`the ORM </topics/db/index>`. Ask on
|
2021-05-20 18:23:36 +08:00
|
|
|
one of :doc:`the support channels </faq/help>` to see if the ORM supports
|
|
|
|
your use case.
|
2018-11-14 07:15:24 +08:00
|
|
|
|
2014-04-25 02:10:03 +08:00
|
|
|
.. warning::
|
|
|
|
|
|
|
|
You should be very careful whenever you write raw SQL. Every time you use
|
|
|
|
it, you should properly escape any parameters that the user can control
|
|
|
|
by using ``params`` in order to protect against SQL injection attacks.
|
|
|
|
Please read more about :ref:`SQL injection protection
|
|
|
|
<sql-injection-protection>`.
|
|
|
|
|
2011-06-10 23:14:36 +08:00
|
|
|
.. _executing-raw-queries:
|
|
|
|
|
2009-12-20 10:46:58 +08:00
|
|
|
Performing raw queries
|
|
|
|
======================
|
|
|
|
|
|
|
|
The ``raw()`` manager method can be used to perform raw SQL queries that
|
|
|
|
return model instances:
|
|
|
|
|
2020-11-29 00:08:27 +08:00
|
|
|
.. method:: Manager.raw(raw_query, params=(), translations=None)
|
2009-12-20 10:46:58 +08:00
|
|
|
|
2014-01-03 03:22:52 +08:00
|
|
|
This method takes a raw SQL query, executes it, and returns a
|
2013-01-01 21:12:42 +08:00
|
|
|
``django.db.models.query.RawQuerySet`` instance. This ``RawQuerySet`` instance
|
2019-06-17 22:54:55 +08:00
|
|
|
can be iterated over like a normal :class:`~django.db.models.query.QuerySet` to
|
|
|
|
provide object instances.
|
2009-12-20 10:46:58 +08:00
|
|
|
|
2014-01-03 03:22:52 +08:00
|
|
|
This is best illustrated with an example. Suppose you have the following model::
|
2009-12-20 10:46:58 +08:00
|
|
|
|
|
|
|
class Person(models.Model):
|
|
|
|
first_name = models.CharField(...)
|
|
|
|
last_name = models.CharField(...)
|
|
|
|
birth_date = models.DateField(...)
|
|
|
|
|
|
|
|
You could then execute custom SQL like so::
|
|
|
|
|
2010-02-24 21:57:02 +08:00
|
|
|
>>> for p in Person.objects.raw('SELECT * FROM myapp_person'):
|
2012-04-29 00:02:01 +08:00
|
|
|
... print(p)
|
2010-02-24 21:57:02 +08:00
|
|
|
John Smith
|
|
|
|
Jane Jones
|
2009-12-20 10:46:58 +08:00
|
|
|
|
2020-05-01 20:37:21 +08:00
|
|
|
This example isn't very exciting -- it's exactly the same as running
|
|
|
|
``Person.objects.all()``. However, ``raw()`` has a bunch of other options that
|
|
|
|
make it very powerful.
|
2011-03-15 03:49:53 +08:00
|
|
|
|
2009-12-20 10:46:58 +08:00
|
|
|
.. admonition:: Model table names
|
|
|
|
|
2015-03-08 00:34:33 +08:00
|
|
|
Where did the name of the ``Person`` table come from in that example?
|
2009-12-20 10:46:58 +08:00
|
|
|
|
|
|
|
By default, Django figures out a database table name by joining the
|
|
|
|
model's "app label" -- the name you used in ``manage.py startapp`` -- to
|
|
|
|
the model's class name, with an underscore between them. In the example
|
|
|
|
we've assumed that the ``Person`` model lives in an app named ``myapp``,
|
|
|
|
so its table would be ``myapp_person``.
|
|
|
|
|
|
|
|
For more details check out the documentation for the
|
|
|
|
:attr:`~Options.db_table` option, which also lets you manually set the
|
|
|
|
database table name.
|
|
|
|
|
2011-03-15 03:49:53 +08:00
|
|
|
.. warning::
|
|
|
|
|
|
|
|
No checking is done on the SQL statement that is passed in to ``.raw()``.
|
|
|
|
Django expects that the statement will return a set of rows from the
|
|
|
|
database, but does nothing to enforce that. If the query does not
|
|
|
|
return rows, a (possibly cryptic) error will result.
|
2009-12-20 10:46:58 +08:00
|
|
|
|
2014-04-21 04:13:41 +08:00
|
|
|
.. warning::
|
|
|
|
|
|
|
|
If you are performing queries on MySQL, note that MySQL's silent type coercion
|
|
|
|
may cause unexpected results when mixing types. If you query on a string
|
|
|
|
type column, but with an integer value, MySQL will coerce the types of all values
|
|
|
|
in the table to an integer before performing the comparison. For example, if your
|
|
|
|
table contains the values ``'abc'``, ``'def'`` and you query for ``WHERE mycolumn=0``,
|
|
|
|
both rows will match. To prevent this, perform the correct typecasting
|
|
|
|
before using the value in a query.
|
|
|
|
|
2020-11-29 00:08:27 +08:00
|
|
|
.. versionchanged:: 3.2
|
|
|
|
|
|
|
|
The default value of the ``params`` argument was changed from ``None`` to
|
|
|
|
an empty tuple.
|
|
|
|
|
2009-12-20 10:46:58 +08:00
|
|
|
Mapping query fields to model fields
|
|
|
|
------------------------------------
|
|
|
|
|
|
|
|
``raw()`` automatically maps fields in the query to fields on the model.
|
|
|
|
|
|
|
|
The order of fields in your query doesn't matter. In other words, both
|
|
|
|
of the following queries work identically::
|
|
|
|
|
|
|
|
>>> Person.objects.raw('SELECT id, first_name, last_name, birth_date FROM myapp_person')
|
|
|
|
...
|
|
|
|
>>> Person.objects.raw('SELECT last_name, birth_date, first_name, id FROM myapp_person')
|
|
|
|
...
|
|
|
|
|
|
|
|
Matching is done by name. This means that you can use SQL's ``AS`` clauses to
|
|
|
|
map fields in the query to model fields. So if you had some other table that
|
|
|
|
had ``Person`` data in it, you could easily map it into ``Person`` instances::
|
|
|
|
|
|
|
|
>>> Person.objects.raw('''SELECT first AS first_name,
|
|
|
|
... last AS last_name,
|
|
|
|
... bd AS birth_date,
|
2014-02-21 02:32:04 +08:00
|
|
|
... pk AS id,
|
2010-11-09 13:04:05 +08:00
|
|
|
... FROM some_other_table''')
|
2009-12-20 10:46:58 +08:00
|
|
|
|
|
|
|
As long as the names match, the model instances will be created correctly.
|
|
|
|
|
|
|
|
Alternatively, you can map fields in the query to model fields using the
|
|
|
|
``translations`` argument to ``raw()``. This is a dictionary mapping names of
|
|
|
|
fields in the query to names of fields on the model. For example, the above
|
|
|
|
query could also be written::
|
|
|
|
|
|
|
|
>>> name_map = {'first': 'first_name', 'last': 'last_name', 'bd': 'birth_date', 'pk': 'id'}
|
|
|
|
>>> Person.objects.raw('SELECT * FROM some_other_table', translations=name_map)
|
|
|
|
|
2010-02-23 13:22:12 +08:00
|
|
|
Index lookups
|
|
|
|
-------------
|
|
|
|
|
|
|
|
``raw()`` supports indexing, so if you need only the first result you can
|
|
|
|
write::
|
|
|
|
|
2014-02-21 02:26:59 +08:00
|
|
|
>>> first_person = Person.objects.raw('SELECT * FROM myapp_person')[0]
|
2010-02-23 13:22:12 +08:00
|
|
|
|
|
|
|
However, the indexing and slicing are not performed at the database level. If
|
2014-01-03 03:22:52 +08:00
|
|
|
you have a large number of ``Person`` objects in your database, it is more
|
2010-02-23 13:22:12 +08:00
|
|
|
efficient to limit the query at the SQL level::
|
|
|
|
|
2014-02-21 02:26:59 +08:00
|
|
|
>>> first_person = Person.objects.raw('SELECT * FROM myapp_person LIMIT 1')[0]
|
2010-02-23 13:22:12 +08:00
|
|
|
|
2009-12-20 10:46:58 +08:00
|
|
|
Deferring model fields
|
|
|
|
----------------------
|
|
|
|
|
|
|
|
Fields may also be left out::
|
|
|
|
|
2010-02-24 21:57:02 +08:00
|
|
|
>>> people = Person.objects.raw('SELECT id, first_name FROM myapp_person')
|
2009-12-20 10:46:58 +08:00
|
|
|
|
2010-08-07 22:26:07 +08:00
|
|
|
The ``Person`` objects returned by this query will be deferred model instances
|
2011-09-30 18:28:39 +08:00
|
|
|
(see :meth:`~django.db.models.query.QuerySet.defer()`). This means that the
|
|
|
|
fields that are omitted from the query will be loaded on demand. For example::
|
2009-12-20 10:46:58 +08:00
|
|
|
|
|
|
|
>>> for p in Person.objects.raw('SELECT id, first_name FROM myapp_person'):
|
2012-04-29 00:02:01 +08:00
|
|
|
... print(p.first_name, # This will be retrieved by the original query
|
|
|
|
... p.last_name) # This will be retrieved on demand
|
2009-12-20 10:46:58 +08:00
|
|
|
...
|
|
|
|
John Smith
|
|
|
|
Jane Jones
|
|
|
|
|
|
|
|
From outward appearances, this looks like the query has retrieved both
|
|
|
|
the first name and last name. However, this example actually issued 3
|
|
|
|
queries. Only the first names were retrieved by the raw() query -- the
|
|
|
|
last names were both retrieved on demand when they were printed.
|
|
|
|
|
|
|
|
There is only one field that you can't leave out - the primary key
|
|
|
|
field. Django uses the primary key to identify model instances, so it
|
2019-11-16 05:20:07 +08:00
|
|
|
must always be included in a raw query. A
|
|
|
|
:class:`~django.core.exceptions.FieldDoesNotExist` exception will be raised if
|
|
|
|
you forget to include the primary key.
|
2009-12-20 10:46:58 +08:00
|
|
|
|
|
|
|
Adding annotations
|
|
|
|
------------------
|
|
|
|
|
|
|
|
You can also execute queries containing fields that aren't defined on the
|
|
|
|
model. For example, we could use `PostgreSQL's age() function`__ to get a list
|
|
|
|
of people with their ages calculated by the database::
|
|
|
|
|
|
|
|
>>> people = Person.objects.raw('SELECT *, age(birth_date) AS age FROM myapp_person')
|
|
|
|
>>> for p in people:
|
2012-04-29 00:02:01 +08:00
|
|
|
... print("%s is %s." % (p.first_name, p.age))
|
2009-12-20 10:46:58 +08:00
|
|
|
John is 37.
|
|
|
|
Jane is 42.
|
|
|
|
...
|
|
|
|
|
2018-11-14 07:15:24 +08:00
|
|
|
You can often avoid using raw SQL to compute annotations by instead using a
|
|
|
|
:ref:`Func() expression <func-expressions>`.
|
|
|
|
|
2019-03-30 09:49:44 +08:00
|
|
|
__ https://www.postgresql.org/docs/current/functions-datetime.html
|
2009-12-20 10:46:58 +08:00
|
|
|
|
|
|
|
Passing parameters into ``raw()``
|
|
|
|
---------------------------------
|
|
|
|
|
|
|
|
If you need to perform parameterized queries, you can use the ``params``
|
|
|
|
argument to ``raw()``::
|
|
|
|
|
|
|
|
>>> lname = 'Doe'
|
|
|
|
>>> Person.objects.raw('SELECT * FROM myapp_person WHERE last_name = %s', [lname])
|
|
|
|
|
2013-06-28 11:15:03 +08:00
|
|
|
``params`` is a list or dictionary of parameters. You'll use ``%s``
|
|
|
|
placeholders in the query string for a list, or ``%(key)s``
|
|
|
|
placeholders for a dictionary (where ``key`` is replaced by a
|
2020-05-01 20:37:21 +08:00
|
|
|
dictionary key), regardless of your database engine. Such placeholders will be
|
|
|
|
replaced with parameters from the ``params`` argument.
|
2013-06-28 11:15:03 +08:00
|
|
|
|
2014-12-06 02:02:59 +08:00
|
|
|
.. note::
|
2013-06-28 11:15:03 +08:00
|
|
|
|
|
|
|
Dictionary params are not supported with the SQLite backend; with
|
|
|
|
this backend, you must pass parameters as a list.
|
2009-12-20 10:46:58 +08:00
|
|
|
|
|
|
|
.. warning::
|
|
|
|
|
2017-11-08 02:07:12 +08:00
|
|
|
**Do not use string formatting on raw queries or quote placeholders in your
|
|
|
|
SQL strings!**
|
2009-12-20 10:46:58 +08:00
|
|
|
|
|
|
|
It's tempting to write the above query as::
|
|
|
|
|
2009-12-21 09:53:39 +08:00
|
|
|
>>> query = 'SELECT * FROM myapp_person WHERE last_name = %s' % lname
|
2009-12-20 10:46:58 +08:00
|
|
|
>>> Person.objects.raw(query)
|
|
|
|
|
2017-11-08 02:07:12 +08:00
|
|
|
You might also think you should write your query like this (with quotes
|
|
|
|
around ``%s``)::
|
2009-12-20 10:46:58 +08:00
|
|
|
|
2017-11-08 02:07:12 +08:00
|
|
|
>>> query = "SELECT * FROM myapp_person WHERE last_name = '%s'"
|
|
|
|
|
|
|
|
**Don't make either of these mistakes.**
|
|
|
|
|
|
|
|
As discussed in :ref:`sql-injection-protection`, using the ``params``
|
|
|
|
argument and leaving the placeholders unquoted protects you from `SQL
|
|
|
|
injection attacks`__, a common exploit where attackers inject arbitrary
|
|
|
|
SQL into your database. If you use string interpolation or quote the
|
|
|
|
placeholder, you're at risk for SQL injection.
|
2009-12-20 10:46:58 +08:00
|
|
|
|
2015-08-08 18:02:32 +08:00
|
|
|
__ https://en.wikipedia.org/wiki/SQL_injection
|
2009-12-20 10:46:58 +08:00
|
|
|
|
2011-06-10 23:14:36 +08:00
|
|
|
.. _executing-custom-sql:
|
|
|
|
|
2009-12-20 10:46:58 +08:00
|
|
|
Executing custom SQL directly
|
|
|
|
=============================
|
|
|
|
|
|
|
|
Sometimes even :meth:`Manager.raw` isn't quite enough: you might need to
|
|
|
|
perform queries that don't map cleanly to models, or directly execute
|
|
|
|
``UPDATE``, ``INSERT``, or ``DELETE`` queries.
|
|
|
|
|
|
|
|
In these cases, you can always access the database directly, routing around
|
|
|
|
the model layer entirely.
|
|
|
|
|
2013-03-03 22:55:11 +08:00
|
|
|
The object ``django.db.connection`` represents the default database
|
|
|
|
connection. To use the database connection, call ``connection.cursor()`` to
|
|
|
|
get a cursor object. Then, call ``cursor.execute(sql, [params])`` to execute
|
|
|
|
the SQL and ``cursor.fetchone()`` or ``cursor.fetchall()`` to return the
|
|
|
|
resulting rows.
|
|
|
|
|
|
|
|
For example::
|
|
|
|
|
|
|
|
from django.db import connection
|
2008-08-24 06:25:40 +08:00
|
|
|
|
2013-05-05 22:22:25 +08:00
|
|
|
def my_custom_sql(self):
|
2016-09-07 18:14:29 +08:00
|
|
|
with connection.cursor() as cursor:
|
|
|
|
cursor.execute("UPDATE bar SET foo = 1 WHERE baz = %s", [self.baz])
|
|
|
|
cursor.execute("SELECT foo FROM bar WHERE baz = %s", [self.baz])
|
|
|
|
row = cursor.fetchone()
|
2009-05-02 15:40:25 +08:00
|
|
|
|
2008-08-24 06:25:40 +08:00
|
|
|
return row
|
|
|
|
|
2017-11-08 02:07:12 +08:00
|
|
|
To protect against SQL injection, you must not include quotes around the ``%s``
|
|
|
|
placeholders in the SQL string.
|
|
|
|
|
2013-03-23 23:09:56 +08:00
|
|
|
Note that if you want to include literal percent signs in the query, you have to
|
|
|
|
double them in the case you are passing parameters::
|
|
|
|
|
|
|
|
cursor.execute("SELECT foo FROM bar WHERE baz = '30%'")
|
2014-02-21 02:32:04 +08:00
|
|
|
cursor.execute("SELECT foo FROM bar WHERE baz = '30%%' AND id = %s", [self.id])
|
2013-03-23 23:09:56 +08:00
|
|
|
|
2012-04-06 17:17:37 +08:00
|
|
|
If you are using :doc:`more than one database </topics/db/multi-db>`, you can
|
|
|
|
use ``django.db.connections`` to obtain the connection (and cursor) for a
|
2010-03-08 11:19:26 +08:00
|
|
|
specific database. ``django.db.connections`` is a dictionary-like
|
2011-05-30 20:11:10 +08:00
|
|
|
object that allows you to retrieve a specific connection using its
|
2010-03-08 11:19:26 +08:00
|
|
|
alias::
|
|
|
|
|
|
|
|
from django.db import connections
|
2017-11-28 21:12:28 +08:00
|
|
|
with connections['my_db_alias'].cursor() as cursor:
|
|
|
|
# Your code here...
|
2010-03-08 11:19:26 +08:00
|
|
|
|
2015-08-29 22:54:51 +08:00
|
|
|
By default, the Python DB API will return results without their field names,
|
|
|
|
which means you end up with a ``list`` of values, rather than a ``dict``. At a
|
|
|
|
small performance and memory cost, you can return results as a ``dict`` by
|
|
|
|
using something like this::
|
2012-09-20 04:39:14 +08:00
|
|
|
|
2011-09-11 13:37:55 +08:00
|
|
|
def dictfetchall(cursor):
|
2015-08-29 22:54:51 +08:00
|
|
|
"Return all rows from a cursor as a dict"
|
2015-09-25 01:17:39 +08:00
|
|
|
columns = [col[0] for col in cursor.description]
|
2011-09-11 13:37:55 +08:00
|
|
|
return [
|
2015-09-25 01:17:39 +08:00
|
|
|
dict(zip(columns, row))
|
2011-09-11 13:37:55 +08:00
|
|
|
for row in cursor.fetchall()
|
|
|
|
]
|
|
|
|
|
2015-08-29 22:54:51 +08:00
|
|
|
Another option is to use :func:`collections.namedtuple` from the Python
|
|
|
|
standard library. A ``namedtuple`` is a tuple-like object that has fields
|
|
|
|
accessible by attribute lookup; it's also indexable and iterable. Results are
|
|
|
|
immutable and accessible by field names or indices, which might be useful::
|
|
|
|
|
|
|
|
from collections import namedtuple
|
|
|
|
|
|
|
|
def namedtuplefetchall(cursor):
|
|
|
|
"Return all rows from a cursor as a namedtuple"
|
|
|
|
desc = cursor.description
|
|
|
|
nt_result = namedtuple('Result', [col[0] for col in desc])
|
|
|
|
return [nt_result(*row) for row in cursor.fetchall()]
|
|
|
|
|
|
|
|
Here is an example of the difference between the three::
|
2011-09-11 13:37:55 +08:00
|
|
|
|
2014-02-21 02:26:59 +08:00
|
|
|
>>> cursor.execute("SELECT id, parent_id FROM test LIMIT 2");
|
2011-09-11 13:37:55 +08:00
|
|
|
>>> cursor.fetchall()
|
2015-08-29 22:54:51 +08:00
|
|
|
((54360982, None), (54360880, None))
|
2012-09-20 04:39:14 +08:00
|
|
|
|
2014-02-21 02:26:59 +08:00
|
|
|
>>> cursor.execute("SELECT id, parent_id FROM test LIMIT 2");
|
2011-09-11 13:37:55 +08:00
|
|
|
>>> dictfetchall(cursor)
|
2015-08-29 22:54:51 +08:00
|
|
|
[{'parent_id': None, 'id': 54360982}, {'parent_id': None, 'id': 54360880}]
|
|
|
|
|
|
|
|
>>> cursor.execute("SELECT id, parent_id FROM test LIMIT 2");
|
|
|
|
>>> results = namedtuplefetchall(cursor)
|
|
|
|
>>> results
|
|
|
|
[Result(id=54360982, parent_id=None), Result(id=54360880, parent_id=None)]
|
|
|
|
>>> results[0].id
|
|
|
|
54360982
|
|
|
|
>>> results[0][0]
|
|
|
|
54360982
|
2011-09-11 13:37:55 +08:00
|
|
|
|
2009-06-17 21:47:39 +08:00
|
|
|
Connections and cursors
|
|
|
|
-----------------------
|
|
|
|
|
2011-09-05 05:17:30 +08:00
|
|
|
``connection`` and ``cursor`` mostly implement the standard Python DB-API
|
2013-03-03 22:55:11 +08:00
|
|
|
described in :pep:`249` — except when it comes to :doc:`transaction handling
|
|
|
|
</topics/db/transactions>`.
|
|
|
|
|
|
|
|
If you're not familiar with the Python DB-API, note that the SQL statement in
|
|
|
|
``cursor.execute()`` uses placeholders, ``"%s"``, rather than adding
|
|
|
|
parameters directly within the SQL. If you use this technique, the underlying
|
|
|
|
database library will automatically escape your parameters as necessary.
|
|
|
|
|
|
|
|
Also note that Django expects the ``"%s"`` placeholder, *not* the ``"?"``
|
|
|
|
placeholder, which is used by the SQLite Python bindings. This is for the sake
|
|
|
|
of consistency and sanity.
|
2013-09-24 08:17:59 +08:00
|
|
|
|
2014-08-18 22:30:44 +08:00
|
|
|
Using a cursor as a context manager::
|
2013-09-24 08:17:59 +08:00
|
|
|
|
|
|
|
with connection.cursor() as c:
|
|
|
|
c.execute(...)
|
|
|
|
|
2014-08-18 22:30:44 +08:00
|
|
|
is equivalent to::
|
2013-09-24 08:17:59 +08:00
|
|
|
|
|
|
|
c = connection.cursor()
|
|
|
|
try:
|
|
|
|
c.execute(...)
|
|
|
|
finally:
|
2014-01-03 03:22:52 +08:00
|
|
|
c.close()
|
2017-08-11 04:21:11 +08:00
|
|
|
|
|
|
|
Calling stored procedures
|
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
|
|
|
2017-08-13 03:06:49 +08:00
|
|
|
.. method:: CursorWrapper.callproc(procname, params=None, kparams=None)
|
2017-08-11 04:21:11 +08:00
|
|
|
|
2017-08-13 03:06:49 +08:00
|
|
|
Calls a database stored procedure with the given name. A sequence
|
|
|
|
(``params``) or dictionary (``kparams``) of input parameters may be
|
|
|
|
provided. Most databases don't support ``kparams``. Of Django's built-in
|
|
|
|
backends, only Oracle supports it.
|
2017-08-11 04:21:11 +08:00
|
|
|
|
|
|
|
For example, given this stored procedure in an Oracle database:
|
|
|
|
|
|
|
|
.. code-block:: sql
|
|
|
|
|
|
|
|
CREATE PROCEDURE "TEST_PROCEDURE"(v_i INTEGER, v_text NVARCHAR2(10)) AS
|
|
|
|
p_i INTEGER;
|
|
|
|
p_text NVARCHAR2(10);
|
|
|
|
BEGIN
|
|
|
|
p_i := v_i;
|
|
|
|
p_text := v_text;
|
|
|
|
...
|
|
|
|
END;
|
|
|
|
|
|
|
|
This will call it::
|
|
|
|
|
|
|
|
with connection.cursor() as cursor:
|
|
|
|
cursor.callproc('test_procedure', [1, 'test'])
|