o
    Th+                     @  sf  d dl mZ d dlmZ d dlmZ ddlmZ ddlmZ ddlm	Z	 dd	lm
Z
 dd
lmZ ddlmZ ddlmZ ddlmZ ddlmZ ddlmZ ddlmZ ddlmZ ddlmZ ddlmZ ddlmZ ddlmZ ddlmZ ddlmZ ddlm Z  ddl!m"Z" dZ#d*d d!Z$G d"d deZG d#d$ d$eZ%G d%d& d&e%Z&G d'd( d(e%Z'd)S )+    )annotations)Any)Optional   )ext   )_OnConflictConstraintT)_OnConflictIndexElementsT)_OnConflictIndexWhereT)_OnConflictSetT)_OnConflictWhereT   )util)	coercions)roles)schema)_DMLTableArgument)_exclusive_against)_generative)ColumnCollection)ReadOnlyColumnCollectionInsert)ClauseElement)KeyedColumnElement)alias)Self)r   inserttabler   returnr   c                 C  s   t | S )as  Construct a PostgreSQL-specific variant :class:`_postgresql.Insert`
    construct.

    .. container:: inherited_member

        The :func:`sqlalchemy.dialects.postgresql.insert` function creates
        a :class:`sqlalchemy.dialects.postgresql.Insert`.  This class is based
        on the dialect-agnostic :class:`_sql.Insert` construct which may
        be constructed using the :func:`_sql.insert` function in
        SQLAlchemy Core.

    The :class:`_postgresql.Insert` construct includes additional methods
    :meth:`_postgresql.Insert.on_conflict_do_update`,
    :meth:`_postgresql.Insert.on_conflict_do_nothing`.

    r   )r    r    b/home/air/segue/gpt/backup/venv/lib/python3.10/site-packages/sqlalchemy/dialects/postgresql/dml.pyr   %   s   r   c                   @  sp   e Zd ZdZdZdZejdddZe	ddd	id
Z
ee
					ddddZee
			ddddZdS ) r   zPostgreSQL-specific implementation of INSERT.

    Adds methods for PG-specific syntaxes such as ON CONFLICT.

    The :class:`_postgresql.Insert` object is created using the
    :func:`sqlalchemy.dialects.postgresql.insert` function.

    
postgresqlFr   6ReadOnlyColumnCollection[str, KeyedColumnElement[Any]]c                 C  s   t | jddjS )ak  Provide the ``excluded`` namespace for an ON CONFLICT statement

        PG's ON CONFLICT clause allows reference to the row that would
        be inserted, known as ``excluded``.  This attribute provides
        all columns in this row to be referenceable.

        .. tip::  The :attr:`_postgresql.Insert.excluded` attribute is an
            instance of :class:`_expression.ColumnCollection`, which provides
            an interface the same as that of the :attr:`_schema.Table.c`
            collection described at :ref:`metadata_tables_and_columns`.
            With this collection, ordinary names are accessible like attributes
            (e.g. ``stmt.excluded.some_column``), but special names and
            dictionary method names should be accessed using indexed access,
            such as ``stmt.excluded["column name"]`` or
            ``stmt.excluded["values"]``.   See the docstring for
            :class:`_expression.ColumnCollection` for further examples.

        .. seealso::

            :ref:`postgresql_insert_on_conflict` - example of how
            to use :attr:`_expression.Insert.excluded`

        excluded)name)r   r   columns)selfr    r    r!   r$   F   s   zInsert.excluded_post_values_clausezCThis Insert construct already has an ON CONFLICT clause established)msgsN
