python腳本: 計算節點建立cgroups綁定虛擬核心,實現計算資源隔離

 
import MySQLdb

import os

import signal


import shlex

import subprocess

from eventlet.green import subprocess as green_subprocess

from eventlet import greenthread

import logging

import string

import time


mysql_host = "10.160.0.120"

mysql_db = "nova"

mysql_user = "nova"

mysql_passwd = "87da3417bb3a42ee"

mysql_port = 3306

mysql_charset = "utf8"

cgroups_hierarchy = "/home/cgroups/cpu"


LOG = logging.getLogger(__name__)

def create_process(cmd, root_helper=None, addl_env=None):
    if root_helper:
        cmd = shlex.split(root_helper) + cmd
    cmd = map(str, cmd)
    LOG.debug(("Running command: %s"), cmd)
    env = os.environ.copy()
    if addl_env:
        env.update(addl_env)
    
    obj = subprocess_popen(cmd, shell=False,
                            stdin=subprocess.PIPE,
                            stdout=subprocess.PIPE,
                            stderr=subprocess.PIPE,
                            env=env)
    
    return obj, cmd


def execute(cmd, root_helper=None, process_input=None, addl_env=None,
            check_exit_code=True, return_stderr=False):
    try:
        obj, cmd = create_process(cmd, root_helper=root_helper,
                                  addl_env=addl_env)
        _stdout, _stderr = (process_input and
                            obj.communicate(process_input) or
                            obj.communicate())
        obj.stdin.close()
        m = ("\nCommand: %(cmd)s\nExit code: %(code)s\nStdout: %(stdout)r\n"
              "Stderr: %(stderr)r") % {'cmd': cmd, 'code': obj.returncode,
                                       'stdout': _stdout, 'stderr': _stderr}
        LOG.debug(m)
        if obj.returncode and check_exit_code:
            raise RuntimeError(m)
    finally:
        greenthread.sleep(0)
    
    return return_stderr and (_stdout, _stderr) or _stdout


def _subprocess_setup():
    signal.signal(signal.SIGPIPE, signal.SIG_DFL)


def subprocess_popen(args, stdin=None, stdout=None, stderr=None, shell=False,
                     env=None):
    return green_subprocess.Popen(args, shell=shell, stdin=stdin, stdout=stdout,
                            stderr=stderr, preexec_fn=_subprocess_setup,
                            close_fds=True, env=env)


def query_db_of_local_vms():
    """ query controller node db for information about vms running on local host """
    conn = None
    try:
        conn = MySQLdb.connect(host=mysql_host,
                               user=mysql_user,
                               passwd=mysql_passwd,
                               db=mysql_db,
                               port=mysql_port,
                               charset=mysql_charset)
    except Exception as e:
        #LOG.error("Fail to connect mysql .")
        
        raise e
    else:

        LOG.info("Connect to mysql .")
    
    local_compute_name = get_host()
    
    sql = "SELECT vcpus,uuid FROM instances WHERE host='%s' AND deleted=0"%local_compute_name
    
    vms = {}
    
    try:
        cursor = conn.cursor()
        cursor.execute(sql)
        result = cursor.fetchall()
        cursor.close()
        conn.close()
        
        for item in result:
            vms.update({item[1]:item[0]})
    except Exception as ex:
        #LOG.error("Exception happens while querying mysql.")
        
        raise ex
    else:
        return vms

def get_host():
    return os.environ['HOSTNAME']


def get_ip():
    """ parse /etc/hosts to get local ip"""
    df = open("/etc/hosts")
    hosts = df.readlines()
    hostname = get_host()
    host_rec = [line for line in hosts if hostname in line and "#" not in line][0]
    return host_rec.split()[0].strip()


def check_cglib():
    """ check required cglibs"""
    check_pkgs = "rpm -qa"
    cglib_kw = "libcgroup"
    cmd = check_pkgs.split()
    pkgs = execute(cmd,root_helper=None)
    
    cglibs = [pkg for pkg in pkgs.split("\n") if cglib_kw in pkg]
    
    if len(cglibs)==0:
        print "libcgroup-x.xx-x.xx.x86_64 not installed.Exit."
        exit(1)


