o
    Ïe^!  ã                   @   s”   d Z ddlmZmZmZmZmZ ddlmZ ddl	m
Z
 ddlmZ dd„ Zdd
d„Zdd„ Zddd„Zddd„ZG dd„ dƒZG dd„ deƒZdS )z Inference in propositional logicé    )ÚAndÚNotÚ	conjunctsÚto_cnfÚBooleanFunction)Úordered)Úsympify)Úimport_modulec              	   C   sV   | du s| du r
| S z| j r| W S | jrt| jd ƒW S t‚ ttfy*   tdƒ‚w )zó
    The symbol in this literal (without the negation).

    Examples
    ========

    >>> from sympy.abc import A
    >>> from sympy.logic.inference import literal_symbol
    >>> literal_symbol(A)
    A
    >>> literal_symbol(~A)
    A

    TFr   z#Argument must be a boolean literal.)Z	is_SymbolZis_NotÚliteral_symbolÚargsÚ
ValueErrorÚAttributeError)Úliteral© r   úED:\Projects\ConvertPro\env\Lib\site-packages\sympy/logic/inference.pyr
   	   s   ÿr
   NFc           	      C   sÒ   |du s|dkrt dƒ}|durd}n
|dkrtdƒ‚d}|dkr+t dƒ}|du r+d}|dkr9dd	lm} || ƒS |dkrHdd	lm} || |ƒS |dkrWdd
lm} || |ƒS |dkrgddlm} || ||ƒS t	‚)aÚ  
    Check satisfiability of a propositional sentence.
    Returns a model when it succeeds.
    Returns {true: true} for trivially true expressions.

    On setting all_models to True, if given expr is satisfiable then
    returns a generator of models. However, if expr is unsatisfiable
    then returns a generator containing the single element False.

    Examples
    ========

    >>> from sympy.abc import A, B
    >>> from sympy.logic.inference import satisfiable
    >>> satisfiable(A & ~B)
    {A: True, B: False}
    >>> satisfiable(A & ~A)
    False
    >>> satisfiable(True)
    {True: True}
    >>> next(satisfiable(A & ~A, all_models=True))
    False
    >>> models = satisfiable((A >> B) & B, all_models=True)
    >>> next(models)
    {A: False, B: True}
    >>> next(models)
    {A: True, B: True}
    >>> def use_models(models):
    ...     for model in models:
    ...         if model:
    ...             # Do something with the model.
    ...             print(model)
    ...         else:
    ...             # Given expr is unsatisfiable.
    ...             print("UNSAT")
    >>> use_models(satisfiable(A >> ~A, all_models=True))
    {A: False}
    >>> use_models(satisfiable(A ^ A, all_models=True))
    UNSAT

    NÚpycosatzpycosat module is not presentZdpll2Z	minisat22ÚpysatZdpllr   )Údpll_satisfiable)Úpycosat_satisfiable)Úminisat22_satisfiable)
r	   ÚImportErrorZsympy.logic.algorithms.dpllr   Zsympy.logic.algorithms.dpll2Z&sympy.logic.algorithms.pycosat_wrapperr   Z(sympy.logic.algorithms.minisat22_wrapperr   ÚNotImplementedError)	ÚexprÚ	algorithmZ
all_modelsZminimalr   r   r   r   r   r   r   r   Úsatisfiable&   s0   *

r   c                 C   s   t t| ƒƒ S )ax  
    Check validity of a propositional sentence.
    A valid propositional sentence is True under every assignment.

    Examples
    ========

    >>> from sympy.abc import A, B
    >>> from sympy.logic.inference import valid
    >>> valid(A | ~A)
    True
    >>> valid(A | B)
    False

    References
    ==========

    .. [1] https://en.wikipedia.org/wiki/Validity

    )r   r   ©r   r   r   r   Úvalido   s   r   c                    sÀ   ddl m‰  d‰‡ ‡‡fdd„‰| ˆv r| S t| ƒ} ˆ| ƒs$td|  ƒ‚|s(i }‡fdd„| ¡ D ƒ}|  |¡}|ˆv r@t|ƒS |r^d	d„ | ¡ D ƒ}t||ƒrXt	|ƒrVd
S dS t
|ƒs^dS dS )a+  
    Returns whether the given assignment is a model or not.

    If the assignment does not specify the value for every proposition,
    this may return None to indicate 'not obvious'.

    Parameters
    ==========

    model : dict, optional, default: {}
        Mapping of symbols to boolean values to indicate assignment.
    deep: boolean, optional, default: False
        Gives the value of the expression under partial assignments
        correctly. May still return None to indicate 'not obvious'.


    Examples
    ========

    >>> from sympy.abc import A, B
    >>> from sympy.logic.inference import pl_true
    >>> pl_true( A & B, {A: True, B: True})
    True
    >>> pl_true(A & B, {A: False})
    False
    >>> pl_true(A & B, {A: True})
    >>> pl_true(A & B, {A: True}, deep=True)
    >>> pl_true(A >> (B >> A))
    >>> pl_true(A >> (B >> A), deep=True)
    True
    >>> pl_true(A & ~A)
    >>> pl_true(A & ~A, deep=True)
    False
    >>> pl_true(A & B & (~A | ~B), {A: True})
    >>> pl_true(A & B & (~A | ~B), {A: True}, deep=True)
    False

    r   )ÚSymbol)TFc                    s<   t | ˆ ƒs	| ˆv rdS t | tƒsdS t‡fdd„| jD ƒƒS )NTFc                 3   s    | ]}ˆ |ƒV  qd S ©Nr   )Ú.0Úarg)Ú	_validater   r   Ú	<genexpr>¸   s   € z-pl_true.<locals>._validate.<locals>.<genexpr>)Ú
