Merge branch 'master' into eucalyptus-devel
authorMarco Yuen <marcoy@gmail.com>
Mon, 25 Jul 2011 16:05:31 +0000 (12:05 -0400)
committerMarco Yuen <marcoy@gmail.com>
Mon, 25 Jul 2011 16:05:31 +0000 (12:05 -0400)
Conflicts:
sfa/managers/aggregate_manager_eucalyptus.py

1  2 
sfa/managers/aggregate_manager_eucalyptus.py

@@@ -2,8 -2,6 +2,8 @@@ from __future__ import with_statemen
  
  import sys
  import os
 +import logging
 +import datetime
  
  import boto
  from boto.ec2.regioninfo import RegionInfo
@@@ -14,17 -12,17 +14,20 @@@ from lxml import etree as E
  from sqlobject import *
  
  from sfa.util.faults import *
- from sfa.util.xrn import urn_to_hrn
+ from sfa.util.xrn import urn_to_hrn, Xrn
  from sfa.util.rspec import RSpec
  from sfa.server.registry import Registries
  from sfa.trust.credential import Credential
  from sfa.plc.api import SfaAPI
  from sfa.util.plxrn import hrn_to_pl_slicename, slicename_to_hrn
  from sfa.util.callids import Callids
+ from sfa.util.sfalogging import sfa_logger
+ from sfa.rspecs.sfa_rspec import sfa_rspec_version
+ from sfa.util.version import version_core
  
 +from threading import Thread
 +from time import sleep
 +
  ##
  # The data structure used to represent a cloud.
  # It contains the cloud name, its ip address, image information,
