D7net
Home
Console
Upload
information
Create File
Create Folder
About
Tools
:
/
proc
/
thread-self
/
root
/
opt
/
alt
/
python37
/
lib
/
python3.7
/
site-packages
/
lvestats
/
lib
/
Filename :
ustate.py
back
Copy
# coding=utf-8 # # Copyright © Cloud Linux GmbH & Cloud Linux Software, Inc 2010-2019 All Rights Reserved # # Licensed under CLOUD LINUX LICENSE AGREEMENT # http://cloudlinux.com/docs/LICENSE.TXT from __future__ import absolute_import from builtins import zip from typing import List, Dict, Optional # pylint: disable=unused-import import logging log = logging.getLogger('ustate') try: # We cann't prescribe a hard dependency on mysql binary in spec-file # because it can be used by different versions import MySQLdb from MySQLdb import OperationalError as MySQLOperationalError except ImportError as e: MySQLdb = None log.debug("Can't import MySQLdb; %s" % str(e)) class MySQLOperationalError(Exception): pass try: from clcommon.cpapi import db_access, dblogin_cplogin_pairs, cpusers from clcommon.cpapi.cpapiexceptions import NoPackage from clcommon.utils import run_command except ImportError: if __name__ == "__main__": pass def _pars_lveps(lveps_output): """ pars /usr/sbin/lveps -c 1 -p -d -n result example returned data: {504: {'CPU': '26%', 'IO': '0', 'MEM': '1', 'EP': '0', 'IOPS': 'N/A', 'PNO': '3', 'TNO': '3', 'TID': {4400: {'CPU': '26%', 'IO': 'N/A', 'MEM': '1', 'CMD': 'md5sum', 'IOPS': 'N/A'}, 4381: {'CPU': '0%', 'IO': 'N/A', 'MEM': '1', 'CMD': 'su', 'IOPS': 'N/A'}, 4382: {'CPU': '0%', 'IO': 'N/A', 'MEM': '1', 'CMD': 'bash', 'IOPS': 'N/A'}}}, 500: {'CPU': '13%', 'IO': '0', 'MEM': '1', 'EP': '0', 'IOPS': 'N/A', 'PNO': '3', 'TNO': '3', 'TID': {4266: {'CPU': '0%', 'IO': 'N/A', 'MEM': '1', 'CMD': 'su', 'IOPS': 'N/A'}, 4299: {'CPU': '13%', 'IO': 'N/A', 'MEM': '1', 'CMD': 'cat', 'IOPS': 'N/A'}, 4267: {'CPU': '0%', 'IO': 'N/A', 'MEM': '1', 'CMD': 'bash', 'IOPS': 'N/A'}}}} example of data manipulation: getting a list of user id >>> lveps_data = _pars_lveps() >>> user_id_list = lveps_data.keys() a list of processes tid particular user id 504 >>> user_tid_list = lveps_data[504]['TID'].keys() getting CPU load user >>> user_cpu = lveps_data[504]['CPU'] getting CPU load specific process >>> lveps_data[504]['TID'][4400] """ lveps_lines = lveps_output.split('\n') header_line = lveps_lines.pop(0) columns_name = header_line.split() # replace columns name to standart replace_col = {'SPEED': 'CPU', 'COM': 'CMD'} columns_name = [replace_col.get(col_name, col_name) for col_name in columns_name] lveps_data = {} user_id = 0 for lveps_line_index, lveps_line in enumerate(lveps_lines): if not lveps_line: continue lveps_line_splited = lveps_line.split(None, len(columns_name)-1) # if the first one is number - use it as lve_id has_id = ((len(lveps_line_splited) == len(columns_name)) and lveps_line_splited[0].isdigit()) if not has_id: lveps_line_splited = lveps_line.split(None, len(columns_name)-2) lveps_line_splited.insert(0, '') if len(lveps_line_splited) != len(columns_name): log.error("lveps output was incorrect: %s", lveps_line, extra={ "data": {"lveps_lines": lveps_lines[max(0, lveps_line_index - 5): lveps_line_index + 15]}}) break lveps_dict_line = dict(zip(columns_name, lveps_line_splited)) if has_id: # FIXME: need to add a filter to a minimum uid user_id = int(lveps_dict_line.pop('ID')) try: del lveps_dict_line['CMD'] except KeyError: pass lveps_data[user_id] = lveps_dict_line lveps_data[user_id]['TID'] = dict() lveps_data[user_id].pop('PID', None) else: lveps_dict_line.pop('EP', None) lveps_dict_line.pop('PNO', None) lveps_dict_line.pop('TNO', None) lveps_dict_line.pop('ID', None) try: lveps_data[user_id]['TID'][int(lveps_dict_line.pop('TID'))] = lveps_dict_line except (ValueError, KeyError) as e: log.error("Can't parse lveps output: %s", str(e)) return lveps_data def get_lveps(): lveps_output = _get_lveps_output() return _pars_lveps(lveps_output) def _get_lveps_output(): lveps_output = run_command([ '/usr/sbin/lveps', '-c', '1', '-p', '-d', '-n', '-o', 'id:10,ep:10,pno:10,pid:15,tno:5,tid:15,cpu:7,mem:15,com:256'], convert_to_str=False) # ignore any non-utf8 symbols here # we use this information only to show user return lveps_output.decode('utf-8', 'replace') class SQLSnapshot(object): def __init__(self): self._mysql_conn = None self._dblogin_cplogin_map = dict() self._db_users = set() # attribute to detect new database users def __enter__(self): self.connect() return self def __exit__(self, exc_type, exc_val, exc_tb): self.close() def connect(self): """ Obtain access data and connect to mysql database """ if not MySQLdb: raise NoPackage('need install mysql server for binding _mysql.so') access = db_access() # pylint:disable=assignment-from-no-return mysql_login = access['login'] mysql_pass = access['pass'] mysql_host = access.get('host', 'localhost') try: self._mysql_conn = MySQLdb.connect(host=mysql_host, user=mysql_login, passwd=mysql_pass, use_unicode=True, charset="utf8mb4") except MySQLdb.OperationalError as e: raise MySQLOperationalError(str(e)) def _refresh_map(self): """ Refresh <database user>:<system user> map """ self._dblogin_cplogin_map = dict(dblogin_cplogin_pairs()) # pylint:disable=assignment-from-no-return def close(self): """ Close Mysql connection """ self._mysql_conn.close() def _raw_processlist(self): result = tuple() try: cursor = self._mysql_conn.cursor() cursor.execute('SHOW FULL PROCESSLIST') result = cursor.fetchall() except MySQLdb.OperationalError as e: log.warning(str(e)) return result def _get_sql_process_list(self): """ Group processlist by database user name :rtype: dict """ process_snapshot = dict() for sql_tuple in self._raw_processlist(): db_username = sql_tuple[1] sql_cmd = sql_tuple[4] # (CMD) type/state of the request, in this case we are only interested 'Query' sql_time = sql_tuple[5] # (Time) Time sql_query = sql_tuple[7] or '' # (SQL-query) sql query, None if no sql query if sql_cmd == 'Sleep': # filter 'Sleep' state continue snapshot_line = [sql_cmd, sql_time, sql_query] # group by database user name grouped_by_user = process_snapshot.get(db_username, list()) grouped_by_user.append(snapshot_line) process_snapshot[db_username] = grouped_by_user return process_snapshot def get(self, cplogin_lst=None): # type: (Optional[List[str]]) -> Dict[str, List] """ :param cplogin_lst: a list of users to retrieve data; None if the data is returned for all users registered in the control panel :return: sql queries for each user """ process_snapshot = self._get_sql_process_list() # refresh map if new database users detect new_db_users = set(process_snapshot.keys()) - self._db_users if new_db_users: self._refresh_map() log.debug('New database user(s) %s detected; ' 'database users map refreshed', str(list(new_db_users))[1:-1]) # refresh self._db_users to detect new users for new_db_user in new_db_users: self._db_users.add(new_db_user) # group and filter by control panel users sql_snapshot = dict() cplogin_lst_ = cplogin_lst or cpusers() or list() for db_username, sql_snap in list(process_snapshot.items()): cp_username = self._dblogin_cplogin_map.get(db_username) if cp_username is not None and cp_username in cplogin_lst_: # group sql snapshots by control panel user sql_snap_ = sql_snapshot.get(cp_username, list()) sql_snap_.extend(sql_snap) sql_snapshot[cp_username] = sql_snap_ return sql_snapshot