isinstancer   Úallr   r   ©r   r!   Úbooleanr   r   r!   ³   s
   
zpl_true.<locals>._validatez$%s is not a valid boolean expressionc                    s   i | ]\}}|ˆ v r||“qS r   r   )r   ÚkÚv)r&   r   r   Ú
<dictcomp>Á   s    zpl_true.<locals>.<dictcomp>c                 S   s   i | ]}|d “qS )Tr   )r   r'   r   r   r   r)   Æ   s    TFN)Zsympy.core.symbolr   r   r   ÚitemsÚsubsÚboolZatomsÚpl_truer   r   )r   ÚmodelÚdeepÚresultr   r%   r   r-   ‡   s.   (

þr-   c                 C   s.   |rt |ƒ}ng }| t| ƒ¡ tt|Ž ƒ S )aø  
    Check whether the given expr_set entail an expr.
    If formula_set is empty then it returns the validity of expr.

    Examples
    ========

    >>> from sympy.abc import A, B, C
    >>> from sympy.logic.inference import entails
    >>> entails(A, [A >> B, B >> C])
    False
    >>> entails(C, [A >> B, B >> C, A])
    True
    >>> entails(A >> B)
    False
    >>> entails(A >> (B >> A))
    True

    References
    ==========

    .. [1] https://en.wikipedia.org/wiki/Logical_consequence

    )ÚlistÚappendr   r   r   )r   Zformula_setr   r   r   ÚentailsÐ   s
   
r3   c                   @   s>   e Zd ZdZddd„Zdd„ Zdd„ Zd	d
„ Zedd„ ƒZ	dS )ÚKBz"Base class for all knowledge basesNc                 C   s   t ƒ | _|r|  |¡ d S d S r   )ÚsetÚclauses_Útell©ÚselfÚsentencer   r   r   Ú__init__ó   s   ÿzKB.__init__c                 C   ó   t ‚r   ©r   r8   r   r   r   r7   ø   ó   zKB.tellc                 C   r<   r   r=   ©r9   Úqueryr   r   r   Úaskû   r>   zKB.askc                 C   r<   r   r=   r8   r   r   r   Úretractþ   r>   z
KB.retractc                 C   s   t t| jƒƒS r   )r1   r   r6   )r9   r   r   r   Úclauses  s   z
KB.clausesr   )
Ú__name__Ú
__module__Ú__qualname__Ú__doc__r;   r7   rA   rB   ÚpropertyrC   r   r   r   r   r4   ñ   s    
r4   c                   @   s(   e Zd ZdZdd„ Zdd„ Zdd„ ZdS )	ÚPropKBz=A KB for Propositional Logic.  Inefficient, with no indexing.c                 C   ó"   t t|ƒƒD ]}| j |¡ qdS )ai  Add the sentence's clauses to the KB

        Examples
        ========

        >>> from sympy.logic.inference import PropKB
        >>> from sympy.abc import x, y
        >>> l = PropKB()
        >>> l.clauses
        []

        >>> l.tell(x | y)
        >>> l.clauses
        [x | y]

        >>> l.tell(y)
        >>> l.clauses
        [y, x | y]

        N)r   r   r6   Úadd©r9   r:   Úcr   r   r   r7   	  ó   ÿzPropKB.tellc                 C   s   t || jƒS )a8  Checks if the query is true given the set of clauses.

        Examples
        ========

        >>> from sympy.logic.inference import PropKB
        >>> from sympy.abc import x, y
        >>> l = PropKB()
        >>> l.tell(x & ~y)
        >>> l.ask(x)
        True
        >>> l.ask(y)
        False

        )r3   r6   r?   r   r   r   rA   !  s   z
PropKB.askc                 C   rJ   )am  Remove the sentence's clauses from the KB

        Examples
        ========

        >>> from sympy.logic.inference import PropKB
        >>> from sympy.abc import x, y
        >>> l = PropKB()
        >>> l.clauses
        []

        >>> l.tell(x | y)
        >>> l.clauses
        [x | y]

        >>> l.retract(x | y)
        >>> l.clauses
        []

        N)r   r   r6   ÚdiscardrL   r   r   r   rB   3  rN   zPropKB.retractN)rD   rE   rF   rG   r7   rA   rB   r   r   r   r   rI     s
    rI   )NFF)NFr   )rG   Zsympy.logic.boolalgr   r   r   r   r   Zsympy.core.sortingr   Zsympy.core.sympifyr   Zsympy.external.importtoolsr	   r
   r   r   r-   r3   r4   rI   r   r   r   r   Ú<module>   s    
I

I!