d539ead5ef9fe12aec0fe09a73c312fc79909f36
[osm/SO.git] / common / python / rift / mano / sdn / accounts.py
1
2 #
3 # Copyright 2017 RIFT.IO Inc
4 #
5 # Licensed under the Apache License, Version 2.0 (the "License");
6 # you may not use this file except in compliance with the License.
7 # You may obtain a copy of the License at
8 #
9 # http://www.apache.org/licenses/LICENSE-2.0
10 #
11 # Unless required by applicable law or agreed to in writing, software
12 # distributed under the License is distributed on an "AS IS" BASIS,
13 # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 # See the License for the specific language governing permissions and
15 # limitations under the License.
16 #
17
18 import sys
19 import asyncio
20 from gi import require_version
21 require_version('RwTypes', '1.0')
22 require_version('RwsdnalYang', '1.0')
23 require_version('RwSdnYang', '1.0')
24
25 from gi.repository import (
26 RwTypes,
27 RwsdnalYang,
28 RwSdnYang,
29 )
30 import rw_peas
31
32 if sys.version_info < (3, 4, 4):
33 asyncio.ensure_future = asyncio.async
34
35
36 class PluginLoadingError(Exception):
37 pass
38
39
40 class SDNAccountCalError(Exception):
41 pass
42
43
44 class SDNAccount(object):
45 def __init__(self, log, rwlog_hdl, account_msg):
46 self._log = log
47 self._account_msg = account_msg.deep_copy()
48
49 self._sdn_plugin = None
50 self._engine = None
51
52 self._sdn = self.plugin.get_interface("Topology")
53 self._sdn.init(rwlog_hdl)
54
55 self._status = RwsdnalYang.SDNAccount_ConnectionStatus(
56 status="unknown",
57 details="Connection status lookup not started"
58 )
59
60 self._validate_task = None
61
62 @property
63 def plugin(self):
64 if self._sdn_plugin is None:
65 try:
66 self._sdn_plugin = rw_peas.PeasPlugin(
67 getattr(self._account_msg, self.account_type).plugin_name,
68 'RwSdn-1.0',
69 )
70
71 except AttributeError as e:
72 raise PluginLoadingError(str(e))
73
74 self._engine, _, _ = self._sdn_plugin()
75
76 return self._sdn_plugin
77
78 def _wrap_status_fn(self, fn, *args, **kwargs):
79 ret = fn(*args, **kwargs)
80 rw_status = ret[0]
81 if rw_status != RwTypes.RwStatus.SUCCESS:
82 msg = "%s returned %s" % (fn.__name__, str(rw_status))
83 self._log.error(msg)
84 raise SDNAccountCalError(msg)
85
86 # If there was only one other return value besides rw_status, then just
87 # return that element. Otherwise return the rest of the return values
88 # as a list.
89 return ret[1] if len(ret) == 2 else ret[1:]
90
91 @property
92 def sdn(self):
93 return self._sdn
94
95 @property
96 def name(self):
97 return self._account_msg.name
98
99 @property
100 def account_msg(self):
101 return self._account_msg
102
103 @property
104 def sdnal_account_msg(self):
105 return RwsdnalYang.SDNAccount.from_dict(
106 self.account_msg.as_dict(),
107 ignore_missing_keys=True,
108 )
109
110 def sdn_account_msg(self, account_dict):
111 self._account_msg = RwSdnYang.SDNAccount.from_dict(account_dict)
112
113 @property
114 def account_type(self):
115 return self._account_msg.account_type
116
117 @property
118 def connection_status(self):
119 return self._status
120
121 def update_from_cfg(self, cfg):
122 self._log.debug("Updating parent SDN Account to %s", cfg)
123
124 raise NotImplementedError("Update SDN account not yet supported")
125
126
127 @asyncio.coroutine
128 def validate_sdn_account_credentials(self, loop):
129 self._log.debug("Validating SDN Account credentials %s", self._account_msg)
130 self._status = RwSdnYang.SDNAccount_ConnectionStatus(
131 status="validating",
132 details="SDN account connection validation in progress"
133 )
134 rwstatus, status = yield from loop.run_in_executor(
135 None,
136 self._sdn.validate_sdn_creds,
137 self.sdnal_account_msg,
138 )
139 if rwstatus == RwTypes.RwStatus.SUCCESS:
140 self._status = RwSdnYang.SDNAccount_ConnectionStatus.from_dict(status.as_dict())
141 else:
142 self._status = RwSdnYang.SDNAccount_ConnectionStatus(
143 status="failure",
144 details="Error when calling SDNAL validate SDN creds"
145 )
146
147 self._log.info("Got SDN account validation response: %s", self._status)
148
149 def start_validate_credentials(self, loop):
150 if self._validate_task is not None:
151 self._validate_task.cancel()
152 self._validate_task = None
153
154 self._validate_task = asyncio.ensure_future(
155 self.validate_sdn_account_credentials(loop),
156 loop=loop
157 )