Spaces:
Paused
Paused
| <html lang="en"> | |
| <head> | |
| <meta http-equiv="Content-Type" content="text/html; charset=utf-8"> | |
| <title>Coverage for tinytroupe/utils/json.py: 11%</title> | |
| <link rel="icon" sizes="32x32" href="favicon_32_cb_c827f16f.png"> | |
| <link rel="stylesheet" href="style_cb_9ff733b0.css" type="text/css"> | |
| <script src="coverage_html_cb_dd2e7eb5.js" defer></script> | |
| </head> | |
| <body class="pyfile"> | |
| <header> | |
| <div class="content"> | |
| <h1> | |
| <span class="text">Coverage for </span><b>tinytroupe / utils / json.py</b>: | |
| <span class="pc_cov">11%</span> | |
| </h1> | |
| <aside id="help_panel_wrapper"> | |
| <input id="help_panel_state" type="checkbox"> | |
| <label for="help_panel_state"> | |
| <img id="keyboard_icon" src="keybd_closed_cb_900cfef5.png" alt="Show/hide keyboard shortcuts"> | |
| </label> | |
| <div id="help_panel"> | |
| <p class="legend">Shortcuts on this page</p> | |
| <div class="keyhelp"> | |
| <p> | |
| <kbd>r</kbd> | |
| <kbd>m</kbd> | |
| <kbd>x</kbd> | |
| toggle line displays | |
| </p> | |
| <p> | |
| <kbd>j</kbd> | |
| <kbd>k</kbd> | |
| next/prev highlighted chunk | |
| </p> | |
| <p> | |
| <kbd>0</kbd> (zero) top of page | |
| </p> | |
| <p> | |
| <kbd>1</kbd> (one) first highlighted chunk | |
| </p> | |
| <p> | |
| <kbd>[</kbd> | |
| <kbd>]</kbd> | |
| prev/next file | |
| </p> | |
| <p> | |
| <kbd>u</kbd> up to the index | |
| </p> | |
| <p> | |
| <kbd>?</kbd> show/hide this help | |
| </p> | |
| </div> | |
| </div> | |
| </aside> | |
| <h2> | |
| <span class="text">167 statements </span> | |
| <button type="button" class="run button_toggle_run" value="run" data-shortcut="r" title="Toggle lines run">18<span class="text"> run</span></button> | |
| <button type="button" class="mis show_mis button_toggle_mis" value="mis" data-shortcut="m" title="Toggle lines missing">149<span class="text"> missing</span></button> | |
| <button type="button" class="exc show_exc button_toggle_exc" value="exc" data-shortcut="x" title="Toggle lines excluded">0<span class="text"> excluded</span></button> | |
| </h2> | |
| <p class="text"> | |
| <a id="prevFileLink" class="nav" href="z_4b3d0d3b603de221_config_py.html">« prev</a> | |
| <a id="indexLink" class="nav" href="index.html">^ index</a> | |
| <a id="nextFileLink" class="nav" href="z_4b3d0d3b603de221_llm_py.html">» next</a> | |
| | |
| <a class="nav" href="https://coverage.readthedocs.io/en/7.13.4">coverage.py v7.13.4</a>, | |
| created at 2026-02-28 17:48 +0000 | |
| </p> | |
| <aside class="hidden"> | |
| <button type="button" class="button_next_chunk" data-shortcut="j"></button> | |
| <button type="button" class="button_prev_chunk" data-shortcut="k"></button> | |
| <button type="button" class="button_top_of_page" data-shortcut="0"></button> | |
| <button type="button" class="button_first_chunk" data-shortcut="1"></button> | |
| <button type="button" class="button_prev_file" data-shortcut="["></button> | |
| <button type="button" class="button_next_file" data-shortcut="]"></button> | |
| <button type="button" class="button_to_index" data-shortcut="u"></button> | |
| <button type="button" class="button_show_hide_help" data-shortcut="?"></button> | |
| </aside> | |
| </div> | |
| </header> | |
| <main id="source"> | |
| <p class="run"><span class="n"><a id="t1" href="#t1">1</a></span><span class="t"><span class="key">import</span> <span class="nam">json</span> </span><span class="r"></span></p> | |
| <p class="run"><span class="n"><a id="t2" href="#t2">2</a></span><span class="t"><span class="key">import</span> <span class="nam">copy</span> </span><span class="r"></span></p> | |
| <p class="run"><span class="n"><a id="t3" href="#t3">3</a></span><span class="t"><span class="key">from</span> <span class="nam">pydantic</span> <span class="key">import</span> <span class="nam">BaseModel</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t4" href="#t4">4</a></span><span class="t"> </span><span class="r"></span></p> | |
| <p class="run"><span class="n"><a id="t5" href="#t5">5</a></span><span class="t"><span class="key">from</span> <span class="nam">tinytroupe</span><span class="op">.</span><span class="nam">utils</span> <span class="key">import</span> <span class="nam">logger</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t6" href="#t6">6</a></span><span class="t"> </span><span class="r"></span></p> | |
| <p class="run"><span class="n"><a id="t7" href="#t7">7</a></span><span class="t"><span class="key">class</span> <span class="nam">JsonSerializableRegistry</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t8" href="#t8">8</a></span><span class="t"> <span class="str">"""</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t9" href="#t9">9</a></span><span class="t"><span class="str"> A mixin class that provides JSON serialization, deserialization, and subclass registration.</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t10" href="#t10">10</a></span><span class="t"><span class="str"> """</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t11" href="#t11">11</a></span><span class="t"> </span><span class="r"></span></p> | |
| <p class="run"><span class="n"><a id="t12" href="#t12">12</a></span><span class="t"> <span class="nam">class_mapping</span> <span class="op">=</span> <span class="op">{</span><span class="op">}</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t13" href="#t13">13</a></span><span class="t"> </span><span class="r"></span></p> | |
| <p class="run"><span class="n"><a id="t14" href="#t14">14</a></span><span class="t"> <span class="key">def</span> <span class="nam">to_json</span><span class="op">(</span><span class="nam">self</span><span class="op">,</span> <span class="nam">include</span><span class="op">:</span> <span class="nam">list</span> <span class="op">=</span> <span class="key">None</span><span class="op">,</span> <span class="nam">suppress</span><span class="op">:</span> <span class="nam">list</span> <span class="op">=</span> <span class="key">None</span><span class="op">,</span> <span class="nam">file_path</span><span class="op">:</span> <span class="nam">str</span> <span class="op">=</span> <span class="key">None</span><span class="op">,</span> </span><span class="r"></span></p> | |
| <p class="run run2"><span class="n"><a id="t15" href="#t15">15</a></span><span class="t"> <span class="nam">serialization_type_field_name</span> <span class="op">=</span> <span class="str">"json_serializable_class_name"</span><span class="op">)</span> <span class="op">-></span> <span class="nam">dict</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t16" href="#t16">16</a></span><span class="t"> <span class="str">"""</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t17" href="#t17">17</a></span><span class="t"><span class="str"> Returns a JSON representation of the object.</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t18" href="#t18">18</a></span><span class="t"><span class="str"> </span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t19" href="#t19">19</a></span><span class="t"><span class="str"> Args:</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t20" href="#t20">20</a></span><span class="t"><span class="str"> include (list, optional): Attributes to include in the serialization. Will override the default behavior.</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t21" href="#t21">21</a></span><span class="t"><span class="str"> suppress (list, optional): Attributes to suppress from the serialization. Will override the default behavior.</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t22" href="#t22">22</a></span><span class="t"><span class="str"> file_path (str, optional): Path to a file where the JSON will be written.</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t23" href="#t23">23</a></span><span class="t"><span class="str"> """</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t24" href="#t24">24</a></span><span class="t"> <span class="com"># Gather all serializable attributes from the class hierarchy</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t25" href="#t25">25</a></span><span class="t"> <span class="nam">serializable_attrs</span> <span class="op">=</span> <span class="nam">set</span><span class="op">(</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t26" href="#t26">26</a></span><span class="t"> <span class="nam">suppress_attrs</span> <span class="op">=</span> <span class="nam">set</span><span class="op">(</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t27" href="#t27">27</a></span><span class="t"> <span class="nam">custom_serializers</span> <span class="op">=</span> <span class="op">{</span><span class="op">}</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t28" href="#t28">28</a></span><span class="t"> <span class="key">for</span> <span class="nam">cls</span> <span class="key">in</span> <span class="nam">self</span><span class="op">.</span><span class="nam">__class__</span><span class="op">.</span><span class="nam">__mro__</span><span class="op">:</span> <span class="com"># Traverse the class hierarchy</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t29" href="#t29">29</a></span><span class="t"> <span class="key">if</span> <span class="nam">hasattr</span><span class="op">(</span><span class="nam">cls</span><span class="op">,</span> <span class="str">'serializable_attributes'</span><span class="op">)</span> <span class="key">and</span> <span class="nam">isinstance</span><span class="op">(</span><span class="nam">cls</span><span class="op">.</span><span class="nam">serializable_attributes</span><span class="op">,</span> <span class="nam">list</span><span class="op">)</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t30" href="#t30">30</a></span><span class="t"> <span class="nam">serializable_attrs</span><span class="op">.</span><span class="nam">update</span><span class="op">(</span><span class="nam">cls</span><span class="op">.</span><span class="nam">serializable_attributes</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t31" href="#t31">31</a></span><span class="t"> <span class="key">if</span> <span class="nam">hasattr</span><span class="op">(</span><span class="nam">cls</span><span class="op">,</span> <span class="str">'suppress_attributes_from_serialization'</span><span class="op">)</span> <span class="key">and</span> <span class="nam">isinstance</span><span class="op">(</span><span class="nam">cls</span><span class="op">.</span><span class="nam">suppress_attributes_from_serialization</span><span class="op">,</span> <span class="nam">list</span><span class="op">)</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t32" href="#t32">32</a></span><span class="t"> <span class="nam">suppress_attrs</span><span class="op">.</span><span class="nam">update</span><span class="op">(</span><span class="nam">cls</span><span class="op">.</span><span class="nam">suppress_attributes_from_serialization</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t33" href="#t33">33</a></span><span class="t"> <span class="key">if</span> <span class="nam">hasattr</span><span class="op">(</span><span class="nam">cls</span><span class="op">,</span> <span class="str">'custom_serializers'</span><span class="op">)</span> <span class="key">and</span> <span class="nam">isinstance</span><span class="op">(</span><span class="nam">cls</span><span class="op">.</span><span class="nam">custom_serializers</span><span class="op">,</span> <span class="nam">dict</span><span class="op">)</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t34" href="#t34">34</a></span><span class="t"> <span class="nam">custom_serializers</span><span class="op">.</span><span class="nam">update</span><span class="op">(</span><span class="nam">cls</span><span class="op">.</span><span class="nam">custom_serializers</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t35" href="#t35">35</a></span><span class="t"> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t36" href="#t36">36</a></span><span class="t"> <span class="com"># Override attributes with method parameters if provided</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t37" href="#t37">37</a></span><span class="t"> <span class="key">if</span> <span class="nam">include</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t38" href="#t38">38</a></span><span class="t"> <span class="nam">serializable_attrs</span> <span class="op">=</span> <span class="nam">set</span><span class="op">(</span><span class="nam">include</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t39" href="#t39">39</a></span><span class="t"> <span class="key">if</span> <span class="nam">suppress</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t40" href="#t40">40</a></span><span class="t"> <span class="nam">suppress_attrs</span><span class="op">.</span><span class="nam">update</span><span class="op">(</span><span class="nam">suppress</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t41" href="#t41">41</a></span><span class="t"> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t42" href="#t42">42</a></span><span class="t"> <span class="key">def</span> <span class="nam">aux_serialize_item</span><span class="op">(</span><span class="nam">item</span><span class="op">)</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t43" href="#t43">43</a></span><span class="t"> <span class="key">if</span> <span class="nam">isinstance</span><span class="op">(</span><span class="nam">item</span><span class="op">,</span> <span class="nam">JsonSerializableRegistry</span><span class="op">)</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t44" href="#t44">44</a></span><span class="t"> <span class="key">return</span> <span class="nam">item</span><span class="op">.</span><span class="nam">to_json</span><span class="op">(</span><span class="nam">serialization_type_field_name</span><span class="op">=</span><span class="nam">serialization_type_field_name</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t45" href="#t45">45</a></span><span class="t"> <span class="key">elif</span> <span class="nam">isinstance</span><span class="op">(</span><span class="nam">item</span><span class="op">,</span> <span class="nam">BaseModel</span><span class="op">)</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t46" href="#t46">46</a></span><span class="t"> <span class="com"># If it's a Pydantic model, convert it to a dict first</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t47" href="#t47">47</a></span><span class="t"> <span class="nam">logger</span><span class="op">.</span><span class="nam">debug</span><span class="op">(</span><span class="fst">f"</span><span class="fst">Serializing Pydantic model: </span><span class="op">{</span><span class="nam">item</span><span class="op">}</span><span class="fst">"</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t48" href="#t48">48</a></span><span class="t"> <span class="key">return</span> <span class="nam">item</span><span class="op">.</span><span class="nam">model_dump</span><span class="op">(</span><span class="nam">mode</span><span class="op">=</span><span class="str">"json"</span><span class="op">,</span> <span class="nam">exclude_unset</span><span class="op">=</span><span class="key">True</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t49" href="#t49">49</a></span><span class="t"> <span class="key">else</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t50" href="#t50">50</a></span><span class="t"> <span class="key">return</span> <span class="nam">copy</span><span class="op">.</span><span class="nam">deepcopy</span><span class="op">(</span><span class="nam">item</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t51" href="#t51">51</a></span><span class="t"> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t52" href="#t52">52</a></span><span class="t"> <span class="nam">result</span> <span class="op">=</span> <span class="op">{</span><span class="nam">serialization_type_field_name</span><span class="op">:</span> <span class="nam">self</span><span class="op">.</span><span class="nam">__class__</span><span class="op">.</span><span class="nam">__name__</span><span class="op">}</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t53" href="#t53">53</a></span><span class="t"> <span class="key">for</span> <span class="nam">attr</span> <span class="key">in</span> <span class="nam">serializable_attrs</span> <span class="key">if</span> <span class="nam">serializable_attrs</span> <span class="key">else</span> <span class="nam">self</span><span class="op">.</span><span class="nam">__dict__</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t54" href="#t54">54</a></span><span class="t"> <span class="key">if</span> <span class="nam">attr</span> <span class="key">not</span> <span class="key">in</span> <span class="nam">suppress_attrs</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t55" href="#t55">55</a></span><span class="t"> <span class="nam">value</span> <span class="op">=</span> <span class="nam">getattr</span><span class="op">(</span><span class="nam">self</span><span class="op">,</span> <span class="nam">attr</span><span class="op">,</span> <span class="key">None</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t56" href="#t56">56</a></span><span class="t"> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t57" href="#t57">57</a></span><span class="t"> <span class="nam">attr_renamed</span> <span class="op">=</span> <span class="nam">self</span><span class="op">.</span><span class="nam">_programmatic_name_to_json_name</span><span class="op">(</span><span class="nam">attr</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t58" href="#t58">58</a></span><span class="t"> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t59" href="#t59">59</a></span><span class="t"> <span class="com"># Check if there's a custom serializer for this attribute</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t60" href="#t60">60</a></span><span class="t"> <span class="key">if</span> <span class="nam">attr</span> <span class="key">in</span> <span class="nam">custom_serializers</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t61" href="#t61">61</a></span><span class="t"> <span class="nam">result</span><span class="op">[</span><span class="nam">attr_renamed</span><span class="op">]</span> <span class="op">=</span> <span class="nam">custom_serializers</span><span class="op">[</span><span class="nam">attr</span><span class="op">]</span><span class="op">(</span><span class="nam">value</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t62" href="#t62">62</a></span><span class="t"> <span class="key">elif</span> <span class="nam">isinstance</span><span class="op">(</span><span class="nam">value</span><span class="op">,</span> <span class="nam">list</span><span class="op">)</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t63" href="#t63">63</a></span><span class="t"> <span class="nam">result</span><span class="op">[</span><span class="nam">attr_renamed</span><span class="op">]</span> <span class="op">=</span> <span class="op">[</span><span class="nam">aux_serialize_item</span><span class="op">(</span><span class="nam">item</span><span class="op">)</span> <span class="key">for</span> <span class="nam">item</span> <span class="key">in</span> <span class="nam">value</span><span class="op">]</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t64" href="#t64">64</a></span><span class="t"> <span class="key">elif</span> <span class="nam">isinstance</span><span class="op">(</span><span class="nam">value</span><span class="op">,</span> <span class="nam">dict</span><span class="op">)</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t65" href="#t65">65</a></span><span class="t"> <span class="nam">result</span><span class="op">[</span><span class="nam">attr_renamed</span><span class="op">]</span> <span class="op">=</span> <span class="op">{</span><span class="nam">k</span><span class="op">:</span> <span class="nam">aux_serialize_item</span><span class="op">(</span><span class="nam">v</span><span class="op">)</span> <span class="key">for</span> <span class="nam">k</span><span class="op">,</span> <span class="nam">v</span> <span class="key">in</span> <span class="nam">value</span><span class="op">.</span><span class="nam">items</span><span class="op">(</span><span class="op">)</span><span class="op">}</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t66" href="#t66">66</a></span><span class="t"> <span class="key">else</span><span class="op">:</span> <span class="com"># isinstance(value, JsonSerializableRegistry) or isinstance(value, BaseModel) or other types</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t67" href="#t67">67</a></span><span class="t"> <span class="nam">result</span><span class="op">[</span><span class="nam">attr_renamed</span><span class="op">]</span> <span class="op">=</span> <span class="nam">aux_serialize_item</span><span class="op">(</span><span class="nam">value</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t68" href="#t68">68</a></span><span class="t"> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t69" href="#t69">69</a></span><span class="t"> <span class="key">if</span> <span class="nam">file_path</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t70" href="#t70">70</a></span><span class="t"> <span class="com"># Create directories if they do not exist</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t71" href="#t71">71</a></span><span class="t"> <span class="key">import</span> <span class="nam">os</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t72" href="#t72">72</a></span><span class="t"> <span class="nam">os</span><span class="op">.</span><span class="nam">makedirs</span><span class="op">(</span><span class="nam">os</span><span class="op">.</span><span class="nam">path</span><span class="op">.</span><span class="nam">dirname</span><span class="op">(</span><span class="nam">file_path</span><span class="op">)</span><span class="op">,</span> <span class="nam">exist_ok</span><span class="op">=</span><span class="key">True</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t73" href="#t73">73</a></span><span class="t"> <span class="key">with</span> <span class="nam">open</span><span class="op">(</span><span class="nam">file_path</span><span class="op">,</span> <span class="str">'w'</span><span class="op">,</span> <span class="nam">encoding</span><span class="op">=</span><span class="str">'utf-8'</span><span class="op">,</span> <span class="nam">errors</span><span class="op">=</span><span class="str">'replace'</span><span class="op">)</span> <span class="key">as</span> <span class="nam">f</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t74" href="#t74">74</a></span><span class="t"> <span class="nam">json</span><span class="op">.</span><span class="nam">dump</span><span class="op">(</span><span class="nam">result</span><span class="op">,</span> <span class="nam">f</span><span class="op">,</span> <span class="nam">indent</span><span class="op">=</span><span class="num">4</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t75" href="#t75">75</a></span><span class="t"> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t76" href="#t76">76</a></span><span class="t"> <span class="key">return</span> <span class="nam">result</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t77" href="#t77">77</a></span><span class="t"> </span><span class="r"></span></p> | |
| <p class="run"><span class="n"><a id="t78" href="#t78">78</a></span><span class="t"> <span class="op">@</span><span class="nam">classmethod</span> </span><span class="r"></span></p> | |
| <p class="run"><span class="n"><a id="t79" href="#t79">79</a></span><span class="t"> <span class="key">def</span> <span class="nam">from_json</span><span class="op">(</span><span class="nam">cls</span><span class="op">,</span> <span class="nam">json_dict_or_path</span><span class="op">,</span> <span class="nam">suppress</span><span class="op">:</span> <span class="nam">list</span> <span class="op">=</span> <span class="key">None</span><span class="op">,</span> </span><span class="r"></span></p> | |
| <p class="run run2"><span class="n"><a id="t80" href="#t80">80</a></span><span class="t"> <span class="nam">serialization_type_field_name</span> <span class="op">=</span> <span class="str">"json_serializable_class_name"</span><span class="op">,</span> </span><span class="r"></span></p> | |
| <p class="run run2"><span class="n"><a id="t81" href="#t81">81</a></span><span class="t"> <span class="nam">post_init_params</span><span class="op">:</span> <span class="nam">dict</span> <span class="op">=</span> <span class="key">None</span><span class="op">)</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t82" href="#t82">82</a></span><span class="t"> <span class="str">"""</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t83" href="#t83">83</a></span><span class="t"><span class="str"> Loads a JSON representation of the object and creates an instance of the class.</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t84" href="#t84">84</a></span><span class="t"><span class="str"> </span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t85" href="#t85">85</a></span><span class="t"><span class="str"> Args:</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t86" href="#t86">86</a></span><span class="t"><span class="str"> json_dict_or_path (dict or str): The JSON dictionary representing the object or a file path to load the JSON from.</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t87" href="#t87">87</a></span><span class="t"><span class="str"> suppress (list, optional): Attributes to suppress from being loaded.</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t88" href="#t88">88</a></span><span class="t"><span class="str"> </span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t89" href="#t89">89</a></span><span class="t"><span class="str"> Returns:</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t90" href="#t90">90</a></span><span class="t"><span class="str"> An instance of the class populated with the data from json_dict_or_path.</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t91" href="#t91">91</a></span><span class="t"><span class="str"> """</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t92" href="#t92">92</a></span><span class="t"> <span class="key">if</span> <span class="nam">isinstance</span><span class="op">(</span><span class="nam">json_dict_or_path</span><span class="op">,</span> <span class="nam">str</span><span class="op">)</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t93" href="#t93">93</a></span><span class="t"> <span class="key">with</span> <span class="nam">open</span><span class="op">(</span><span class="nam">json_dict_or_path</span><span class="op">,</span> <span class="str">'r'</span><span class="op">,</span> <span class="nam">encoding</span><span class="op">=</span><span class="str">'utf-8'</span><span class="op">,</span> <span class="nam">errors</span><span class="op">=</span><span class="str">'replace'</span><span class="op">)</span> <span class="key">as</span> <span class="nam">f</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t94" href="#t94">94</a></span><span class="t"> <span class="nam">json_dict</span> <span class="op">=</span> <span class="nam">json</span><span class="op">.</span><span class="nam">load</span><span class="op">(</span><span class="nam">f</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t95" href="#t95">95</a></span><span class="t"> <span class="key">else</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t96" href="#t96">96</a></span><span class="t"> <span class="nam">json_dict</span> <span class="op">=</span> <span class="nam">json_dict_or_path</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t97" href="#t97">97</a></span><span class="t"> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t98" href="#t98">98</a></span><span class="t"> <span class="nam">subclass_name</span> <span class="op">=</span> <span class="nam">json_dict</span><span class="op">.</span><span class="nam">get</span><span class="op">(</span><span class="nam">serialization_type_field_name</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t99" href="#t99">99</a></span><span class="t"> <span class="nam">target_class</span> <span class="op">=</span> <span class="nam">cls</span><span class="op">.</span><span class="nam">class_mapping</span><span class="op">.</span><span class="nam">get</span><span class="op">(</span><span class="nam">subclass_name</span><span class="op">,</span> <span class="nam">cls</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t100" href="#t100">100</a></span><span class="t"> <span class="nam">instance</span> <span class="op">=</span> <span class="nam">target_class</span><span class="op">.</span><span class="nam">__new__</span><span class="op">(</span><span class="nam">target_class</span><span class="op">)</span> <span class="com"># Create an instance without calling __init__</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t101" href="#t101">101</a></span><span class="t"> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t102" href="#t102">102</a></span><span class="t"> <span class="com"># Gather all serializable attributes from the class hierarchy</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t103" href="#t103">103</a></span><span class="t"> <span class="nam">serializable_attrs</span> <span class="op">=</span> <span class="nam">set</span><span class="op">(</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t104" href="#t104">104</a></span><span class="t"> <span class="nam">custom_deserializers</span> <span class="op">=</span> <span class="op">{</span><span class="op">}</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t105" href="#t105">105</a></span><span class="t"> <span class="nam">suppress_attrs</span> <span class="op">=</span> <span class="nam">set</span><span class="op">(</span><span class="nam">suppress</span><span class="op">)</span> <span class="key">if</span> <span class="nam">suppress</span> <span class="key">else</span> <span class="nam">set</span><span class="op">(</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t106" href="#t106">106</a></span><span class="t"> <span class="key">for</span> <span class="nam">target_mro</span> <span class="key">in</span> <span class="nam">target_class</span><span class="op">.</span><span class="nam">__mro__</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t107" href="#t107">107</a></span><span class="t"> <span class="key">if</span> <span class="nam">hasattr</span><span class="op">(</span><span class="nam">target_mro</span><span class="op">,</span> <span class="str">'serializable_attributes'</span><span class="op">)</span> <span class="key">and</span> <span class="nam">isinstance</span><span class="op">(</span><span class="nam">target_mro</span><span class="op">.</span><span class="nam">serializable_attributes</span><span class="op">,</span> <span class="nam">list</span><span class="op">)</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t108" href="#t108">108</a></span><span class="t"> <span class="nam">serializable_attrs</span><span class="op">.</span><span class="nam">update</span><span class="op">(</span><span class="nam">target_mro</span><span class="op">.</span><span class="nam">serializable_attributes</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t109" href="#t109">109</a></span><span class="t"> <span class="key">if</span> <span class="nam">hasattr</span><span class="op">(</span><span class="nam">target_mro</span><span class="op">,</span> <span class="str">'custom_deserializers'</span><span class="op">)</span> <span class="key">and</span> <span class="nam">isinstance</span><span class="op">(</span><span class="nam">target_mro</span><span class="op">.</span><span class="nam">custom_deserializers</span><span class="op">,</span> <span class="nam">dict</span><span class="op">)</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t110" href="#t110">110</a></span><span class="t"> <span class="nam">custom_deserializers</span><span class="op">.</span><span class="nam">update</span><span class="op">(</span><span class="nam">target_mro</span><span class="op">.</span><span class="nam">custom_deserializers</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t111" href="#t111">111</a></span><span class="t"> <span class="key">if</span> <span class="nam">hasattr</span><span class="op">(</span><span class="nam">target_mro</span><span class="op">,</span> <span class="str">'suppress_attributes_from_serialization'</span><span class="op">)</span> <span class="key">and</span> <span class="nam">isinstance</span><span class="op">(</span><span class="nam">target_mro</span><span class="op">.</span><span class="nam">suppress_attributes_from_serialization</span><span class="op">,</span> <span class="nam">list</span><span class="op">)</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t112" href="#t112">112</a></span><span class="t"> <span class="nam">suppress_attrs</span><span class="op">.</span><span class="nam">update</span><span class="op">(</span><span class="nam">target_mro</span><span class="op">.</span><span class="nam">suppress_attributes_from_serialization</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t113" href="#t113">113</a></span><span class="t"> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t114" href="#t114">114</a></span><span class="t"> <span class="com"># Assign values only for serializable attributes if specified, otherwise assign everything</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t115" href="#t115">115</a></span><span class="t"> <span class="key">for</span> <span class="nam">key</span> <span class="key">in</span> <span class="nam">serializable_attrs</span> <span class="key">if</span> <span class="nam">serializable_attrs</span> <span class="key">else</span> <span class="nam">json_dict</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t116" href="#t116">116</a></span><span class="t"> <span class="nam">key_in_json</span> <span class="op">=</span> <span class="nam">cls</span><span class="op">.</span><span class="nam">_programmatic_name_to_json_name</span><span class="op">(</span><span class="nam">key</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t117" href="#t117">117</a></span><span class="t"> <span class="key">if</span> <span class="nam">key_in_json</span> <span class="key">in</span> <span class="nam">json_dict</span> <span class="key">and</span> <span class="nam">key</span> <span class="key">not</span> <span class="key">in</span> <span class="nam">suppress_attrs</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t118" href="#t118">118</a></span><span class="t"> <span class="nam">value</span> <span class="op">=</span> <span class="nam">json_dict</span><span class="op">[</span><span class="nam">key_in_json</span><span class="op">]</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t119" href="#t119">119</a></span><span class="t"> <span class="key">if</span> <span class="nam">key</span> <span class="key">in</span> <span class="nam">custom_deserializers</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t120" href="#t120">120</a></span><span class="t"> <span class="com"># Use custom initializer if provided</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t121" href="#t121">121</a></span><span class="t"> <span class="nam">setattr</span><span class="op">(</span><span class="nam">instance</span><span class="op">,</span> <span class="nam">key</span><span class="op">,</span> <span class="nam">custom_deserializers</span><span class="op">[</span><span class="nam">key</span><span class="op">]</span><span class="op">(</span><span class="nam">value</span><span class="op">)</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t122" href="#t122">122</a></span><span class="t"> <span class="key">elif</span> <span class="nam">isinstance</span><span class="op">(</span><span class="nam">value</span><span class="op">,</span> <span class="nam">dict</span><span class="op">)</span> <span class="key">and</span> <span class="nam">serialization_type_field_name</span> <span class="key">in</span> <span class="nam">value</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t123" href="#t123">123</a></span><span class="t"> <span class="com"># Assume it's another JsonSerializableRegistry object</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t124" href="#t124">124</a></span><span class="t"> <span class="nam">setattr</span><span class="op">(</span><span class="nam">instance</span><span class="op">,</span> <span class="nam">key</span><span class="op">,</span> <span class="nam">JsonSerializableRegistry</span><span class="op">.</span><span class="nam">from_json</span><span class="op">(</span><span class="nam">value</span><span class="op">,</span> <span class="nam">serialization_type_field_name</span><span class="op">=</span><span class="nam">serialization_type_field_name</span><span class="op">)</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t125" href="#t125">125</a></span><span class="t"> <span class="key">elif</span> <span class="nam">isinstance</span><span class="op">(</span><span class="nam">value</span><span class="op">,</span> <span class="nam">list</span><span class="op">)</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t126" href="#t126">126</a></span><span class="t"> <span class="com"># Handle collections, recursively deserialize if items are JsonSerializableRegistry objects</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t127" href="#t127">127</a></span><span class="t"> <span class="nam">deserialized_collection</span> <span class="op">=</span> <span class="op">[</span><span class="op">]</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t128" href="#t128">128</a></span><span class="t"> <span class="key">for</span> <span class="nam">item</span> <span class="key">in</span> <span class="nam">value</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t129" href="#t129">129</a></span><span class="t"> <span class="key">if</span> <span class="nam">isinstance</span><span class="op">(</span><span class="nam">item</span><span class="op">,</span> <span class="nam">dict</span><span class="op">)</span> <span class="key">and</span> <span class="nam">serialization_type_field_name</span> <span class="key">in</span> <span class="nam">item</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t130" href="#t130">130</a></span><span class="t"> <span class="nam">deserialized_collection</span><span class="op">.</span><span class="nam">append</span><span class="op">(</span><span class="nam">JsonSerializableRegistry</span><span class="op">.</span><span class="nam">from_json</span><span class="op">(</span><span class="nam">item</span><span class="op">,</span> <span class="nam">serialization_type_field_name</span><span class="op">=</span><span class="nam">serialization_type_field_name</span><span class="op">)</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t131" href="#t131">131</a></span><span class="t"> <span class="key">else</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t132" href="#t132">132</a></span><span class="t"> <span class="nam">deserialized_collection</span><span class="op">.</span><span class="nam">append</span><span class="op">(</span><span class="nam">copy</span><span class="op">.</span><span class="nam">deepcopy</span><span class="op">(</span><span class="nam">item</span><span class="op">)</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t133" href="#t133">133</a></span><span class="t"> <span class="nam">setattr</span><span class="op">(</span><span class="nam">instance</span><span class="op">,</span> <span class="nam">key</span><span class="op">,</span> <span class="nam">deserialized_collection</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t134" href="#t134">134</a></span><span class="t"> <span class="key">else</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t135" href="#t135">135</a></span><span class="t"> <span class="nam">setattr</span><span class="op">(</span><span class="nam">instance</span><span class="op">,</span> <span class="nam">key</span><span class="op">,</span> <span class="nam">copy</span><span class="op">.</span><span class="nam">deepcopy</span><span class="op">(</span><span class="nam">value</span><span class="op">)</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t136" href="#t136">136</a></span><span class="t"> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t137" href="#t137">137</a></span><span class="t"> <span class="com"># Call post-deserialization initialization if available</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t138" href="#t138">138</a></span><span class="t"> <span class="key">if</span> <span class="nam">hasattr</span><span class="op">(</span><span class="nam">instance</span><span class="op">,</span> <span class="str">'_post_deserialization_init'</span><span class="op">)</span> <span class="key">and</span> <span class="nam">callable</span><span class="op">(</span><span class="nam">instance</span><span class="op">.</span><span class="nam">_post_deserialization_init</span><span class="op">)</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t139" href="#t139">139</a></span><span class="t"> <span class="nam">post_init_params</span> <span class="op">=</span> <span class="nam">post_init_params</span> <span class="key">if</span> <span class="nam">post_init_params</span> <span class="key">else</span> <span class="op">{</span><span class="op">}</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t140" href="#t140">140</a></span><span class="t"> <span class="nam">instance</span><span class="op">.</span><span class="nam">_post_deserialization_init</span><span class="op">(</span><span class="op">**</span><span class="nam">post_init_params</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t141" href="#t141">141</a></span><span class="t"> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t142" href="#t142">142</a></span><span class="t"> <span class="key">return</span> <span class="nam">instance</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t143" href="#t143">143</a></span><span class="t"> </span><span class="r"></span></p> | |
| <p class="run"><span class="n"><a id="t144" href="#t144">144</a></span><span class="t"> <span class="key">def</span> <span class="nam">__init_subclass__</span><span class="op">(</span><span class="nam">cls</span><span class="op">,</span> <span class="op">**</span><span class="nam">kwargs</span><span class="op">)</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t145" href="#t145">145</a></span><span class="t"> <span class="nam">super</span><span class="op">(</span><span class="op">)</span><span class="op">.</span><span class="nam">__init_subclass__</span><span class="op">(</span><span class="op">**</span><span class="nam">kwargs</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t146" href="#t146">146</a></span><span class="t"> <span class="com"># Register the subclass using its name as the key</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t147" href="#t147">147</a></span><span class="t"> <span class="nam">JsonSerializableRegistry</span><span class="op">.</span><span class="nam">class_mapping</span><span class="op">[</span><span class="nam">cls</span><span class="op">.</span><span class="nam">__name__</span><span class="op">]</span> <span class="op">=</span> <span class="nam">cls</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t148" href="#t148">148</a></span><span class="t"> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t149" href="#t149">149</a></span><span class="t"> <span class="com"># Automatically extend serializable attributes and custom initializers from parent classes </span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t150" href="#t150">150</a></span><span class="t"> <span class="key">if</span> <span class="nam">hasattr</span><span class="op">(</span><span class="nam">cls</span><span class="op">,</span> <span class="str">'serializable_attributes'</span><span class="op">)</span> <span class="key">and</span> <span class="nam">isinstance</span><span class="op">(</span><span class="nam">cls</span><span class="op">.</span><span class="nam">serializable_attributes</span><span class="op">,</span> <span class="nam">list</span><span class="op">)</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t151" href="#t151">151</a></span><span class="t"> <span class="key">for</span> <span class="nam">base</span> <span class="key">in</span> <span class="nam">cls</span><span class="op">.</span><span class="nam">__bases__</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t152" href="#t152">152</a></span><span class="t"> <span class="key">if</span> <span class="nam">hasattr</span><span class="op">(</span><span class="nam">base</span><span class="op">,</span> <span class="str">'serializable_attributes'</span><span class="op">)</span> <span class="key">and</span> <span class="nam">isinstance</span><span class="op">(</span><span class="nam">base</span><span class="op">.</span><span class="nam">serializable_attributes</span><span class="op">,</span> <span class="nam">list</span><span class="op">)</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t153" href="#t153">153</a></span><span class="t"> <span class="nam">cls</span><span class="op">.</span><span class="nam">serializable_attributes</span> <span class="op">=</span> <span class="nam">list</span><span class="op">(</span><span class="nam">set</span><span class="op">(</span><span class="nam">base</span><span class="op">.</span><span class="nam">serializable_attributes</span> <span class="op">+</span> <span class="nam">cls</span><span class="op">.</span><span class="nam">serializable_attributes</span><span class="op">)</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t154" href="#t154">154</a></span><span class="t"> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t155" href="#t155">155</a></span><span class="t"> <span class="key">if</span> <span class="nam">hasattr</span><span class="op">(</span><span class="nam">cls</span><span class="op">,</span> <span class="str">'suppress_attributes_from_serialization'</span><span class="op">)</span> <span class="key">and</span> <span class="nam">isinstance</span><span class="op">(</span><span class="nam">cls</span><span class="op">.</span><span class="nam">suppress_attributes_from_serialization</span><span class="op">,</span> <span class="nam">list</span><span class="op">)</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t156" href="#t156">156</a></span><span class="t"> <span class="key">for</span> <span class="nam">base</span> <span class="key">in</span> <span class="nam">cls</span><span class="op">.</span><span class="nam">__bases__</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t157" href="#t157">157</a></span><span class="t"> <span class="key">if</span> <span class="nam">hasattr</span><span class="op">(</span><span class="nam">base</span><span class="op">,</span> <span class="str">'suppress_attributes_from_serialization'</span><span class="op">)</span> <span class="key">and</span> <span class="nam">isinstance</span><span class="op">(</span><span class="nam">base</span><span class="op">.</span><span class="nam">suppress_attributes_from_serialization</span><span class="op">,</span> <span class="nam">list</span><span class="op">)</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t158" href="#t158">158</a></span><span class="t"> <span class="nam">cls</span><span class="op">.</span><span class="nam">suppress_attributes_from_serialization</span> <span class="op">=</span> <span class="nam">list</span><span class="op">(</span><span class="nam">set</span><span class="op">(</span><span class="nam">base</span><span class="op">.</span><span class="nam">suppress_attributes_from_serialization</span> <span class="op">+</span> <span class="nam">cls</span><span class="op">.</span><span class="nam">suppress_attributes_from_serialization</span><span class="op">)</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t159" href="#t159">159</a></span><span class="t"> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t160" href="#t160">160</a></span><span class="t"> <span class="key">if</span> <span class="nam">hasattr</span><span class="op">(</span><span class="nam">cls</span><span class="op">,</span> <span class="str">'custom_deserializers'</span><span class="op">)</span> <span class="key">and</span> <span class="nam">isinstance</span><span class="op">(</span><span class="nam">cls</span><span class="op">.</span><span class="nam">custom_deserializers</span><span class="op">,</span> <span class="nam">dict</span><span class="op">)</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t161" href="#t161">161</a></span><span class="t"> <span class="key">for</span> <span class="nam">base</span> <span class="key">in</span> <span class="nam">cls</span><span class="op">.</span><span class="nam">__bases__</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t162" href="#t162">162</a></span><span class="t"> <span class="key">if</span> <span class="nam">hasattr</span><span class="op">(</span><span class="nam">base</span><span class="op">,</span> <span class="str">'custom_deserializers'</span><span class="op">)</span> <span class="key">and</span> <span class="nam">isinstance</span><span class="op">(</span><span class="nam">base</span><span class="op">.</span><span class="nam">custom_deserializers</span><span class="op">,</span> <span class="nam">dict</span><span class="op">)</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t163" href="#t163">163</a></span><span class="t"> <span class="nam">base_initializers</span> <span class="op">=</span> <span class="nam">base</span><span class="op">.</span><span class="nam">custom_deserializers</span><span class="op">.</span><span class="nam">copy</span><span class="op">(</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t164" href="#t164">164</a></span><span class="t"> <span class="nam">base_initializers</span><span class="op">.</span><span class="nam">update</span><span class="op">(</span><span class="nam">cls</span><span class="op">.</span><span class="nam">custom_deserializers</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t165" href="#t165">165</a></span><span class="t"> <span class="nam">cls</span><span class="op">.</span><span class="nam">custom_deserializers</span> <span class="op">=</span> <span class="nam">base_initializers</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t166" href="#t166">166</a></span><span class="t"> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t167" href="#t167">167</a></span><span class="t"> <span class="key">if</span> <span class="nam">hasattr</span><span class="op">(</span><span class="nam">cls</span><span class="op">,</span> <span class="str">'custom_serializers'</span><span class="op">)</span> <span class="key">and</span> <span class="nam">isinstance</span><span class="op">(</span><span class="nam">cls</span><span class="op">.</span><span class="nam">custom_serializers</span><span class="op">,</span> <span class="nam">dict</span><span class="op">)</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t168" href="#t168">168</a></span><span class="t"> <span class="key">for</span> <span class="nam">base</span> <span class="key">in</span> <span class="nam">cls</span><span class="op">.</span><span class="nam">__bases__</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t169" href="#t169">169</a></span><span class="t"> <span class="key">if</span> <span class="nam">hasattr</span><span class="op">(</span><span class="nam">base</span><span class="op">,</span> <span class="str">'custom_serializers'</span><span class="op">)</span> <span class="key">and</span> <span class="nam">isinstance</span><span class="op">(</span><span class="nam">base</span><span class="op">.</span><span class="nam">custom_serializers</span><span class="op">,</span> <span class="nam">dict</span><span class="op">)</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t170" href="#t170">170</a></span><span class="t"> <span class="nam">base_serializers</span> <span class="op">=</span> <span class="nam">base</span><span class="op">.</span><span class="nam">custom_serializers</span><span class="op">.</span><span class="nam">copy</span><span class="op">(</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t171" href="#t171">171</a></span><span class="t"> <span class="nam">base_serializers</span><span class="op">.</span><span class="nam">update</span><span class="op">(</span><span class="nam">cls</span><span class="op">.</span><span class="nam">custom_serializers</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t172" href="#t172">172</a></span><span class="t"> <span class="nam">cls</span><span class="op">.</span><span class="nam">custom_serializers</span> <span class="op">=</span> <span class="nam">base_serializers</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t173" href="#t173">173</a></span><span class="t"> </span><span class="r"></span></p> | |
| <p class="run"><span class="n"><a id="t174" href="#t174">174</a></span><span class="t"> <span class="key">def</span> <span class="nam">_post_deserialization_init</span><span class="op">(</span><span class="nam">self</span><span class="op">,</span> <span class="op">**</span><span class="nam">kwargs</span><span class="op">)</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t175" href="#t175">175</a></span><span class="t"> <span class="com"># if there's a _post_init method, call it after deserialization</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t176" href="#t176">176</a></span><span class="t"> <span class="key">if</span> <span class="nam">hasattr</span><span class="op">(</span><span class="nam">self</span><span class="op">,</span> <span class="str">'_post_init'</span><span class="op">)</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t177" href="#t177">177</a></span><span class="t"> <span class="nam">self</span><span class="op">.</span><span class="nam">_post_init</span><span class="op">(</span><span class="op">**</span><span class="nam">kwargs</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t178" href="#t178">178</a></span><span class="t"> </span><span class="r"></span></p> | |
| <p class="run"><span class="n"><a id="t179" href="#t179">179</a></span><span class="t"> <span class="op">@</span><span class="nam">classmethod</span> </span><span class="r"></span></p> | |
| <p class="run"><span class="n"><a id="t180" href="#t180">180</a></span><span class="t"> <span class="key">def</span> <span class="nam">_programmatic_name_to_json_name</span><span class="op">(</span><span class="nam">cls</span><span class="op">,</span> <span class="nam">name</span><span class="op">)</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t181" href="#t181">181</a></span><span class="t"> <span class="str">"""</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t182" href="#t182">182</a></span><span class="t"><span class="str"> Converts a programmatic name to a JSON name by converting it to snake case.</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t183" href="#t183">183</a></span><span class="t"><span class="str"> """</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t184" href="#t184">184</a></span><span class="t"> <span class="key">if</span> <span class="nam">hasattr</span><span class="op">(</span><span class="nam">cls</span><span class="op">,</span> <span class="str">'serializable_attributes_renaming'</span><span class="op">)</span> <span class="key">and</span> <span class="nam">isinstance</span><span class="op">(</span><span class="nam">cls</span><span class="op">.</span><span class="nam">serializable_attributes_renaming</span><span class="op">,</span> <span class="nam">dict</span><span class="op">)</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t185" href="#t185">185</a></span><span class="t"> <span class="key">return</span> <span class="nam">cls</span><span class="op">.</span><span class="nam">serializable_attributes_renaming</span><span class="op">.</span><span class="nam">get</span><span class="op">(</span><span class="nam">name</span><span class="op">,</span> <span class="nam">name</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t186" href="#t186">186</a></span><span class="t"> <span class="key">return</span> <span class="nam">name</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t187" href="#t187">187</a></span><span class="t"> </span><span class="r"></span></p> | |
| <p class="run"><span class="n"><a id="t188" href="#t188">188</a></span><span class="t"> <span class="op">@</span><span class="nam">classmethod</span> </span><span class="r"></span></p> | |
| <p class="run"><span class="n"><a id="t189" href="#t189">189</a></span><span class="t"> <span class="key">def</span> <span class="nam">_json_name_to_programmatic_name</span><span class="op">(</span><span class="nam">cls</span><span class="op">,</span> <span class="nam">name</span><span class="op">)</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t190" href="#t190">190</a></span><span class="t"> <span class="str">"""</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t191" href="#t191">191</a></span><span class="t"><span class="str"> Converts a JSON name to a programmatic name.</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t192" href="#t192">192</a></span><span class="t"><span class="str"> """</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t193" href="#t193">193</a></span><span class="t"> <span class="key">if</span> <span class="nam">hasattr</span><span class="op">(</span><span class="nam">cls</span><span class="op">,</span> <span class="str">'serializable_attributes_renaming'</span><span class="op">)</span> <span class="key">and</span> <span class="nam">isinstance</span><span class="op">(</span><span class="nam">cls</span><span class="op">.</span><span class="nam">serializable_attributes_renaming</span><span class="op">,</span> <span class="nam">dict</span><span class="op">)</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t194" href="#t194">194</a></span><span class="t"> <span class="nam">reverse_rename</span> <span class="op">=</span> <span class="op">{</span><span class="op">}</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t195" href="#t195">195</a></span><span class="t"> <span class="key">for</span> <span class="nam">k</span><span class="op">,</span> <span class="nam">v</span> <span class="key">in</span> <span class="nam">cls</span><span class="op">.</span><span class="nam">serializable_attributes_renaming</span><span class="op">.</span><span class="nam">items</span><span class="op">(</span><span class="op">)</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t196" href="#t196">196</a></span><span class="t"> <span class="key">if</span> <span class="nam">v</span> <span class="key">in</span> <span class="nam">reverse_rename</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t197" href="#t197">197</a></span><span class="t"> <span class="key">raise</span> <span class="nam">ValueError</span><span class="op">(</span><span class="fst">f"</span><span class="fst">Duplicate value '</span><span class="op">{</span><span class="nam">v</span><span class="op">}</span><span class="fst">' found in serializable_attributes_renaming.</span><span class="fst">"</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t198" href="#t198">198</a></span><span class="t"> <span class="nam">reverse_rename</span><span class="op">[</span><span class="nam">v</span><span class="op">]</span> <span class="op">=</span> <span class="nam">k</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t199" href="#t199">199</a></span><span class="t"> <span class="key">return</span> <span class="nam">reverse_rename</span><span class="op">.</span><span class="nam">get</span><span class="op">(</span><span class="nam">name</span><span class="op">,</span> <span class="nam">name</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t200" href="#t200">200</a></span><span class="t"> <span class="key">return</span> <span class="nam">name</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t201" href="#t201">201</a></span><span class="t"> </span><span class="r"></span></p> | |
| <p class="run"><span class="n"><a id="t202" href="#t202">202</a></span><span class="t"><span class="key">def</span> <span class="nam">post_init</span><span class="op">(</span><span class="nam">cls</span><span class="op">)</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t203" href="#t203">203</a></span><span class="t"> <span class="str">"""</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t204" href="#t204">204</a></span><span class="t"><span class="str"> Decorator to enforce a post-initialization method call in a class, if it has one.</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t205" href="#t205">205</a></span><span class="t"><span class="str"> The method must be named `_post_init`.</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t206" href="#t206">206</a></span><span class="t"><span class="str"> """</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t207" href="#t207">207</a></span><span class="t"> <span class="nam">original_init</span> <span class="op">=</span> <span class="nam">cls</span><span class="op">.</span><span class="nam">__init__</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t208" href="#t208">208</a></span><span class="t"> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t209" href="#t209">209</a></span><span class="t"> <span class="key">def</span> <span class="nam">new_init</span><span class="op">(</span><span class="nam">self</span><span class="op">,</span> <span class="op">*</span><span class="nam">args</span><span class="op">,</span> <span class="op">**</span><span class="nam">kwargs</span><span class="op">)</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t210" href="#t210">210</a></span><span class="t"> <span class="nam">original_init</span><span class="op">(</span><span class="nam">self</span><span class="op">,</span> <span class="op">*</span><span class="nam">args</span><span class="op">,</span> <span class="op">**</span><span class="nam">kwargs</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t211" href="#t211">211</a></span><span class="t"> <span class="key">if</span> <span class="nam">hasattr</span><span class="op">(</span><span class="nam">cls</span><span class="op">,</span> <span class="str">'_post_init'</span><span class="op">)</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t212" href="#t212">212</a></span><span class="t"> <span class="nam">cls</span><span class="op">.</span><span class="nam">_post_init</span><span class="op">(</span><span class="nam">self</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t213" href="#t213">213</a></span><span class="t"> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t214" href="#t214">214</a></span><span class="t"> <span class="nam">cls</span><span class="op">.</span><span class="nam">__init__</span> <span class="op">=</span> <span class="nam">new_init</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t215" href="#t215">215</a></span><span class="t"> <span class="key">return</span> <span class="nam">cls</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t216" href="#t216">216</a></span><span class="t"> </span><span class="r"></span></p> | |
| <p class="run"><span class="n"><a id="t217" href="#t217">217</a></span><span class="t"><span class="key">def</span> <span class="nam">merge_dicts</span><span class="op">(</span><span class="nam">current</span><span class="op">,</span> <span class="nam">additions</span><span class="op">,</span> <span class="nam">overwrite</span><span class="op">=</span><span class="key">False</span><span class="op">,</span> <span class="nam">error_on_conflict</span><span class="op">=</span><span class="key">True</span><span class="op">,</span> <span class="nam">remove_duplicates</span><span class="op">=</span><span class="key">True</span><span class="op">)</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t218" href="#t218">218</a></span><span class="t"> <span class="str">"""</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t219" href="#t219">219</a></span><span class="t"><span class="str"> Merges two dictionaries and returns a new dictionary. Works as follows:</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t220" href="#t220">220</a></span><span class="t"><span class="str"> - If a key exists in the additions dictionary but not in the current dictionary, it is added.</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t221" href="#t221">221</a></span><span class="t"><span class="str"> - If a key maps to None in the current dictionary, it is replaced by the value in the additions dictionary.</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t222" href="#t222">222</a></span><span class="t"><span class="str"> - If a key exists in both dictionaries and the values are dictionaries, the function is called recursively.</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t223" href="#t223">223</a></span><span class="t"><span class="str"> - If a key exists in both dictionaries and the values are lists, the lists are concatenated and duplicates are removed</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t224" href="#t224">224</a></span><span class="t"><span class="str"> (if remove_duplicates is True).</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t225" href="#t225">225</a></span><span class="t"><span class="str"> - If the values are of different types, an exception is raised.</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t226" href="#t226">226</a></span><span class="t"><span class="str"> - If the values are of the same type but not both lists/dictionaries, the value from the additions dictionary overwrites the value in the current dictionary based on the overwrite parameter.</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t227" href="#t227">227</a></span><span class="t"><span class="str"> </span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t228" href="#t228">228</a></span><span class="t"><span class="str"> Parameters:</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t229" href="#t229">229</a></span><span class="t"><span class="str"> - current (dict): The original dictionary.</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t230" href="#t230">230</a></span><span class="t"><span class="str"> - additions (dict): The dictionary with values to add.</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t231" href="#t231">231</a></span><span class="t"><span class="str"> - overwrite (bool): Whether to overwrite values if they are of the same type but not both lists/dictionaries.</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t232" href="#t232">232</a></span><span class="t"><span class="str"> - error_on_conflict (bool): Whether to raise an error if there is a conflict and overwrite is False.</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t233" href="#t233">233</a></span><span class="t"><span class="str"> - remove_duplicates (bool): Whether to remove duplicates from lists when merging.</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t234" href="#t234">234</a></span><span class="t"><span class="str"> </span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t235" href="#t235">235</a></span><span class="t"><span class="str"> Returns:</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t236" href="#t236">236</a></span><span class="t"><span class="str"> - dict: A new dictionary with merged values.</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t237" href="#t237">237</a></span><span class="t"><span class="str"> """</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t238" href="#t238">238</a></span><span class="t"> <span class="nam">merged</span> <span class="op">=</span> <span class="nam">current</span><span class="op">.</span><span class="nam">copy</span><span class="op">(</span><span class="op">)</span> <span class="com"># Create a copy of the current dictionary to avoid altering it</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t239" href="#t239">239</a></span><span class="t"> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t240" href="#t240">240</a></span><span class="t"> <span class="key">for</span> <span class="nam">key</span> <span class="key">in</span> <span class="nam">additions</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t241" href="#t241">241</a></span><span class="t"> <span class="key">if</span> <span class="nam">key</span> <span class="key">in</span> <span class="nam">merged</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t242" href="#t242">242</a></span><span class="t"> <span class="com"># If the current value is None, directly assign the new value</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t243" href="#t243">243</a></span><span class="t"> <span class="key">if</span> <span class="nam">merged</span><span class="op">[</span><span class="nam">key</span><span class="op">]</span> <span class="key">is</span> <span class="key">None</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t244" href="#t244">244</a></span><span class="t"> <span class="nam">merged</span><span class="op">[</span><span class="nam">key</span><span class="op">]</span> <span class="op">=</span> <span class="nam">additions</span><span class="op">[</span><span class="nam">key</span><span class="op">]</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t245" href="#t245">245</a></span><span class="t"> <span class="com"># If both values are dictionaries, merge them recursively</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t246" href="#t246">246</a></span><span class="t"> <span class="key">elif</span> <span class="nam">isinstance</span><span class="op">(</span><span class="nam">merged</span><span class="op">[</span><span class="nam">key</span><span class="op">]</span><span class="op">,</span> <span class="nam">dict</span><span class="op">)</span> <span class="key">and</span> <span class="nam">isinstance</span><span class="op">(</span><span class="nam">additions</span><span class="op">[</span><span class="nam">key</span><span class="op">]</span><span class="op">,</span> <span class="nam">dict</span><span class="op">)</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t247" href="#t247">247</a></span><span class="t"> <span class="nam">merged</span><span class="op">[</span><span class="nam">key</span><span class="op">]</span> <span class="op">=</span> <span class="nam">merge_dicts</span><span class="op">(</span><span class="nam">merged</span><span class="op">[</span><span class="nam">key</span><span class="op">]</span><span class="op">,</span> <span class="nam">additions</span><span class="op">[</span><span class="nam">key</span><span class="op">]</span><span class="op">,</span> <span class="nam">overwrite</span><span class="op">,</span> <span class="nam">error_on_conflict</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t248" href="#t248">248</a></span><span class="t"> <span class="com"># If both values are lists, concatenate them and remove duplicates</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t249" href="#t249">249</a></span><span class="t"> <span class="key">elif</span> <span class="nam">isinstance</span><span class="op">(</span><span class="nam">merged</span><span class="op">[</span><span class="nam">key</span><span class="op">]</span><span class="op">,</span> <span class="nam">list</span><span class="op">)</span> <span class="key">and</span> <span class="nam">isinstance</span><span class="op">(</span><span class="nam">additions</span><span class="op">[</span><span class="nam">key</span><span class="op">]</span><span class="op">,</span> <span class="nam">list</span><span class="op">)</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t250" href="#t250">250</a></span><span class="t"> <span class="nam">merged</span><span class="op">[</span><span class="nam">key</span><span class="op">]</span><span class="op">.</span><span class="nam">extend</span><span class="op">(</span><span class="nam">additions</span><span class="op">[</span><span class="nam">key</span><span class="op">]</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t251" href="#t251">251</a></span><span class="t"> <span class="com"># Remove duplicates while preserving order</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t252" href="#t252">252</a></span><span class="t"> <span class="key">if</span> <span class="nam">remove_duplicates</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t253" href="#t253">253</a></span><span class="t"> <span class="nam">merged</span><span class="op">[</span><span class="nam">key</span><span class="op">]</span> <span class="op">=</span> <span class="nam">remove_duplicate_items</span><span class="op">(</span><span class="nam">merged</span><span class="op">[</span><span class="nam">key</span><span class="op">]</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t254" href="#t254">254</a></span><span class="t"> <span class="com"># If the values are of different types, raise an exception</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t255" href="#t255">255</a></span><span class="t"> <span class="key">elif</span> <span class="nam">type</span><span class="op">(</span><span class="nam">merged</span><span class="op">[</span><span class="nam">key</span><span class="op">]</span><span class="op">)</span> <span class="op">!=</span> <span class="nam">type</span><span class="op">(</span><span class="nam">additions</span><span class="op">[</span><span class="nam">key</span><span class="op">]</span><span class="op">)</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t256" href="#t256">256</a></span><span class="t"> <span class="key">raise</span> <span class="nam">TypeError</span><span class="op">(</span><span class="fst">f"</span><span class="fst">Cannot merge different types: </span><span class="op">{</span><span class="nam">type</span><span class="op">(</span><span class="nam">merged</span><span class="op">[</span><span class="nam">key</span><span class="op">]</span><span class="op">)</span><span class="op">}</span><span class="fst"> and </span><span class="op">{</span><span class="nam">type</span><span class="op">(</span><span class="nam">additions</span><span class="op">[</span><span class="nam">key</span><span class="op">]</span><span class="op">)</span><span class="op">}</span><span class="fst"> for key '</span><span class="op">{</span><span class="nam">key</span><span class="op">}</span><span class="fst">'</span><span class="fst">"</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t257" href="#t257">257</a></span><span class="t"> <span class="com"># If the values are of the same type but not both lists/dictionaries, decide based on the overwrite parameter</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t258" href="#t258">258</a></span><span class="t"> <span class="key">else</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t259" href="#t259">259</a></span><span class="t"> <span class="key">if</span> <span class="nam">overwrite</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t260" href="#t260">260</a></span><span class="t"> <span class="nam">merged</span><span class="op">[</span><span class="nam">key</span><span class="op">]</span> <span class="op">=</span> <span class="nam">additions</span><span class="op">[</span><span class="nam">key</span><span class="op">]</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t261" href="#t261">261</a></span><span class="t"> <span class="key">elif</span> <span class="nam">merged</span><span class="op">[</span><span class="nam">key</span><span class="op">]</span> <span class="op">!=</span> <span class="nam">additions</span><span class="op">[</span><span class="nam">key</span><span class="op">]</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t262" href="#t262">262</a></span><span class="t"> <span class="key">if</span> <span class="nam">error_on_conflict</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t263" href="#t263">263</a></span><span class="t"> <span class="key">raise</span> <span class="nam">ValueError</span><span class="op">(</span><span class="fst">f"</span><span class="fst">Conflict at key '</span><span class="op">{</span><span class="nam">key</span><span class="op">}</span><span class="fst">': overwrite is set to False and values are different.</span><span class="fst">"</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t264" href="#t264">264</a></span><span class="t"> <span class="key">else</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t265" href="#t265">265</a></span><span class="t"> <span class="key">continue</span> <span class="com"># Ignore the conflict and continue</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t266" href="#t266">266</a></span><span class="t"> <span class="key">else</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t267" href="#t267">267</a></span><span class="t"> <span class="com"># If the key is not present in merged, add it from additions</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t268" href="#t268">268</a></span><span class="t"> <span class="nam">merged</span><span class="op">[</span><span class="nam">key</span><span class="op">]</span> <span class="op">=</span> <span class="nam">additions</span><span class="op">[</span><span class="nam">key</span><span class="op">]</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t269" href="#t269">269</a></span><span class="t"> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t270" href="#t270">270</a></span><span class="t"> <span class="key">return</span> <span class="nam">merged</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t271" href="#t271">271</a></span><span class="t"> </span><span class="r"></span></p> | |
| <p class="run"><span class="n"><a id="t272" href="#t272">272</a></span><span class="t"><span class="key">def</span> <span class="nam">remove_duplicate_items</span><span class="op">(</span><span class="nam">lst</span><span class="op">)</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t273" href="#t273">273</a></span><span class="t"> <span class="str">"""</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t274" href="#t274">274</a></span><span class="t"><span class="str"> Removes duplicates from a list while preserving order.</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t275" href="#t275">275</a></span><span class="t"><span class="str"> Handles unhashable elements by using a list comprehension.</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t276" href="#t276">276</a></span><span class="t"> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t277" href="#t277">277</a></span><span class="t"><span class="str"> Parameters:</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t278" href="#t278">278</a></span><span class="t"><span class="str"> - lst (list): The list to remove duplicates from.</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t279" href="#t279">279</a></span><span class="t"> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t280" href="#t280">280</a></span><span class="t"><span class="str"> Returns:</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t281" href="#t281">281</a></span><span class="t"><span class="str"> - list: A new list with duplicates removed.</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t282" href="#t282">282</a></span><span class="t"><span class="str"> """</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t283" href="#t283">283</a></span><span class="t"> <span class="nam">seen</span> <span class="op">=</span> <span class="op">[</span><span class="op">]</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t284" href="#t284">284</a></span><span class="t"> <span class="nam">result</span> <span class="op">=</span> <span class="op">[</span><span class="op">]</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t285" href="#t285">285</a></span><span class="t"> <span class="key">for</span> <span class="nam">item</span> <span class="key">in</span> <span class="nam">lst</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t286" href="#t286">286</a></span><span class="t"> <span class="key">if</span> <span class="nam">isinstance</span><span class="op">(</span><span class="nam">item</span><span class="op">,</span> <span class="nam">dict</span><span class="op">)</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t287" href="#t287">287</a></span><span class="t"> <span class="com"># Convert dict to a frozenset of its items to make it hashable</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t288" href="#t288">288</a></span><span class="t"> <span class="nam">item_key</span> <span class="op">=</span> <span class="nam">frozenset</span><span class="op">(</span><span class="nam">item</span><span class="op">.</span><span class="nam">items</span><span class="op">(</span><span class="op">)</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t289" href="#t289">289</a></span><span class="t"> <span class="key">else</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t290" href="#t290">290</a></span><span class="t"> <span class="nam">item_key</span> <span class="op">=</span> <span class="nam">item</span> </span><span class="r"></span></p> | |
| <p class="pln"><span class="n"><a id="t291" href="#t291">291</a></span><span class="t"> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t292" href="#t292">292</a></span><span class="t"> <span class="key">if</span> <span class="nam">item_key</span> <span class="key">not</span> <span class="key">in</span> <span class="nam">seen</span><span class="op">:</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t293" href="#t293">293</a></span><span class="t"> <span class="nam">seen</span><span class="op">.</span><span class="nam">append</span><span class="op">(</span><span class="nam">item_key</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t294" href="#t294">294</a></span><span class="t"> <span class="nam">result</span><span class="op">.</span><span class="nam">append</span><span class="op">(</span><span class="nam">item</span><span class="op">)</span> </span><span class="r"></span></p> | |
| <p class="mis show_mis"><span class="n"><a id="t295" href="#t295">295</a></span><span class="t"> <span class="key">return</span> <span class="nam">result</span> </span><span class="r"></span></p> | |
| </main> | |
| <footer> | |
| <div class="content"> | |
| <p> | |
| <a class="nav" href="z_4b3d0d3b603de221_config_py.html">« prev</a> | |
| <a class="nav" href="index.html">^ index</a> | |
| <a class="nav" href="z_4b3d0d3b603de221_llm_py.html">» next</a> | |
| | |
| <a class="nav" href="https://coverage.readthedocs.io/en/7.13.4">coverage.py v7.13.4</a>, | |
| created at 2026-02-28 17:48 +0000 | |
| </p> | |
| </div> | |
| </footer> | |
| </body> | |
| </html> | |