constraintr   index_elementsr	   index_wherer
   set_r   wherer   r   c                 C  s   t |||||| _| S )aS  
        Specifies a DO UPDATE SET action for ON CONFLICT clause.

        Either the ``constraint`` or ``index_elements`` argument is
        required, but only one of these can be specified.

        :param constraint:
         The name of a unique or exclusion constraint on the table,
         or the constraint object itself if it has a .name attribute.

        :param index_elements:
         A sequence consisting of string column names, :class:`_schema.Column`
         objects, or other column expression objects that will be used
         to infer a target index.

        :param index_where:
         Additional WHERE criterion that can be used to infer a
         conditional target index.

        :param set\_:
         A dictionary or other mapping object
         where the keys are either names of columns in the target table,
         or :class:`_schema.Column` objects or other ORM-mapped columns
         matching that of the target table, and expressions or literals
         as values, specifying the ``SET`` actions to take.

         .. versionadded:: 1.4 The
            :paramref:`_postgresql.Insert.on_conflict_do_update.set_`
            parameter supports :class:`_schema.Column` objects from the target
            :class:`_schema.Table` as keys.

         .. warning:: This dictionary does **not** take into account
            Python-specified default UPDATE values or generation functions,
            e.g. those specified using :paramref:`_schema.Column.onupdate`.
            These values will not be exercised for an ON CONFLICT style of
            UPDATE, unless they are manually specified in the
            :paramref:`.Insert.on_conflict_do_update.set_` dictionary.

        :param where:
         Optional argument. If present, can be a literal SQL
         string or an acceptable expression for a ``WHERE`` clause
         that restricts the rows affected by ``DO UPDATE SET``. Rows
         not meeting the ``WHERE`` condition will not be updated
         (effectively a ``DO NOTHING`` for those rows).


        .. seealso::

            :ref:`postgresql_insert_on_conflict`

        )OnConflictDoUpdater(   r'   r*   r+   r,   r-   r.   r    r    r!   on_conflict_do_updatek   s   =
zInsert.on_conflict_do_updatec                 C  s   t |||| _| S )a	  
        Specifies a DO NOTHING action for ON CONFLICT clause.

        The ``constraint`` and ``index_elements`` arguments
        are optional, but only one of these can be specified.

        :param constraint:
         The name of a unique or exclusion constraint on the table,
         or the constraint object itself if it has a .name attribute.

        :param index_elements:
         A sequence consisting of string column names, :class:`_schema.Column`
         objects, or other column expression objects that will be used
         to infer a target index.

        :param index_where:
         Additional WHERE criterion that can be used to infer a
         conditional target index.

        .. seealso::

            :ref:`postgresql_insert_on_conflict`

        )OnConflictDoNothingr(   r'   r*   r+   r,   r    r    r!   on_conflict_do_nothing   s    zInsert.on_conflict_do_nothing)r   r#   NNNNN)r*   r   r+   r	   r,   r
   r-   r   r.   r   r   r   NNN)r*   r   r+   r	   r,   r
   r   r   )__name__
__module____qualname____doc__stringify_dialectinherit_cacher   memoized_propertyr$   r   _on_conflict_exclusiver   r1   r4   r    r    r    r!   r   9   s2    	@c                   @  s<   e Zd ZU dZded< ded< ded< 			ddddZdS )OnConflictClauser"   zOptional[str]constraint_targetr	   inferred_target_elementsr
   inferred_target_whereclauseNr*   r   r+   r,   c                 C  s   |d urt |tst |tjtjfrt|dp|}|d ur^|d ur%tdt |tr4|| _d | _	d | _
n*t |tjrF|j}|jd d}nt |tjrS|j}|j}n|j}|jd d}|d urmd | _|| _	|| _
d S |d u r|d  | _ | _	| _
d S d S )Nr%   z8'constraint' and 'index_elements' are mutually exclusiver"   r.   )
isinstancestrr   
Constraintr   ExcludeConstraintgetattr
ValueErrorr@   rA   rB   Indexexpressionsdialect_optionsgetr&   r.   r3   r    r    r!   __init__   sH   




zOnConflictClause.__init__r6   )r*   r   r+   r	   r,   r
   )r7   r8   r9   r;   __annotations__rM   r    r    r    r!   r?      s   
 r?   c                   @  s   e Zd ZdZdS )r2   r4   N)r7   r8   r9   __visit_name__r    r    r    r!   r2   	  s    r2   c                      s.   e Zd ZdZ					dd fddZ  ZS )r/   r1   Nr*   r   r+   r	   r,   r
   r-   r   r.   r   c                   s   t  j|||d | jd u r| jd u rtdt|tr#|s"tdnt|tr-t|}ntddd | D | _	|| _
d S )N)r*   r+   r,   zVEither constraint or index_elements, but not both, must be specified unless DO NOTHINGz*set parameter dictionary must not be emptyzqset parameter must be a non-empty dictionary or a ColumnCollection such as the `.c.` collection of a Table objectc                 S  s"   g | ]\}}t tj||fqS r    )r   expectr   DMLColumnRole).0keyvaluer    r    r!   
<listcomp>2  s    z/OnConflictDoUpdate.__init__.<locals>.<listcomp>)superrM   rA   r@   rH   rC   dictr   itemsupdate_values_to_setupdate_whereclauser0   	__class__r    r!   rM     s.   





zOnConflictDoUpdate.__init__r5   )
r*   r   r+   r	   r,   r
   r-   r   r.   r   )r7   r8   r9   rO   rM   __classcell__r    r    r[   r!   r/     s    r/   N)r   r   r   r   )(
__future__r   typingr   r    r   _typingr   r	   r
   r   r   r   sqlr   r   r   sql._typingr   sql.baser   r   r   r   sql.dmlr   StandardInsertsql.elementsr   r   sql.expressionr   util.typingr   __all__r   r?   r2   r/   r    r    r    r!   <module>   s<   
 6