U
    9vf                     @   sF   d Z ddlmZ ddlZddgZejd	ddZe d
ddZdS )aY  Routines to find the boundary of a set of nodes.

An edge boundary is a set of edges, each of which has exactly one
endpoint in a given set of nodes (or, in the case of directed graphs,
the set of edges whose source node is in the set).

A node boundary of a set *S* of nodes is the set of (out-)neighbors of
nodes in *S* that are outside *S*.

    )chainNedge_boundarynode_boundaryFc                    st    fdd|D    r. j|||d}n j||d}|dkrXfdd|D S t|fdd|D S )	a  Returns the edge boundary of `nbunch1`.

    The *edge boundary* of a set *S* with respect to a set *T* is the
    set of edges (*u*, *v*) such that *u* is in *S* and *v* is in *T*.
    If *T* is not specified, it is assumed to be the set of all nodes
    not in *S*.

    Parameters
    ----------
    G : NetworkX graph

    nbunch1 : iterable
        Iterable of nodes in the graph representing the set of nodes
        whose edge boundary will be returned. (This is the set *S* from
        the definition above.)

    nbunch2 : iterable
        Iterable of nodes representing the target (or "exterior") set of
        nodes. (This is the set *T* from the definition above.) If not
        specified, this is assumed to be the set of all nodes in `G`
        not in `nbunch1`.

    keys : bool
        This parameter has the same meaning as in
        :meth:`MultiGraph.edges`.

    data : bool or object
        This parameter has the same meaning as in
        :meth:`MultiGraph.edges`.

    default : object
        This parameter has the same meaning as in
        :meth:`MultiGraph.edges`.

    Returns
    -------
    iterator
        An iterator over the edges in the boundary of `nbunch1` with
        respect to `nbunch2`. If `keys`, `data`, or `default`
        are specified and `G` is a multigraph, then edges are returned
        with keys and/or data, as in :meth:`MultiGraph.edges`.

    Examples
    --------
    >>> G = nx.wheel_graph(6)

    When nbunch2=None:

    >>> list(nx.edge_boundary(G, (1, 3)))
    [(1, 0), (1, 2), (1, 5), (3, 0), (3, 2), (3, 4)]

    When nbunch2 is given:

    >>> list(nx.edge_boundary(G, (1, 3), (2, 0)))
    [(1, 0), (1, 2), (3, 0), (3, 2)]

    Notes
    -----
    Any element of `nbunch` that is not in the graph `G` will be
    ignored.

    `nbunch1` and `nbunch2` are usually meant to be disjoint, but in
    the interest of speed and generality, that is not required here.

    c                    s   h | ]}| kr|qS  r   .0nGr   @/tmp/pip-unpacked-wheel-_lngutwb/networkx/algorithms/boundary.py	<setcomp>U   s      z edge_boundary.<locals>.<setcomp>)datakeysdefault)r   r   Nc                 3   s*   | ]"}|d   k|d  kA r|V  qdS r      Nr   r   e)nset1r   r   	<genexpr>d   s      z edge_boundary.<locals>.<genexpr>c                 3   sB   | ]:}|d   kr|d ks6|d  kr|d  kr|V  qdS r   r   r   )r   nset2r   r   r   f   s      )Zis_multigraphedgesset)r
   nbunch1nbunch2r   r   r   r   r   )r
   r   r   r   r      s    Cc                    sJ    fdd|D }t t fdd|D | }|dk	rF|t |M }|S )ac  Returns the node boundary of `nbunch1`.

    The *node boundary* of a set *S* with respect to a set *T* is the
    set of nodes *v* in *T* such that for some *u* in *S*, there is an
    edge joining *u* to *v*. If *T* is not specified, it is assumed to
    be the set of all nodes not in *S*.

    Parameters
    ----------
    G : NetworkX graph

    nbunch1 : iterable
        Iterable of nodes in the graph representing the set of nodes
        whose node boundary will be returned. (This is the set *S* from
        the definition above.)

    nbunch2 : iterable
        Iterable of nodes representing the target (or "exterior") set of
        nodes. (This is the set *T* from the definition above.) If not
        specified, this is assumed to be the set of all nodes in `G`
        not in `nbunch1`.

    Returns
    -------
    set
        The node boundary of `nbunch1` with respect to `nbunch2`.

    Examples
    --------
    >>> G = nx.wheel_graph(6)

    When nbunch2=None:

    >>> list(nx.node_boundary(G, (3, 4)))
    [0, 2, 5]

    When nbunch2 is given:

    >>> list(nx.node_boundary(G, (3, 4), (0, 1, 5)))
    [0, 5]

    Notes
    -----
    Any element of `nbunch` that is not in the graph `G` will be
    ignored.

    `nbunch1` and `nbunch2` are usually meant to be disjoint, but in
    the interest of speed and generality, that is not required here.

    c                    s   h | ]}| kr|qS r   r   r   r	   r   r   r      s      z node_boundary.<locals>.<setcomp>c                 3   s   | ]} | V  qd S )Nr   )r   vr	   r   r   r      s     z node_boundary.<locals>.<genexpr>N)r   r   from_iterable)r
   r   r   r   Zbdyr   r	   r   r   m   s
    4 )NFFN)N)	__doc__	itertoolsr   ZnetworkxZnx__all__	_dispatchr   r   r   r   r   r   <module>   s   
Z