@@@ -37,18 -35,10 +40,18 @@@ cloud = {
  #
  EUCALYPTUS_RSPEC_SCHEMA='/etc/sfa/eucalyptus.rng'
  
 -# Quick hack
 -sys.stderr = file('/var/log/euca_agg.log', 'a+')
  api = SfaAPI()
  
 +##
 +# Meta data of an instance.
 +#
 +class Meta(SQLObject):
 +    instance   = SingleJoin('EucaInstance')
 +    state      = StringCol(default = 'new')
 +    pub_addr   = StringCol(default = None)
 +    pri_addr   = StringCol(default = None)
 +    start_time = DateTimeCol(default = None)
 +
  ##
  # A representation of an Eucalyptus instance. This is a support class
  # for instance <-> slice mapping.
@@@ -60,8 -50,7 +63,8 @@@ class EucaInstance(SQLObject)
      ramdisk_id  = StringCol()
      inst_type   = StringCol()
      key_pair    = StringCol()
 -    slice = ForeignKey('Slice')
 +    slice       = ForeignKey('Slice')
 +    meta        = ForeignKey('Meta')
  
      ##
      # Contacts Eucalyptus and tries to reserve this instance.
      # @param pubKeys A list of public keys for the instance.
      #
      def reserveInstance(self, botoConn, pubKeys):
 -        print >>sys.stderr, 'Reserving an instance: image: %s, kernel: ' \
 -                            '%s, ramdisk: %s, type: %s, key: %s' % \
 -                            (self.image_id, self.kernel_id, self.ramdisk_id, 
 -                             self.inst_type, self.key_pair)
 +        logger = logging.getLogger('EucaAggregate')
 +        logger.info('Reserving an instance: image: %s, kernel: ' \
 +                    '%s, ramdisk: %s, type: %s, key: %s' % \
 +                    (self.image_id, self.kernel_id, self.ramdisk_id,
 +                    self.inst_type, self.key_pair))
  
          # XXX The return statement is for testing. REMOVE in production
          #return
@@@ -93,7 -81,7 +96,7 @@@
          except EC2ResponseError, ec2RespErr:
              errTree = ET.fromstring(ec2RespErr.body)
              msg = errTree.find('.//Message')
 -            print >>sys.stderr, msg.text
 +            logger.error(msg.text)
              self.destroySelf()
  
  ##
@@@ -109,16 -97,10 +112,16 @@@ class Slice(SQLObject)
  # Initialize the aggregate manager by reading a configuration file.
  #
  def init_server():
 +    logger = logging.getLogger('EucaAggregate')
 +    fileHandler = logging.FileHandler('/var/log/euca.log')
 +    fileHandler.setFormatter(logging.Formatter('%(asctime)s - %(levelname)s - %(message)s'))
 +    logger.addHandler(fileHandler)
 +    logger.setLevel(logging.DEBUG)
 +
      configParser = ConfigParser()
      configParser.read(['/etc/sfa/eucalyptus_aggregate.conf', 'eucalyptus_aggregate.conf'])
      if len(configParser.sections()) < 1:
 -        print >>sys.stderr, 'No cloud defined in the config file'
 +        logger.error('No cloud defined in the config file')
          raise Exception('Cannot find cloud definition in configuration file.')
  
      # Only read the first section.
          detail = {'imageID' : i.id, 'kernelID' : i.kernel_id, 'ramdiskID' : i.ramdisk_id}
          cloud['imageBundles'][name] = detail
  
 -    # Initialize sqlite3 database.
 +    # Initialize sqlite3 database and tables.
      dbPath = '/etc/sfa/db'
      dbName = 'euca_aggregate.db'
  
      if not os.path.isdir(dbPath):
 -        print >>sys.stderr, '%s not found. Creating directory ...' % dbPath
 +        logger.info('%s not found. Creating directory ...' % dbPath)
          os.mkdir(dbPath)
  
      conn = connectionForURI('sqlite://%s/%s' % (dbPath, dbName))
      sqlhub.processConnection = conn
      Slice.createTable(ifNotExists=True)
      EucaInstance.createTable(ifNotExists=True)
 +    IP.createTable(ifNotExists=True)
 +
 +    # Start the update thread to keep track of the meta data
 +    # about Eucalyptus instance.
 +    Thread(target=updateMeta).start()
  
      # Make sure the schema exists.
      if not os.path.exists(EUCALYPTUS_RSPEC_SCHEMA):
          err = 'Cannot location schema at %s' % EUCALYPTUS_RSPEC_SCHEMA
 -        print >>sys.stderr, err
 +        logger.error(err)
          raise Exception(err)
  
  ##
@@@ -182,11 -159,10 +185,11 @@@ def getEucaConnection()
      useSSL    = False
      srvPath   = '/'
      eucaPort  = 8773
 +    logger    = logging.getLogger('EucaAggregate')
  
      if not accessKey or not secretKey or not eucaURL:
 -        print >>sys.stderr, 'Please set ALL of the required environment ' \
 -                            'variables by sourcing the eucarc file.'
 +        logger.error('Please set ALL of the required environment ' \
 +                     'variables by sourcing the eucarc file.')
          return None
      
      # Split the url into parts
  # @param sliceHRN The hunman readable name of the slice.
  # @return sting()
  #
- def getKeysForSlice(sliceHRN):
-     logger = logging.getLogger('EucaAggregate')
-     try:
-         # convert hrn to slice name
-         plSliceName = hrn_to_pl_slicename(sliceHRN)
-     except IndexError, e:
-         logger.error('Invalid slice name (%s)' % sliceHRN)
+ def getKeysForSlice(api, sliceHRN):
++    logger   = logging.getLogger('EucaAggregate')
+     cred     = api.getCredential()
+     registry = api.registries[api.hrn]
+     keys     = []
+     # Get the slice record
+     records = registry.Resolve(sliceHRN, cred)
+     if not records:
 -        print >>sys.stderr, 'Cannot find any record for slice %s' % sliceHRN
++        logging.warn('Cannot find any record for slice %s' % sliceHRN)
          return []
  
-     # Get the slice's information
-     sliceData = api.plshell.GetSlices(api.plauth, {'name':plSliceName})
-     if not sliceData:
-         logger.warn('Cannot get any data for slice %s' % plSliceName)
-         return []
+     # Find who can log into this slice
+     persons = records[0]['persons']
  
-     # It should only return a list with len = 1
-     sliceData = sliceData[0]
+     # Extract the keys from persons records
+     for p in persons:
+         sliceUser = registry.Resolve(p, cred)
+         userKeys = sliceUser[0]['keys']
+         keys += userKeys
  
-     keys = []
-     person_ids = sliceData['person_ids']
-     if not person_ids: 
-         logger.warn('No users in slice %s' % sliceHRN)
-         return []
-     persons = api.plshell.GetPersons(api.plauth, person_ids)
-     for person in persons:
-         pkeys = api.plshell.GetKeys(api.plauth, person['key_ids'])
-         for key in pkeys:
-             keys.append(key['key'])
-  
      return ''.join(keys)
  
  ##
@@@ -366,9 -332,8 +360,9 @@@ class EucaRSpecBuilder(object)
      # Generates the RSpec.
      #
      def toXML(self):
 +        logger = logging.getLogger('EucaAggregate')
          if not self.cloudInfo:
 -            print >>sys.stderr, 'No cloud information'
 +            logger.error('No cloud information')
              return ''
  
          xml = self.eucaRSpec
@@@ -429,17 -394,17 +423,18 @@@ def ListResources(api, creds, options, 
      # get slice's hrn from options
      xrn = options.get('geni_slice_urn', '')
      hrn, type = urn_to_hrn(xrn)
 +    logger = logging.getLogger('EucaAggregate')
  
      # get hrn of the original caller
      origin_hrn = options.get('origin_hrn', None)
      if not origin_hrn:
-         origin_hrn = Credential(string=creds[0]).get_gid_caller().get_hrn()
+         origin_hrn = Credential(string=creds).get_gid_caller().get_hrn()
+         # origin_hrn = Credential(string=creds[0]).get_gid_caller().get_hrn()
  
      conn = getEucaConnection()
  
      if not conn:
 -        print >>sys.stderr, 'Error: Cannot create a connection to Eucalyptus'
 +        logger.error('Cannot create a connection to Eucalyptus')
          return 'Cannot create a connection to Eucalyptus'
  
      try:
      except EC2ResponseError, ec2RespErr:
          errTree = ET.fromstring(ec2RespErr.body)
          errMsgE = errTree.find('.//Message')
 -        print >>sys.stderr, errMsgE.text
 +        logger.error(errMsgE.text)
  
      rspec = EucaRSpecBuilder(cloud).toXML()
  
@@@ -525,17 -490,21 +520,22 @@@ def CreateSliver(api, xrn, creds, xml, 
  
      global cloud
      hrn = urn_to_hrn(xrn)[0]
 +    logger = logging.getLogger('EucaAggregate')
  
      conn = getEucaConnection()
      if not conn:
 -        print >>sys.stderr, 'Error: Cannot create a connection to Eucalyptus'
 +        logger.error('Cannot create a connection to Eucalyptus')
          return ""
  
      # Validate RSpec
      schemaXML = ET.parse(EUCALYPTUS_RSPEC_SCHEMA)
      rspecValidator = ET.RelaxNG(schemaXML)
      rspecXML = ET.XML(xml)
+     for network in rspecXML.iterfind("./network"):
+         if network.get('id') != cloud['name']:
+             # Throw away everything except my own RSpec
+             # sfa_logger().error("CreateSliver: deleting %s from rspec"%network.get('id'))
+             network.getparent().remove(network)
      if not rspecValidator(rspecXML):
          error = rspecValidator.error_log.last_error
          message = '%s (line %s)' % (error.message, error.line) 
      pendingRmInst = []
      for sliceInst in s.instances:
          pendingRmInst.append(sliceInst.instance_id)
-     existingInstGroup = rspecXML.findall('.//euca_instances')
+     existingInstGroup = rspecXML.findall(".//euca_instances")
      for instGroup in existingInstGroup:
          for existingInst in instGroup:
              if existingInst.get('id') in pendingRmInst:
                  pendingRmInst.remove(existingInst.get('id'))
      for inst in pendingRmInst:
 -        print >>sys.stderr, 'Instance %s will be terminated' % inst
 +        logger.debug('Instance %s will be terminated' % inst)
          dbInst = EucaInstance.select(EucaInstance.q.instance_id == inst).getOne(None)
 -        dbInst.destroySelf()
 +        # Only change the state but do not remove the entry from the DB.
 +        dbInst.meta.state = 'deleted'
 +        #dbInst.destroySelf()
      conn.terminate_instances(pendingRmInst)
  
      # Process new instance requests
-     requests = rspecXML.findall('.//request')
+     requests = rspecXML.findall(".//request")
      if requests:
          # Get all the public keys associate with slice.
-         pubKeys = getKeysForSlice(s.slice_hrn)
+         pubKeys = getKeysForSlice(api, s.slice_hrn)
 -        print >>sys.stderr, "Passing the following keys to the instance:\n%s" % pubKeys
 -        sys.stderr.flush()
 +        logger.debug('Passing the following keys to the instance:\n%s' % pubKeys)
      for req in requests:
          vmTypeElement = req.getparent()
          instType = vmTypeElement.get('name')
          
          bundleName = req.find('bundle').text
          if not cloud['imageBundles'][bundleName]:
 -            print >>sys.stderr, 'Cannot find bundle %s' % bundleName
 +            logger.error('Cannot find bundle %s' % bundleName)
          bundleInfo = cloud['imageBundles'][bundleName]
          instKernel  = bundleInfo['kernelID']
          instDiskImg = bundleInfo['imageID']
  
          # Create the instances
          for i in range(0, numInst):
 -            eucaInst = EucaInstance(slice = s, 
 -                                    kernel_id = instKernel,
 -                                    image_id = instDiskImg,
 +            eucaInst = EucaInstance(slice      = s,
 +                                    kernel_id  = instKernel,
 +                                    image_id   = instDiskImg,
                                      ramdisk_id = instRamDisk,
 -                                    key_pair = instKey,
 -                                    inst_type = instType)
 +                                    key_pair   = instKey,
 +                                    inst_type  = instType,
 +                                    meta       = Meta(start_time=datetime.datetime.now()))
              eucaInst.reserveInstance(conn, pubKeys)
  
      # xxx - should return altered rspec 
      # with enough data for the client to understand what's happened
      return xml
  
 +##
 +# A thread that will update the meta data.
 +#
 +def updateMeta():
 +    logger = logging.getLogger('EucaAggregate')
 +    while True:
 +        sleep(120)
 +
 +        # Get IDs of the instances that don't have IPs yet.
 +        dbResults = Meta.select(
 +                      AND(Meta.q.pri_addr == None,
 +                          Meta.q.state    != 'deleted')
 +                    )
 +        dbResults = list(dbResults)
 +        logger.debug('[update thread] dbResults: %s' % dbResults)
 +        instids = []
 +        for r in dbResults:
 +            instids.append(r.instance.instance_id)
 +        logger.debug('[update thread] Instance Id: %s' % ', '.join(instids))
 +
 +        # Get instance information from Eucalyptus
 +        conn = getEucaConnection()
 +        vmInstances = []
 +        reservations = conn.get_all_instances(instids)
 +        for reservation in reservations:
 +            vmInstances += reservation.instances
 +
 +        # Check the IPs
 +        instIPs = [ {'id':i.id, 'pri_addr':i.private_dns_name, 'pub_addr':i.public_dns_name}
 +                    for i in vmInstances if i.private_dns_name != '0.0.0.0' ]
 +        logger.debug('[update thread] IP dict: %s' % str(instIPs))
 +
 +        # Update the local DB
 +        for ipData in instIPs:
 +            dbInst = EucaInstance.select(EucaInstance.q.instance_id == ipData['id']).getOne(None)
 +            if not dbInst:
 +                logger.info('[update thread] Could not find %s in DB' % ipData['id'])
 +                continue
 +            dbInst.meta.pri_addr = ipData['pri_addr']
 +            dbInst.meta.pub_addr = ipData['pub_addr']
 +            dbInst.meta.state    = 'running'
 +
+ def GetVersion(api):
+     xrn=Xrn(api.hrn)
+     request_rspec_versions = [dict(sfa_rspec_version)]
+     ad_rspec_versions = [dict(sfa_rspec_version)]
+     version_more = {'interface':'aggregate',
+                     'testbed':'myplc',
+                     'hrn':xrn.get_hrn(),
+                     'request_rspec_versions': request_rspec_versions,
+                     'ad_rspec_versions': ad_rspec_versions,
+                     'default_ad_rspec': dict(sfa_rspec_version)
+                     }
+     return version_core(version_more)
  def main():
      init_server()
  
  
      #rspec = ListResources('euca', 'planetcloud.pc.test', 'planetcloud.pc.marcoy', 'test_euca')
      #print rspec
-     print getKeysForSlice('gc.gc.test1')
+     server_key_file = '/var/lib/sfa/authorities/server.key'
+     server_cert_file = '/var/lib/sfa/authorities/server.cert'
+     api = SfaAPI(key_file = server_key_file, cert_file = server_cert_file, interface='aggregate')
+     print getKeysForSlice(api, 'gc.gc.test1')
  
  if __name__ == "__main__":
      main()