def init_cgroups():
    
    """ensure cgrouplib installed"""
    check_cglib()
    
    """ create cgroups base architecture """
    """clear old cgroups settings mounted on cgroups_hierarchy """
    cg_clear()
    
    """create base cgroups hierarchy"""
    create_hierarchy = ("mkdir -p %s"%cgroups_hierarchy).split()
    execute(create_hierarchy,root_helper=None)
    
    """mount target subsystem,that's cpuset,to established hierarchy."""
    mount_cg = ("mount -t cgroup -o cpuset cpuset2015  %s"%cgroups_hierarchy).split()
    execute(mount_cg,root_helper=None)


def cpuinfo():
    """ get cpu counts and memory nodes """
    cpuinfo = "lscpu"
    cmd = cpuinfo.split()
    retv = execute(cmd,root_helper=None)
    memNodes = retv.count("NUMA node")-1
    cpus = string.atoi([line for line in retv.split("\n") if "CPU(s)" in line][0].split(":")[1].strip())
    return {"cpu_count":cpus,"memNodes":memNodes}


def assign_cores():
    """ cal out schemes of core binding for instances running on local host"""
    vmDetails = query_db_of_local_vms()
    cpuinfos = cpuinfo()
    def assign(require,all):
        tmp = {}
        i=0
        for k,v in require.iteritems():
                    cores = [p%all for p in xrange(i,i+v)]
                    i=i+v
                    tmp.update({k:{"cores":cores,"memNodes":"0-%d"%(cpuinfos["memNodes"]-1)}})
        return tmp
    vmDetails = assign(vmDetails,cpuinfos["cpu_count"])
    return vmDetails


def get_pids(vmDetails):
    query_pids = "ps aux"
    cmd = query_pids.split()
    retv = execute(cmd,root_helper=None)
    qemu_kvm_processes =[p for p in retv.split("\n") if "qemu-kvm" in p and len(p)>100]
    tmp = {} 
    for vmDetail in vmDetails:
        tmp.update({vmDetail:vmDetails[vmDetail]})
        for qkp in qemu_kvm_processes:
            if vmDetail not in qkp:
                continue                
            else:
                pid = string.atoi(qkp.split()[1]) 
                tmp[vmDetail]["pid"] = pid
    return tmp
 

def create_cgs_for_instances(vmDetails):
    for item in vmDetails:
        detail = vmDetails[item]
        if "pid" not in detail or "cores" not in detail or "memNodes" not in detail:
            print "Instance %s invalid"%item
            continue
        else:
            """ create private group for each instance"""
            instance_cg = "%s/%s"%(cgroups_hierarchy,item)
            create_cg_for_instance = ("mkdir -p %s"%instance_cg).split()
            execute(create_cg_for_instance,root_helper=None)
            setcpu_txt = "%s"%detail["cores"][0]
            if len(detail["cores"]) > 1:
                for core_num in detail["cores"][1:]:
                    setcpu_txt = "%s,%s"%(setcpu_txt,core_num)
     
            setmem_txt = detail["memNodes"]
            
            fd_setcpu = open("%s/cpuset.cpus"%instance_cg,"w")
            fd_setcpu.write(setcpu_txt)
            fd_setcpu.close()
            fd_setmem = open("%s/cpuset.mems"%instance_cg,"w")
            fd_setmem.write(setmem_txt)
            fd_setmem.close()
            
            print detail["pid"]
            fd_task = open("%s/tasks"%instance_cg,"w")
            fd_task.write("%s"%detail["pid"])
            fd_task.close()
           

def cg_clear():
    """ destroy existing cgroups """
    cgclear = "umount -t cgroup %s"%cgroups_hierarchy
    cmd = cgclear.split()
    try:
        execute(cmd,root_helper=None)
    except Exception as e:
        pass


def periodic_task():
    init_cgroups()
    hosted_vms = assign_cores()
    hosted_vms = get_pids(hosted_vms)
    create_cgs_for_instances(hosted_vms)
 
if __name__ == "__main__":
    periodic_task() 
    while True:
        time.sleep(3*60)
        print "Loop Job."
        periodic_task()

該腳本做用是:鏈接數據庫查詢出運行在當前宿主下的客戶機的cpu配額,而後再根據當前宿主的CPU信息,作出虛擬核心的分配;建立cgroups 綁定虛擬核心,實現資源隔離.node

該腳本後臺運行:python

nohup python /path/to/this/scripy  &
相關文章
相關標籤/搜索