PageRenderTime 88ms CodeModel.GetById 19ms RepoModel.GetById 0ms app.codeStats 1ms

/SQLAlchemy-0.7.8/doc/orm/relationships.html

#
HTML | 1735 lines | 1601 code | 134 blank | 0 comment | 0 complexity | f1e1f20d4eb8b175a9244080d8c9f17c MD5 | raw file
  1. <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN"
  2. "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
  3. <html>
  4. <head>
  5. <meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
  6. <title>
  7. Relationship Configuration
  8. &mdash;
  9. SQLAlchemy 0.7 Documentation
  10. </title>
  11. <link rel="stylesheet" href="../_static/pygments.css" type="text/css" />
  12. <link rel="stylesheet" href="../_static/docs.css" type="text/css" />
  13. <script type="text/javascript">
  14. var DOCUMENTATION_OPTIONS = {
  15. URL_ROOT: '../',
  16. VERSION: '0.7.8',
  17. COLLAPSE_MODINDEX: false,
  18. FILE_SUFFIX: '.html'
  19. };
  20. </script>
  21. <script type="text/javascript" src="../_static/jquery.js"></script>
  22. <script type="text/javascript" src="../_static/underscore.js"></script>
  23. <script type="text/javascript" src="../_static/doctools.js"></script>
  24. <script type="text/javascript" src="../_static/init.js"></script>
  25. <link rel="index" title="Index" href="../genindex.html" />
  26. <link rel="search" title="Search" href="../search.html" />
  27. <link rel="copyright" title="Copyright" href="../copyright.html" />
  28. <link rel="top" title="SQLAlchemy 0.7 Documentation" href="../index.html" />
  29. <link rel="up" title="SQLAlchemy ORM" href="index.html" />
  30. <link rel="next" title="Collection Configuration and Techniques" href="collections.html" />
  31. <link rel="prev" title="Mapper Configuration" href="mapper_config.html" />
  32. </head>
  33. <body>
  34. <div id="docs-container">
  35. <div id="docs-header">
  36. <h1>SQLAlchemy 0.7 Documentation</h1>
  37. <div id="docs-search">
  38. Search:
  39. <form class="search" action="../search.html" method="get">
  40. <input type="text" name="q" size="18" /> <input type="submit" value="Search" />
  41. <input type="hidden" name="check_keywords" value="yes" />
  42. <input type="hidden" name="area" value="default" />
  43. </form>
  44. </div>
  45. <div id="docs-version-header">
  46. Release: <span class="version-num">0.7.8</span> | Release Date: June 16, 2012
  47. </div>
  48. </div>
  49. <div id="docs-top-navigation">
  50. <div id="docs-top-page-control" class="docs-navigation-links">
  51. <ul>
  52. <li>Prev:
  53. <a href="mapper_config.html" title="previous chapter">Mapper Configuration</a>
  54. </li>
  55. <li>Next:
  56. <a href="collections.html" title="next chapter">Collection Configuration and Techniques</a>
  57. </li>
  58. <li>
  59. <a href="../contents.html">Table of Contents</a> |
  60. <a href="../genindex.html">Index</a>
  61. | <a href="../_sources/orm/relationships.txt">view source
  62. </li>
  63. </ul>
  64. </div>
  65. <div id="docs-navigation-banner">
  66. <a href="../index.html">SQLAlchemy 0.7 Documentation</a>
  67. ť <a href="index.html" title="SQLAlchemy ORM">SQLAlchemy ORM</a>
  68. ť
  69. Relationship Configuration
  70. <h2>
  71. Relationship Configuration
  72. </h2>
  73. </div>
  74. </div>
  75. <div id="docs-body-container">
  76. <div id="docs-sidebar">
  77. <h3><a href="../index.html">Table of Contents</a></h3>
  78. <ul>
  79. <li><a class="reference internal" href="#">Relationship Configuration</a><ul>
  80. <li><a class="reference internal" href="#basic-relational-patterns">Basic Relational Patterns</a><ul>
  81. <li><a class="reference internal" href="#one-to-many">One To Many</a></li>
  82. <li><a class="reference internal" href="#many-to-one">Many To One</a></li>
  83. <li><a class="reference internal" href="#one-to-one">One To One</a></li>
  84. <li><a class="reference internal" href="#many-to-many">Many To Many</a></li>
  85. <li><a class="reference internal" href="#association-object">Association Object</a></li>
  86. </ul>
  87. </li>
  88. <li><a class="reference internal" href="#adjacency-list-relationships">Adjacency List Relationships</a><ul>
  89. <li><a class="reference internal" href="#self-referential-query-strategies">Self-Referential Query Strategies</a></li>
  90. <li><a class="reference internal" href="#configuring-self-referential-eager-loading">Configuring Self-Referential Eager Loading</a></li>
  91. </ul>
  92. </li>
  93. <li><a class="reference internal" href="#linking-relationships-with-backref">Linking Relationships with Backref</a><ul>
  94. <li><a class="reference internal" href="#backref-arguments">Backref Arguments</a></li>
  95. <li><a class="reference internal" href="#one-way-backrefs">One Way Backrefs</a></li>
  96. </ul>
  97. </li>
  98. <li><a class="reference internal" href="#setting-the-primaryjoin-and-secondaryjoin">Setting the primaryjoin and secondaryjoin</a><ul>
  99. <li><a class="reference internal" href="#specifying-alternate-join-conditions">Specifying Alternate Join Conditions</a></li>
  100. <li><a class="reference internal" href="#self-referential-many-to-many-relationship">Self-Referential Many-to-Many Relationship</a></li>
  101. <li><a class="reference internal" href="#specifying-foreign-keys">Specifying Foreign Keys</a></li>
  102. <li><a class="reference internal" href="#building-query-enabled-properties">Building Query-Enabled Properties</a></li>
  103. </ul>
  104. </li>
  105. <li><a class="reference internal" href="#rows-that-point-to-themselves-mutually-dependent-rows">Rows that point to themselves / Mutually Dependent Rows</a></li>
  106. <li><a class="reference internal" href="#mutable-primary-keys-update-cascades">Mutable Primary Keys / Update Cascades</a></li>
  107. <li><a class="reference internal" href="#relationships-api">Relationships API</a></li>
  108. </ul>
  109. </li>
  110. </ul>
  111. <h4>Previous Topic</h4>
  112. <p>
  113. <a href="mapper_config.html" title="previous chapter">Mapper Configuration</a>
  114. </p>
  115. <h4>Next Topic</h4>
  116. <p>
  117. <a href="collections.html" title="next chapter">Collection Configuration and Techniques</a>
  118. </p>
  119. <h4>Quick Search</h4>
  120. <p>
  121. <form class="search" action="../search.html" method="get">
  122. <input type="text" name="q" size="18" /> <input type="submit" value="Search" />
  123. <input type="hidden" name="check_keywords" value="yes" />
  124. <input type="hidden" name="area" value="default" />
  125. </form>
  126. </p>
  127. </div>
  128. <div id="docs-body" class="withsidebar" >
  129. <span class="target" id="module-sqlalchemy.orm"></span><div class="section" id="relationship-configuration">
  130. <span id="relationship-config-toplevel"></span><h1>Relationship Configuration<a class="headerlink" href="#relationship-configuration" title="Permalink to this headline">ś</a></h1>
  131. <p>This section describes the <a class="reference internal" href="#sqlalchemy.orm.relationship" title="sqlalchemy.orm.relationship"><tt class="xref py py-func docutils literal"><span class="pre">relationship()</span></tt></a> function and in depth discussion
  132. of its usage. The reference material here continues into the next section,
  133. <a class="reference internal" href="collections.html"><em>Collection Configuration and Techniques</em></a>, which has additional detail on configuration
  134. of collections via <a class="reference internal" href="#sqlalchemy.orm.relationship" title="sqlalchemy.orm.relationship"><tt class="xref py py-func docutils literal"><span class="pre">relationship()</span></tt></a>.</p>
  135. <div class="section" id="basic-relational-patterns">
  136. <span id="relationship-patterns"></span><h2>Basic Relational Patterns<a class="headerlink" href="#basic-relational-patterns" title="Permalink to this headline">ś</a></h2>
  137. <p>A quick walkthrough of the basic relational patterns.</p>
  138. <p>The imports used for each of the following sections is as follows:</p>
  139. <div class="highlight-python"><div class="highlight"><pre><span class="kn">from</span> <span class="nn">sqlalchemy</span> <span class="kn">import</span> <span class="n">Table</span><span class="p">,</span> <span class="n">Column</span><span class="p">,</span> <span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span>
  140. <span class="kn">from</span> <span class="nn">sqlalchemy.orm</span> <span class="kn">import</span> <span class="n">relationship</span><span class="p">,</span> <span class="n">backref</span>
  141. <span class="kn">from</span> <span class="nn">sqlalchemy.ext.declarative</span> <span class="kn">import</span> <span class="n">declarative_base</span>
  142. <span class="n">Base</span> <span class="o">=</span> <span class="n">declarative_base</span><span class="p">()</span></pre></div>
  143. </div>
  144. <div class="section" id="one-to-many">
  145. <h3>One To Many<a class="headerlink" href="#one-to-many" title="Permalink to this headline">ś</a></h3>
  146. <p>A one to many relationship places a foreign key on the child table referencing
  147. the parent. <a class="reference internal" href="#sqlalchemy.orm.relationship" title="sqlalchemy.orm.relationship"><tt class="xref py py-func docutils literal"><span class="pre">relationship()</span></tt></a> is then specified on the parent, as referencing
  148. a collection of items represented by the child:</p>
  149. <div class="highlight-python"><div class="highlight"><pre><span class="k">class</span> <span class="nc">Parent</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  150. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;parent&#39;</span>
  151. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  152. <span class="n">children</span> <span class="o">=</span> <span class="n">relationship</span><span class="p">(</span><span class="s">&quot;Child&quot;</span><span class="p">)</span>
  153. <span class="k">class</span> <span class="nc">Child</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  154. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;child&#39;</span>
  155. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  156. <span class="n">parent_id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">(</span><span class="s">&#39;parent.id&#39;</span><span class="p">))</span></pre></div>
  157. </div>
  158. <p>To establish a bidirectional relationship in one-to-many, where the &#8220;reverse&#8221;
  159. side is a many to one, specify the <tt class="docutils literal"><span class="pre">backref</span></tt> option:</p>
  160. <div class="highlight-python"><div class="highlight"><pre><span class="k">class</span> <span class="nc">Parent</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  161. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;parent&#39;</span>
  162. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  163. <span class="n">children</span> <span class="o">=</span> <span class="n">relationship</span><span class="p">(</span><span class="s">&quot;Child&quot;</span><span class="p">,</span> <span class="n">backref</span><span class="o">=</span><span class="s">&quot;parent&quot;</span><span class="p">)</span>
  164. <span class="k">class</span> <span class="nc">Child</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  165. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;child&#39;</span>
  166. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  167. <span class="n">parent_id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">(</span><span class="s">&#39;parent.id&#39;</span><span class="p">))</span></pre></div>
  168. </div>
  169. <p><tt class="docutils literal"><span class="pre">Child</span></tt> will get a <tt class="docutils literal"><span class="pre">parent</span></tt> attribute with many-to-one semantics.</p>
  170. </div>
  171. <div class="section" id="many-to-one">
  172. <h3>Many To One<a class="headerlink" href="#many-to-one" title="Permalink to this headline">ś</a></h3>
  173. <p>Many to one places a foreign key in the parent table referencing the child.
  174. <a class="reference internal" href="#sqlalchemy.orm.relationship" title="sqlalchemy.orm.relationship"><tt class="xref py py-func docutils literal"><span class="pre">relationship()</span></tt></a> is declared on the parent, where a new scalar-holding
  175. attribute will be created:</p>
  176. <div class="highlight-python"><div class="highlight"><pre><span class="k">class</span> <span class="nc">Parent</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  177. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;parent&#39;</span>
  178. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  179. <span class="n">child_id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">(</span><span class="s">&#39;child.id&#39;</span><span class="p">))</span>
  180. <span class="n">child</span> <span class="o">=</span> <span class="n">relationship</span><span class="p">(</span><span class="s">&quot;Child&quot;</span><span class="p">)</span>
  181. <span class="k">class</span> <span class="nc">Child</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  182. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;child&#39;</span>
  183. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span></pre></div>
  184. </div>
  185. <p>Bidirectional behavior is achieved by specifying <tt class="docutils literal"><span class="pre">backref=&quot;parents&quot;</span></tt>,
  186. which will place a one-to-many collection on the <tt class="docutils literal"><span class="pre">Child</span></tt> class:</p>
  187. <div class="highlight-python"><div class="highlight"><pre><span class="k">class</span> <span class="nc">Parent</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  188. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;parent&#39;</span>
  189. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  190. <span class="n">child_id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">(</span><span class="s">&#39;child.id&#39;</span><span class="p">))</span>
  191. <span class="n">child</span> <span class="o">=</span> <span class="n">relationship</span><span class="p">(</span><span class="s">&quot;Child&quot;</span><span class="p">,</span> <span class="n">backref</span><span class="o">=</span><span class="s">&quot;parents&quot;</span><span class="p">)</span></pre></div>
  192. </div>
  193. </div>
  194. <div class="section" id="one-to-one">
  195. <h3>One To One<a class="headerlink" href="#one-to-one" title="Permalink to this headline">ś</a></h3>
  196. <p>One To One is essentially a bidirectional relationship with a scalar
  197. attribute on both sides. To achieve this, the <tt class="docutils literal"><span class="pre">uselist=False</span></tt> flag indicates
  198. the placement of a scalar attribute instead of a collection on the &#8220;many&#8221; side
  199. of the relationship. To convert one-to-many into one-to-one:</p>
  200. <div class="highlight-python"><div class="highlight"><pre><span class="k">class</span> <span class="nc">Parent</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  201. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;parent&#39;</span>
  202. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  203. <span class="n">child</span> <span class="o">=</span> <span class="n">relationship</span><span class="p">(</span><span class="s">&quot;Child&quot;</span><span class="p">,</span> <span class="n">uselist</span><span class="o">=</span><span class="bp">False</span><span class="p">,</span> <span class="n">backref</span><span class="o">=</span><span class="s">&quot;parent&quot;</span><span class="p">)</span>
  204. <span class="k">class</span> <span class="nc">Child</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  205. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;child&#39;</span>
  206. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  207. <span class="n">parent_id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">(</span><span class="s">&#39;parent.id&#39;</span><span class="p">))</span></pre></div>
  208. </div>
  209. <p>Or to turn a one-to-many backref into one-to-one, use the <a class="reference internal" href="#sqlalchemy.orm.backref" title="sqlalchemy.orm.backref"><tt class="xref py py-func docutils literal"><span class="pre">backref()</span></tt></a> function
  210. to provide arguments for the reverse side:</p>
  211. <div class="highlight-python"><div class="highlight"><pre><span class="k">class</span> <span class="nc">Parent</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  212. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;parent&#39;</span>
  213. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  214. <span class="n">child_id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">(</span><span class="s">&#39;child.id&#39;</span><span class="p">))</span>
  215. <span class="n">child</span> <span class="o">=</span> <span class="n">relationship</span><span class="p">(</span><span class="s">&quot;Child&quot;</span><span class="p">,</span> <span class="n">backref</span><span class="o">=</span><span class="n">backref</span><span class="p">(</span><span class="s">&quot;parent&quot;</span><span class="p">,</span> <span class="n">uselist</span><span class="o">=</span><span class="bp">False</span><span class="p">))</span>
  216. <span class="k">class</span> <span class="nc">Child</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  217. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;child&#39;</span>
  218. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span></pre></div>
  219. </div>
  220. </div>
  221. <div class="section" id="many-to-many">
  222. <span id="relationships-many-to-many"></span><h3>Many To Many<a class="headerlink" href="#many-to-many" title="Permalink to this headline">ś</a></h3>
  223. <p>Many to Many adds an association table between two classes. The association
  224. table is indicated by the <tt class="docutils literal"><span class="pre">secondary</span></tt> argument to
  225. <a class="reference internal" href="#sqlalchemy.orm.relationship" title="sqlalchemy.orm.relationship"><tt class="xref py py-func docutils literal"><span class="pre">relationship()</span></tt></a>. Usually, the <a class="reference internal" href="../core/schema.html#sqlalchemy.schema.Table" title="sqlalchemy.schema.Table"><tt class="xref py py-class docutils literal"><span class="pre">Table</span></tt></a> uses the <a class="reference internal" href="../core/schema.html#sqlalchemy.schema.MetaData" title="sqlalchemy.schema.MetaData"><tt class="xref py py-class docutils literal"><span class="pre">MetaData</span></tt></a>
  226. object associated with the declarative base class, so that the <a class="reference internal" href="../core/schema.html#sqlalchemy.schema.ForeignKey" title="sqlalchemy.schema.ForeignKey"><tt class="xref py py-class docutils literal"><span class="pre">ForeignKey</span></tt></a>
  227. directives can locate the remote tables with which to link:</p>
  228. <div class="highlight-python"><div class="highlight"><pre><span class="n">association_table</span> <span class="o">=</span> <span class="n">Table</span><span class="p">(</span><span class="s">&#39;association&#39;</span><span class="p">,</span> <span class="n">Base</span><span class="o">.</span><span class="n">metadata</span><span class="p">,</span>
  229. <span class="n">Column</span><span class="p">(</span><span class="s">&#39;left_id&#39;</span><span class="p">,</span> <span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">(</span><span class="s">&#39;left.id&#39;</span><span class="p">)),</span>
  230. <span class="n">Column</span><span class="p">(</span><span class="s">&#39;right_id&#39;</span><span class="p">,</span> <span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">(</span><span class="s">&#39;right.id&#39;</span><span class="p">))</span>
  231. <span class="p">)</span>
  232. <span class="k">class</span> <span class="nc">Parent</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  233. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;left&#39;</span>
  234. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  235. <span class="n">children</span> <span class="o">=</span> <span class="n">relationship</span><span class="p">(</span><span class="s">&quot;Child&quot;</span><span class="p">,</span>
  236. <span class="n">secondary</span><span class="o">=</span><span class="n">association_table</span><span class="p">)</span>
  237. <span class="k">class</span> <span class="nc">Child</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  238. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;right&#39;</span>
  239. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span></pre></div>
  240. </div>
  241. <p>For a bidirectional relationship, both sides of the relationship contain a
  242. collection. The <tt class="docutils literal"><span class="pre">backref</span></tt> keyword will automatically use
  243. the same <tt class="docutils literal"><span class="pre">secondary</span></tt> argument for the reverse relationship:</p>
  244. <div class="highlight-python"><div class="highlight"><pre><span class="n">association_table</span> <span class="o">=</span> <span class="n">Table</span><span class="p">(</span><span class="s">&#39;association&#39;</span><span class="p">,</span> <span class="n">Base</span><span class="o">.</span><span class="n">metadata</span><span class="p">,</span>
  245. <span class="n">Column</span><span class="p">(</span><span class="s">&#39;left_id&#39;</span><span class="p">,</span> <span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">(</span><span class="s">&#39;left.id&#39;</span><span class="p">)),</span>
  246. <span class="n">Column</span><span class="p">(</span><span class="s">&#39;right_id&#39;</span><span class="p">,</span> <span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">(</span><span class="s">&#39;right.id&#39;</span><span class="p">))</span>
  247. <span class="p">)</span>
  248. <span class="k">class</span> <span class="nc">Parent</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  249. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;left&#39;</span>
  250. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  251. <span class="n">children</span> <span class="o">=</span> <span class="n">relationship</span><span class="p">(</span><span class="s">&quot;Child&quot;</span><span class="p">,</span>
  252. <span class="n">secondary</span><span class="o">=</span><span class="n">association_table</span><span class="p">,</span>
  253. <span class="n">backref</span><span class="o">=</span><span class="s">&quot;parents&quot;</span><span class="p">)</span>
  254. <span class="k">class</span> <span class="nc">Child</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  255. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;right&#39;</span>
  256. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span></pre></div>
  257. </div>
  258. <p>The <tt class="docutils literal"><span class="pre">secondary</span></tt> argument of <a class="reference internal" href="#sqlalchemy.orm.relationship" title="sqlalchemy.orm.relationship"><tt class="xref py py-func docutils literal"><span class="pre">relationship()</span></tt></a> also accepts a callable
  259. that returns the ultimate argument, which is evaluated only when mappers are
  260. first used. Using this, we can define the <tt class="docutils literal"><span class="pre">association_table</span></tt> at a later
  261. point, as long as it&#8217;s available to the callable after all module initialization
  262. is complete:</p>
  263. <div class="highlight-python"><div class="highlight"><pre><span class="k">class</span> <span class="nc">Parent</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  264. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;left&#39;</span>
  265. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  266. <span class="n">children</span> <span class="o">=</span> <span class="n">relationship</span><span class="p">(</span><span class="s">&quot;Child&quot;</span><span class="p">,</span>
  267. <span class="n">secondary</span><span class="o">=</span><span class="k">lambda</span><span class="p">:</span> <span class="n">association_table</span><span class="p">,</span>
  268. <span class="n">backref</span><span class="o">=</span><span class="s">&quot;parents&quot;</span><span class="p">)</span></pre></div>
  269. </div>
  270. <p>With the declarative extension in use, the traditional &#8220;string name of the table&#8221;
  271. is accepted as well, matching the name of the table as stored in <tt class="docutils literal"><span class="pre">Base.metadata.tables</span></tt>:</p>
  272. <div class="highlight-python"><div class="highlight"><pre><span class="k">class</span> <span class="nc">Parent</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  273. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;left&#39;</span>
  274. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  275. <span class="n">children</span> <span class="o">=</span> <span class="n">relationship</span><span class="p">(</span><span class="s">&quot;Child&quot;</span><span class="p">,</span>
  276. <span class="n">secondary</span><span class="o">=</span><span class="s">&quot;association&quot;</span><span class="p">,</span>
  277. <span class="n">backref</span><span class="o">=</span><span class="s">&quot;parents&quot;</span><span class="p">)</span></pre></div>
  278. </div>
  279. </div>
  280. <div class="section" id="association-object">
  281. <span id="association-pattern"></span><h3>Association Object<a class="headerlink" href="#association-object" title="Permalink to this headline">ś</a></h3>
  282. <p>The association object pattern is a variant on many-to-many: it&#8217;s
  283. used when your association table contains additional columns beyond those
  284. which are foreign keys to the left and right tables. Instead of using the
  285. <tt class="docutils literal"><span class="pre">secondary</span></tt> argument, you map a new class directly to the association table.
  286. The left side of the relationship references the association object via
  287. one-to-many, and the association class references the right side via
  288. many-to-one. Below we illustrate an association table mapped to the
  289. <tt class="docutils literal"><span class="pre">Association</span></tt> class which includes a column called <tt class="docutils literal"><span class="pre">extra_data</span></tt>,
  290. which is a string value that is stored along with each association
  291. between <tt class="docutils literal"><span class="pre">Parent</span></tt> and <tt class="docutils literal"><span class="pre">Child</span></tt>:</p>
  292. <div class="highlight-python"><div class="highlight"><pre><span class="k">class</span> <span class="nc">Association</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  293. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;association&#39;</span>
  294. <span class="n">left_id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">(</span><span class="s">&#39;left.id&#39;</span><span class="p">),</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  295. <span class="n">right_id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">(</span><span class="s">&#39;right.id&#39;</span><span class="p">),</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  296. <span class="n">extra_data</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">String</span><span class="p">(</span><span class="mi">50</span><span class="p">))</span>
  297. <span class="n">child</span> <span class="o">=</span> <span class="n">relationship</span><span class="p">(</span><span class="s">&quot;Child&quot;</span><span class="p">)</span>
  298. <span class="k">class</span> <span class="nc">Parent</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  299. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;left&#39;</span>
  300. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  301. <span class="n">children</span> <span class="o">=</span> <span class="n">relationship</span><span class="p">(</span><span class="s">&quot;Association&quot;</span><span class="p">)</span>
  302. <span class="k">class</span> <span class="nc">Child</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  303. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;right&#39;</span>
  304. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span></pre></div>
  305. </div>
  306. <p>The bidirectional version adds backrefs to both relationships:</p>
  307. <div class="highlight-python"><div class="highlight"><pre><span class="k">class</span> <span class="nc">Association</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  308. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;association&#39;</span>
  309. <span class="n">left_id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">(</span><span class="s">&#39;left.id&#39;</span><span class="p">),</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  310. <span class="n">right_id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">(</span><span class="s">&#39;right.id&#39;</span><span class="p">),</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  311. <span class="n">extra_data</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">String</span><span class="p">(</span><span class="mi">50</span><span class="p">))</span>
  312. <span class="n">child</span> <span class="o">=</span> <span class="n">relationship</span><span class="p">(</span><span class="s">&quot;Child&quot;</span><span class="p">,</span> <span class="n">backref</span><span class="o">=</span><span class="s">&quot;parent_assocs&quot;</span><span class="p">)</span>
  313. <span class="k">class</span> <span class="nc">Parent</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  314. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;left&#39;</span>
  315. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  316. <span class="n">children</span> <span class="o">=</span> <span class="n">relationship</span><span class="p">(</span><span class="s">&quot;Association&quot;</span><span class="p">,</span> <span class="n">backref</span><span class="o">=</span><span class="s">&quot;parent&quot;</span><span class="p">)</span>
  317. <span class="k">class</span> <span class="nc">Child</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  318. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;right&#39;</span>
  319. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span></pre></div>
  320. </div>
  321. <p>Working with the association pattern in its direct form requires that child
  322. objects are associated with an association instance before being appended to
  323. the parent; similarly, access from parent to child goes through the
  324. association object:</p>
  325. <div class="highlight-python"><div class="highlight"><pre><span class="c"># create parent, append a child via association</span>
  326. <span class="n">p</span> <span class="o">=</span> <span class="n">Parent</span><span class="p">()</span>
  327. <span class="n">a</span> <span class="o">=</span> <span class="n">Association</span><span class="p">(</span><span class="n">extra_data</span><span class="o">=</span><span class="s">&quot;some data&quot;</span><span class="p">)</span>
  328. <span class="n">a</span><span class="o">.</span><span class="n">child</span> <span class="o">=</span> <span class="n">Child</span><span class="p">()</span>
  329. <span class="n">p</span><span class="o">.</span><span class="n">children</span><span class="o">.</span><span class="n">append</span><span class="p">(</span><span class="n">a</span><span class="p">)</span>
  330. <span class="c"># iterate through child objects via association, including association</span>
  331. <span class="c"># attributes</span>
  332. <span class="k">for</span> <span class="n">assoc</span> <span class="ow">in</span> <span class="n">p</span><span class="o">.</span><span class="n">children</span><span class="p">:</span>
  333. <span class="k">print</span> <span class="n">assoc</span><span class="o">.</span><span class="n">extra_data</span>
  334. <span class="k">print</span> <span class="n">assoc</span><span class="o">.</span><span class="n">child</span></pre></div>
  335. </div>
  336. <p>To enhance the association object pattern such that direct
  337. access to the <tt class="docutils literal"><span class="pre">Association</span></tt> object is optional, SQLAlchemy
  338. provides the <a class="reference internal" href="extensions/associationproxy.html"><em>Association Proxy</em></a> extension. This
  339. extension allows the configuration of attributes which will
  340. access two &#8220;hops&#8221; with a single access, one &#8220;hop&#8221; to the
  341. associated object, and a second to a target attribute.</p>
  342. <div class="admonition note">
  343. <p class="first admonition-title">Note</p>
  344. <p class="last">When using the association object pattern, it is
  345. advisable that the association-mapped table not be used
  346. as the <tt class="docutils literal"><span class="pre">secondary</span></tt> argument on a <a class="reference internal" href="#sqlalchemy.orm.relationship" title="sqlalchemy.orm.relationship"><tt class="xref py py-func docutils literal"><span class="pre">relationship()</span></tt></a>
  347. elsewhere, unless that <a class="reference internal" href="#sqlalchemy.orm.relationship" title="sqlalchemy.orm.relationship"><tt class="xref py py-func docutils literal"><span class="pre">relationship()</span></tt></a> contains
  348. the option <tt class="docutils literal"><span class="pre">viewonly=True</span></tt>. SQLAlchemy otherwise
  349. may attempt to emit redundant INSERT and DELETE
  350. statements on the same table, if similar state is detected
  351. on the related attribute as well as the associated
  352. object.</p>
  353. </div>
  354. </div>
  355. </div>
  356. <div class="section" id="adjacency-list-relationships">
  357. <h2>Adjacency List Relationships<a class="headerlink" href="#adjacency-list-relationships" title="Permalink to this headline">ś</a></h2>
  358. <p>The <strong>adjacency list</strong> pattern is a common relational pattern whereby a table
  359. contains a foreign key reference to itself. This is the most common
  360. way to represent hierarchical data in flat tables. Other methods
  361. include <strong>nested sets</strong>, sometimes called &#8220;modified preorder&#8221;,
  362. as well as <strong>materialized path</strong>. Despite the appeal that modified preorder
  363. has when evaluated for its fluency within SQL queries, the adjacency list model is
  364. probably the most appropriate pattern for the large majority of hierarchical
  365. storage needs, for reasons of concurrency, reduced complexity, and that
  366. modified preorder has little advantage over an application which can fully
  367. load subtrees into the application space.</p>
  368. <p>In this example, we&#8217;ll work with a single mapped
  369. class called <tt class="docutils literal"><span class="pre">Node</span></tt>, representing a tree structure:</p>
  370. <div class="highlight-python"><div class="highlight"><pre><span class="k">class</span> <span class="nc">Node</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  371. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;node&#39;</span>
  372. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  373. <span class="n">parent_id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">(</span><span class="s">&#39;node.id&#39;</span><span class="p">))</span>
  374. <span class="n">data</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">String</span><span class="p">(</span><span class="mi">50</span><span class="p">))</span>
  375. <span class="n">children</span> <span class="o">=</span> <span class="n">relationship</span><span class="p">(</span><span class="s">&quot;Node&quot;</span><span class="p">)</span></pre></div>
  376. </div>
  377. <p>With this structure, a graph such as the following:</p>
  378. <div class="highlight-python"><pre>root --+---&gt; child1
  379. +---&gt; child2 --+--&gt; subchild1
  380. | +--&gt; subchild2
  381. +---&gt; child3</pre>
  382. </div>
  383. <p>Would be represented with data such as:</p>
  384. <div class="highlight-python"><pre>id parent_id data
  385. --- ------- ----
  386. 1 NULL root
  387. 2 1 child1
  388. 3 1 child2
  389. 4 3 subchild1
  390. 5 3 subchild2
  391. 6 1 child3</pre>
  392. </div>
  393. <p>The <a class="reference internal" href="#sqlalchemy.orm.relationship" title="sqlalchemy.orm.relationship"><tt class="xref py py-func docutils literal"><span class="pre">relationship()</span></tt></a> configuration here works in the
  394. same way as a &#8220;normal&#8221; one-to-many relationship, with the
  395. exception that the &#8220;direction&#8221;, i.e. whether the relationship
  396. is one-to-many or many-to-one, is assumed by default to
  397. be one-to-many. To establish the relationship as many-to-one,
  398. an extra directive is added known as <tt class="docutils literal"><span class="pre">remote_side</span></tt>, which
  399. is a <a class="reference internal" href="../core/schema.html#sqlalchemy.schema.Column" title="sqlalchemy.schema.Column"><tt class="xref py py-class docutils literal"><span class="pre">Column</span></tt></a> or collection of <a class="reference internal" href="../core/schema.html#sqlalchemy.schema.Column" title="sqlalchemy.schema.Column"><tt class="xref py py-class docutils literal"><span class="pre">Column</span></tt></a> objects
  400. that indicate those which should be considered to be &#8220;remote&#8221;:</p>
  401. <div class="highlight-python"><div class="highlight"><pre><span class="k">class</span> <span class="nc">Node</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  402. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;node&#39;</span>
  403. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  404. <span class="n">parent_id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">(</span><span class="s">&#39;node.id&#39;</span><span class="p">))</span>
  405. <span class="n">data</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">String</span><span class="p">(</span><span class="mi">50</span><span class="p">))</span>
  406. <span class="n">parent</span> <span class="o">=</span> <span class="n">relationship</span><span class="p">(</span><span class="s">&quot;Node&quot;</span><span class="p">,</span> <span class="n">remote_side</span><span class="o">=</span><span class="p">[</span><span class="nb">id</span><span class="p">])</span></pre></div>
  407. </div>
  408. <p>Where above, the <tt class="docutils literal"><span class="pre">id</span></tt> column is applied as the <tt class="docutils literal"><span class="pre">remote_side</span></tt>
  409. of the <tt class="docutils literal"><span class="pre">parent</span></tt> <a class="reference internal" href="#sqlalchemy.orm.relationship" title="sqlalchemy.orm.relationship"><tt class="xref py py-func docutils literal"><span class="pre">relationship()</span></tt></a>, thus establishing
  410. <tt class="docutils literal"><span class="pre">parent_id</span></tt> as the &#8220;local&#8221; side, and the relationship
  411. then behaves as a many-to-one.</p>
  412. <p>As always, both directions can be combined into a bidirectional
  413. relationship using the <a class="reference internal" href="#sqlalchemy.orm.backref" title="sqlalchemy.orm.backref"><tt class="xref py py-func docutils literal"><span class="pre">backref()</span></tt></a> function:</p>
  414. <div class="highlight-python"><div class="highlight"><pre><span class="k">class</span> <span class="nc">Node</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  415. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;node&#39;</span>
  416. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  417. <span class="n">parent_id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">(</span><span class="s">&#39;node.id&#39;</span><span class="p">))</span>
  418. <span class="n">data</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">String</span><span class="p">(</span><span class="mi">50</span><span class="p">))</span>
  419. <span class="n">children</span> <span class="o">=</span> <span class="n">relationship</span><span class="p">(</span><span class="s">&quot;Node&quot;</span><span class="p">,</span>
  420. <span class="n">backref</span><span class="o">=</span><span class="n">backref</span><span class="p">(</span><span class="s">&#39;parent&#39;</span><span class="p">,</span> <span class="n">remote_side</span><span class="o">=</span><span class="p">[</span><span class="nb">id</span><span class="p">])</span>
  421. <span class="p">)</span></pre></div>
  422. </div>
  423. <p>There are several examples included with SQLAlchemy illustrating
  424. self-referential strategies; these include <a class="reference internal" href="examples.html#examples-adjacencylist"><em>Adjacency List</em></a> and
  425. <a class="reference internal" href="examples.html#examples-xmlpersistence"><em>XML Persistence</em></a>.</p>
  426. <div class="section" id="self-referential-query-strategies">
  427. <h3>Self-Referential Query Strategies<a class="headerlink" href="#self-referential-query-strategies" title="Permalink to this headline">ś</a></h3>
  428. <p>Querying of self-referential structures works like any other query:</p>
  429. <div class="highlight-python"><div class="highlight"><pre><span class="c"># get all nodes named &#39;child2&#39;</span>
  430. <span class="n">session</span><span class="o">.</span><span class="n">query</span><span class="p">(</span><span class="n">Node</span><span class="p">)</span><span class="o">.</span><span class="n">filter</span><span class="p">(</span><span class="n">Node</span><span class="o">.</span><span class="n">data</span><span class="o">==</span><span class="s">&#39;child2&#39;</span><span class="p">)</span></pre></div>
  431. </div>
  432. <p>However extra care is needed when attempting to join along
  433. the foreign key from one level of the tree to the next. In SQL,
  434. a join from a table to itself requires that at least one side of the
  435. expression be &#8220;aliased&#8221; so that it can be unambiguously referred to.</p>
  436. <p>Recall from <a class="reference internal" href="tutorial.html#ormtutorial-aliases"><em>Using Aliases</em></a> in the ORM tutorial that the
  437. <a class="reference internal" href="query.html#sqlalchemy.orm.aliased" title="sqlalchemy.orm.aliased"><tt class="xref py py-class docutils literal"><span class="pre">orm.aliased</span></tt></a> construct is normally used to provide an &#8220;alias&#8221; of
  438. an ORM entity. Joining from <tt class="docutils literal"><span class="pre">Node</span></tt> to itself using this technique
  439. looks like:</p>
  440. <div class="highlight-python+sql"><div class="highlight"><pre><span class="kn">from</span> <span class="nn">sqlalchemy.orm</span> <span class="kn">import</span> <span class="n">aliased</span>
  441. <span class="n">nodealias</span> <span class="o">=</span> <span class="n">aliased</span><span class="p">(</span><span class="n">Node</span><span class="p">)</span>
  442. <a href='#' class='sql_link'>sql</a><span class="n">session</span><span class="o">.</span><span class="n">query</span><span class="p">(</span><span class="n">Node</span><span class="p">)</span><span class="o">.</span><span class="n">filter</span><span class="p">(</span><span class="n">Node</span><span class="o">.</span><span class="n">data</span><span class="o">==</span><span class="s">&#39;subchild1&#39;</span><span class="p">)</span><span class="o">.</span>\
  443. <span class="n">join</span><span class="p">(</span><span class="n">nodealias</span><span class="p">,</span> <span class="n">Node</span><span class="o">.</span><span class="n">parent</span><span class="p">)</span><span class="o">.</span>\
  444. <span class="nb">filter</span><span class="p">(</span><span class="n">nodealias</span><span class="o">.</span><span class="n">data</span><span class="o">==</span><span class="s">&quot;child2&quot;</span><span class="p">)</span><span class="o">.</span>\
  445. <span class="nb">all</span><span class="p">()</span>
  446. <div class='popup_sql'>SELECT node.id AS node_id,
  447. node.parent_id AS node_parent_id,
  448. node.data AS node_data
  449. FROM node JOIN node AS node_1
  450. ON node.parent_id = node_1.id
  451. WHERE node.data = ?
  452. AND node_1.data = ?
  453. ['subchild1', 'child2']</div></pre></div>
  454. </div>
  455. <p><a class="reference internal" href="query.html#sqlalchemy.orm.query.Query.join" title="sqlalchemy.orm.query.Query.join"><tt class="xref py py-meth docutils literal"><span class="pre">Query.join()</span></tt></a> also includes a feature known as <tt class="docutils literal"><span class="pre">aliased=True</span></tt> that
  456. can shorten the verbosity self-referential joins, at the expense
  457. of query flexibility. This feature
  458. performs a similar &#8220;aliasing&#8221; step to that above, without the need for an
  459. explicit entity. Calls to <a class="reference internal" href="query.html#sqlalchemy.orm.query.Query.filter" title="sqlalchemy.orm.query.Query.filter"><tt class="xref py py-meth docutils literal"><span class="pre">Query.filter()</span></tt></a> and similar subsequent to
  460. the aliased join will <strong>adapt</strong> the <tt class="docutils literal"><span class="pre">Node</span></tt> entity to be that of the alias:</p>
  461. <div class="highlight-python+sql"><div class="highlight"><pre><a href='#' class='sql_link'>sql</a><span class="n">session</span><span class="o">.</span><span class="n">query</span><span class="p">(</span><span class="n">Node</span><span class="p">)</span><span class="o">.</span><span class="n">filter</span><span class="p">(</span><span class="n">Node</span><span class="o">.</span><span class="n">data</span><span class="o">==</span><span class="s">&#39;subchild1&#39;</span><span class="p">)</span><span class="o">.</span>\
  462. <span class="n">join</span><span class="p">(</span><span class="n">Node</span><span class="o">.</span><span class="n">parent</span><span class="p">,</span> <span class="n">aliased</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span><span class="o">.</span>\
  463. <span class="nb">filter</span><span class="p">(</span><span class="n">Node</span><span class="o">.</span><span class="n">data</span><span class="o">==</span><span class="s">&#39;child2&#39;</span><span class="p">)</span><span class="o">.</span>\
  464. <span class="nb">all</span><span class="p">()</span>
  465. <div class='popup_sql'>SELECT node.id AS node_id,
  466. node.parent_id AS node_parent_id,
  467. node.data AS node_data
  468. FROM node
  469. JOIN node AS node_1 ON node_1.id = node.parent_id
  470. WHERE node.data = ? AND node_1.data = ?
  471. ['subchild1', 'child2']</div></pre></div>
  472. </div>
  473. <p>To add criterion to multiple points along a longer join, add <tt class="docutils literal"><span class="pre">from_joinpoint=True</span></tt>
  474. to the additional <a class="reference internal" href="query.html#sqlalchemy.orm.query.Query.join" title="sqlalchemy.orm.query.Query.join"><tt class="xref py py-meth docutils literal"><span class="pre">join()</span></tt></a> calls:</p>
  475. <div class="highlight-python+sql"><div class="highlight"><pre><span class="c"># get all nodes named &#39;subchild1&#39; with a</span>
  476. <span class="c"># parent named &#39;child2&#39; and a grandparent &#39;root&#39;</span>
  477. <a href='#' class='sql_link'>sql</a><span class="n">session</span><span class="o">.</span><span class="n">query</span><span class="p">(</span><span class="n">Node</span><span class="p">)</span><span class="o">.</span>\
  478. <span class="nb">filter</span><span class="p">(</span><span class="n">Node</span><span class="o">.</span><span class="n">data</span><span class="o">==</span><span class="s">&#39;subchild1&#39;</span><span class="p">)</span><span class="o">.</span>\
  479. <span class="n">join</span><span class="p">(</span><span class="n">Node</span><span class="o">.</span><span class="n">parent</span><span class="p">,</span> <span class="n">aliased</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span><span class="o">.</span>\
  480. <span class="nb">filter</span><span class="p">(</span><span class="n">Node</span><span class="o">.</span><span class="n">data</span><span class="o">==</span><span class="s">&#39;child2&#39;</span><span class="p">)</span><span class="o">.</span>\
  481. <span class="n">join</span><span class="p">(</span><span class="n">Node</span><span class="o">.</span><span class="n">parent</span><span class="p">,</span> <span class="n">aliased</span><span class="o">=</span><span class="bp">True</span><span class="p">,</span> <span class="n">from_joinpoint</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span><span class="o">.</span>\
  482. <span class="nb">filter</span><span class="p">(</span><span class="n">Node</span><span class="o">.</span><span class="n">data</span><span class="o">==</span><span class="s">&#39;root&#39;</span><span class="p">)</span><span class="o">.</span>\
  483. <span class="nb">all</span><span class="p">()</span>
  484. <div class='popup_sql'>SELECT node.id AS node_id,
  485. node.parent_id AS node_parent_id,
  486. node.data AS node_data
  487. FROM node
  488. JOIN node AS node_1 ON node_1.id = node.parent_id
  489. JOIN node AS node_2 ON node_2.id = node_1.parent_id
  490. WHERE node.data = ?
  491. AND node_1.data = ?
  492. AND node_2.data = ?
  493. ['subchild1', 'child2', 'root']</div></pre></div>
  494. </div>
  495. <p><a class="reference internal" href="query.html#sqlalchemy.orm.query.Query.reset_joinpoint" title="sqlalchemy.orm.query.Query.reset_joinpoint"><tt class="xref py py-meth docutils literal"><span class="pre">Query.reset_joinpoint()</span></tt></a> will also remove the &#8220;aliasing&#8221; from filtering
  496. calls:</p>
  497. <div class="highlight-python"><div class="highlight"><pre><span class="n">session</span><span class="o">.</span><span class="n">query</span><span class="p">(</span><span class="n">Node</span><span class="p">)</span><span class="o">.</span>\
  498. <span class="n">join</span><span class="p">(</span><span class="n">Node</span><span class="o">.</span><span class="n">children</span><span class="p">,</span> <span class="n">aliased</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span><span class="o">.</span>\
  499. <span class="nb">filter</span><span class="p">(</span><span class="n">Node</span><span class="o">.</span><span class="n">data</span> <span class="o">==</span> <span class="s">&#39;foo&#39;</span><span class="p">)</span><span class="o">.</span>\
  500. <span class="n">reset_joinpoint</span><span class="p">()</span><span class="o">.</span>\
  501. <span class="nb">filter</span><span class="p">(</span><span class="n">Node</span><span class="o">.</span><span class="n">data</span> <span class="o">==</span> <span class="s">&#39;bar&#39;</span><span class="p">)</span></pre></div>
  502. </div>
  503. <p>For an example of using <tt class="docutils literal"><span class="pre">aliased=True</span></tt> to arbitrarily join along a chain of self-referential
  504. nodes, see <a class="reference internal" href="examples.html#examples-xmlpersistence"><em>XML Persistence</em></a>.</p>
  505. </div>
  506. <div class="section" id="configuring-self-referential-eager-loading">
  507. <h3>Configuring Self-Referential Eager Loading<a class="headerlink" href="#configuring-self-referential-eager-loading" title="Permalink to this headline">ś</a></h3>
  508. <p>Eager loading of relationships occurs using joins or outerjoins from parent to
  509. child table during a normal query operation, such that the parent and its
  510. immediate child collection or reference can be populated from a single SQL
  511. statement, or a second statement for all immediate child collections.
  512. SQLAlchemy&#8217;s joined and subquery eager loading use aliased tables in all cases
  513. when joining to related items, so are compatible with self-referential
  514. joining. However, to use eager loading with a self-referential relationship,
  515. SQLAlchemy needs to be told how many levels deep it should join and/or query;
  516. otherwise the eager load will not take place at all. This depth setting is
  517. configured via <tt class="docutils literal"><span class="pre">join_depth</span></tt>:</p>
  518. <div class="highlight-python+sql"><div class="highlight"><pre><span class="k">class</span> <span class="nc">Node</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  519. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;node&#39;</span>
  520. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  521. <span class="n">parent_id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">(</span><span class="s">&#39;node.id&#39;</span><span class="p">))</span>
  522. <span class="n">data</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">String</span><span class="p">(</span><span class="mi">50</span><span class="p">))</span>
  523. <span class="n">children</span> <span class="o">=</span> <span class="n">relationship</span><span class="p">(</span><span class="s">&quot;Node&quot;</span><span class="p">,</span>
  524. <span class="n">lazy</span><span class="o">=</span><span class="s">&quot;joined&quot;</span><span class="p">,</span>
  525. <span class="n">join_depth</span><span class="o">=</span><span class="mi">2</span><span class="p">)</span>
  526. <a href='#' class='sql_link'>sql</a><span class="n">session</span><span class="o">.</span><span class="n">query</span><span class="p">(</span><span class="n">Node</span><span class="p">)</span><span class="o">.</span><span class="n">all</span><span class="p">()</span>
  527. <div class='popup_sql'>SELECT node_1.id AS node_1_id,
  528. node_1.parent_id AS node_1_parent_id,
  529. node_1.data AS node_1_data,
  530. node_2.id AS node_2_id,
  531. node_2.parent_id AS node_2_parent_id,
  532. node_2.data AS node_2_data,
  533. node.id AS node_id,
  534. node.parent_id AS node_parent_id,
  535. node.data AS node_data
  536. FROM node
  537. LEFT OUTER JOIN node AS node_2
  538. ON node.id = node_2.parent_id
  539. LEFT OUTER JOIN node AS node_1
  540. ON node_2.id = node_1.parent_id
  541. []</div></pre></div>
  542. </div>
  543. </div>
  544. </div>
  545. <div class="section" id="linking-relationships-with-backref">
  546. <span id="relationships-backref"></span><h2>Linking Relationships with Backref<a class="headerlink" href="#linking-relationships-with-backref" title="Permalink to this headline">ś</a></h2>
  547. <p>The <tt class="docutils literal"><span class="pre">backref</span></tt> keyword argument was first introduced in <a class="reference internal" href="tutorial.html"><em>Object Relational Tutorial</em></a>, and has been
  548. mentioned throughout many of the examples here. What does it actually do ? Let&#8217;s start
  549. with the canonical <tt class="docutils literal"><span class="pre">User</span></tt> and <tt class="docutils literal"><span class="pre">Address</span></tt> scenario:</p>
  550. <div class="highlight-python"><div class="highlight"><pre><span class="kn">from</span> <span class="nn">sqlalchemy</span> <span class="kn">import</span> <span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">,</span> <span class="n">String</span><span class="p">,</span> <span class="n">Column</span>
  551. <span class="kn">from</span> <span class="nn">sqlalchemy.ext.declarative</span> <span class="kn">import</span> <span class="n">declarative_base</span>
  552. <span class="kn">from</span> <span class="nn">sqlalchemy.orm</span> <span class="kn">import</span> <span class="n">relationship</span>
  553. <span class="n">Base</span> <span class="o">=</span> <span class="n">declarative_base</span><span class="p">()</span>
  554. <span class="k">class</span> <span class="nc">User</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  555. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;user&#39;</span>
  556. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  557. <span class="n">name</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">String</span><span class="p">)</span>
  558. <span class="n">addresses</span> <span class="o">=</span> <span class="n">relationship</span><span class="p">(</span><span class="s">&quot;Address&quot;</span><span class="p">,</span> <span class="n">backref</span><span class="o">=</span><span class="s">&quot;user&quot;</span><span class="p">)</span>
  559. <span class="k">class</span> <span class="nc">Address</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  560. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;address&#39;</span>
  561. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  562. <span class="n">email</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">String</span><span class="p">)</span>
  563. <span class="n">user_id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">(</span><span class="s">&#39;user.id&#39;</span><span class="p">))</span></pre></div>
  564. </div>
  565. <p>The above configuration establishes a collection of <tt class="docutils literal"><span class="pre">Address</span></tt> objects on <tt class="docutils literal"><span class="pre">User</span></tt> called
  566. <tt class="docutils literal"><span class="pre">User.addresses</span></tt>. It also establishes a <tt class="docutils literal"><span class="pre">.user</span></tt> attribute on <tt class="docutils literal"><span class="pre">Address</span></tt> which will
  567. refer to the parent <tt class="docutils literal"><span class="pre">User</span></tt> object.</p>
  568. <p>In fact, the <tt class="docutils literal"><span class="pre">backref</span></tt> keyword is only a common shortcut for placing a second
  569. <tt class="docutils literal"><span class="pre">relationship</span></tt> onto the <tt class="docutils literal"><span class="pre">Address</span></tt> mapping, including the establishment
  570. of an event listener on both sides which will mirror attribute operations
  571. in both directions. The above configuration is equivalent to:</p>
  572. <div class="highlight-python"><div class="highlight"><pre><span class="kn">from</span> <span class="nn">sqlalchemy</span> <span class="kn">import</span> <span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">,</span> <span class="n">String</span><span class="p">,</span> <span class="n">Column</span>
  573. <span class="kn">from</span> <span class="nn">sqlalchemy.ext.declarative</span> <span class="kn">import</span> <span class="n">declarative_base</span>
  574. <span class="kn">from</span> <span class="nn">sqlalchemy.orm</span> <span class="kn">import</span> <span class="n">relationship</span>
  575. <span class="n">Base</span> <span class="o">=</span> <span class="n">declarative_base</span><span class="p">()</span>
  576. <span class="k">class</span> <span class="nc">User</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  577. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;user&#39;</span>
  578. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  579. <span class="n">name</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">String</span><span class="p">)</span>
  580. <span class="n">addresses</span> <span class="o">=</span> <span class="n">relationship</span><span class="p">(</span><span class="s">&quot;Address&quot;</span><span class="p">,</span> <span class="n">back_populates</span><span class="o">=</span><span class="s">&quot;user&quot;</span><span class="p">)</span>
  581. <span class="k">class</span> <span class="nc">Address</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  582. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;address&#39;</span>
  583. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  584. <span class="n">email</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">String</span><span class="p">)</span>
  585. <span class="n">user_id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">(</span><span class="s">&#39;user.id&#39;</span><span class="p">))</span>
  586. <span class="n">user</span> <span class="o">=</span> <span class="n">relationship</span><span class="p">(</span><span class="s">&quot;User&quot;</span><span class="p">,</span> <span class="n">back_populates</span><span class="o">=</span><span class="s">&quot;addresses&quot;</span><span class="p">)</span></pre></div>
  587. </div>
  588. <p>Above, we add a <tt class="docutils literal"><span class="pre">.user</span></tt> relationship to <tt class="docutils literal"><span class="pre">Address</span></tt> explicitly. On
  589. both relationships, the <tt class="docutils literal"><span class="pre">back_populates</span></tt> directive tells each relationship
  590. about the other one, indicating that they should establish &#8220;bidirectional&#8221;
  591. behavior between each other. The primary effect of this configuration
  592. is that the relationship adds event handlers to both attributes
  593. which have the behavior of &#8220;when an append or set event occurs here, set ourselves
  594. onto the incoming attribute using this particular attribute name&#8221;.
  595. The behavior is illustrated as follows. Start with a <tt class="docutils literal"><span class="pre">User</span></tt> and an <tt class="docutils literal"><span class="pre">Address</span></tt>
  596. instance. The <tt class="docutils literal"><span class="pre">.addresses</span></tt> collection is empty, and the <tt class="docutils literal"><span class="pre">.user</span></tt> attribute
  597. is <tt class="docutils literal"><span class="pre">None</span></tt>:</p>
  598. <div class="highlight-python"><div class="highlight"><pre><span class="gp">&gt;&gt;&gt; </span><span class="n">u1</span> <span class="o">=</span> <span class="n">User</span><span class="p">()</span>
  599. <span class="gp">&gt;&gt;&gt; </span><span class="n">a1</span> <span class="o">=</span> <span class="n">Address</span><span class="p">()</span>
  600. <span class="gp">&gt;&gt;&gt; </span><span class="n">u1</span><span class="o">.</span><span class="n">addresses</span>
  601. <span class="go">[]</span>
  602. <span class="gp">&gt;&gt;&gt; </span><span class="k">print</span> <span class="n">a1</span><span class="o">.</span><span class="n">user</span>
  603. <span class="go">None</span></pre></div>
  604. </div>
  605. <p>However, once the <tt class="docutils literal"><span class="pre">Address</span></tt> is appended to the <tt class="docutils literal"><span class="pre">u1.addresses</span></tt> collection,
  606. both the collection and the scalar attribute have been populated:</p>
  607. <div class="highlight-python"><div class="highlight"><pre><span class="gp">&gt;&gt;&gt; </span><span class="n">u1</span><span class="o">.</span><span class="n">addresses</span><span class="o">.</span><span class="n">append</span><span class="p">(</span><span class="n">a1</span><span class="p">)</span>
  608. <span class="gp">&gt;&gt;&gt; </span><span class="n">u1</span><span class="o">.</span><span class="n">addresses</span>
  609. <span class="go">[&lt;__main__.Address object at 0x12a6ed0&gt;]</span>
  610. <span class="gp">&gt;&gt;&gt; </span><span class="n">a1</span><span class="o">.</span><span class="n">user</span>
  611. <span class="go">&lt;__main__.User object at 0x12a6590&gt;</span></pre></div>
  612. </div>
  613. <p>This behavior of course works in reverse for removal operations as well, as well
  614. as for equivalent operations on both sides. Such as
  615. when <tt class="docutils literal"><span class="pre">.user</span></tt> is set again to <tt class="docutils literal"><span class="pre">None</span></tt>, the <tt class="docutils literal"><span class="pre">Address</span></tt> object is removed
  616. from the reverse collection:</p>
  617. <div class="highlight-python"><div class="highlight"><pre><span class="gp">&gt;&gt;&gt; </span><span class="n">a1</span><span class="o">.</span><span class="n">user</span> <span class="o">=</span> <span class="bp">None</span>
  618. <span class="gp">&gt;&gt;&gt; </span><span class="n">u1</span><span class="o">.</span><span class="n">addresses</span>
  619. <span class="go">[]</span></pre></div>
  620. </div>
  621. <p>The manipulation of the <tt class="docutils literal"><span class="pre">.addresses</span></tt> collection and the <tt class="docutils literal"><span class="pre">.user</span></tt> attribute
  622. occurs entirely in Python without any interaction with the SQL database.
  623. Without this behavior, the proper state would be apparent on both sides once the
  624. data has been flushed to the database, and later reloaded after a commit or
  625. expiration operation occurs. The <tt class="docutils literal"><span class="pre">backref</span></tt>/<tt class="docutils literal"><span class="pre">back_populates</span></tt> behavior has the advantage
  626. that common bidirectional operations can reflect the correct state without requiring
  627. a database round trip.</p>
  628. <p>Remember, when the <tt class="docutils literal"><span class="pre">backref</span></tt> keyword is used on a single relationship, it&#8217;s
  629. exactly the same as if the above two relationships were created individually
  630. using <tt class="docutils literal"><span class="pre">back_populates</span></tt> on each.</p>
  631. <div class="section" id="backref-arguments">
  632. <h3>Backref Arguments<a class="headerlink" href="#backref-arguments" title="Permalink to this headline">ś</a></h3>
  633. <p>We&#8217;ve established that the <tt class="docutils literal"><span class="pre">backref</span></tt> keyword is merely a shortcut for building
  634. two individual <a class="reference internal" href="#sqlalchemy.orm.relationship" title="sqlalchemy.orm.relationship"><tt class="xref py py-func docutils literal"><span class="pre">relationship()</span></tt></a> constructs that refer to each other. Part of
  635. the behavior of this shortcut is that certain configurational arguments applied to
  636. the <a class="reference internal" href="#sqlalchemy.orm.relationship" title="sqlalchemy.orm.relationship"><tt class="xref py py-func docutils literal"><span class="pre">relationship()</span></tt></a>
  637. will also be applied to the other direction - namely those arguments that describe
  638. the relationship at a schema level, and are unlikely to be different in the reverse
  639. direction. The usual case
  640. here is a many-to-many <a class="reference internal" href="#sqlalchemy.orm.relationship" title="sqlalchemy.orm.relationship"><tt class="xref py py-func docutils literal"><span class="pre">relationship()</span></tt></a> that has a <tt class="docutils literal"><span class="pre">secondary</span></tt> argument,
  641. or a one-to-many or many-to-one which has a <tt class="docutils literal"><span class="pre">primaryjoin</span></tt> argument (the
  642. <tt class="docutils literal"><span class="pre">primaryjoin</span></tt> argument is discussed in <a class="reference internal" href="#relationship-primaryjoin"><em>Setting the primaryjoin and secondaryjoin</em></a>). Such
  643. as if we limited the list of <tt class="docutils literal"><span class="pre">Address</span></tt> objects to those which start with &#8220;tony&#8221;:</p>
  644. <div class="highlight-python"><div class="highlight"><pre><span class="kn">from</span> <span class="nn">sqlalchemy</span> <span class="kn">import</span> <span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">,</span> <span class="n">String</span><span class="p">,</span> <span class="n">Column</span>
  645. <span class="kn">from</span> <span class="nn">sqlalchemy.ext.declarative</span> <span class="kn">import</span> <span class="n">declarative_base</span>
  646. <span class="kn">from</span> <span class="nn">sqlalchemy.orm</span> <span class="kn">import</span> <span class="n">relationship</span>
  647. <span class="n">Base</span> <span class="o">=</span> <span class="n">declarative_base</span><span class="p">()</span>
  648. <span class="k">class</span> <span class="nc">User</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  649. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;user&#39;</span>
  650. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  651. <span class="n">name</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">String</span><span class="p">)</span>
  652. <span class="n">addresses</span> <span class="o">=</span> <span class="n">relationship</span><span class="p">(</span><span class="s">&quot;Address&quot;</span><span class="p">,</span>
  653. <span class="n">primaryjoin</span><span class="o">=</span><span class="s">&quot;and_(User.id==Address.user_id, &quot;</span>
  654. <span class="s">&quot;Address.email.startswith(&#39;tony&#39;))&quot;</span><span class="p">,</span>
  655. <span class="n">backref</span><span class="o">=</span><span class="s">&quot;user&quot;</span><span class="p">)</span>
  656. <span class="k">class</span> <span class="nc">Address</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  657. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;address&#39;</span>
  658. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  659. <span class="n">email</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">String</span><span class="p">)</span>
  660. <span class="n">user_id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">(</span><span class="s">&#39;user.id&#39;</span><span class="p">))</span></pre></div>
  661. </div>
  662. <p>We can observe, by inspecting the resulting property, that both sides
  663. of the relationship have this join condition applied:</p>
  664. <div class="highlight-python"><div class="highlight"><pre><span class="gp">&gt;&gt;&gt; </span><span class="k">print</span> <span class="n">User</span><span class="o">.</span><span class="n">addresses</span><span class="o">.</span><span class="n">property</span><span class="o">.</span><span class="n">primaryjoin</span>
  665. <span class="go">&quot;user&quot;.id = address.user_id AND address.email LIKE :email_1 || &#39;%%&#39;</span>
  666. <span class="go">&gt;&gt;&gt;</span>
  667. <span class="gp">&gt;&gt;&gt; </span><span class="k">print</span> <span class="n">Address</span><span class="o">.</span><span class="n">user</span><span class="o">.</span><span class="n">property</span><span class="o">.</span><span class="n">primaryjoin</span>
  668. <span class="go">&quot;user&quot;.id = address.user_id AND address.email LIKE :email_1 || &#39;%%&#39;</span>
  669. <span class="go">&gt;&gt;&gt;</span></pre></div>
  670. </div>
  671. <p>This reuse of arguments should pretty much do the &#8220;right thing&#8221; - it uses
  672. only arguments that are applicable, and in the case of a many-to-many
  673. relationship, will reverse the usage of <tt class="docutils literal"><span class="pre">primaryjoin</span></tt> and <tt class="docutils literal"><span class="pre">secondaryjoin</span></tt>
  674. to correspond to the other direction (see the example in <a class="reference internal" href="#self-referential-many-to-many"><em>Self-Referential Many-to-Many Relationship</em></a>
  675. for this).</p>
  676. <p>It&#8217;s very often the case however that we&#8217;d like to specify arguments that
  677. are specific to just the side where we happened to place the &#8220;backref&#8221;.
  678. This includes <a class="reference internal" href="#sqlalchemy.orm.relationship" title="sqlalchemy.orm.relationship"><tt class="xref py py-func docutils literal"><span class="pre">relationship()</span></tt></a> arguments like <tt class="docutils literal"><span class="pre">lazy</span></tt>, <tt class="docutils literal"><span class="pre">remote_side</span></tt>,
  679. <tt class="docutils literal"><span class="pre">cascade</span></tt> and <tt class="docutils literal"><span class="pre">cascade_backrefs</span></tt>. For this case we use the <a class="reference internal" href="#sqlalchemy.orm.backref" title="sqlalchemy.orm.backref"><tt class="xref py py-func docutils literal"><span class="pre">backref()</span></tt></a>
  680. function in place of a string:</p>
  681. <div class="highlight-python"><div class="highlight"><pre><span class="c"># &lt;other imports&gt;</span>
  682. <span class="kn">from</span> <span class="nn">sqlalchemy.orm</span> <span class="kn">import</span> <span class="n">backref</span>
  683. <span class="k">class</span> <span class="nc">User</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  684. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;user&#39;</span>
  685. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  686. <span class="n">name</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">String</span><span class="p">)</span>
  687. <span class="n">addresses</span> <span class="o">=</span> <span class="n">relationship</span><span class="p">(</span><span class="s">&quot;Address&quot;</span><span class="p">,</span>
  688. <span class="n">backref</span><span class="o">=</span><span class="n">backref</span><span class="p">(</span><span class="s">&quot;user&quot;</span><span class="p">,</span> <span class="n">lazy</span><span class="o">=</span><span class="s">&quot;joined&quot;</span><span class="p">))</span></pre></div>
  689. </div>
  690. <p>Where above, we placed a <tt class="docutils literal"><span class="pre">lazy=&quot;joined&quot;</span></tt> directive only on the <tt class="docutils literal"><span class="pre">Address.user</span></tt>
  691. side, indicating that when a query against <tt class="docutils literal"><span class="pre">Address</span></tt> is made, a join to the <tt class="docutils literal"><span class="pre">User</span></tt>
  692. entity should be made automatically which will populate the <tt class="docutils literal"><span class="pre">.user</span></tt> attribute of each
  693. returned <tt class="docutils literal"><span class="pre">Address</span></tt>. The <a class="reference internal" href="#sqlalchemy.orm.backref" title="sqlalchemy.orm.backref"><tt class="xref py py-func docutils literal"><span class="pre">backref()</span></tt></a> function formatted the arguments we gave
  694. it into a form that is interpreted by the receiving <a class="reference internal" href="#sqlalchemy.orm.relationship" title="sqlalchemy.orm.relationship"><tt class="xref py py-func docutils literal"><span class="pre">relationship()</span></tt></a> as additional
  695. arguments to be applied to the new relationship it creates.</p>
  696. </div>
  697. <div class="section" id="one-way-backrefs">
  698. <h3>One Way Backrefs<a class="headerlink" href="#one-way-backrefs" title="Permalink to this headline">ś</a></h3>
  699. <p>An unusual case is that of the &#8220;one way backref&#8221;. This is where the &#8220;back-populating&#8221;
  700. behavior of the backref is only desirable in one direction. An example of this
  701. is a collection which contains a filtering <tt class="docutils literal"><span class="pre">primaryjoin</span></tt> condition. We&#8217;d like to append
  702. items to this collection as needed, and have them populate the &#8220;parent&#8221; object on the
  703. incoming object. However, we&#8217;d also like to have items that are not part of the collection,
  704. but still have the same &#8220;parent&#8221; association - these items should never be in the
  705. collection.</p>
  706. <p>Taking our previous example, where we established a <tt class="docutils literal"><span class="pre">primaryjoin</span></tt> that limited the
  707. collection only to <tt class="docutils literal"><span class="pre">Address</span></tt> objects whose email address started with the word <tt class="docutils literal"><span class="pre">tony</span></tt>,
  708. the usual backref behavior is that all items populate in both directions. We wouldn&#8217;t
  709. want this behavior for a case like the following:</p>
  710. <div class="highlight-python"><div class="highlight"><pre><span class="gp">&gt;&gt;&gt; </span><span class="n">u1</span> <span class="o">=</span> <span class="n">User</span><span class="p">()</span>
  711. <span class="gp">&gt;&gt;&gt; </span><span class="n">a1</span> <span class="o">=</span> <span class="n">Address</span><span class="p">(</span><span class="n">email</span><span class="o">=</span><span class="s">&#39;mary&#39;</span><span class="p">)</span>
  712. <span class="gp">&gt;&gt;&gt; </span><span class="n">a1</span><span class="o">.</span><span class="n">user</span> <span class="o">=</span> <span class="n">u1</span>
  713. <span class="gp">&gt;&gt;&gt; </span><span class="n">u1</span><span class="o">.</span><span class="n">addresses</span>
  714. <span class="go">[&lt;__main__.Address object at 0x1411910&gt;]</span></pre></div>
  715. </div>
  716. <p>Above, the <tt class="docutils literal"><span class="pre">Address</span></tt> object that doesn&#8217;t match the criterion of &#8220;starts with &#8216;tony&#8217;&#8221;
  717. is present in the <tt class="docutils literal"><span class="pre">addresses</span></tt> collection of <tt class="docutils literal"><span class="pre">u1</span></tt>. After these objects are flushed,
  718. the transaction committed and their attributes expired for a re-load, the <tt class="docutils literal"><span class="pre">addresses</span></tt>
  719. collection will hit the database on next access and no longer have this <tt class="docutils literal"><span class="pre">Address</span></tt> object
  720. present, due to the filtering condition. But we can do away with this unwanted side
  721. of the &#8220;backref&#8221; behavior on the Python side by using two separate <a class="reference internal" href="#sqlalchemy.orm.relationship" title="sqlalchemy.orm.relationship"><tt class="xref py py-func docutils literal"><span class="pre">relationship()</span></tt></a> constructs,
  722. placing <tt class="docutils literal"><span class="pre">back_populates</span></tt> only on one side:</p>
  723. <div class="highlight-python"><div class="highlight"><pre><span class="kn">from</span> <span class="nn">sqlalchemy</span> <span class="kn">import</span> <span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">,</span> <span class="n">String</span><span class="p">,</span> <span class="n">Column</span>
  724. <span class="kn">from</span> <span class="nn">sqlalchemy.ext.declarative</span> <span class="kn">import</span> <span class="n">declarative_base</span>
  725. <span class="kn">from</span> <span class="nn">sqlalchemy.orm</span> <span class="kn">import</span> <span class="n">relationship</span>
  726. <span class="n">Base</span> <span class="o">=</span> <span class="n">declarative_base</span><span class="p">()</span>
  727. <span class="k">class</span> <span class="nc">User</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  728. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;user&#39;</span>
  729. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  730. <span class="n">name</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">String</span><span class="p">)</span>
  731. <span class="n">addresses</span> <span class="o">=</span> <span class="n">relationship</span><span class="p">(</span><span class="s">&quot;Address&quot;</span><span class="p">,</span>
  732. <span class="n">primaryjoin</span><span class="o">=</span><span class="s">&quot;and_(User.id==Address.user_id, &quot;</span>
  733. <span class="s">&quot;Address.email.startswith(&#39;tony&#39;))&quot;</span><span class="p">,</span>
  734. <span class="n">back_populates</span><span class="o">=</span><span class="s">&quot;user&quot;</span><span class="p">)</span>
  735. <span class="k">class</span> <span class="nc">Address</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  736. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;address&#39;</span>
  737. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  738. <span class="n">email</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">String</span><span class="p">)</span>
  739. <span class="n">user_id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">(</span><span class="s">&#39;user.id&#39;</span><span class="p">))</span>
  740. <span class="n">user</span> <span class="o">=</span> <span class="n">relationship</span><span class="p">(</span><span class="s">&quot;User&quot;</span><span class="p">)</span></pre></div>
  741. </div>
  742. <p>With the above scenario, appending an <tt class="docutils literal"><span class="pre">Address</span></tt> object to the <tt class="docutils literal"><span class="pre">.addresses</span></tt>
  743. collection of a <tt class="docutils literal"><span class="pre">User</span></tt> will always establish the <tt class="docutils literal"><span class="pre">.user</span></tt> attribute on that
  744. <tt class="docutils literal"><span class="pre">Address</span></tt>:</p>
  745. <div class="highlight-python"><div class="highlight"><pre><span class="gp">&gt;&gt;&gt; </span><span class="n">u1</span> <span class="o">=</span> <span class="n">User</span><span class="p">()</span>
  746. <span class="gp">&gt;&gt;&gt; </span><span class="n">a1</span> <span class="o">=</span> <span class="n">Address</span><span class="p">(</span><span class="n">email</span><span class="o">=</span><span class="s">&#39;tony&#39;</span><span class="p">)</span>
  747. <span class="gp">&gt;&gt;&gt; </span><span class="n">u1</span><span class="o">.</span><span class="n">addresses</span><span class="o">.</span><span class="n">append</span><span class="p">(</span><span class="n">a1</span><span class="p">)</span>
  748. <span class="gp">&gt;&gt;&gt; </span><span class="n">a1</span><span class="o">.</span><span class="n">user</span>
  749. <span class="go">&lt;__main__.User object at 0x1411850&gt;</span></pre></div>
  750. </div>
  751. <p>However, applying a <tt class="docutils literal"><span class="pre">User</span></tt> to the <tt class="docutils literal"><span class="pre">.user</span></tt> attribute of an <tt class="docutils literal"><span class="pre">Address</span></tt>,
  752. will not append the <tt class="docutils literal"><span class="pre">Address</span></tt> object to the collection:</p>
  753. <div class="highlight-python"><div class="highlight"><pre><span class="gp">&gt;&gt;&gt; </span><span class="n">a2</span> <span class="o">=</span> <span class="n">Address</span><span class="p">(</span><span class="n">email</span><span class="o">=</span><span class="s">&#39;mary&#39;</span><span class="p">)</span>
  754. <span class="gp">&gt;&gt;&gt; </span><span class="n">a2</span><span class="o">.</span><span class="n">user</span> <span class="o">=</span> <span class="n">u1</span>
  755. <span class="gp">&gt;&gt;&gt; </span><span class="n">a2</span> <span class="ow">in</span> <span class="n">u1</span><span class="o">.</span><span class="n">addresses</span>
  756. <span class="go">False</span></pre></div>
  757. </div>
  758. <p>Of course, we&#8217;ve disabled some of the usefulness of <tt class="docutils literal"><span class="pre">backref</span></tt> here, in that
  759. when we do append an <tt class="docutils literal"><span class="pre">Address</span></tt> that corresponds to the criteria of <tt class="docutils literal"><span class="pre">email.startswith('tony')</span></tt>,
  760. it won&#8217;t show up in the <tt class="docutils literal"><span class="pre">User.addresses</span></tt> collection until the session is flushed,
  761. and the attributes reloaded after a commit or expire operation. While we could
  762. consider an attribute event that checks this criterion in Python, this starts
  763. to cross the line of duplicating too much SQL behavior in Python. The backref behavior
  764. itself is only a slight transgression of this philosophy - SQLAlchemy tries to keep
  765. these to a minimum overall.</p>
  766. </div>
  767. </div>
  768. <div class="section" id="setting-the-primaryjoin-and-secondaryjoin">
  769. <span id="relationship-primaryjoin"></span><h2>Setting the primaryjoin and secondaryjoin<a class="headerlink" href="#setting-the-primaryjoin-and-secondaryjoin" title="Permalink to this headline">ś</a></h2>
  770. <p>A common scenario arises when we attempt to relate two
  771. classes together, where there exist multiple ways to join the
  772. two tables.</p>
  773. <p>Consider a <tt class="docutils literal"><span class="pre">Customer</span></tt> class that contains two foreign keys to an <tt class="docutils literal"><span class="pre">Address</span></tt>
  774. class:</p>
  775. <div class="highlight-python"><div class="highlight"><pre><span class="kn">from</span> <span class="nn">sqlalchemy</span> <span class="kn">import</span> <span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">,</span> <span class="n">String</span><span class="p">,</span> <span class="n">Column</span>
  776. <span class="kn">from</span> <span class="nn">sqlalchemy.ext.declarative</span> <span class="kn">import</span> <span class="n">declarative_base</span>
  777. <span class="kn">from</span> <span class="nn">sqlalchemy.orm</span> <span class="kn">import</span> <span class="n">relationship</span>
  778. <span class="n">Base</span> <span class="o">=</span> <span class="n">declarative_base</span><span class="p">()</span>
  779. <span class="k">class</span> <span class="nc">Customer</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  780. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;customer&#39;</span>
  781. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  782. <span class="n">name</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">String</span><span class="p">)</span>
  783. <span class="n">billing_address_id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">(</span><span class="s">&quot;address.id&quot;</span><span class="p">))</span>
  784. <span class="n">shipping_address_id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">(</span><span class="s">&quot;address.id&quot;</span><span class="p">))</span>
  785. <span class="n">billing_address</span> <span class="o">=</span> <span class="n">relationship</span><span class="p">(</span><span class="s">&quot;Address&quot;</span><span class="p">)</span>
  786. <span class="n">shipping_address</span> <span class="o">=</span> <span class="n">relationship</span><span class="p">(</span><span class="s">&quot;Address&quot;</span><span class="p">)</span>
  787. <span class="k">class</span> <span class="nc">Address</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  788. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;address&#39;</span>
  789. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  790. <span class="n">street</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">String</span><span class="p">)</span>
  791. <span class="n">city</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">String</span><span class="p">)</span>
  792. <span class="n">state</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">String</span><span class="p">)</span>
  793. <span class="nb">zip</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">String</span><span class="p">)</span></pre></div>
  794. </div>
  795. <p>The above mapping, when we attempt to use it, will produce the error:</p>
  796. <div class="highlight-python"><pre>sqlalchemy.exc.ArgumentError: Could not determine join condition between
  797. parent/child tables on relationship Customer.billing_address. Specify a
  798. 'primaryjoin' expression. If 'secondary' is present, 'secondaryjoin' is
  799. needed as well.</pre>
  800. </div>
  801. <p>What this error means is that if you have a <tt class="docutils literal"><span class="pre">Customer</span></tt> object, and wish
  802. to load in an associated <tt class="docutils literal"><span class="pre">Address</span></tt>, there is the choice of retrieving
  803. the <tt class="docutils literal"><span class="pre">Address</span></tt> referred to by the <tt class="docutils literal"><span class="pre">billing_address_id</span></tt> column or the one
  804. referred to by the <tt class="docutils literal"><span class="pre">shipping_address_id</span></tt> column. The <a class="reference internal" href="#sqlalchemy.orm.relationship" title="sqlalchemy.orm.relationship"><tt class="xref py py-func docutils literal"><span class="pre">relationship()</span></tt></a>,
  805. as it is, cannot determine its full configuration. The examples at
  806. <a class="reference internal" href="#relationship-patterns"><em>Basic Relational Patterns</em></a> didn&#8217;t have this issue, because in each of those examples
  807. there was only <strong>one</strong> way to refer to the related table.</p>
  808. <p>To resolve this issue, <a class="reference internal" href="#sqlalchemy.orm.relationship" title="sqlalchemy.orm.relationship"><tt class="xref py py-func docutils literal"><span class="pre">relationship()</span></tt></a> accepts an argument named
  809. <tt class="docutils literal"><span class="pre">primaryjoin</span></tt> which accepts a Python-based SQL expression, using the system described
  810. at <a class="reference internal" href="../core/tutorial.html"><em>SQL Expression Language Tutorial</em></a>, that describes how the two tables should be joined
  811. together. When using the declarative system, we often will specify this Python
  812. expression within a string, which is late-evaluated by the mapping configuration
  813. system so that it has access to the full namespace of available classes:</p>
  814. <div class="highlight-python"><div class="highlight"><pre><span class="k">class</span> <span class="nc">Customer</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  815. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;customer&#39;</span>
  816. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  817. <span class="n">name</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">String</span><span class="p">)</span>
  818. <span class="n">billing_address_id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">(</span><span class="s">&quot;address.id&quot;</span><span class="p">))</span>
  819. <span class="n">shipping_address_id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">(</span><span class="s">&quot;address.id&quot;</span><span class="p">))</span>
  820. <span class="n">billing_address</span> <span class="o">=</span> <span class="n">relationship</span><span class="p">(</span><span class="s">&quot;Address&quot;</span><span class="p">,</span>
  821. <span class="n">primaryjoin</span><span class="o">=</span><span class="s">&quot;Address.id==Customer.billing_address_id&quot;</span><span class="p">)</span>
  822. <span class="n">shipping_address</span> <span class="o">=</span> <span class="n">relationship</span><span class="p">(</span><span class="s">&quot;Address&quot;</span><span class="p">,</span>
  823. <span class="n">primaryjoin</span><span class="o">=</span><span class="s">&quot;Address.id==Customer.shipping_address_id&quot;</span><span class="p">)</span></pre></div>
  824. </div>
  825. <p>Above, loading the <tt class="docutils literal"><span class="pre">Customer.billing_address</span></tt> relationship from a <tt class="docutils literal"><span class="pre">Customer</span></tt>
  826. object will use the value present in <tt class="docutils literal"><span class="pre">billing_address_id</span></tt> in order to
  827. identify the row in <tt class="docutils literal"><span class="pre">Address</span></tt> to be loaded; similarly, <tt class="docutils literal"><span class="pre">shipping_address_id</span></tt>
  828. is used for the <tt class="docutils literal"><span class="pre">shipping_address</span></tt> relationship. The linkage of the two
  829. columns also plays a role during persistence; the newly generated primary key
  830. of a just-inserted <tt class="docutils literal"><span class="pre">Address</span></tt> object will be copied into the appropriate
  831. foreign key column of an associated <tt class="docutils literal"><span class="pre">Customer</span></tt> object during a flush.</p>
  832. <div class="section" id="specifying-alternate-join-conditions">
  833. <h3>Specifying Alternate Join Conditions<a class="headerlink" href="#specifying-alternate-join-conditions" title="Permalink to this headline">ś</a></h3>
  834. <p>The open-ended nature of <tt class="docutils literal"><span class="pre">primaryjoin</span></tt> also allows us to customize how
  835. related items are loaded. In the example below, using the <tt class="docutils literal"><span class="pre">User</span></tt> class
  836. as well as an <tt class="docutils literal"><span class="pre">Address</span></tt> class which stores a street address, we
  837. create a relationship <tt class="docutils literal"><span class="pre">boston_addresses</span></tt> which will only
  838. load those <tt class="docutils literal"><span class="pre">Address</span></tt> objects which specify a city of &#8220;Boston&#8221;:</p>
  839. <div class="highlight-python"><div class="highlight"><pre><span class="kn">from</span> <span class="nn">sqlalchemy</span> <span class="kn">import</span> <span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">,</span> <span class="n">String</span><span class="p">,</span> <span class="n">Column</span>
  840. <span class="kn">from</span> <span class="nn">sqlalchemy.ext.declarative</span> <span class="kn">import</span> <span class="n">declarative_base</span>
  841. <span class="kn">from</span> <span class="nn">sqlalchemy.orm</span> <span class="kn">import</span> <span class="n">relationship</span>
  842. <span class="n">Base</span> <span class="o">=</span> <span class="n">declarative_base</span><span class="p">()</span>
  843. <span class="k">class</span> <span class="nc">User</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  844. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;user&#39;</span>
  845. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  846. <span class="n">name</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">String</span><span class="p">)</span>
  847. <span class="n">addresses</span> <span class="o">=</span> <span class="n">relationship</span><span class="p">(</span><span class="s">&quot;Address&quot;</span><span class="p">,</span>
  848. <span class="n">primaryjoin</span><span class="o">=</span><span class="s">&quot;and_(User.id==Address.user_id, &quot;</span>
  849. <span class="s">&quot;Address.city==&#39;Boston&#39;)&quot;</span><span class="p">)</span>
  850. <span class="k">class</span> <span class="nc">Address</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  851. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;address&#39;</span>
  852. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  853. <span class="n">user_id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">(</span><span class="s">&#39;user.id&#39;</span><span class="p">))</span>
  854. <span class="n">street</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">String</span><span class="p">)</span>
  855. <span class="n">city</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">String</span><span class="p">)</span>
  856. <span class="n">state</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">String</span><span class="p">)</span>
  857. <span class="nb">zip</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">String</span><span class="p">)</span></pre></div>
  858. </div>
  859. <p>Within this string SQL expression, we made use of the <a class="reference internal" href="../core/expression_api.html#sqlalchemy.sql.expression.and_" title="sqlalchemy.sql.expression.and_"><tt class="xref py py-func docutils literal"><span class="pre">and_()</span></tt></a> conjunction construct to establish
  860. two distinct predicates for the join condition - joining both the <tt class="docutils literal"><span class="pre">User.id</span></tt> and
  861. <tt class="docutils literal"><span class="pre">Address.user_id</span></tt> columns to each other, as well as limiting rows in <tt class="docutils literal"><span class="pre">Address</span></tt>
  862. to just <tt class="docutils literal"><span class="pre">city='Boston'</span></tt>. When using Declarative, rudimentary SQL functions like
  863. <a class="reference internal" href="../core/expression_api.html#sqlalchemy.sql.expression.and_" title="sqlalchemy.sql.expression.and_"><tt class="xref py py-func docutils literal"><span class="pre">and_()</span></tt></a> are automatically available in the evaluated namespace of a string
  864. <a class="reference internal" href="#sqlalchemy.orm.relationship" title="sqlalchemy.orm.relationship"><tt class="xref py py-func docutils literal"><span class="pre">relationship()</span></tt></a> argument.</p>
  865. <p>When using classical mappings, we have the advantage of the <a class="reference internal" href="../core/schema.html#sqlalchemy.schema.Table" title="sqlalchemy.schema.Table"><tt class="xref py py-class docutils literal"><span class="pre">Table</span></tt></a> objects
  866. already being present when the mapping is defined, so that the SQL expression
  867. can be created immediately:</p>
  868. <div class="highlight-python"><div class="highlight"><pre><span class="kn">from</span> <span class="nn">sqlalchemy.orm</span> <span class="kn">import</span> <span class="n">relationship</span><span class="p">,</span> <span class="n">mapper</span>
  869. <span class="k">class</span> <span class="nc">User</span><span class="p">(</span><span class="nb">object</span><span class="p">):</span>
  870. <span class="k">pass</span>
  871. <span class="k">class</span> <span class="nc">Address</span><span class="p">(</span><span class="nb">object</span><span class="p">):</span>
  872. <span class="k">pass</span>
  873. <span class="n">mapper</span><span class="p">(</span><span class="n">Address</span><span class="p">,</span> <span class="n">addresses_table</span><span class="p">)</span>
  874. <span class="n">mapper</span><span class="p">(</span><span class="n">User</span><span class="p">,</span> <span class="n">users_table</span><span class="p">,</span> <span class="n">properties</span><span class="o">=</span><span class="p">{</span>
  875. <span class="s">&#39;boston_addresses&#39;</span><span class="p">:</span> <span class="n">relationship</span><span class="p">(</span><span class="n">Address</span><span class="p">,</span> <span class="n">primaryjoin</span><span class="o">=</span>
  876. <span class="n">and_</span><span class="p">(</span><span class="n">users_table</span><span class="o">.</span><span class="n">c</span><span class="o">.</span><span class="n">id</span><span class="o">==</span><span class="n">addresses_table</span><span class="o">.</span><span class="n">c</span><span class="o">.</span><span class="n">user_id</span><span class="p">,</span>
  877. <span class="n">addresses_table</span><span class="o">.</span><span class="n">c</span><span class="o">.</span><span class="n">city</span><span class="o">==</span><span class="s">&#39;Boston&#39;</span><span class="p">))</span>
  878. <span class="p">})</span></pre></div>
  879. </div>
  880. <p>Note that the custom criteria we use in a <tt class="docutils literal"><span class="pre">primaryjoin</span></tt> is generally only significant
  881. when SQLAlchemy is rendering SQL in order to load or represent this relationship.
  882. That is, it&#8217;s used
  883. in the SQL statement that&#8217;s emitted in order to perform a per-attribute lazy load, or when a join is
  884. constructed at query time, such as via <a class="reference internal" href="query.html#sqlalchemy.orm.query.Query.join" title="sqlalchemy.orm.query.Query.join"><tt class="xref py py-meth docutils literal"><span class="pre">Query.join()</span></tt></a>, or via the eager &#8220;joined&#8221; or &#8220;subquery&#8221;
  885. styles of loading. When in-memory objects are being manipulated, we can place any <tt class="docutils literal"><span class="pre">Address</span></tt> object
  886. we&#8217;d like into the <tt class="docutils literal"><span class="pre">boston_addresses</span></tt> collection, regardless of what the value of the <tt class="docutils literal"><span class="pre">.city</span></tt>
  887. attribute is. The objects will remain present in the collection until the attribute is expired
  888. and re-loaded from the database where the criterion is applied. When
  889. a flush occurs, the objects inside of <tt class="docutils literal"><span class="pre">boston_addresses</span></tt> will be flushed unconditionally, assigning
  890. value of the primary key <tt class="docutils literal"><span class="pre">user.id</span></tt> column onto the foreign-key-holding <tt class="docutils literal"><span class="pre">address.user_id</span></tt> column
  891. for each row. The <tt class="docutils literal"><span class="pre">city</span></tt> criteria has no effect here, as the flush process only cares about synchronizing primary
  892. key values into referencing foreign key values.</p>
  893. </div>
  894. <div class="section" id="self-referential-many-to-many-relationship">
  895. <span id="self-referential-many-to-many"></span><h3>Self-Referential Many-to-Many Relationship<a class="headerlink" href="#self-referential-many-to-many-relationship" title="Permalink to this headline">ś</a></h3>
  896. <p>Many to many relationships can be customized by one or both of <tt class="docutils literal"><span class="pre">primaryjoin</span></tt>
  897. and <tt class="docutils literal"><span class="pre">secondaryjoin</span></tt> - the latter is significant for a relationship that
  898. specifies a many-to-many reference using the <tt class="docutils literal"><span class="pre">secondary</span></tt> argument.
  899. A common situation which involves the usage of <tt class="docutils literal"><span class="pre">primaryjoin</span></tt> and <tt class="docutils literal"><span class="pre">secondaryjoin</span></tt>
  900. is when establishing a many-to-many relationship from a class to itself, as shown below:</p>
  901. <div class="highlight-python"><div class="highlight"><pre><span class="kn">from</span> <span class="nn">sqlalchemy</span> <span class="kn">import</span> <span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">,</span> <span class="n">String</span><span class="p">,</span> <span class="n">Column</span><span class="p">,</span> <span class="n">Table</span>
  902. <span class="kn">from</span> <span class="nn">sqlalchemy.ext.declarative</span> <span class="kn">import</span> <span class="n">declarative_base</span>
  903. <span class="kn">from</span> <span class="nn">sqlalchemy.orm</span> <span class="kn">import</span> <span class="n">relationship</span>
  904. <span class="n">Base</span> <span class="o">=</span> <span class="n">declarative_base</span><span class="p">()</span>
  905. <span class="n">node_to_node</span> <span class="o">=</span> <span class="n">Table</span><span class="p">(</span><span class="s">&quot;node_to_node&quot;</span><span class="p">,</span> <span class="n">Base</span><span class="o">.</span><span class="n">metadata</span><span class="p">,</span>
  906. <span class="n">Column</span><span class="p">(</span><span class="s">&quot;left_node_id&quot;</span><span class="p">,</span> <span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">(</span><span class="s">&quot;node.id&quot;</span><span class="p">),</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">),</span>
  907. <span class="n">Column</span><span class="p">(</span><span class="s">&quot;right_node_id&quot;</span><span class="p">,</span> <span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">(</span><span class="s">&quot;node.id&quot;</span><span class="p">),</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  908. <span class="p">)</span>
  909. <span class="k">class</span> <span class="nc">Node</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  910. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;node&#39;</span>
  911. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  912. <span class="n">label</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">String</span><span class="p">)</span>
  913. <span class="n">right_nodes</span> <span class="o">=</span> <span class="n">relationship</span><span class="p">(</span><span class="s">&quot;Node&quot;</span><span class="p">,</span>
  914. <span class="n">secondary</span><span class="o">=</span><span class="n">node_to_node</span><span class="p">,</span>
  915. <span class="n">primaryjoin</span><span class="o">=</span><span class="nb">id</span><span class="o">==</span><span class="n">node_to_node</span><span class="o">.</span><span class="n">c</span><span class="o">.</span><span class="n">left_node_id</span><span class="p">,</span>
  916. <span class="n">secondaryjoin</span><span class="o">=</span><span class="nb">id</span><span class="o">==</span><span class="n">node_to_node</span><span class="o">.</span><span class="n">c</span><span class="o">.</span><span class="n">right_node_id</span><span class="p">,</span>
  917. <span class="n">backref</span><span class="o">=</span><span class="s">&quot;left_nodes&quot;</span>
  918. <span class="p">)</span></pre></div>
  919. </div>
  920. <p>Where above, SQLAlchemy can&#8217;t know automatically which columns should connect
  921. to which for the <tt class="docutils literal"><span class="pre">right_nodes</span></tt> and <tt class="docutils literal"><span class="pre">left_nodes</span></tt> relationships. The <tt class="docutils literal"><span class="pre">primaryjoin</span></tt>
  922. and <tt class="docutils literal"><span class="pre">secondaryjoin</span></tt> arguments establish how we&#8217;d like to join to the association table.
  923. In the Declarative form above, as we are declaring these conditions within the Python
  924. block that corresponds to the <tt class="docutils literal"><span class="pre">Node</span></tt> class, the <tt class="docutils literal"><span class="pre">id</span></tt> variable is available directly
  925. as the <tt class="docutils literal"><span class="pre">Column</span></tt> object we wish to join with.</p>
  926. <p>A classical mapping situation here is similar, where <tt class="docutils literal"><span class="pre">node_to_node</span></tt> can be joined
  927. to <tt class="docutils literal"><span class="pre">node.c.id</span></tt>:</p>
  928. <div class="highlight-python"><div class="highlight"><pre><span class="kn">from</span> <span class="nn">sqlalchemy</span> <span class="kn">import</span> <span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">,</span> <span class="n">String</span><span class="p">,</span> <span class="n">Column</span><span class="p">,</span> <span class="n">Table</span><span class="p">,</span> <span class="n">MetaData</span>
  929. <span class="kn">from</span> <span class="nn">sqlalchemy.orm</span> <span class="kn">import</span> <span class="n">relationship</span><span class="p">,</span> <span class="n">mapper</span>
  930. <span class="n">metadata</span> <span class="o">=</span> <span class="n">MetaData</span><span class="p">()</span>
  931. <span class="n">node_to_node</span> <span class="o">=</span> <span class="n">Table</span><span class="p">(</span><span class="s">&quot;node_to_node&quot;</span><span class="p">,</span> <span class="n">metadata</span><span class="p">,</span>
  932. <span class="n">Column</span><span class="p">(</span><span class="s">&quot;left_node_id&quot;</span><span class="p">,</span> <span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">(</span><span class="s">&quot;node.id&quot;</span><span class="p">),</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">),</span>
  933. <span class="n">Column</span><span class="p">(</span><span class="s">&quot;right_node_id&quot;</span><span class="p">,</span> <span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">(</span><span class="s">&quot;node.id&quot;</span><span class="p">),</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  934. <span class="p">)</span>
  935. <span class="n">node</span> <span class="o">=</span> <span class="n">Table</span><span class="p">(</span><span class="s">&quot;node&quot;</span><span class="p">,</span> <span class="n">metadata</span><span class="p">,</span>
  936. <span class="n">Column</span><span class="p">(</span><span class="s">&#39;id&#39;</span><span class="p">,</span> <span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">),</span>
  937. <span class="n">Column</span><span class="p">(</span><span class="s">&#39;label&#39;</span><span class="p">,</span> <span class="n">String</span><span class="p">)</span>
  938. <span class="p">)</span>
  939. <span class="k">class</span> <span class="nc">Node</span><span class="p">(</span><span class="nb">object</span><span class="p">):</span>
  940. <span class="k">pass</span>
  941. <span class="n">mapper</span><span class="p">(</span><span class="n">Node</span><span class="p">,</span> <span class="n">node</span><span class="p">,</span> <span class="n">properties</span><span class="o">=</span><span class="p">{</span>
  942. <span class="s">&#39;right_nodes&#39;</span><span class="p">:</span><span class="n">relationship</span><span class="p">(</span><span class="n">Node</span><span class="p">,</span>
  943. <span class="n">secondary</span><span class="o">=</span><span class="n">node_to_node</span><span class="p">,</span>
  944. <span class="n">primaryjoin</span><span class="o">=</span><span class="n">node</span><span class="o">.</span><span class="n">c</span><span class="o">.</span><span class="n">id</span><span class="o">==</span><span class="n">node_to_node</span><span class="o">.</span><span class="n">c</span><span class="o">.</span><span class="n">left_node_id</span><span class="p">,</span>
  945. <span class="n">secondaryjoin</span><span class="o">=</span><span class="n">node</span><span class="o">.</span><span class="n">c</span><span class="o">.</span><span class="n">id</span><span class="o">==</span><span class="n">node_to_node</span><span class="o">.</span><span class="n">c</span><span class="o">.</span><span class="n">right_node_id</span><span class="p">,</span>
  946. <span class="n">backref</span><span class="o">=</span><span class="s">&quot;left_nodes&quot;</span>
  947. <span class="p">)})</span></pre></div>
  948. </div>
  949. <p>Note that in both examples, the <tt class="docutils literal"><span class="pre">backref</span></tt> keyword specifies a <tt class="docutils literal"><span class="pre">left_nodes</span></tt>
  950. backref - when <a class="reference internal" href="#sqlalchemy.orm.relationship" title="sqlalchemy.orm.relationship"><tt class="xref py py-func docutils literal"><span class="pre">relationship()</span></tt></a> creates the second relationship in the reverse
  951. direction, it&#8217;s smart enough to reverse the <tt class="docutils literal"><span class="pre">primaryjoin</span></tt> and <tt class="docutils literal"><span class="pre">secondaryjoin</span></tt> arguments.</p>
  952. </div>
  953. <div class="section" id="specifying-foreign-keys">
  954. <h3>Specifying Foreign Keys<a class="headerlink" href="#specifying-foreign-keys" title="Permalink to this headline">ś</a></h3>
  955. <p>When using <tt class="docutils literal"><span class="pre">primaryjoin</span></tt> and <tt class="docutils literal"><span class="pre">secondaryjoin</span></tt>, SQLAlchemy also needs to be
  956. aware of which columns in the relationship reference the other. In most cases,
  957. a <a class="reference internal" href="../core/schema.html#sqlalchemy.schema.Table" title="sqlalchemy.schema.Table"><tt class="xref py py-class docutils literal"><span class="pre">Table</span></tt></a> construct will have
  958. <a class="reference internal" href="../core/schema.html#sqlalchemy.schema.ForeignKey" title="sqlalchemy.schema.ForeignKey"><tt class="xref py py-class docutils literal"><span class="pre">ForeignKey</span></tt></a> constructs which take care of this;
  959. however, in the case of reflected tables on a database that does not report
  960. FKs (like MySQL ISAM) or when using join conditions on columns that don&#8217;t have
  961. foreign keys, the <a class="reference internal" href="#sqlalchemy.orm.relationship" title="sqlalchemy.orm.relationship"><tt class="xref py py-func docutils literal"><span class="pre">relationship()</span></tt></a> needs to be told
  962. specifically which columns are &#8220;foreign&#8221; using the <tt class="docutils literal"><span class="pre">foreign_keys</span></tt>
  963. collection:</p>
  964. <div class="highlight-python+sql"><div class="highlight"><pre><span class="k">class</span> <span class="nc">Address</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  965. <span class="n">__table__</span> <span class="o">=</span> <span class="n">addresses_table</span>
  966. <span class="k">class</span> <span class="nc">User</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  967. <span class="n">__table__</span> <span class="o">=</span> <span class="n">users_table</span>
  968. <span class="n">addresses</span> <span class="o">=</span> <span class="n">relationship</span><span class="p">(</span><span class="n">Address</span><span class="p">,</span>
  969. <span class="n">primaryjoin</span><span class="o">=</span>
  970. <span class="n">users_table</span><span class="o">.</span><span class="n">c</span><span class="o">.</span><span class="n">user_id</span><span class="o">==</span><span class="n">addresses_table</span><span class="o">.</span><span class="n">c</span><span class="o">.</span><span class="n">user_id</span><span class="p">,</span>
  971. <span class="n">foreign_keys</span><span class="o">=</span><span class="p">[</span><span class="n">addresses_table</span><span class="o">.</span><span class="n">c</span><span class="o">.</span><span class="n">user_id</span><span class="p">])</span></pre></div>
  972. </div>
  973. </div>
  974. <div class="section" id="building-query-enabled-properties">
  975. <h3>Building Query-Enabled Properties<a class="headerlink" href="#building-query-enabled-properties" title="Permalink to this headline">ś</a></h3>
  976. <p>Very ambitious custom join conditions may fail to be directly persistable, and
  977. in some cases may not even load correctly. To remove the persistence part of
  978. the equation, use the flag <tt class="docutils literal"><span class="pre">viewonly=True</span></tt> on the
  979. <a class="reference internal" href="#sqlalchemy.orm.relationship" title="sqlalchemy.orm.relationship"><tt class="xref py py-func docutils literal"><span class="pre">relationship()</span></tt></a>, which establishes it as a read-only
  980. attribute (data written to the collection will be ignored on flush()).
  981. However, in extreme cases, consider using a regular Python property in
  982. conjunction with <a class="reference internal" href="query.html#sqlalchemy.orm.query.Query" title="sqlalchemy.orm.query.Query"><tt class="xref py py-class docutils literal"><span class="pre">Query</span></tt></a> as follows:</p>
  983. <div class="highlight-python+sql"><div class="highlight"><pre><span class="k">class</span> <span class="nc">User</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  984. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;user&#39;</span>
  985. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  986. <span class="k">def</span> <span class="nf">_get_addresses</span><span class="p">(</span><span class="bp">self</span><span class="p">):</span>
  987. <span class="k">return</span> <span class="n">object_session</span><span class="p">(</span><span class="bp">self</span><span class="p">)</span><span class="o">.</span><span class="n">query</span><span class="p">(</span><span class="n">Address</span><span class="p">)</span><span class="o">.</span><span class="n">with_parent</span><span class="p">(</span><span class="bp">self</span><span class="p">)</span><span class="o">.</span><span class="n">filter</span><span class="p">(</span><span class="o">...</span><span class="p">)</span><span class="o">.</span><span class="n">all</span><span class="p">()</span>
  988. <span class="n">addresses</span> <span class="o">=</span> <span class="nb">property</span><span class="p">(</span><span class="n">_get_addresses</span><span class="p">)</span></pre></div>
  989. </div>
  990. </div>
  991. </div>
  992. <div class="section" id="rows-that-point-to-themselves-mutually-dependent-rows">
  993. <span id="post-update"></span><h2>Rows that point to themselves / Mutually Dependent Rows<a class="headerlink" href="#rows-that-point-to-themselves-mutually-dependent-rows" title="Permalink to this headline">ś</a></h2>
  994. <p>This is a very specific case where relationship() must perform an INSERT and a
  995. second UPDATE in order to properly populate a row (and vice versa an UPDATE
  996. and DELETE in order to delete without violating foreign key constraints). The
  997. two use cases are:</p>
  998. <ul class="simple">
  999. <li>A table contains a foreign key to itself, and a single row will
  1000. have a foreign key value pointing to its own primary key.</li>
  1001. <li>Two tables each contain a foreign key referencing the other
  1002. table, with a row in each table referencing the other.</li>
  1003. </ul>
  1004. <p>For example:</p>
  1005. <div class="highlight-python"><pre> user
  1006. ---------------------------------
  1007. user_id name related_user_id
  1008. 1 'ed' 1</pre>
  1009. </div>
  1010. <p>Or:</p>
  1011. <div class="highlight-python"><pre> widget entry
  1012. ------------------------------------------- ---------------------------------
  1013. widget_id name favorite_entry_id entry_id name widget_id
  1014. 1 'somewidget' 5 5 'someentry' 1</pre>
  1015. </div>
  1016. <p>In the first case, a row points to itself. Technically, a database that uses
  1017. sequences such as PostgreSQL or Oracle can INSERT the row at once using a
  1018. previously generated value, but databases which rely upon autoincrement-style
  1019. primary key identifiers cannot. The <a class="reference internal" href="#sqlalchemy.orm.relationship" title="sqlalchemy.orm.relationship"><tt class="xref py py-func docutils literal"><span class="pre">relationship()</span></tt></a>
  1020. always assumes a &#8220;parent/child&#8221; model of row population during flush, so
  1021. unless you are populating the primary key/foreign key columns directly,
  1022. <a class="reference internal" href="#sqlalchemy.orm.relationship" title="sqlalchemy.orm.relationship"><tt class="xref py py-func docutils literal"><span class="pre">relationship()</span></tt></a> needs to use two statements.</p>
  1023. <p>In the second case, the &#8220;widget&#8221; row must be inserted before any referring
  1024. &#8220;entry&#8221; rows, but then the &#8220;favorite_entry_id&#8221; column of that &#8220;widget&#8221; row
  1025. cannot be set until the &#8220;entry&#8221; rows have been generated. In this case, it&#8217;s
  1026. typically impossible to insert the &#8220;widget&#8221; and &#8220;entry&#8221; rows using just two
  1027. INSERT statements; an UPDATE must be performed in order to keep foreign key
  1028. constraints fulfilled. The exception is if the foreign keys are configured as
  1029. &#8220;deferred until commit&#8221; (a feature some databases support) and if the
  1030. identifiers were populated manually (again essentially bypassing
  1031. <a class="reference internal" href="#sqlalchemy.orm.relationship" title="sqlalchemy.orm.relationship"><tt class="xref py py-func docutils literal"><span class="pre">relationship()</span></tt></a>).</p>
  1032. <p>To enable the usage of a supplementary UPDATE statement,
  1033. we use the <tt class="docutils literal"><span class="pre">post_update</span></tt> option
  1034. of <a class="reference internal" href="#sqlalchemy.orm.relationship" title="sqlalchemy.orm.relationship"><tt class="xref py py-func docutils literal"><span class="pre">relationship()</span></tt></a>. This specifies that the linkage between the
  1035. two rows should be created using an UPDATE statement after both rows
  1036. have been INSERTED; it also causes the rows to be de-associated with
  1037. each other via UPDATE before a DELETE is emitted. The flag should
  1038. be placed on just <em>one</em> of the relationships, preferably the
  1039. many-to-one side. Below we illustrate
  1040. a complete example, including two <a class="reference internal" href="../core/schema.html#sqlalchemy.schema.ForeignKey" title="sqlalchemy.schema.ForeignKey"><tt class="xref py py-class docutils literal"><span class="pre">ForeignKey</span></tt></a> constructs, one which
  1041. specifies <tt class="docutils literal"><span class="pre">use_alter=True</span></tt> to help with emitting CREATE TABLE statements:</p>
  1042. <div class="highlight-python"><div class="highlight"><pre><span class="kn">from</span> <span class="nn">sqlalchemy</span> <span class="kn">import</span> <span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">,</span> <span class="n">Column</span>
  1043. <span class="kn">from</span> <span class="nn">sqlalchemy.ext.declarative</span> <span class="kn">import</span> <span class="n">declarative_base</span>
  1044. <span class="kn">from</span> <span class="nn">sqlalchemy.orm</span> <span class="kn">import</span> <span class="n">relationship</span>
  1045. <span class="n">Base</span> <span class="o">=</span> <span class="n">declarative_base</span><span class="p">()</span>
  1046. <span class="k">class</span> <span class="nc">Entry</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  1047. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;entry&#39;</span>
  1048. <span class="n">entry_id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  1049. <span class="n">widget_id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">(</span><span class="s">&#39;widget.widget_id&#39;</span><span class="p">))</span>
  1050. <span class="n">name</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">String</span><span class="p">(</span><span class="mi">50</span><span class="p">))</span>
  1051. <span class="k">class</span> <span class="nc">Widget</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  1052. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;widget&#39;</span>
  1053. <span class="n">widget_id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  1054. <span class="n">favorite_entry_id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span>
  1055. <span class="n">ForeignKey</span><span class="p">(</span><span class="s">&#39;entry.entry_id&#39;</span><span class="p">,</span>
  1056. <span class="n">use_alter</span><span class="o">=</span><span class="bp">True</span><span class="p">,</span>
  1057. <span class="n">name</span><span class="o">=</span><span class="s">&quot;fk_favorite_entry&quot;</span><span class="p">))</span>
  1058. <span class="n">name</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">String</span><span class="p">(</span><span class="mi">50</span><span class="p">))</span>
  1059. <span class="n">entries</span> <span class="o">=</span> <span class="n">relationship</span><span class="p">(</span><span class="n">Entry</span><span class="p">,</span> <span class="n">primaryjoin</span><span class="o">=</span>
  1060. <span class="n">widget_id</span><span class="o">==</span><span class="n">Entry</span><span class="o">.</span><span class="n">widget_id</span><span class="p">)</span>
  1061. <span class="n">favorite_entry</span> <span class="o">=</span> <span class="n">relationship</span><span class="p">(</span><span class="n">Entry</span><span class="p">,</span>
  1062. <span class="n">primaryjoin</span><span class="o">=</span>
  1063. <span class="n">favorite_entry_id</span><span class="o">==</span><span class="n">Entry</span><span class="o">.</span><span class="n">entry_id</span><span class="p">,</span>
  1064. <span class="n">post_update</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span></pre></div>
  1065. </div>
  1066. <p>When a structure against the above configuration is flushed, the &#8220;widget&#8221; row will be
  1067. INSERTed minus the &#8220;favorite_entry_id&#8221; value, then all the &#8220;entry&#8221; rows will
  1068. be INSERTed referencing the parent &#8220;widget&#8221; row, and then an UPDATE statement
  1069. will populate the &#8220;favorite_entry_id&#8221; column of the &#8220;widget&#8221; table (it&#8217;s one
  1070. row at a time for the time being):</p>
  1071. <div class="highlight-pycon+sql"><div class="highlight"><pre><span class="gp">&gt;&gt;&gt; </span><span class="n">w1</span> <span class="o">=</span> <span class="n">Widget</span><span class="p">(</span><span class="n">name</span><span class="o">=</span><span class="s">&#39;somewidget&#39;</span><span class="p">)</span>
  1072. <span class="gp">&gt;&gt;&gt; </span><span class="n">e1</span> <span class="o">=</span> <span class="n">Entry</span><span class="p">(</span><span class="n">name</span><span class="o">=</span><span class="s">&#39;someentry&#39;</span><span class="p">)</span>
  1073. <span class="gp">&gt;&gt;&gt; </span><span class="n">w1</span><span class="o">.</span><span class="n">favorite_entry</span> <span class="o">=</span> <span class="n">e1</span>
  1074. <span class="gp">&gt;&gt;&gt; </span><span class="n">w1</span><span class="o">.</span><span class="n">entries</span> <span class="o">=</span> <span class="p">[</span><span class="n">e1</span><span class="p">]</span>
  1075. <span class="gp">&gt;&gt;&gt; </span><span class="n">session</span><span class="o">.</span><span class="n">add_all</span><span class="p">([</span><span class="n">w1</span><span class="p">,</span> <span class="n">e1</span><span class="p">])</span>
  1076. <a href='#' class='sql_link'>sql</a><span class="gp">&gt;&gt;&gt; </span><span class="n">session</span><span class="o">.</span><span class="n">commit</span><span class="p">()</span>
  1077. <div class='popup_sql'>BEGIN (implicit)
  1078. INSERT INTO widget (favorite_entry_id, name) VALUES (?, ?)
  1079. (None, 'somewidget')
  1080. INSERT INTO entry (widget_id, name) VALUES (?, ?)
  1081. (1, 'someentry')
  1082. UPDATE widget SET favorite_entry_id=? WHERE widget.widget_id = ?
  1083. (1, 1)
  1084. COMMIT</div></pre></div>
  1085. </div>
  1086. <p>An additional configuration we can specify is to supply a more
  1087. comprehensive foreign key constraint on <tt class="docutils literal"><span class="pre">Widget</span></tt>, such that
  1088. it&#8217;s guaranteed that <tt class="docutils literal"><span class="pre">favorite_entry_id</span></tt> refers to an <tt class="docutils literal"><span class="pre">Entry</span></tt>
  1089. that also refers to this <tt class="docutils literal"><span class="pre">Widget</span></tt>. We can use a composite foreign key,
  1090. as illustrated below:</p>
  1091. <div class="highlight-python"><div class="highlight"><pre><span class="kn">from</span> <span class="nn">sqlalchemy</span> <span class="kn">import</span> <span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">,</span> <span class="n">String</span><span class="p">,</span> \
  1092. <span class="n">Column</span><span class="p">,</span> <span class="n">UniqueConstraint</span><span class="p">,</span> <span class="n">ForeignKeyConstraint</span>
  1093. <span class="kn">from</span> <span class="nn">sqlalchemy.ext.declarative</span> <span class="kn">import</span> <span class="n">declarative_base</span>
  1094. <span class="kn">from</span> <span class="nn">sqlalchemy.orm</span> <span class="kn">import</span> <span class="n">relationship</span>
  1095. <span class="n">Base</span> <span class="o">=</span> <span class="n">declarative_base</span><span class="p">()</span>
  1096. <span class="k">class</span> <span class="nc">Entry</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  1097. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;entry&#39;</span>
  1098. <span class="n">entry_id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  1099. <span class="n">widget_id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">ForeignKey</span><span class="p">(</span><span class="s">&#39;widget.widget_id&#39;</span><span class="p">))</span>
  1100. <span class="n">name</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">String</span><span class="p">(</span><span class="mi">50</span><span class="p">))</span>
  1101. <span class="n">__table_args__</span> <span class="o">=</span> <span class="p">(</span>
  1102. <span class="n">UniqueConstraint</span><span class="p">(</span><span class="s">&quot;entry_id&quot;</span><span class="p">,</span> <span class="s">&quot;widget_id&quot;</span><span class="p">),</span>
  1103. <span class="p">)</span>
  1104. <span class="k">class</span> <span class="nc">Widget</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  1105. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;widget&#39;</span>
  1106. <span class="n">widget_id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">autoincrement</span><span class="o">=</span><span class="s">&#39;ignore_fk&#39;</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  1107. <span class="n">favorite_entry_id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">)</span>
  1108. <span class="n">name</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">String</span><span class="p">(</span><span class="mi">50</span><span class="p">))</span>
  1109. <span class="n">__table_args__</span> <span class="o">=</span> <span class="p">(</span>
  1110. <span class="n">ForeignKeyConstraint</span><span class="p">(</span>
  1111. <span class="p">[</span><span class="s">&quot;widget_id&quot;</span><span class="p">,</span> <span class="s">&quot;favorite_entry_id&quot;</span><span class="p">],</span>
  1112. <span class="p">[</span><span class="s">&quot;entry.widget_id&quot;</span><span class="p">,</span> <span class="s">&quot;entry.entry_id&quot;</span><span class="p">],</span>
  1113. <span class="n">name</span><span class="o">=</span><span class="s">&quot;fk_favorite_entry&quot;</span><span class="p">,</span> <span class="n">use_alter</span><span class="o">=</span><span class="bp">True</span>
  1114. <span class="p">),</span>
  1115. <span class="p">)</span>
  1116. <span class="n">entries</span> <span class="o">=</span> <span class="n">relationship</span><span class="p">(</span><span class="n">Entry</span><span class="p">,</span> <span class="n">primaryjoin</span><span class="o">=</span>
  1117. <span class="n">widget_id</span><span class="o">==</span><span class="n">Entry</span><span class="o">.</span><span class="n">widget_id</span><span class="p">,</span>
  1118. <span class="n">foreign_keys</span><span class="o">=</span><span class="n">Entry</span><span class="o">.</span><span class="n">widget_id</span><span class="p">)</span>
  1119. <span class="n">favorite_entry</span> <span class="o">=</span> <span class="n">relationship</span><span class="p">(</span><span class="n">Entry</span><span class="p">,</span>
  1120. <span class="n">primaryjoin</span><span class="o">=</span>
  1121. <span class="n">favorite_entry_id</span><span class="o">==</span><span class="n">Entry</span><span class="o">.</span><span class="n">entry_id</span><span class="p">,</span>
  1122. <span class="n">foreign_keys</span><span class="o">=</span><span class="n">favorite_entry_id</span><span class="p">,</span>
  1123. <span class="n">post_update</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span></pre></div>
  1124. </div>
  1125. <p>The above mapping features a composite <a class="reference internal" href="../core/schema.html#sqlalchemy.schema.ForeignKeyConstraint" title="sqlalchemy.schema.ForeignKeyConstraint"><tt class="xref py py-class docutils literal"><span class="pre">ForeignKeyConstraint</span></tt></a>
  1126. bridging the <tt class="docutils literal"><span class="pre">widget_id</span></tt> and <tt class="docutils literal"><span class="pre">favorite_entry_id</span></tt> columns. To ensure
  1127. that <tt class="docutils literal"><span class="pre">Widget.widget_id</span></tt> remains an &#8220;autoincrementing&#8221; column we specify
  1128. <tt class="docutils literal"><span class="pre">autoincrement='ignore_fk'</span></tt> on <a class="reference internal" href="../core/schema.html#sqlalchemy.schema.Column" title="sqlalchemy.schema.Column"><tt class="xref py py-class docutils literal"><span class="pre">Column</span></tt></a>, and additionally on each
  1129. <a class="reference internal" href="#sqlalchemy.orm.relationship" title="sqlalchemy.orm.relationship"><tt class="xref py py-func docutils literal"><span class="pre">relationship()</span></tt></a> we must limit those columns considered as part of
  1130. the foreign key for the purposes of joining and cross-population.</p>
  1131. <p class="versionadded">
  1132. <span class="versionmodified">New in version 0.7.4: </span><tt class="docutils literal"><span class="pre">autoincrement='ignore_fk'</span></tt> on <a class="reference internal" href="../core/schema.html#sqlalchemy.schema.Column" title="sqlalchemy.schema.Column"><tt class="xref py py-class docutils literal"><span class="pre">Column</span></tt></a>.</p>
  1133. </div>
  1134. <div class="section" id="mutable-primary-keys-update-cascades">
  1135. <span id="passive-updates"></span><h2>Mutable Primary Keys / Update Cascades<a class="headerlink" href="#mutable-primary-keys-update-cascades" title="Permalink to this headline">ś</a></h2>
  1136. <p>When the primary key of an entity changes, related items
  1137. which reference the primary key must also be updated as
  1138. well. For databases which enforce referential integrity,
  1139. it&#8217;s required to use the database&#8217;s ON UPDATE CASCADE
  1140. functionality in order to propagate primary key changes
  1141. to referenced foreign keys - the values cannot be out
  1142. of sync for any moment.</p>
  1143. <p>For databases that don&#8217;t support this, such as SQLite and
  1144. MySQL without their referential integrity options turned
  1145. on, the <tt class="docutils literal"><span class="pre">passive_updates</span></tt> flag can
  1146. be set to <tt class="docutils literal"><span class="pre">False</span></tt>, most preferably on a one-to-many or
  1147. many-to-many <a class="reference internal" href="#sqlalchemy.orm.relationship" title="sqlalchemy.orm.relationship"><tt class="xref py py-func docutils literal"><span class="pre">relationship()</span></tt></a>, which instructs
  1148. SQLAlchemy to issue UPDATE statements individually for
  1149. objects referenced in the collection, loading them into
  1150. memory if not already locally present. The
  1151. <tt class="docutils literal"><span class="pre">passive_updates</span></tt> flag can also be <tt class="docutils literal"><span class="pre">False</span></tt> in
  1152. conjunction with ON UPDATE CASCADE functionality,
  1153. although in that case the unit of work will be issuing
  1154. extra SELECT and UPDATE statements unnecessarily.</p>
  1155. <p>A typical mutable primary key setup might look like:</p>
  1156. <div class="highlight-python"><div class="highlight"><pre><span class="k">class</span> <span class="nc">User</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  1157. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;user&#39;</span>
  1158. <span class="n">username</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">String</span><span class="p">(</span><span class="mi">50</span><span class="p">),</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  1159. <span class="n">fullname</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">String</span><span class="p">(</span><span class="mi">100</span><span class="p">))</span>
  1160. <span class="c"># passive_updates=False *only* needed if the database</span>
  1161. <span class="c"># does not implement ON UPDATE CASCADE</span>
  1162. <span class="n">addresses</span> <span class="o">=</span> <span class="n">relationship</span><span class="p">(</span><span class="s">&quot;Address&quot;</span><span class="p">,</span> <span class="n">passive_updates</span><span class="o">=</span><span class="bp">False</span><span class="p">)</span>
  1163. <span class="k">class</span> <span class="nc">Address</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  1164. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;address&#39;</span>
  1165. <span class="n">email</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">String</span><span class="p">(</span><span class="mi">50</span><span class="p">),</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  1166. <span class="n">username</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">String</span><span class="p">(</span><span class="mi">50</span><span class="p">),</span>
  1167. <span class="n">ForeignKey</span><span class="p">(</span><span class="s">&#39;user.username&#39;</span><span class="p">,</span> <span class="n">onupdate</span><span class="o">=</span><span class="s">&quot;cascade&quot;</span><span class="p">)</span>
  1168. <span class="p">)</span></pre></div>
  1169. </div>
  1170. <p><tt class="docutils literal"><span class="pre">passive_updates</span></tt> is set to <tt class="docutils literal"><span class="pre">True</span></tt> by default,
  1171. indicating that ON UPDATE CASCADE is expected to be in
  1172. place in the usual case for foreign keys that expect
  1173. to have a mutating parent key.</p>
  1174. <p><tt class="docutils literal"><span class="pre">passive_updates=False</span></tt> may be configured on any
  1175. direction of relationship, i.e. one-to-many, many-to-one,
  1176. and many-to-many, although it is much more effective when
  1177. placed just on the one-to-many or many-to-many side.
  1178. Configuring the <tt class="docutils literal"><span class="pre">passive_updates=False</span></tt> only on the
  1179. many-to-one side will have only a partial effect, as the
  1180. unit of work searches only through the current identity
  1181. map for objects that may be referencing the one with a
  1182. mutating primary key, not throughout the database.</p>
  1183. </div>
  1184. <div class="section" id="relationships-api">
  1185. <h2>Relationships API<a class="headerlink" href="#relationships-api" title="Permalink to this headline">ś</a></h2>
  1186. <dl class="function">
  1187. <dt id="sqlalchemy.orm.relationship">
  1188. <tt class="descclassname">sqlalchemy.orm.</tt><tt class="descname">relationship</tt><big>(</big><em>argument</em>, <em>secondary=None</em>, <em>**kwargs</em><big>)</big><a class="headerlink" href="#sqlalchemy.orm.relationship" title="Permalink to this definition">ś</a></dt>
  1189. <dd><p>Provide a relationship of a primary Mapper to a secondary Mapper.</p>
  1190. <p class="versionchanged">
  1191. <span class="versionmodified">Changed in version 0.6: </span><a class="reference internal" href="#sqlalchemy.orm.relationship" title="sqlalchemy.orm.relationship"><tt class="xref py py-func docutils literal"><span class="pre">relationship()</span></tt></a> is historically known as <a class="reference internal" href="#sqlalchemy.orm.relation" title="sqlalchemy.orm.relation"><tt class="xref py py-func docutils literal"><span class="pre">relation()</span></tt></a>.</p>
  1192. <p>This corresponds to a parent-child or associative table relationship. The
  1193. constructed class is an instance of <a class="reference internal" href="internals.html#sqlalchemy.orm.properties.RelationshipProperty" title="sqlalchemy.orm.properties.RelationshipProperty"><tt class="xref py py-class docutils literal"><span class="pre">RelationshipProperty</span></tt></a>.</p>
  1194. <p>A typical <a class="reference internal" href="#sqlalchemy.orm.relationship" title="sqlalchemy.orm.relationship"><tt class="xref py py-func docutils literal"><span class="pre">relationship()</span></tt></a>, used in a classical mapping:</p>
  1195. <div class="highlight-python"><div class="highlight"><pre><span class="n">mapper</span><span class="p">(</span><span class="n">Parent</span><span class="p">,</span> <span class="n">properties</span><span class="o">=</span><span class="p">{</span>
  1196. <span class="s">&#39;children&#39;</span><span class="p">:</span> <span class="n">relationship</span><span class="p">(</span><span class="n">Child</span><span class="p">)</span>
  1197. <span class="p">})</span></pre></div>
  1198. </div>
  1199. <p>Some arguments accepted by <a class="reference internal" href="#sqlalchemy.orm.relationship" title="sqlalchemy.orm.relationship"><tt class="xref py py-func docutils literal"><span class="pre">relationship()</span></tt></a> optionally accept a
  1200. callable function, which when called produces the desired value.
  1201. The callable is invoked by the parent <a class="reference internal" href="mapper_config.html#sqlalchemy.orm.mapper.Mapper" title="sqlalchemy.orm.mapper.Mapper"><tt class="xref py py-class docutils literal"><span class="pre">Mapper</span></tt></a> at &#8220;mapper initialization&#8221;
  1202. time, which happens only when mappers are first used, and is assumed
  1203. to be after all mappings have been constructed. This can be used
  1204. to resolve order-of-declaration and other dependency issues, such as
  1205. if <tt class="docutils literal"><span class="pre">Child</span></tt> is declared below <tt class="docutils literal"><span class="pre">Parent</span></tt> in the same file:</p>
  1206. <div class="highlight-python"><div class="highlight"><pre><span class="n">mapper</span><span class="p">(</span><span class="n">Parent</span><span class="p">,</span> <span class="n">properties</span><span class="o">=</span><span class="p">{</span>
  1207. <span class="s">&quot;children&quot;</span><span class="p">:</span><span class="n">relationship</span><span class="p">(</span><span class="k">lambda</span><span class="p">:</span> <span class="n">Child</span><span class="p">,</span>
  1208. <span class="n">order_by</span><span class="o">=</span><span class="k">lambda</span><span class="p">:</span> <span class="n">Child</span><span class="o">.</span><span class="n">id</span><span class="p">)</span>
  1209. <span class="p">})</span></pre></div>
  1210. </div>
  1211. <p>When using the <a class="reference internal" href="extensions/declarative.html"><em>Declarative</em></a> extension, the Declarative
  1212. initializer allows string arguments to be passed to <a class="reference internal" href="#sqlalchemy.orm.relationship" title="sqlalchemy.orm.relationship"><tt class="xref py py-func docutils literal"><span class="pre">relationship()</span></tt></a>.
  1213. These string arguments are converted into callables that evaluate
  1214. the string as Python code, using the Declarative
  1215. class-registry as a namespace. This allows the lookup of related
  1216. classes to be automatic via their string name, and removes the need to import
  1217. related classes at all into the local module space:</p>
  1218. <div class="highlight-python"><div class="highlight"><pre><span class="kn">from</span> <span class="nn">sqlalchemy.ext.declarative</span> <span class="kn">import</span> <span class="n">declarative_base</span>
  1219. <span class="n">Base</span> <span class="o">=</span> <span class="n">declarative_base</span><span class="p">()</span>
  1220. <span class="k">class</span> <span class="nc">Parent</span><span class="p">(</span><span class="n">Base</span><span class="p">):</span>
  1221. <span class="n">__tablename__</span> <span class="o">=</span> <span class="s">&#39;parent&#39;</span>
  1222. <span class="nb">id</span> <span class="o">=</span> <span class="n">Column</span><span class="p">(</span><span class="n">Integer</span><span class="p">,</span> <span class="n">primary_key</span><span class="o">=</span><span class="bp">True</span><span class="p">)</span>
  1223. <span class="n">children</span> <span class="o">=</span> <span class="n">relationship</span><span class="p">(</span><span class="s">&quot;Child&quot;</span><span class="p">,</span> <span class="n">order_by</span><span class="o">=</span><span class="s">&quot;Child.id&quot;</span><span class="p">)</span></pre></div>
  1224. </div>
  1225. <p>A full array of examples and reference documentation regarding
  1226. <a class="reference internal" href="#sqlalchemy.orm.relationship" title="sqlalchemy.orm.relationship"><tt class="xref py py-func docutils literal"><span class="pre">relationship()</span></tt></a> is at <a class="reference internal" href="#"><em>Relationship Configuration</em></a>.</p>
  1227. <table class="docutils field-list" frame="void" rules="none">
  1228. <col class="field-name" />
  1229. <col class="field-body" />
  1230. <tbody valign="top">
  1231. <tr class="field-odd field"><th class="field-name">Parameters:</th><td class="field-body"><ul class="first last simple">
  1232. <li><strong>argument</strong> &#8211; <p>a mapped class, or actual <a class="reference internal" href="mapper_config.html#sqlalchemy.orm.mapper.Mapper" title="sqlalchemy.orm.mapper.Mapper"><tt class="xref py py-class docutils literal"><span class="pre">Mapper</span></tt></a> instance, representing the target of
  1233. the relationship.</p>
  1234. <p><tt class="docutils literal"><span class="pre">argument</span></tt> may also be passed as a callable function
  1235. which is evaluated at mapper initialization time, and may be passed as a
  1236. Python-evaluable string when using Declarative.</p>
  1237. </li>
  1238. <li><strong>secondary</strong> &#8211; <p>for a many-to-many relationship, specifies the intermediary
  1239. table, and is an instance of <a class="reference internal" href="../core/schema.html#sqlalchemy.schema.Table" title="sqlalchemy.schema.Table"><tt class="xref py py-class docutils literal"><span class="pre">Table</span></tt></a>. The <tt class="docutils literal"><span class="pre">secondary</span></tt> keyword
  1240. argument should generally only
  1241. be used for a table that is not otherwise expressed in any class
  1242. mapping, unless this relationship is declared as view only, otherwise
  1243. conflicting persistence operations can occur.</p>
  1244. <p><tt class="docutils literal"><span class="pre">secondary</span></tt> may
  1245. also be passed as a callable function which is evaluated at
  1246. mapper initialization time.</p>
  1247. </li>
  1248. <li><strong>active_history=False</strong> &#8211; When <tt class="docutils literal"><span class="pre">True</span></tt>, indicates that the &#8220;previous&#8221; value for a
  1249. many-to-one reference should be loaded when replaced, if
  1250. not already loaded. Normally, history tracking logic for
  1251. simple many-to-ones only needs to be aware of the &#8220;new&#8221;
  1252. value in order to perform a flush. This flag is available
  1253. for applications that make use of
  1254. <a class="reference internal" href="session.html#sqlalchemy.orm.attributes.get_history" title="sqlalchemy.orm.attributes.get_history"><tt class="xref py py-func docutils literal"><span class="pre">attributes.get_history()</span></tt></a> which also need to know
  1255. the &#8220;previous&#8221; value of the attribute.</li>
  1256. <li><strong>backref</strong> &#8211; indicates the string name of a property to be placed on the related
  1257. mapper&#8217;s class that will handle this relationship in the other
  1258. direction. The other property will be created automatically
  1259. when the mappers are configured. Can also be passed as a
  1260. <a class="reference internal" href="#sqlalchemy.orm.backref" title="sqlalchemy.orm.backref"><tt class="xref py py-func docutils literal"><span class="pre">backref()</span></tt></a> object to control the configuration of the
  1261. new relationship.</li>
  1262. <li><strong>back_populates</strong> &#8211; Takes a string name and has the same meaning as <tt class="docutils literal"><span class="pre">backref</span></tt>,
  1263. except the complementing property is <strong>not</strong> created automatically,
  1264. and instead must be configured explicitly on the other mapper. The
  1265. complementing property should also indicate <tt class="docutils literal"><span class="pre">back_populates</span></tt>
  1266. to this relationship to ensure proper functioning.</li>
  1267. <li><strong>cascade</strong> &#8211; <blockquote>
  1268. <div>a comma-separated list of cascade rules which determines how
  1269. Session operations should be &#8220;cascaded&#8221; from parent to child.
  1270. This defaults to <tt class="docutils literal"><span class="pre">False</span></tt>, which means the default cascade
  1271. should be used. The default value is <tt class="docutils literal"><span class="pre">&quot;save-update,</span> <span class="pre">merge&quot;</span></tt>.<p>Available cascades are:</p>
  1272. <ul>
  1273. <li><tt class="docutils literal"><span class="pre">save-update</span></tt> - cascade the <a class="reference internal" href="session.html#sqlalchemy.orm.session.Session.add" title="sqlalchemy.orm.session.Session.add"><tt class="xref py py-meth docutils literal"><span class="pre">Session.add()</span></tt></a>
  1274. operation. This cascade applies both to future and
  1275. past calls to <a class="reference internal" href="session.html#sqlalchemy.orm.session.Session.add" title="sqlalchemy.orm.session.Session.add"><tt class="xref py py-meth docutils literal"><span class="pre">add()</span></tt></a>,
  1276. meaning new items added to a collection or scalar relationship
  1277. get placed into the same session as that of the parent, and
  1278. also applies to items which have been removed from this
  1279. relationship but are still part of unflushed history.</li>
  1280. <li><tt class="docutils literal"><span class="pre">merge</span></tt> - cascade the <a class="reference internal" href="session.html#sqlalchemy.orm.session.Session.merge" title="sqlalchemy.orm.session.Session.merge"><tt class="xref py py-meth docutils literal"><span class="pre">merge()</span></tt></a>
  1281. operation</li>
  1282. <li><tt class="docutils literal"><span class="pre">expunge</span></tt> - cascade the <a class="reference internal" href="session.html#sqlalchemy.orm.session.Session.expunge" title="sqlalchemy.orm.session.Session.expunge"><tt class="xref py py-meth docutils literal"><span class="pre">Session.expunge()</span></tt></a>
  1283. operation</li>
  1284. <li><tt class="docutils literal"><span class="pre">delete</span></tt> - cascade the <a class="reference internal" href="session.html#sqlalchemy.orm.session.Session.delete" title="sqlalchemy.orm.session.Session.delete"><tt class="xref py py-meth docutils literal"><span class="pre">Session.delete()</span></tt></a>
  1285. operation</li>
  1286. <li><tt class="docutils literal"><span class="pre">delete-orphan</span></tt> - if an item of the child&#8217;s type is
  1287. detached from its parent, mark it for deletion.<p class="versionchanged">
  1288. <span class="versionmodified">Changed in version 0.7: </span>This option does not prevent
  1289. a new instance of the child object from being persisted
  1290. without a parent to start with; to constrain against
  1291. that case, ensure the child&#8217;s foreign key column(s)
  1292. is configured as NOT NULL</p>
  1293. </li>
  1294. <li><tt class="docutils literal"><span class="pre">refresh-expire</span></tt> - cascade the <a class="reference internal" href="session.html#sqlalchemy.orm.session.Session.expire" title="sqlalchemy.orm.session.Session.expire"><tt class="xref py py-meth docutils literal"><span class="pre">Session.expire()</span></tt></a>
  1295. and <a class="reference internal" href="session.html#sqlalchemy.orm.session.Session.refresh" title="sqlalchemy.orm.session.Session.refresh"><tt class="xref py py-meth docutils literal"><span class="pre">refresh()</span></tt></a> operations</li>
  1296. <li><tt class="docutils literal"><span class="pre">all</span></tt> - shorthand for &#8220;save-update,merge, refresh-expire,
  1297. expunge, delete&#8221;</li>
  1298. </ul>
  1299. </div></blockquote>
  1300. <p>See the section <a class="reference internal" href="session.html#unitofwork-cascades"><em>Cascades</em></a> for more background
  1301. on configuring cascades.</p>
  1302. </li>
  1303. <li><strong>cascade_backrefs=True</strong> &#8211; <p>a boolean value indicating if the <tt class="docutils literal"><span class="pre">save-update</span></tt> cascade should
  1304. operate along an assignment event intercepted by a backref.
  1305. When set to <tt class="docutils literal"><span class="pre">False</span></tt>,
  1306. the attribute managed by this relationship will not cascade
  1307. an incoming transient object into the session of a
  1308. persistent parent, if the event is received via backref.</p>
  1309. <p>That is:</p>
  1310. <div class="highlight-python"><div class="highlight"><pre><span class="n">mapper</span><span class="p">(</span><span class="n">A</span><span class="p">,</span> <span class="n">a_table</span><span class="p">,</span> <span class="n">properties</span><span class="o">=</span><span class="p">{</span>
  1311. <span class="s">&#39;bs&#39;</span><span class="p">:</span><span class="n">relationship</span><span class="p">(</span><span class="n">B</span><span class="p">,</span> <span class="n">backref</span><span class="o">=</span><span class="s">&quot;a&quot;</span><span class="p">,</span> <span class="n">cascade_backrefs</span><span class="o">=</span><span class="bp">False</span><span class="p">)</span>
  1312. <span class="p">})</span></pre></div>
  1313. </div>
  1314. <p>If an <tt class="docutils literal"><span class="pre">A()</span></tt> is present in the session, assigning it to
  1315. the &#8220;a&#8221; attribute on a transient <tt class="docutils literal"><span class="pre">B()</span></tt> will not place
  1316. the <tt class="docutils literal"><span class="pre">B()</span></tt> into the session. To set the flag in the other
  1317. direction, i.e. so that <tt class="docutils literal"><span class="pre">A().bs.append(B())</span></tt> won&#8217;t add
  1318. a transient <tt class="docutils literal"><span class="pre">A()</span></tt> into the session for a persistent <tt class="docutils literal"><span class="pre">B()</span></tt>:</p>
  1319. <div class="highlight-python"><div class="highlight"><pre><span class="n">mapper</span><span class="p">(</span><span class="n">A</span><span class="p">,</span> <span class="n">a_table</span><span class="p">,</span> <span class="n">properties</span><span class="o">=</span><span class="p">{</span>
  1320. <span class="s">&#39;bs&#39;</span><span class="p">:</span><span class="n">relationship</span><span class="p">(</span><span class="n">B</span><span class="p">,</span>
  1321. <span class="n">backref</span><span class="o">=</span><span class="n">backref</span><span class="p">(</span><span class="s">&quot;a&quot;</span><span class="p">,</span> <span class="n">cascade_backrefs</span><span class="o">=</span><span class="bp">False</span><span class="p">)</span>
  1322. <span class="p">)</span>
  1323. <span class="p">})</span></pre></div>
  1324. </div>
  1325. <p>See the section <a class="reference internal" href="session.html#unitofwork-cascades"><em>Cascades</em></a> for more background
  1326. on configuring cascades.</p>
  1327. </li>
  1328. <li><strong>collection_class</strong> &#8211; a class or callable that returns a new list-holding object. will
  1329. be used in place of a plain list for storing elements.
  1330. Behavior of this attribute is described in detail at
  1331. <a class="reference internal" href="collections.html#custom-collections"><em>Customizing Collection Access</em></a>.</li>
  1332. <li><strong>comparator_factory</strong> &#8211; a class which extends <a class="reference internal" href="internals.html#sqlalchemy.orm.properties.RelationshipProperty.Comparator" title="sqlalchemy.orm.properties.RelationshipProperty.Comparator"><tt class="xref py py-class docutils literal"><span class="pre">RelationshipProperty.Comparator</span></tt></a> which
  1333. provides custom SQL clause generation for comparison operations.</li>
  1334. <li><strong>doc</strong> &#8211; docstring which will be applied to the resulting descriptor.</li>
  1335. <li><strong>extension</strong> &#8211; an <a class="reference internal" href="deprecated.html#sqlalchemy.orm.interfaces.AttributeExtension" title="sqlalchemy.orm.interfaces.AttributeExtension"><tt class="xref py py-class docutils literal"><span class="pre">AttributeExtension</span></tt></a> instance, or list of extensions,
  1336. which will be prepended to the list of attribute listeners for
  1337. the resulting descriptor placed on the class.
  1338. <strong>Deprecated.</strong> Please see <a class="reference internal" href="events.html#sqlalchemy.orm.events.AttributeEvents" title="sqlalchemy.orm.events.AttributeEvents"><tt class="xref py py-class docutils literal"><span class="pre">AttributeEvents</span></tt></a>.</li>
  1339. <li><strong>foreign_keys</strong> &#8211; <p>a list of columns which are to be used as &#8220;foreign key&#8221; columns.
  1340. Normally, <a class="reference internal" href="#sqlalchemy.orm.relationship" title="sqlalchemy.orm.relationship"><tt class="xref py py-func docutils literal"><span class="pre">relationship()</span></tt></a> uses the <a class="reference internal" href="../core/schema.html#sqlalchemy.schema.ForeignKey" title="sqlalchemy.schema.ForeignKey"><tt class="xref py py-class docutils literal"><span class="pre">ForeignKey</span></tt></a>
  1341. and <a class="reference internal" href="../core/schema.html#sqlalchemy.schema.ForeignKeyConstraint" title="sqlalchemy.schema.ForeignKeyConstraint"><tt class="xref py py-class docutils literal"><span class="pre">ForeignKeyConstraint</span></tt></a> objects present within the
  1342. mapped or secondary <a class="reference internal" href="../core/schema.html#sqlalchemy.schema.Table" title="sqlalchemy.schema.Table"><tt class="xref py py-class docutils literal"><span class="pre">Table</span></tt></a> to determine the &#8220;foreign&#8221; side of
  1343. the join condition. This is used to construct SQL clauses in order
  1344. to load objects, as well as to &#8220;synchronize&#8221; values from
  1345. primary key columns to referencing foreign key columns.
  1346. The <tt class="docutils literal"><span class="pre">foreign_keys</span></tt> parameter overrides the notion of what&#8217;s
  1347. &#8220;foreign&#8221; in the table metadata, allowing the specification
  1348. of a list of <a class="reference internal" href="../core/schema.html#sqlalchemy.schema.Column" title="sqlalchemy.schema.Column"><tt class="xref py py-class docutils literal"><span class="pre">Column</span></tt></a> objects that should be considered
  1349. part of the foreign key.</p>
  1350. <p>There are only two use cases for <tt class="docutils literal"><span class="pre">foreign_keys</span></tt> - one, when it is not
  1351. convenient for <a class="reference internal" href="../core/schema.html#sqlalchemy.schema.Table" title="sqlalchemy.schema.Table"><tt class="xref py py-class docutils literal"><span class="pre">Table</span></tt></a> metadata to contain its own foreign key
  1352. metadata (which should be almost never, unless reflecting a large amount of
  1353. tables from a MySQL MyISAM schema, or a schema that doesn&#8217;t actually
  1354. have foreign keys on it). The other is for extremely
  1355. rare and exotic composite foreign key setups where some columns
  1356. should artificially not be considered as foreign.</p>
  1357. <p><tt class="docutils literal"><span class="pre">foreign_keys</span></tt> may also be passed as a callable function
  1358. which is evaluated at mapper initialization time, and may be passed as a
  1359. Python-evaluable string when using Declarative.</p>
  1360. </li>
  1361. <li><strong>innerjoin=False</strong> &#8211; <p>when <tt class="docutils literal"><span class="pre">True</span></tt>, joined eager loads will use an inner join to join
  1362. against related tables instead of an outer join. The purpose
  1363. of this option is generally one of performance, as inner joins
  1364. generally perform better than outer joins. Another reason can be
  1365. the use of <tt class="docutils literal"><span class="pre">with_lockmode</span></tt>, which does not support outer joins.</p>
  1366. <p>This flag can be set to <tt class="docutils literal"><span class="pre">True</span></tt> when the relationship references an
  1367. object via many-to-one using local foreign keys that are not nullable,
  1368. or when the reference is one-to-one or a collection that is guaranteed
  1369. to have one or at least one entry.</p>
  1370. </li>
  1371. <li><strong>join_depth</strong> &#8211; when non-<tt class="docutils literal"><span class="pre">None</span></tt>, an integer value indicating how many levels
  1372. deep &#8220;eager&#8221; loaders should join on a self-referring or cyclical
  1373. relationship. The number counts how many times the same Mapper
  1374. shall be present in the loading condition along a particular join
  1375. branch. When left at its default of <tt class="docutils literal"><span class="pre">None</span></tt>, eager loaders
  1376. will stop chaining when they encounter a the same target mapper
  1377. which is already higher up in the chain. This option applies
  1378. both to joined- and subquery- eager loaders.</li>
  1379. <li><strong>lazy=&#8217;select&#8217;</strong> &#8211; <p>specifies
  1380. how the related items should be loaded. Default value is
  1381. <tt class="docutils literal"><span class="pre">select</span></tt>. Values include:</p>
  1382. <ul>
  1383. <li><tt class="docutils literal"><span class="pre">select</span></tt> - items should be loaded lazily when the property is first
  1384. accessed, using a separate SELECT statement, or identity map
  1385. fetch for simple many-to-one references.</li>
  1386. <li><tt class="docutils literal"><span class="pre">immediate</span></tt> - items should be loaded as the parents are loaded,
  1387. using a separate SELECT statement, or identity map fetch for
  1388. simple many-to-one references.<p class="versionadded">
  1389. <span class="versionmodified">New in version 0.6.5.</span></p>
  1390. </li>
  1391. <li><tt class="docutils literal"><span class="pre">joined</span></tt> - items should be loaded &#8220;eagerly&#8221; in the same query as
  1392. that of the parent, using a JOIN or LEFT OUTER JOIN. Whether
  1393. the join is &#8220;outer&#8221; or not is determined by the <tt class="docutils literal"><span class="pre">innerjoin</span></tt>
  1394. parameter.</li>
  1395. <li><tt class="docutils literal"><span class="pre">subquery</span></tt> - items should be loaded &#8220;eagerly&#8221; within the same
  1396. query as that of the parent, using a second SQL statement
  1397. which issues a JOIN to a subquery of the original
  1398. statement.</li>
  1399. <li><tt class="docutils literal"><span class="pre">noload</span></tt> - no loading should occur at any time. This is to
  1400. support &#8220;write-only&#8221; attributes, or attributes which are
  1401. populated in some manner specific to the application.</li>
  1402. <li><tt class="docutils literal"><span class="pre">dynamic</span></tt> - the attribute will return a pre-configured
  1403. <a class="reference internal" href="query.html#sqlalchemy.orm.query.Query" title="sqlalchemy.orm.query.Query"><tt class="xref py py-class docutils literal"><span class="pre">Query</span></tt></a> object for all read
  1404. operations, onto which further filtering operations can be
  1405. applied before iterating the results. See
  1406. the section <a class="reference internal" href="collections.html#dynamic-relationship"><em>Dynamic Relationship Loaders</em></a> for more details.</li>
  1407. <li>True - a synonym for &#8216;select&#8217;</li>
  1408. <li>False - a synonym for &#8216;joined&#8217;</li>
  1409. <li>None - a synonym for &#8216;noload&#8217;</li>
  1410. </ul>
  1411. <p>Detailed discussion of loader strategies is at <a class="reference internal" href="loading.html"><em>Relationship Loading Techniques</em></a>.</p>
  1412. </li>
  1413. <li><strong>load_on_pending=False</strong> &#8211; <p>Indicates loading behavior for transient or pending parent objects.</p>
  1414. <p>When set to <tt class="docutils literal"><span class="pre">True</span></tt>, causes the lazy-loader to
  1415. issue a query for a parent object that is not persistent, meaning it has
  1416. never been flushed. This may take effect for a pending object when
  1417. autoflush is disabled, or for a transient object that has been
  1418. &#8220;attached&#8221; to a <a class="reference internal" href="session.html#sqlalchemy.orm.session.Session" title="sqlalchemy.orm.session.Session"><tt class="xref py py-class docutils literal"><span class="pre">Session</span></tt></a> but is not part of its pending
  1419. collection. Attachment of transient objects to the session without
  1420. moving to the &#8220;pending&#8221; state is not a supported behavior at this time.</p>
  1421. <p>Note that the load of related objects on a pending or transient object
  1422. also does not trigger any attribute change events - no user-defined
  1423. events will be emitted for these attributes, and if and when the
  1424. object is ultimately flushed, only the user-specific foreign key
  1425. attributes will be part of the modified state.</p>
  1426. <p>The load_on_pending flag does not improve behavior
  1427. when the ORM is used normally - object references should be constructed
  1428. at the object level, not at the foreign key level, so that they
  1429. are present in an ordinary way before flush() proceeds. This flag
  1430. is not not intended for general use.</p>
  1431. <p>New in 0.6.5.</p>
  1432. </li>
  1433. <li><strong>order_by</strong> &#8211; <p>indicates the ordering that should be applied when loading these
  1434. items. <tt class="docutils literal"><span class="pre">order_by</span></tt> is expected to refer to one of the <a class="reference internal" href="../core/schema.html#sqlalchemy.schema.Column" title="sqlalchemy.schema.Column"><tt class="xref py py-class docutils literal"><span class="pre">Column</span></tt></a>
  1435. objects to which the target class is mapped, or
  1436. the attribute itself bound to the target class which refers
  1437. to the column.</p>
  1438. <p><tt class="docutils literal"><span class="pre">order_by</span></tt> may also be passed as a callable function
  1439. which is evaluated at mapper initialization time, and may be passed as a
  1440. Python-evaluable string when using Declarative.</p>
  1441. </li>
  1442. <li><strong>passive_deletes=False</strong> &#8211; <p>Indicates loading behavior during delete operations.</p>
  1443. <p>A value of True indicates that unloaded child items should not
  1444. be loaded during a delete operation on the parent. Normally,
  1445. when a parent item is deleted, all child items are loaded so
  1446. that they can either be marked as deleted, or have their
  1447. foreign key to the parent set to NULL. Marking this flag as
  1448. True usually implies an ON DELETE &lt;CASCADE|SET NULL&gt; rule is in
  1449. place which will handle updating/deleting child rows on the
  1450. database side.</p>
  1451. <p>Additionally, setting the flag to the string value &#8216;all&#8217; will
  1452. disable the &#8220;nulling out&#8221; of the child foreign keys, when there
  1453. is no delete or delete-orphan cascade enabled. This is
  1454. typically used when a triggering or error raise scenario is in
  1455. place on the database side. Note that the foreign key
  1456. attributes on in-session child objects will not be changed
  1457. after a flush occurs so this is a very special use-case
  1458. setting.</p>
  1459. </li>
  1460. <li><strong>passive_updates=True</strong> &#8211; <p>Indicates loading and INSERT/UPDATE/DELETE behavior when the
  1461. source of a foreign key value changes (i.e. an &#8220;on update&#8221;
  1462. cascade), which are typically the primary key columns of the
  1463. source row.</p>
  1464. <p>When True, it is assumed that ON UPDATE CASCADE is configured on
  1465. the foreign key in the database, and that the database will
  1466. handle propagation of an UPDATE from a source column to
  1467. dependent rows. Note that with databases which enforce
  1468. referential integrity (i.e. PostgreSQL, MySQL with InnoDB tables),
  1469. ON UPDATE CASCADE is required for this operation. The
  1470. relationship() will update the value of the attribute on related
  1471. items which are locally present in the session during a flush.</p>
  1472. <p>When False, it is assumed that the database does not enforce
  1473. referential integrity and will not be issuing its own CASCADE
  1474. operation for an update. The relationship() will issue the
  1475. appropriate UPDATE statements to the database in response to the
  1476. change of a referenced key, and items locally present in the
  1477. session during a flush will also be refreshed.</p>
  1478. <p>This flag should probably be set to False if primary key changes
  1479. are expected and the database in use doesn&#8217;t support CASCADE
  1480. (i.e. SQLite, MySQL MyISAM tables).</p>
  1481. <p>Also see the passive_updates flag on <tt class="docutils literal"><span class="pre">mapper()</span></tt>.</p>
  1482. <p>A future SQLAlchemy release will provide a &#8220;detect&#8221; feature for
  1483. this flag.</p>
  1484. </li>
  1485. <li><strong>post_update</strong> &#8211; this indicates that the relationship should be handled by a
  1486. second UPDATE statement after an INSERT or before a
  1487. DELETE. Currently, it also will issue an UPDATE after the
  1488. instance was UPDATEd as well, although this technically should
  1489. be improved. This flag is used to handle saving bi-directional
  1490. dependencies between two individual rows (i.e. each row
  1491. references the other), where it would otherwise be impossible to
  1492. INSERT or DELETE both rows fully since one row exists before the
  1493. other. Use this flag when a particular mapping arrangement will
  1494. incur two rows that are dependent on each other, such as a table
  1495. that has a one-to-many relationship to a set of child rows, and
  1496. also has a column that references a single child row within that
  1497. list (i.e. both tables contain a foreign key to each other). If
  1498. a <tt class="docutils literal"><span class="pre">flush()</span></tt> operation returns an error that a &#8220;cyclical
  1499. dependency&#8221; was detected, this is a cue that you might want to
  1500. use <tt class="docutils literal"><span class="pre">post_update</span></tt> to &#8220;break&#8221; the cycle.</li>
  1501. <li><strong>primaryjoin</strong> &#8211; <p>a SQL expression that will be used as the primary
  1502. join of this child object against the parent object, or in a
  1503. many-to-many relationship the join of the primary object to the
  1504. association table. By default, this value is computed based on the
  1505. foreign key relationships of the parent and child tables (or association
  1506. table).</p>
  1507. <p><tt class="docutils literal"><span class="pre">primaryjoin</span></tt> may also be passed as a callable function
  1508. which is evaluated at mapper initialization time, and may be passed as a
  1509. Python-evaluable string when using Declarative.</p>
  1510. </li>
  1511. <li><strong>remote_side</strong> &#8211; <p>used for self-referential relationships, indicates the column or
  1512. list of columns that form the &#8220;remote side&#8221; of the relationship.</p>
  1513. <p><tt class="docutils literal"><span class="pre">remote_side</span></tt> may also be passed as a callable function
  1514. which is evaluated at mapper initialization time, and may be passed as a
  1515. Python-evaluable string when using Declarative.</p>
  1516. </li>
  1517. <li><strong>query_class</strong> &#8211; a <a class="reference internal" href="query.html#sqlalchemy.orm.query.Query" title="sqlalchemy.orm.query.Query"><tt class="xref py py-class docutils literal"><span class="pre">Query</span></tt></a> subclass that will be used as the base of the
  1518. &#8220;appender query&#8221; returned by a &#8220;dynamic&#8221; relationship, that
  1519. is, a relationship that specifies <tt class="docutils literal"><span class="pre">lazy=&quot;dynamic&quot;</span></tt> or was
  1520. otherwise constructed using the <a class="reference internal" href="#sqlalchemy.orm.dynamic_loader" title="sqlalchemy.orm.dynamic_loader"><tt class="xref py py-func docutils literal"><span class="pre">orm.dynamic_loader()</span></tt></a>
  1521. function.</li>
  1522. <li><strong>secondaryjoin</strong> &#8211; <p>a SQL expression that will be used as the join of
  1523. an association table to the child object. By default, this value is
  1524. computed based on the foreign key relationships of the association and
  1525. child tables.</p>
  1526. <p><tt class="docutils literal"><span class="pre">secondaryjoin</span></tt> may also be passed as a callable function
  1527. which is evaluated at mapper initialization time, and may be passed as a
  1528. Python-evaluable string when using Declarative.</p>
  1529. </li>
  1530. <li><strong>single_parent=(True|False)</strong> &#8211; when True, installs a validator which will prevent objects
  1531. from being associated with more than one parent at a time.
  1532. This is used for many-to-one or many-to-many relationships that
  1533. should be treated either as one-to-one or one-to-many. Its
  1534. usage is optional unless delete-orphan cascade is also
  1535. set on this relationship(), in which case its required.</li>
  1536. <li><strong>uselist=(True|False)</strong> &#8211; a boolean that indicates if this property should be loaded as a
  1537. list or a scalar. In most cases, this value is determined
  1538. automatically by <tt class="docutils literal"><span class="pre">relationship()</span></tt>, based on the type and direction
  1539. of the relationship - one to many forms a list, many to one
  1540. forms a scalar, many to many is a list. If a scalar is desired
  1541. where normally a list would be present, such as a bi-directional
  1542. one-to-one relationship, set uselist to False.</li>
  1543. <li><strong>viewonly=False</strong> &#8211; when set to True, the relationship is used only for loading objects
  1544. within the relationship, and has no effect on the unit-of-work
  1545. flush process. Relationships with viewonly can specify any kind of
  1546. join conditions to provide additional views of related objects
  1547. onto a parent object. Note that the functionality of a viewonly
  1548. relationship has its limits - complicated join conditions may
  1549. not compile into eager or lazy loaders properly. If this is the
  1550. case, use an alternative method.</li>
  1551. </ul>
  1552. </td>
  1553. </tr>
  1554. </tbody>
  1555. </table>
  1556. </dd></dl>
  1557. <dl class="function">
  1558. <dt id="sqlalchemy.orm.backref">
  1559. <tt class="descclassname">sqlalchemy.orm.</tt><tt class="descname">backref</tt><big>(</big><em>name</em>, <em>**kwargs</em><big>)</big><a class="headerlink" href="#sqlalchemy.orm.backref" title="Permalink to this definition">ś</a></dt>
  1560. <dd><p>Create a back reference with explicit keyword arguments, which are the same
  1561. arguments one can send to <a class="reference internal" href="#sqlalchemy.orm.relationship" title="sqlalchemy.orm.relationship"><tt class="xref py py-func docutils literal"><span class="pre">relationship()</span></tt></a>.</p>
  1562. <p>Used with the <tt class="docutils literal"><span class="pre">backref</span></tt> keyword argument to <a class="reference internal" href="#sqlalchemy.orm.relationship" title="sqlalchemy.orm.relationship"><tt class="xref py py-func docutils literal"><span class="pre">relationship()</span></tt></a> in
  1563. place of a string argument, e.g.:</p>
  1564. <div class="highlight-python"><pre>'items':relationship(SomeItem, backref=backref('parent', lazy='subquery'))</pre>
  1565. </div>
  1566. </dd></dl>
  1567. <dl class="function">
  1568. <dt id="sqlalchemy.orm.relation">
  1569. <tt class="descclassname">sqlalchemy.orm.</tt><tt class="descname">relation</tt><big>(</big><em>*arg</em>, <em>**kw</em><big>)</big><a class="headerlink" href="#sqlalchemy.orm.relation" title="Permalink to this definition">ś</a></dt>
  1570. <dd><p>A synonym for <a class="reference internal" href="#sqlalchemy.orm.relationship" title="sqlalchemy.orm.relationship"><tt class="xref py py-func docutils literal"><span class="pre">relationship()</span></tt></a>.</p>
  1571. </dd></dl>
  1572. <dl class="function">
  1573. <dt id="sqlalchemy.orm.dynamic_loader">
  1574. <tt class="descclassname">sqlalchemy.orm.</tt><tt class="descname">dynamic_loader</tt><big>(</big><em>argument</em>, <em>**kw</em><big>)</big><a class="headerlink" href="#sqlalchemy.orm.dynamic_loader" title="Permalink to this definition">ś</a></dt>
  1575. <dd><p>Construct a dynamically-loading mapper property.</p>
  1576. <p>This is essentially the same as
  1577. using the <tt class="docutils literal"><span class="pre">lazy='dynamic'</span></tt> argument with <a class="reference internal" href="#sqlalchemy.orm.relationship" title="sqlalchemy.orm.relationship"><tt class="xref py py-func docutils literal"><span class="pre">relationship()</span></tt></a>:</p>
  1578. <div class="highlight-python"><div class="highlight"><pre><span class="n">dynamic_loader</span><span class="p">(</span><span class="n">SomeClass</span><span class="p">)</span>
  1579. <span class="c"># is the same as</span>
  1580. <span class="n">relationship</span><span class="p">(</span><span class="n">SomeClass</span><span class="p">,</span> <span class="n">lazy</span><span class="o">=</span><span class="s">&quot;dynamic&quot;</span><span class="p">)</span></pre></div>
  1581. </div>
  1582. <p>See the section <a class="reference internal" href="collections.html#dynamic-relationship"><em>Dynamic Relationship Loaders</em></a> for more details
  1583. on dynamic loading.</p>
  1584. </dd></dl>
  1585. </div>
  1586. </div>
  1587. </div>
  1588. </div>
  1589. <div id="docs-bottom-navigation" class="docs-navigation-links">
  1590. Previous:
  1591. <a href="mapper_config.html" title="previous chapter">Mapper Configuration</a>
  1592. Next:
  1593. <a href="collections.html" title="next chapter">Collection Configuration and Techniques</a>
  1594. <div id="docs-copyright">
  1595. &copy; <a href="../copyright.html">Copyright</a> 2007-2012, the SQLAlchemy authors and contributors.
  1596. Created using <a href="http://sphinx.pocoo.org/">Sphinx</a> 1.1.3.
  1597. </div>
  1598. </div>
  1599. </div>
  1600. </body>
  1601. </html>