blob: 73459f7110123340c4b713029483e3e31e495001 [file] [log] [blame]
palsus5ddcded2021-02-09 18:23:03 +00001# -*- coding: utf-8 -*-
2
3# Copyright 2018 Whitestack, LLC
4# *************************************************************
5
6# This file is part of OSM Monitoring module
7# All Rights Reserved to Whitestack, LLC
8
9# Licensed under the Apache License, Version 2.0 (the "License"); you may
10# not use this file except in compliance with the License. You may obtain
11# a copy of the License at
12
13# http://www.apache.org/licenses/LICENSE-2.0
14
15# Unless required by applicable law or agreed to in writing, software
16# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
17# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
18# License for the specific language governing permissions and limitations
19# under the License.
20# For those usages not covered by the Apache License, Version 2.0 please
21# contact: glavado@whitestack.com or fbravo@whitestack.com
22##
23import logging
24import requests
25import base64
26import json
27from osm_mon.core.config import Config
28
29log = logging.getLogger(__name__)
30
31
32class GrafanaBackend:
33 def __init__(self, config: Config):
34 self.conf = config
35 self.url = config.get('grafana', 'url')
36 grafana_user = config.get("grafana", "user")
37 grafana_password = config.get("grafana", "password")
38 self.headers = {
39 'content-type': "application/json",
40 'authorization': "Basic %s" % base64.b64encode(
41 (grafana_user + ":" + grafana_password).encode("utf-8")).decode()
42 }
43
44 def get_all_dashboard_uids(self):
45 # Gets only dashboards that were automated by OSM (with tag 'osm_automated')
46 response = requests.request("GET", self.url + "/api/search?tag=osm_automated", headers=self.headers)
47 dashboards = response.json()
48 dashboard_uids = []
49 for dashboard in dashboards:
50 dashboard_uids.append(dashboard['uid'])
51 log.debug("Searching for all dashboard uids: %s", dashboard_uids)
52 return dashboard_uids
53
54 def get_dashboard_status(self, uid):
55 response = requests.request("GET", self.url + "/api/dashboards/uid/" + uid, headers=self.headers)
56 log.debug("Searching for dashboard result: %s", response.text)
57 return response
58
59 def create_dashboard(self, uid, name, json_file, project_name=None):
60 try:
61 with open(json_file) as f:
62 dashboard_data = f.read()
63
64 dashboard_data = dashboard_data.replace('OSM_ID', uid).replace('OSM_NAME', name)
Atul Agarwal9006f6d2021-01-06 13:29:19 +000065 dashboard_json_data = json.loads(dashboard_data)
palsusf0d75662021-02-14 19:06:33 +000066 # Get folder id
Atul Agarwal9006f6d2021-01-06 13:29:19 +000067 if project_name:
68 folder_name = project_name
69 else:
70 folder_name = name
71 response_folder_id = requests.request(
72 "GET", self.url + "/api/folders/{}".format(folder_name), headers=self.headers)
73 if response_folder_id.status_code == 200:
74 folder_id = json.loads(response_folder_id.text)["id"]
75 dashboard_json_data["folderId"] = folder_id
76 dashboard_json_data["overwrite"] = False
palsus5ddcded2021-02-09 18:23:03 +000077
Atul Agarwal282c9c92021-03-16 11:02:14 +000078 response = self.send_request_for_creating_dashboard(dashboard_json_data)
Atul Agarwal9006f6d2021-01-06 13:29:19 +000079
Atul Agarwal282c9c92021-03-16 11:02:14 +000080 # Admin dashboard will be created if already exists. Rest will remain same.
palsusa6bce212021-03-03 20:07:07 +000081 if json.loads(response.text).get("status") == "name-exists":
Atul Agarwal282c9c92021-03-16 11:02:14 +000082 # Delete any previous project-admin dashboard if it already exist.
83 if name == 'admin':
84 self.delete_admin_dashboard()
85 response = self.send_request_for_creating_dashboard(dashboard_json_data)
86 else:
87 return
palsusa6bce212021-03-03 20:07:07 +000088
palsusf0d75662021-02-14 19:06:33 +000089 # Get team id
palsus5ddcded2021-02-09 18:23:03 +000090 if project_name is not None:
91 name = project_name
92 response_team = requests.request(
93 "GET", self.url + "/api/teams/search?name={}".format(name), headers=self.headers)
palsusf0d75662021-02-14 19:06:33 +000094
palsusa6bce212021-03-03 20:07:07 +000095 # Remove default permissions of admin user's dashboard so that it is not visible to non-admin users
96 if len(json.loads(response_team.text)["teams"]) == 0:
palsusf0d75662021-02-14 19:06:33 +000097 # As team information is not available so it is admin user
98 dahboard_id = json.loads(response.text)["id"]
palsusf0d75662021-02-14 19:06:33 +000099 requests.request(
100 "POST", self.url + "/api/dashboards/id/{}/permissions".format(dahboard_id),
101 headers=self.headers)
102
Atul Agarwal9006f6d2021-01-06 13:29:19 +0000103 log.info("Dashboard %s is created in Grafana", name)
palsus5ddcded2021-02-09 18:23:03 +0000104 return response
105 except Exception:
106 log.exception("Exception processing message: ")
107
Atul Agarwal282c9c92021-03-16 11:02:14 +0000108 def send_request_for_creating_dashboard(self, dashboard_data):
109 response = requests.request(
110 "POST", self.url + "/api/dashboards/db/", data=json.dumps(dashboard_data), headers=self.headers)
111 return response
112
palsus5ddcded2021-02-09 18:23:03 +0000113 def delete_dashboard(self, uid):
114 response = requests.request("DELETE", self.url + "/api/dashboards/uid/" + uid, headers=self.headers)
Atul Agarwal9006f6d2021-01-06 13:29:19 +0000115 log.debug("Dashboard %s deleted from Grafana", uid)
palsus5ddcded2021-02-09 18:23:03 +0000116 return response
117
Atul Agarwal282c9c92021-03-16 11:02:14 +0000118 def delete_admin_dashboard(self):
119 requests.request(
120 "DELETE", self.url + "/api/dashboards/db/osm-project-status-admin", headers=self.headers)
121 log.debug("Dashboard osm-project-status-admin deleted from Grafana")
122
palsus5ddcded2021-02-09 18:23:03 +0000123 def create_grafana_users(self, user):
124 email = "{}@osm.etsi.org".format(user)
125 user_payload = {
126 "name": user,
127 "email": email,
128 "login": user,
129 "password": user,
130 }
131 response_users = requests.request("POST", self.url + "/api/admin/users/", json=user_payload,
132 headers=self.headers)
133 json_data = json.loads(response_users.text)
134 url = "/api/org/users/{}/".format(json_data["id"])
135 permission_payload = {"role": "Editor", }
Atul Agarwal9006f6d2021-01-06 13:29:19 +0000136 requests.request("PATCH", self.url + url, json=permission_payload, headers=self.headers)
137 log.info("New user %s created in Grafana", user)
palsus5ddcded2021-02-09 18:23:03 +0000138 return response_users
139
palsusf0d75662021-02-14 19:06:33 +0000140 # Create Grafana team with member
palsus5ddcded2021-02-09 18:23:03 +0000141 def create_grafana_teams_members(self, project_name, user_name, is_admin, proj_list):
palsusf0d75662021-02-14 19:06:33 +0000142 # Check if user exist in Grafana
palsus5ddcded2021-02-09 18:23:03 +0000143 user_response = requests.request("GET", self.url + "/api/users/lookup?loginOrEmail={}".format(user_name),
144 headers=self.headers)
145 user_obj = json.loads(user_response.text)
146 if user_response.status_code != 200:
147 user_response = self.create_grafana_users(user_name)
148 user_obj = json.loads(user_response.text)
149
150 user_id = user_obj["id"]
151
palsusf0d75662021-02-14 19:06:33 +0000152 # Get teams for user
palsus5ddcded2021-02-09 18:23:03 +0000153 team_objs = requests.request("GET", self.url + "/api/users/{}/teams".format(user_id), headers=self.headers)
154 team_obj = json.loads(team_objs.text)
155 team_list = []
156 if len(team_obj):
157 for team in team_obj:
158 team_list.append(team["name"])
159
160 proj_unlink = set(team_list) - set(proj_list)
161 for prj in proj_unlink:
162 response_team = requests.request("GET", self.url + "/api/teams/search?name={}".format(prj),
163 headers=self.headers)
164 team_id = json.loads(response_team.text)["teams"][0]["id"]
165 requests.request("DELETE", self.url + "/api/teams/{}/members/{}".format(team_id, user_id),
166 headers=self.headers)
167 if project_name != "admin":
palsusf0d75662021-02-14 19:06:33 +0000168 # Add member to team
palsus5ddcded2021-02-09 18:23:03 +0000169 response_team = requests.request("GET", self.url + "/api/teams/search?name={}".format(project_name),
170 headers=self.headers)
171
palsusf0d75662021-02-14 19:06:33 +0000172 # Search if team in Grafana corresponding to the project already exists
palsus5ddcded2021-02-09 18:23:03 +0000173 if not json.loads(response_team.text)["teams"]:
palsus5ddcded2021-02-09 18:23:03 +0000174 self.create_grafana_teams(project_name)
175 response_team = requests.request("GET", self.url + "/api/teams/search?name={}".format(project_name),
176 headers=self.headers)
177 team_id = json.loads(response_team.text)["teams"][0]["id"]
178 if project_name not in team_list:
palsusf0d75662021-02-14 19:06:33 +0000179 # Create a team in Grafana corresponding to the project as it doesn't exist
palsus5ddcded2021-02-09 18:23:03 +0000180 member_payload = {
181 "userId": user_id
182 }
183 requests.request("POST", self.url + "/api/teams/{}/members".format(team_id), json=member_payload,
184 headers=self.headers)
palsusf0d75662021-02-14 19:06:33 +0000185 # Check if user role or project name is admin
186 if is_admin or project_name == 'admin':
187 # Give admin righsts to user
palsus5ddcded2021-02-09 18:23:03 +0000188 url = "/api/org/users/{}/".format(user_id)
189 permission_payload = {"role": "Admin", }
190 requests.request("PATCH", self.url + url, json=permission_payload, headers=self.headers)
Atul Agarwal9006f6d2021-01-06 13:29:19 +0000191 log.info("User %s is assigned Admin permission", user_name)
192 else:
palsusf0d75662021-02-14 19:06:33 +0000193 # Give editor rights to user
Atul Agarwal9006f6d2021-01-06 13:29:19 +0000194 url = "/api/org/users/{}/".format(user_id)
195 permission_payload = {"role": "Editor", }
196 requests.request("PATCH", self.url + url, json=permission_payload, headers=self.headers)
197 log.info("User %s is assigned Editor permission", user_name)
palsus5ddcded2021-02-09 18:23:03 +0000198
palsusf0d75662021-02-14 19:06:33 +0000199 # Create team in Grafana
palsus5ddcded2021-02-09 18:23:03 +0000200 def create_grafana_teams(self, team_name):
201 team_payload = {"name": team_name, }
Atul Agarwal9006f6d2021-01-06 13:29:19 +0000202 requests.request("POST", self.url + "/api/teams", json=team_payload, headers=self.headers)
203 log.info("New team %s created in Grafana", team_name)
204
palsusf0d75662021-02-14 19:06:33 +0000205 # Create folder in Grafana
Atul Agarwal9006f6d2021-01-06 13:29:19 +0000206 def create_grafana_folders(self, folder_name):
207 folder_payload = {"uid": folder_name, "title": folder_name}
208 requests.request("POST", self.url + "/api/folders", json=folder_payload, headers=self.headers)
209 log.info("Dashboard folder %s created", folder_name)
palsus5ddcded2021-02-09 18:23:03 +0000210
palsusa6bce212021-03-03 20:07:07 +0000211 response_team = requests.request("GET", self.url + "/api/teams/search?name={}".format(folder_name),
212 headers=self.headers)
213 # Create team if it doesn't already exists
214 if len(json.loads(response_team.text)["teams"]) == 0:
215 self.create_grafana_teams(folder_name)
216 response_team = requests.request("GET", self.url + "/api/teams/search?name={}".format(folder_name),
217 headers=self.headers)
218 # Assign required permission to the team's folder
219 team_id = json.loads(response_team.text)["teams"][0]["id"]
220 permission_data = {"items": [{"teamId": team_id, "permission": 2}, ]}
221 requests.request("POST", self.url + "/api/folders/{}/permissions".format(folder_name),
222 json=permission_data, headers=self.headers)
223
Atul Agarwal517bd7e2021-03-15 09:16:09 +0000224 # delete user from grafana
palsus5ddcded2021-02-09 18:23:03 +0000225 def delete_grafana_users(self, user_name):
palsusf0d75662021-02-14 19:06:33 +0000226 # Get user id
palsus5ddcded2021-02-09 18:23:03 +0000227 response_id = requests.request("GET", self.url + "/api/users/lookup?loginOrEmail={}".format(user_name),
228 headers=self.headers)
229 try:
230 user_id = json.loads(response_id.text)["id"]
231 except Exception:
232 log.exception("Exception processing message: ")
palsusf0d75662021-02-14 19:06:33 +0000233 # Delete user
palsus5ddcded2021-02-09 18:23:03 +0000234 response = requests.request("DELETE", self.url + "/api/admin/users/{}".format(user_id), headers=self.headers)
Atul Agarwal9006f6d2021-01-06 13:29:19 +0000235 log.info("User %s deleted in Grafana", user_name)
palsus5ddcded2021-02-09 18:23:03 +0000236 return response
237
Atul Agarwal517bd7e2021-03-15 09:16:09 +0000238 # delete team from grafana
palsus5ddcded2021-02-09 18:23:03 +0000239 def delete_grafana_team(self, project_name):
palsusf0d75662021-02-14 19:06:33 +0000240 # Delete Grafana folder
Atul Agarwal9006f6d2021-01-06 13:29:19 +0000241 requests.request("DELETE", self.url + "/api/folders/{}".format(project_name),
242 headers=self.headers)
palsusf0d75662021-02-14 19:06:33 +0000243 # Delete Grafana team
palsus5ddcded2021-02-09 18:23:03 +0000244 team_obj = requests.request("GET", self.url + "/api/teams/search?name={}".format(project_name),
245 headers=self.headers)
246 team_id = json.loads(team_obj.text)["teams"][0]["id"]
247 response = requests.request("DELETE", self.url + "/api/teams/{}".format(team_id), headers=self.headers)
Atul Agarwal9006f6d2021-01-06 13:29:19 +0000248 log.info("Team %s deleted in Grafana", project_name)
palsus5ddcded2021-02-09 18:23:03 +0000249 return response
250
Atul Agarwal517bd7e2021-03-15 09:16:09 +0000251 # update grafana team
palsus5ddcded2021-02-09 18:23:03 +0000252 def update_grafana_teams(self, project_new_name, project_old_name):
253 team_obj = requests.request("GET", self.url + "/api/teams/search?name={}".format(project_old_name),
254 headers=self.headers)
255 team_id = json.loads(team_obj.text)["teams"][0]["id"]
256 data = {"name": project_new_name, }
257 response = requests.request("PUT", self.url + "/api/teams/{}".format(team_id), json=data, headers=self.headers)
258 log.info("Grafana team updated %s", response.text)
Atul Agarwal9006f6d2021-01-06 13:29:19 +0000259 return response
Atul Agarwal517bd7e2021-03-15 09:16:09 +0000260
261 # remove member from grafana team
262 def remove_grafana_team_member(self, user_name, project_data):
263 # Get user id
264 response_id = requests.request("GET", self.url + "/api/users/lookup?loginOrEmail={}".format(user_name),
265 headers=self.headers)
266 user_id = json.loads(response_id.text)["id"]
267 for project in project_data:
268 # Get team id
269 team_obj = requests.request("GET", self.url + "/api/teams/search?name={}".format(project['project']),
270 headers=self.headers)
271 team_id = json.loads(team_obj.text)["teams"][0]["id"]
272 response = requests.request("DELETE", self.url + "/api/teams/{}/members/{}".format(team_id, user_id),
273 headers=self.headers)
274 return response