# vim: tabstop=4 shiftwidth=4 softtabstop=4
# Copyright (c) 2011 X.commerce, a business unit of eBay Inc.
# Copyright 2010 United States Government as represented by the
# Administrator of the National Aeronautics and Space Administration.
# All Rights Reserved.
#
#    Licensed under the Apache License, Version 2.0 (the "License"); you may
#    not use this file except in compliance with the License. You may obtain
#    a copy of the License at
#
#         http://www.apache.org/licenses/LICENSE-2.0
#
#    Unless required by applicable law or agreed to in writing, software
#    distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
#    WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
#    License for the specific language governing permissions and limitations
#    under the License.
#    Single virtual machine io limit , reader 70/MB , writer 50/MB
"""
    curl -d '{"auth": {"tenantName": "admin", "passwordCredentials": {"username": "admin", "password": "admin"}}}' -H "Content-type: application/json" http://127.0.0.1:35357/v2.0/tokens | python -m json.tool
    curl -v -d '{"ioctlAction": {"instance_name":"instance-0000000c","limit_reader":"40MB","limit_writer":"60MB"}}' -i http://127.0.0.1:8774/v2/ba5a6a86a8224a70ba0ffde747d8a724/servers/e6b9cae7-8732-405d-9aaf-575643b4bbef/action -X POST -H "X-Auth-Project-Id: ba5a6a86a8224a70ba0ffde747d8a724" -H "Accept: application/json" -H "X-Auth-Token: 6f0c37595f384fd4a18766825cf49013" -H "Content-Type: application/json"
"""
import time,sys,os,libvirt,traceback,re,subprocess,json
from nova import log as logging
from nova import flags
from nova.periodic.blkio  import config
from nova import utils
from nova.periodic.blkio import utils as blkio_utils
FLAGS = flags.FLAGS
LOG = logging.getLogger(__name__)
class ActionIoctl(object):
    def __init__(self,instance_name,limit_reader,limit_writer):
        self.conn = libvirt.open(None)
        self.limitReader = int(limit_reader[0:-2])
        self.limitWriter = int(limit_writer[0:-2])
        self.name = instance_name
        self.path = FLAGS.instances_path
    def diskDeviceNum(self,disk):
        """ Func info:  Returns the number of hard disk,other is 8:0 """
        if disk == '/dev/sda1' or disk == '/dev/hda1' or disk[:-1] == '/dev/sda' or disk[:-1] == '/dev/hda':
            return '8:0'
        elif disk == '/dev/sdb1' or disk == '/dev/hdb1' or disk[:-1] == '/dev/sdb' or disk[:-1] == '/dev/hdb':
            return '8:16'
        elif disk == '/dev/sdc1' or disk == '/dev/hdc1' or disk[:-1] == '/dev/sdc' or disk[:-1] == '/dev/hdc':
            return '8:32'
        elif disk == '/dev/sdd1' or disk == '/dev/hdd1' or disk[:-1] == '/dev/sdd' or disk[:-1] == '/dev/hdd':
            return '8:48'
        else:
            return '8:16'
    def vmsPIDall(self):
        """ Getting all vm port, return to """
        pid = list()
        try:
            cmds = """ps -ef |grep uuid|grep -v grep |awk '{print $2}'"""
            pid=os.popen(cmds).readlines()
            return pid
        except Exception,e:
            LOG.error(_('[-william-] Error %s'%e))
            return
    def cgroupPath(self):
        '''
        Func info : Return Cgroup path ,Because the system is not the same,
                    the Cgroup path is different also.
                    By default, have RedHat, Centos, Ubuntu  linux  ...
            osType : The current system version , And return to ...
            Cpath  : Cgroup path
        '''
        osType = open('/etc/issue').readlines()
        for i in osType:
            if re.match('Ubuntu',i):
                Cpath = '/sys/fs/cgroup/'
                return Cpath
            elif re.match('CentOS',i):
                Cpath = '/cgroup/'
                return Cpath
            elif re.match('Red Hat',i):
                Cpath = '/cgroup/'
                return Cpath
            else:
                return '/sys/fs/cgroup/'
    def exeCmd(self,cmds):
        ''' Perform system command '''
        LOG.info(_('exeCmd  %s'%cmds))
        try:
            exec_process = subprocess.Popen(cmds,
                            stdin = subprocess.PIPE,
                            stdout = subprocess.PIPE,
                            stderr = subprocess.PIPE,
                            shell = True)
            exec_process.wait()
            rest = exec_process.stdout.read()
            err = exec_process.stderr.read()
            if err == '':
                return rest
            else:
                LOG.error(_('[-william-] return values is null'))
        except Exception,e:
            LOG.error(_('[-william-] Error %s') %e)
            return
    ''' Geting vmName name  and pid '''
    def get_vm_pid(self,name):
        pid = self.conn.listDomainsID()
        try:
            pid = os.popen("""ps -ef |grep %s |grep -v grep |awk '{print $2}'""" %name).readlines()
            return int(pid[0])
        except:
            LOG.error(_('[-william-] Error %s' \
                                        %traceback.format_exc()))
            return
    def _vm_pids(self):
        try:
            return os.popen('pidof kvm').read().split()
        except:
            return
    def work(self):
        """
        Func info :  working func ................
        Cpath : is _Cpath function return values ...
        path :  The current virtual machine IMG file path
        vm_count: All of the current node number of virtual machines
        size :  Limit size
        mountInfo : mounting point
        """
        Cpath = self.cgroupPath()
        vm_pid = self.get_vm_pid(self.name)
        LOG.info(_('[- william -] %s , %s' %(self.name,vm_pid)))
        r_size = self.limitReader * 1000 * 1000
        w_size = self.limitWriter * 1000 * 1000
        device = blkio_utils.instance_path(self.path)
        diskDeviceNum = self.diskDeviceNum(device)
        vmTaskPath = "%s/blkio/%s"%(Cpath,self.name)
        utils.execute("mkdir","-p",vmTaskPath,run_as_root=True)
        utils.execute("tee" , "%s/blkio.throttle.read_bps_device"%vmTaskPath,\
                      process_input = "%s %s"%(diskDeviceNum,r_size),\
                      run_as_root=True)
        utils.execute("tee" , "%s/blkio.throttle.write_bps_device"%vmTaskPath, \
                      process_input = "%s %s"%(diskDeviceNum,w_size) ,\
                      run_as_root=True)
        LOG.info(_(""" [-william-] echo  '%s, %s, %s , %s , %s ' write oK """
                    %(self.path , device, diskDeviceNum, w_size, r_size)))
        utils.execute('tee', '%s/tasks'%vmTaskPath,\
                      process_input= str(vm_pid) ,\
                      run_as_root=True)
        LOG.info(_('[- william-] append pid %s in task' %vm_pid))
if __name__ == "__main__":
    sc = ActionIoctl('instance-0000000c','40MB','30MB')
    sc.work()