-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathabstract_graph.py
executable file
·176 lines (154 loc) · 6.2 KB
/
abstract_graph.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
# This software is Copyright 2014 The Regents of the University of
# California. All Rights Reserved.
#
# Permission to copy, modify, and distribute this software and its
# documentation for educational, research and non-profit purposes, without fee,
# and without a written agreement is hereby granted, provided that the above
# copyright notice, this paragraph and the following three paragraphs appear
# in all copies.
#
# Permission to make commercial use of this software may be obtained by
# contacting:
# Technology Transfer Office
# 9500 Gilman Drive, Mail Code 0910
# University of California
# La Jolla, CA 92093-0910
# (858) 534-5815
# invent@ucsd.edu
#
# This software program and documentation are copyrighted by The Regents of the
# University of California. The software program and documentation are supplied
# "as is", without any accompanying services from The Regents. The Regents does
# not warrant that the operation of the program will be uninterrupted or
# error-free. The end-user understands that the program was developed for
# research purposes and is advised not to rely exclusively on the program for
# any reason.
#
# IN NO EVENT SHALL THE UNIVERSITY OF CALIFORNIA BE LIABLE TO
# ANY PARTY FOR DIRECT, INDIRECT, SPECIAL, INCIDENTAL, OR
# CONSEQUENTIAL DAMAGES, INCLUDING LOST PROFITS, ARISING
# OUT OF THE USE OF THIS SOFTWARE AND ITS DOCUMENTATION,
# EVEN IF THE UNIVERSITY OF CALIFORNIA HAS BEEN ADVISED OF
# THE POSSIBILITY OF SUCH DAMAGE. THE UNIVERSITY OF
# CALIFORNIA SPECIFICALLY DISCLAIMS ANY WARRANTIES,
# INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
# MERCHANTABILITY AND FITNESSFOR A PARTICULAR PURPOSE.
# THE SOFTWARE PROVIDED HEREUNDER IS ON AN "AS IS" BASIS, AND THE UNIVERSITY OF
# CALIFORNIA HAS NO OBLIGATIONS TO PROVIDE MAINTENANCE, SUPPORT, UPDATES,
# ENHANCEMENTS, OR MODIFICATIONS.
#Author: Viraj Deshpande
#Contact: virajbdeshpande@gmail.com
# This file defines classes and methods for an abstract undirected graph, vertex and edge.
import logging
class abstract_vertex(object):
"""Class describing a graph vertex.
Attributes:
elist: List of abstract_edges
vid: (optional) ID for the abstract_vertex
graph: (optional) abstract_graph to which the vertex belongs"""
def __init__(self, vid=0, graph=None):
"""Initiate vertex with optional vid and graph"""
self.elist = []
self.vid = vid # vertexid
self.graph = graph
if self.vid == 0 and self.graph is not None:
self.vid = self.graph.next_vid()
if self.graph is not None:
if vid in graph.vs:
raise Exception("Adding with duplicate vid")
self.graph.include_vertex(self)
def neighbors(self):
"""Return list of vertices connected to abstract_vertex by a direct edge"""
return [e.v2 for e in self.elist]
def __hash__(self):
"""Return hash based on vid to allow to efficiently check for presence of vid in graph, etc"""
return self.vid
def __repr__(self):
"""Vertex is represented by vid"""
return str(self.vid)
class abstract_edge(object):
"""Class describing a graph edge.
Attributes:
v1, v2: Ordered pair of vertices connected by the edge
eid: (optional) ID for the abstract_edge
graph: (optional) abstract_graph to which the vertex belongs."""
def __init__(self, v1, v2, eid=0, graph=None, update_vertices=True):
"""Initiate edge
Arguments: v1, v2, (optional)eid, (optional) graph.
update_vertices: (optional True/False) to update vertices to include edge in v1.elist, v2.elist. (default=True)"""
self.v1, self.v2 = v1, v2
self.eid = eid
self.graph = graph
if self.eid == 0 and self.graph is not None:
self.eid = self.graph.next_eid()
if self.graph is not None:
if eid in self.graph.es:
raise Exception("Adding edge with duplicate eid")
self.graph.include_edge(self)
if update_vertices:
if v1.graph is not v2.graph:
raise Exception("Adding edge between vertices of different graphs.")
if graph is not None and v1.graph is not graph:
raise Exception("Edge in different graph than vertex.")
if graph is None and v1.graph is not None:
graph = v1.graph
v1.elist.append(self)
v2.elist.append(self)
def neighbor(self, v):
"""Given a vertex, return its neighbor along the edge"""
if v == self.v1:
return self.v2
if v == self.v2:
return self.v1
raise Exception("Edge not connected to vertex")
def __hash__(self):
"""Return hash based on eid to allow to efficiently check for presence of eid in graph, etc"""
return self.eid
def length(self):
"""Not implemented"""
pass
def __repr__(self):
"""String representation of the form v1<->v2."""
return str(self.v1) + '<->' + str(self.v2)
class abstract_graph(object):
"""Class describing a graph.
Attributes:
vs: Dictionary from vid/key to vertex
es: Dictionary from eid/key to edge
max_vid: (internal) max_vid, used to assign vid for new vertex. Suggested to use function next_vid.
max_eid: (internal) max_eid, used to assign eid for new edge. Suggested to use function next_eid."""
def __init__(self):
"""Initiate empty graph"""
self.es = {} # key -->edges
self.vs = {} # key -->vertices
#self.logger = logging.getLogger('Algae')
self.max_eid = 1
self.max_vid = 1
def include_vertex(self, v):
"""Include orphan abstract_vertex in graph and update vertex.graph to point to self"""
if v.vid in self.vs and self.vs[v.vid] is not v:
raise "Adding vertex with duplicate vid"
if v.graph is not None and v.graph is not self:
raise "Adding vertex from another graph"
if v.graph is None:
v.graph = self
self.vs[v.vid] = v
def include_edge(self, e):
"""Include orphan abstract_edge in graph and update edge.graph to point to self. Vertices should be updated separately"""
if e.eid in self.es and self.es[e.eid] is not e:
raise "Adding edge with duplicate eid"
if e.graph is not None and e.graph is not self:
raise "Adding edge from another graph"
if e.graph is None:
e.graph = self
self.es[e.eid] = e
def next_eid (self):
"""Find the next eid available for assignment to new edge"""
while self.max_eid in self.es or -1 * self.max_eid in self.es:
self.max_eid += 1
return self.max_eid
def next_vid (self):
"""Find the next vid available for assignment to new vertex"""
while self.max_vid in self.vs or -1 * self.max_vid in self.vs:
self.max_vid += 1
return self.max_vid