blob: c23e47c8b434c218c976cd721bacd818239d43d5 [file] [log] [blame]
Scott Bakerbba67b62019-01-28 17:38:21 -08001# Copyright 2017-present Open Networking Foundation
2#
3# Licensed under the Apache License, Version 2.0 (the "License");
4# you may not use this file except in compliance with the License.
5# You may obtain a copy of the License at
6#
7# http://www.apache.org/licenses/LICENSE-2.0
8#
9# Unless required by applicable law or agreed to in writing, software
10# distributed under the License is distributed on an "AS IS" BASIS,
11# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12# See the License for the specific language governing permissions and
13# limitations under the License.
14
15
16from __future__ import print_function
17from xossynchronizer.modelaccessor import *
18from xossynchronizer.dependency_walker_new import *
19from xossynchronizer.policy import Policy
20
21import imp
22import pdb
23import time
24import traceback
25
26
27class XOSPolicyEngine(object):
28 def __init__(self, policies_dir, log):
29 self.model_policies = self.load_model_policies(policies_dir)
30 self.policies_by_name = {}
31 self.policies_by_class = {}
32 self.log = log
33
34 for policy in self.model_policies:
35 if policy.model_name not in self.policies_by_name:
36 self.policies_by_name[policy.model_name] = []
37 self.policies_by_name[policy.model_name].append(policy)
38
39 if policy.model not in self.policies_by_class:
40 self.policies_by_class[policy.model] = []
41 self.policies_by_class[policy.model].append(policy)
42
43 def update_wp(self, d, o):
44 try:
45 save_fields = []
46 if d.write_protect != o.write_protect:
47 d.write_protect = o.write_protect
48 save_fields.append("write_protect")
49 if save_fields:
50 d.save(update_fields=save_fields)
51 except AttributeError as e:
52 raise e
53
54 def update_dep(self, d, o):
55 try:
56 print("Trying to update %s" % d)
57 save_fields = []
58 if d.updated < o.updated:
59 save_fields = ["updated"]
60
61 if save_fields:
62 d.save(update_fields=save_fields)
63 except AttributeError as e:
64 log.exception("AttributeError in update_dep", e=e)
65 raise e
66 except Exception as e:
67 log.exception("Exception in update_dep", e=e)
68
69 def delete_if_inactive(self, d, o):
70 try:
71 d.delete()
72 print("Deleted %s (%s)" % (d, d.__class__.__name__))
73 except BaseException:
74 pass
75 return
76
77 def load_model_policies(self, policies_dir):
78 policies = []
79 for fn in os.listdir(policies_dir):
80 if fn.startswith("test"):
81 # don't try to import unit tests!
82 continue
83 pathname = os.path.join(policies_dir, fn)
84 if (
85 os.path.isfile(pathname)
86 and fn.endswith(".py")
87 and (fn != "__init__.py")
88 ):
89 module = imp.load_source(fn[:-3], pathname)
90 for classname in dir(module):
91 c = getattr(module, classname, None)
92
93 # make sure 'c' is a descendent of Policy and has a
94 # provides field (this eliminates the abstract base classes
95 # since they don't have a provides)
96
97 if (
98 inspect.isclass(c)
99 and issubclass(c, Policy)
100 and hasattr(c, "model_name")
101 and (c not in policies)
102 ):
103 if not c.model_name:
104 log.info(
105 "load_model_policies: skipping model policy",
106 classname=classname,
107 )
108 continue
109 if not model_accessor.has_model_class(c.model_name):
110 log.error(
111 "load_model_policies: unable to find model policy",
112 classname=classname,
113 model=c.model_name,
114 )
115 c.model = model_accessor.get_model_class(c.model_name)
116 policies.append(c)
117
118 log.info("Loaded model policies", policies=policies)
119 return policies
120
121 def execute_model_policy(self, instance, action):
122 # These are the models whose children get deleted when they are
123 delete_policy_models = ["Slice", "Instance", "Network"]
124 sender_name = getattr(instance, "model_name", instance.__class__.__name__)
125
126 # if (action != "deleted"):
127 # walk_inv_deps(self.update_dep, instance)
128 # walk_deps(self.update_wp, instance)
129 # elif (sender_name in delete_policy_models):
130 # walk_inv_deps(self.delete_if_inactive, instance)
131
132 policies_failed = False
133 for policy in self.policies_by_name.get(sender_name, None):
134 method_name = "handle_%s" % action
135 if hasattr(policy, method_name):
136 try:
137 log.debug(
138 "MODEL POLICY: calling handler",
139 sender_name=sender_name,
140 instance=instance,
141 policy=policy.__name__,
142 method=method_name,
143 )
144 getattr(policy(), method_name)(instance)
145 log.debug(
146 "MODEL POLICY: completed handler",
147 sender_name=sender_name,
148 instance=instance,
149 policy_name=policy.__name__,
150 method=method_name,
151 )
152 except Exception as e:
153 log.exception("MODEL POLICY: Exception when running handler", e=e)
154 policies_failed = True
155
156 try:
157 instance.policy_status = "%s" % traceback.format_exc(limit=1)
158 instance.policy_code = 2
159 instance.save(update_fields=["policy_status", "policy_code"])
160 except Exception as e:
161 log.exception(
162 "MODEL_POLICY: Exception when storing policy_status", e=e
163 )
164
165 if not policies_failed:
166 try:
167 instance.policed = max(instance.updated, instance.changed_by_step)
168 instance.policy_status = "done"
169 instance.policy_code = 1
170
171 instance.save(update_fields=["policed", "policy_status", "policy_code"])
172
173 if hasattr(policy, "after_policy_save"):
174 policy().after_policy_save(instance)
175
176 log.info("MODEL_POLICY: Saved", o=instance)
177 except BaseException:
178 log.exception(
179 "MODEL POLICY: Object failed to update policed timestamp",
180 instance=instance,
181 )
182
183 def noop(self, o, p):
184 pass
185
186 def run(self):
187 while True:
188 start = time.time()
189 try:
190 self.run_policy_once()
191 except Exception as e:
192 log.exception("MODEL_POLICY: Exception in run()", e=e)
193 if time.time() - start < 5:
194 time.sleep(5)
195
196 # TODO: This loop is different from the synchronizer event_loop, but they both do mostly the same thing. Look for
197 # ways to combine them.
198
199 def run_policy_once(self):
200 models = self.policies_by_class.keys()
201
202 model_accessor.check_db_connection_okay()
203
204 objects = model_accessor.fetch_policies(models, False)
205 deleted_objects = model_accessor.fetch_policies(models, True)
206
207 for o in objects:
208 if o.deleted:
209 # This shouldn't happen, but previous code was examining o.deleted. Verify.
210 continue
211 if not o.policed:
212 self.execute_model_policy(o, "create")
213 else:
214 self.execute_model_policy(o, "update")
215
216 for o in deleted_objects:
217 self.execute_model_policy(o, "delete")
218
219 try:
220 model_accessor.reset_queries()
221 except Exception as e:
222 # this shouldn't happen, but in case it does, catch it...
223 log.exception("MODEL POLICY: exception in reset_queries", e)