forked from JunyiJ/Fault-Tree-Python
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathFault_tree_dict_time.py
148 lines (130 loc) · 3.69 KB
/
Fault_tree_dict_time.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
# Using dictionary(dict) to realize the fault tree function
# For example dict['Root']=['AND','E1','E2']
# All leaf nodes' names are stored in a list called leaves, for example ['A1','A2']
# To mannually test the code
# Build_FT('tree.txt') tree.txt is the xml file
# MCS(n,k)
# It will print the k cut set after running n times
import xml.etree.ElementTree as ET
from random import choice
import sys
import copy
import time
dict = {} # The dictionary to store all tree node information
dict_all={}
leaves = [] # The list to store the names of leaf node
nonleaves = []
def Build_FT(filename):
"""Store tree nodes in a dictionary dict and store leaf node names in list leaves"""
tree = ET.parse(filename)
root = tree.getroot()
global dict
global leaves
global nonleaves
for n in root:
name=n.attrib['id']
#nonleaves.append(name)
if name not in dict:
dict[name]=[]
for i in range(len(n)):
if i==0:
if n[i].text.upper()=="AND":
dict[name].append(1)
else:
dict[name].append(0)
else:
dict[name].append(n[i].text)
if n[i].text not in dict: # Also create entry for kid nodes
dict[n[i].text]=[]
"""Test whether a node is leaf node, if yes add it to list leaves"""
for i in dict:
dict_all[i]=-1
if not dict[i]:
leaves.append(i)
def MCS(n,k):
"""Using Monte Carlo algorith to find minimal cut set
Run the code n times and print the first k cut set
"""
global dict_all
dict_val=copy.deepcopy(dict_all)
#start_time = time.time()
final = {} # Store all result with the count as key. For example final[1]=[[1,0,0],[0,1,1]]
seq = [] # Store the count with no duplication
for i in range(n):
leaf={} # leaf is the dictionary to store the random value of each leaf
#count=0
for i in leaves:
leaf[i] = choice([0,1])
dict_val[i]=leaf[i]
#count += leaf[i]
result = Cal_FT(dict_val)
'''
if result:
cutset = []
for i in leaves:
cutset.append(str(leaf[i]))
cutset="".join(cutset)
if cutset not in final:
final[cutset]=count
final_sorted=sorted(zip(final.values(),final.keys())) #Order the cutset by its count
for i in range(k): #Print the first k result
cutset=list(final_sorted[i][1])
result=[]
for index in range(len(cutset)):
if cutset[index] is "1":
result.append(leaves[index])
print result
#end_time=time.time()
#print "Running time is", end_time-start_time
'''
def DFS():
"""
Using depth first search to find out all non-leaf nodes
Cal_FT2 will use this node list to calculate the root value without recurrion
"""
global nonleaves
dict_cp=copy.deepcopy(dict)
nonleaves=["Root"]
waitlist=["Root"]
while waitlist:
peek=waitlist[-1]
if len(dict_cp[peek])>=2:
sub_node=dict_cp[peek].pop()
if dict[sub_node]:
nonleaves.append(sub_node)
waitlist.append(sub_node)
else:
waitlist.pop()
def Cal_FT(dict_val):
global dict
"""Using non recursion method to calculate the value in Root node"""
nonleaves_cp=copy.deepcopy(nonleaves)
while nonleaves_cp:
cur_node=nonleaves_cp.pop()
if dict_val[cur_node]==-1:
result = dict[cur_node][0]
if dict[cur_node][0]:
for i in dict[cur_node][1:]:
result*=dict_val[i]
if not result:
break
else:
for i in dict[cur_node][1:]:
result+=dict_val[i]
if result:
break
dict_val[cur_node]=result
return dict_val["Root"]
starttime=time.time()
argv=sys.argv
filename=argv[1]
n=int(argv[2])
k=int(argv[3])
Build_FT(filename)
DFS()
MCS(n,k)
endtime=time.time()
timetxt="Script: Fault_tree_dict. Running time for %s is %s seconds\n"%(filename,endtime-starttime)
with open("running_time.txt","a") as f:
f.write(timetxt)
print(timetxt)