Merge "Add Progress Update feature to new compass-core" into dev/experimental
This commit is contained in:
commit
814f35d22e
113
compass/actions/update_progress.py
Normal file
113
compass/actions/update_progress.py
Normal file
@ -0,0 +1,113 @@
|
|||||||
|
# Copyright 2014 Huawei Technologies Co. Ltd
|
||||||
|
#
|
||||||
|
# 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.
|
||||||
|
|
||||||
|
"""Module to update status and installing progress of the given cluster.
|
||||||
|
|
||||||
|
.. moduleauthor:: Xiaodong Wang <xiaodongwang@huawei.com>
|
||||||
|
"""
|
||||||
|
import logging
|
||||||
|
|
||||||
|
from compass.actions import util
|
||||||
|
from compass.db.api import database
|
||||||
|
from compass.db import models
|
||||||
|
from compass.log_analyzor import progress_calculator
|
||||||
|
from compass.utils import setting_wrapper as setting
|
||||||
|
|
||||||
|
|
||||||
|
def _cluster_filter(cluster):
|
||||||
|
"""filter cluster."""
|
||||||
|
if not cluster.state:
|
||||||
|
logging.error('there is no state for cluster %s',
|
||||||
|
cluster.id)
|
||||||
|
return False
|
||||||
|
|
||||||
|
if cluster.state.state != 'INSTALLING':
|
||||||
|
logging.error('the cluster %s state %s is not installing',
|
||||||
|
cluster.id, cluster.state.state)
|
||||||
|
return False
|
||||||
|
|
||||||
|
return True
|
||||||
|
|
||||||
|
|
||||||
|
def _host_filter(host):
|
||||||
|
"""filter host."""
|
||||||
|
if not host.state:
|
||||||
|
logging.error('there is no state for host %s',
|
||||||
|
host.id)
|
||||||
|
return False
|
||||||
|
|
||||||
|
if host.state.state != 'INSTALLING':
|
||||||
|
logging.error('the host %s state %s is not installing',
|
||||||
|
host.id, host.state.state)
|
||||||
|
return False
|
||||||
|
|
||||||
|
return True
|
||||||
|
|
||||||
|
|
||||||
|
def update_progress(cluster_hosts):
|
||||||
|
"""Update status and installing progress of the given cluster.
|
||||||
|
|
||||||
|
:param cluster_hosts: clusters and hosts in each cluster to update.
|
||||||
|
:type cluster_hosts: dict of int or str to list of int or str
|
||||||
|
|
||||||
|
.. note::
|
||||||
|
The function should be called out of the database session scope.
|
||||||
|
In the function, it will update the database cluster_state and
|
||||||
|
host_state table for the deploying cluster and hosts.
|
||||||
|
|
||||||
|
The function will also query log_progressing_history table to get
|
||||||
|
the lastest installing progress and the position of log it has
|
||||||
|
processed in the last run. The function uses these information to
|
||||||
|
avoid recalculate the progress from the beginning of the log file.
|
||||||
|
After the progress got updated, these information will be stored back
|
||||||
|
to the log_progressing_history for next time run.
|
||||||
|
"""
|
||||||
|
with util.lock('log_progressing', blocking=False) as lock:
|
||||||
|
if not lock:
|
||||||
|
logging.error(
|
||||||
|
'failed to acquire lock to calculate installation progress')
|
||||||
|
return
|
||||||
|
|
||||||
|
logging.info('update installing progress of cluster_hosts: %s',
|
||||||
|
cluster_hosts)
|
||||||
|
os_names = {}
|
||||||
|
distributed_systems = {}
|
||||||
|
with database.session() as session:
|
||||||
|
cluster = session.query(models.Cluster).first()
|
||||||
|
clusterid = cluster.id
|
||||||
|
|
||||||
|
adapter = cluster.adapter
|
||||||
|
os_installer = adapter.os_installer
|
||||||
|
os_installer_name = os_installer.instance_name
|
||||||
|
package_installer = adapter.package_installer
|
||||||
|
package_installer_name = package_installer.instance_name
|
||||||
|
|
||||||
|
distributed_system = cluster.distributed_system
|
||||||
|
distributed_system_name = distributed_system.name
|
||||||
|
|
||||||
|
host = session.query(models.Host).first()
|
||||||
|
os_name = host.os_name
|
||||||
|
os_names[clusterid] = os_name
|
||||||
|
distributed_systems[clusterid] = distributed_system_name
|
||||||
|
|
||||||
|
clusterhosts = cluster.clusterhosts
|
||||||
|
hostids = [clusterhost.host.id for clusterhost in clusterhosts]
|
||||||
|
cluster_hosts.update({clusterid: hostids})
|
||||||
|
|
||||||
|
progress_calculator.update_progress(
|
||||||
|
os_installer_name,
|
||||||
|
os_names,
|
||||||
|
package_installer_name,
|
||||||
|
distributed_systems,
|
||||||
|
cluster_hosts)
|
@ -48,3 +48,66 @@ def lock(lock_name, blocking=True, timeout=10):
|
|||||||
instance_lock.acquired_until = 0
|
instance_lock.acquired_until = 0
|
||||||
instance_lock.release()
|
instance_lock.release()
|
||||||
logging.debug('released lock %s', lock_name)
|
logging.debug('released lock %s', lock_name)
|
||||||
|
|
||||||
|
"""
|
||||||
|
def update_cluster_hosts(cluster_hosts,
|
||||||
|
cluster_filter=None, host_filter=None):
|
||||||
|
session = database.current_session()
|
||||||
|
os_versions = {}
|
||||||
|
target_systems = {}
|
||||||
|
updated_cluster_hosts = {}
|
||||||
|
clusters = session.query(models.Cluster).all()
|
||||||
|
for cluster in clusters:
|
||||||
|
if cluster_hosts and (
|
||||||
|
cluster.id not in cluster_hosts and
|
||||||
|
str(cluster.id) not in cluster_hosts and
|
||||||
|
cluster.name not in cluster_hosts
|
||||||
|
):
|
||||||
|
logging.debug('ignore cluster %s sinc it is not in %s',
|
||||||
|
cluster.id, cluster_hosts)
|
||||||
|
continue
|
||||||
|
|
||||||
|
adapter = cluster.adapter
|
||||||
|
if not cluster.adapter:
|
||||||
|
logging.error('there is no adapter for cluster %s',
|
||||||
|
cluster.id)
|
||||||
|
continue
|
||||||
|
|
||||||
|
if cluster_filter and not cluster_filter(cluster):
|
||||||
|
logging.debug('filter cluster %s', cluster.id)
|
||||||
|
continue
|
||||||
|
|
||||||
|
updated_cluster_hosts[cluster.id] = []
|
||||||
|
os_versions[cluster.id] = 'CentOS-6.5-x86_64'
|
||||||
|
target_systems[cluster.id] = 'openstack'
|
||||||
|
|
||||||
|
if cluster.id in cluster_hosts:
|
||||||
|
hosts = cluster_hosts[cluster.id]
|
||||||
|
elif str(cluster.id) in cluster_hosts:
|
||||||
|
hosts = cluster_hosts[str(cluster.id)]
|
||||||
|
elif cluster.name in cluster_hosts:
|
||||||
|
hosts = cluster_hosts[cluster.name]
|
||||||
|
else:
|
||||||
|
hosts = []
|
||||||
|
|
||||||
|
if not hosts:
|
||||||
|
hosts = [host.id for host in cluster.hosts]
|
||||||
|
|
||||||
|
for host in cluster.hosts:
|
||||||
|
if (
|
||||||
|
host.id not in hosts and
|
||||||
|
str(host.id) not in hosts and
|
||||||
|
host.hostname not in hosts
|
||||||
|
):
|
||||||
|
logging.debug('ignore host %s which is not in %s',
|
||||||
|
host.id, hosts)
|
||||||
|
continue
|
||||||
|
|
||||||
|
if host_filter and not host_filter(host):
|
||||||
|
logging.debug('filter host %s', host.id)
|
||||||
|
continue
|
||||||
|
|
||||||
|
updated_cluster_hosts[cluster.id].append(host.id)
|
||||||
|
|
||||||
|
return (updated_cluster_hosts, os_versions, target_systems)
|
||||||
|
"""
|
||||||
|
@ -155,6 +155,6 @@ def get_adapters_internal(session):
|
|||||||
else:
|
else:
|
||||||
logging.info(
|
logging.info(
|
||||||
'ignore adapter %s since it is not deployable',
|
'ignore adapter %s since it is not deployable',
|
||||||
adapter_dict
|
adapter.to_dict()
|
||||||
)
|
)
|
||||||
return adapter_mapping
|
return adapter_mapping
|
||||||
|
@ -1204,7 +1204,7 @@ def update_cluster_host_state(
|
|||||||
@utils.supported_filters(
|
@utils.supported_filters(
|
||||||
optional_support_keys=UPDATED_CLUSTERHOST_STATE_FIELDS
|
optional_support_keys=UPDATED_CLUSTERHOST_STATE_FIELDS
|
||||||
)
|
)
|
||||||
@database.run_in_session()
|
## @database.run_in_session()
|
||||||
@user_api.check_user_permission_in_session(
|
@user_api.check_user_permission_in_session(
|
||||||
permission.PERMISSION_UPDATE_CLUSTERHOST_STATE
|
permission.PERMISSION_UPDATE_CLUSTERHOST_STATE
|
||||||
)
|
)
|
||||||
@ -1223,7 +1223,7 @@ def update_clusterhost_state(
|
|||||||
@utils.supported_filters(
|
@utils.supported_filters(
|
||||||
optional_support_keys=UPDATED_CLUSTER_STATE_FIELDS
|
optional_support_keys=UPDATED_CLUSTER_STATE_FIELDS
|
||||||
)
|
)
|
||||||
@database.run_in_session()
|
## @database.run_in_session()
|
||||||
@user_api.check_user_permission_in_session(
|
@user_api.check_user_permission_in_session(
|
||||||
permission.PERMISSION_UPDATE_CLUSTER_STATE
|
permission.PERMISSION_UPDATE_CLUSTER_STATE
|
||||||
)
|
)
|
||||||
|
@ -589,7 +589,7 @@ def get_host_state(session, getter, host_id, **kwargs):
|
|||||||
|
|
||||||
|
|
||||||
@utils.supported_filters(UPDATED_STATE_FIELDS)
|
@utils.supported_filters(UPDATED_STATE_FIELDS)
|
||||||
@database.run_in_session()
|
## @database.run_in_session()
|
||||||
@user_api.check_user_permission_in_session(
|
@user_api.check_user_permission_in_session(
|
||||||
permission.PERMISSION_UPDATE_HOST_STATE
|
permission.PERMISSION_UPDATE_HOST_STATE
|
||||||
)
|
)
|
||||||
|
@ -21,6 +21,7 @@ import simplejson as json
|
|||||||
from sqlalchemy import BigInteger
|
from sqlalchemy import BigInteger
|
||||||
from sqlalchemy import Boolean
|
from sqlalchemy import Boolean
|
||||||
from sqlalchemy import Column
|
from sqlalchemy import Column
|
||||||
|
from sqlalchemy import ColumnDefault
|
||||||
from sqlalchemy import DateTime
|
from sqlalchemy import DateTime
|
||||||
from sqlalchemy import Enum
|
from sqlalchemy import Enum
|
||||||
from sqlalchemy.ext.declarative import declarative_base
|
from sqlalchemy.ext.declarative import declarative_base
|
||||||
@ -1885,3 +1886,46 @@ class Network(BASE, TimestampMixin, HelperMixin):
|
|||||||
raise exception.InvalidParameter(
|
raise exception.InvalidParameter(
|
||||||
'subnet %s format is uncorrect' % self.subnet
|
'subnet %s format is uncorrect' % self.subnet
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class LogProgressingHistory(BASE):
|
||||||
|
"""host installing log history for each file.
|
||||||
|
|
||||||
|
:param id: int, identity as primary key.
|
||||||
|
:param pathname: str, the full path of the installing log file. unique.
|
||||||
|
:param position: int, the position of the log file it has processed.
|
||||||
|
:param partial_line: str, partial line of the log.
|
||||||
|
:param progressing: float, indicate the installing progress between 0 to 1.
|
||||||
|
:param message: str, str, the installing message.
|
||||||
|
:param severity: Enum, the installing message severity.
|
||||||
|
('ERROR', 'WARNING', 'INFO')
|
||||||
|
:param line_matcher_name: str, the line matcher name of the log processor.
|
||||||
|
:param update_timestamp: datetime, the latest timestamp the entry updated.
|
||||||
|
"""
|
||||||
|
__tablename__ = 'log_progressing_history'
|
||||||
|
id = Column(Integer, primary_key=True)
|
||||||
|
pathname = Column(String(80), unique=True)
|
||||||
|
position = Column(Integer, ColumnDefault(0))
|
||||||
|
partial_line = Column(Text)
|
||||||
|
percentage = Column(Float, ColumnDefault(0.0))
|
||||||
|
message = Column(Text)
|
||||||
|
severity = Column(Enum('ERROR', 'WARNING', 'INFO'), ColumnDefault('INFO'))
|
||||||
|
line_matcher_name = Column(String(80), ColumnDefault('start'))
|
||||||
|
update_timestamp = Column(DateTime, default=datetime.datetime.now(),
|
||||||
|
onupdate=datetime.datetime.now())
|
||||||
|
|
||||||
|
def __init__(self, **kwargs):
|
||||||
|
super(LogProgressingHistory, self).__init__(**kwargs)
|
||||||
|
|
||||||
|
def __repr__(self):
|
||||||
|
return (
|
||||||
|
'LogProgressingHistory[%r: position %r,'
|
||||||
|
'partial_line %r,percentage %r,message %r,'
|
||||||
|
'severity %r]'
|
||||||
|
) % (
|
||||||
|
self.pathname, self.position,
|
||||||
|
self.partial_line,
|
||||||
|
self.percentage,
|
||||||
|
self.message,
|
||||||
|
self.severity
|
||||||
|
)
|
||||||
|
@ -19,11 +19,19 @@
|
|||||||
import logging
|
import logging
|
||||||
import re
|
import re
|
||||||
|
|
||||||
from compass.db import database
|
from compass.db.api import cluster as cluster_api
|
||||||
from compass.db.model import Cluster
|
from compass.db.api import database
|
||||||
from compass.db.model import ClusterHost
|
from compass.db.api import host as host_api
|
||||||
|
from compass.db.api import user as user_api
|
||||||
|
|
||||||
|
from compass.db.models import Cluster
|
||||||
|
from compass.db.models import ClusterHost
|
||||||
|
from compass.db.models import Host
|
||||||
|
|
||||||
from compass.log_analyzor.line_matcher import Progress
|
from compass.log_analyzor.line_matcher import Progress
|
||||||
|
|
||||||
|
import datetime
|
||||||
|
|
||||||
|
|
||||||
class AdapterItemMatcher(object):
|
class AdapterItemMatcher(object):
|
||||||
"""Progress matcher for the os installing or package installing."""
|
"""Progress matcher for the os installing or package installing."""
|
||||||
@ -162,155 +170,101 @@ class AdapterMatcher(object):
|
|||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def _get_host_progress(cls, hostid):
|
def _get_host_progress(cls, hostid):
|
||||||
"""Get Host Progress from database.
|
"""Get Host Progress from HostState."""
|
||||||
|
|
||||||
.. notes::
|
|
||||||
The function should be called in database session.
|
|
||||||
"""
|
|
||||||
session = database.current_session()
|
session = database.current_session()
|
||||||
host = session.query(
|
host = session.query(
|
||||||
ClusterHost
|
Host
|
||||||
).filter_by(id=hostid).first()
|
).filter_by(id=hostid).first()
|
||||||
if not host:
|
if not host:
|
||||||
logging.error(
|
logging.error(
|
||||||
'there is no host for %s in ClusterHost', hostid)
|
'there is no host for %s in Host', hostid)
|
||||||
return None, None, None
|
return None, None, None
|
||||||
|
|
||||||
if not host.state:
|
if not host.state:
|
||||||
logging.error('there is no related HostState for %s',
|
logging.error('there is no related HostState for %s',
|
||||||
hostid)
|
hostid)
|
||||||
return host.fullname, None, None
|
return host.name, None, None
|
||||||
|
|
||||||
"""
|
|
||||||
return (
|
return (
|
||||||
host.fullname,
|
host.name,
|
||||||
host.state.state,
|
host.state.state,
|
||||||
Progress(host.state.progress,
|
Progress(host.state.percentage,
|
||||||
host.state.message,
|
host.state.message,
|
||||||
host.state.severity))
|
host.state.severity))
|
||||||
"""
|
|
||||||
return {
|
|
||||||
'os': (
|
|
||||||
host.fullname,
|
|
||||||
host.state.state,
|
|
||||||
Progress(host.state.os_progress,
|
|
||||||
host.state.os_message,
|
|
||||||
host.state.os_severity)),
|
|
||||||
'package': (
|
|
||||||
host.fullname,
|
|
||||||
host.state.state,
|
|
||||||
Progress(host.state.progress,
|
|
||||||
host.state.message,
|
|
||||||
host.state.severity))}
|
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def _update_host_os_progress(cls, hostid, os_progress):
|
def _get_clusterhost_progress(cls, hostid):
|
||||||
"""Update host progress to database.
|
"""Get ClusterHost progress from ClusterHostState."""
|
||||||
|
|
||||||
|
session = database.current_session()
|
||||||
|
clusterhost = session.query(
|
||||||
|
ClusterHost
|
||||||
|
).filter_by(id=hostid).first()
|
||||||
|
if not clusterhost:
|
||||||
|
logging.error(
|
||||||
|
'there is no clusterhost for %s in ClusterHost',
|
||||||
|
hostid
|
||||||
|
)
|
||||||
|
return None, None, None
|
||||||
|
|
||||||
|
if not clusterhost.state:
|
||||||
|
logging.error(
|
||||||
|
'there is no related ClusterHostState for %s',
|
||||||
|
hostid
|
||||||
|
)
|
||||||
|
return clusterhost.name, None, None
|
||||||
|
|
||||||
|
return (
|
||||||
|
clusterhost.name,
|
||||||
|
clusterhost.state.state,
|
||||||
|
Progress(clusterhost.state.percentage,
|
||||||
|
clusterhost.state.message,
|
||||||
|
clusterhost.state.severity))
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def _update_host_progress(cls, hostid, host_progress, updater):
|
||||||
|
"""Updates host progress to db."""
|
||||||
|
|
||||||
.. note::
|
|
||||||
The function should be called in database session.
|
|
||||||
"""
|
|
||||||
session = database.current_session()
|
session = database.current_session()
|
||||||
host = session.query(
|
host = session.query(
|
||||||
ClusterHost).filter_by(id=hostid).first()
|
Host).filter_by(id=hostid).first()
|
||||||
if not host:
|
if not host:
|
||||||
logging.error(
|
logging.error(
|
||||||
'there is no host for %s in ClusterHost', hostid)
|
'there is no host for %s in table Host',
|
||||||
return
|
hostid
|
||||||
|
)
|
||||||
|
|
||||||
if not host.state:
|
if not host.state:
|
||||||
logging.error(
|
logging.error(
|
||||||
'there is no related HostState for %s', hostid)
|
'there is no related HostState for %s',
|
||||||
return
|
hostid
|
||||||
|
)
|
||||||
|
|
||||||
logging.debug('os progress: %s', os_progress.progress)
|
if host.state.percentage > host_progress.progress:
|
||||||
|
|
||||||
if host.state.os_progress > os_progress.progress:
|
|
||||||
logging.error(
|
logging.error(
|
||||||
'host %s os_progress is not increased '
|
'host %s progress has not been increased'
|
||||||
'from %s to %s',
|
' from %s to $s',
|
||||||
hostid, host.state, os_progress)
|
hostid, host.state, host_progress
|
||||||
|
)
|
||||||
return
|
return
|
||||||
if (
|
|
||||||
host.state.os_progress == os_progress.progress and
|
if (host.state.percentage == host_progress.progress and
|
||||||
host.state.os_message == os_progress.message
|
host.state.message == host_progress.message):
|
||||||
):
|
|
||||||
logging.info(
|
logging.info(
|
||||||
'ignore update host %s progress %s to %s',
|
'host %s update ignored due to same progress'
|
||||||
hostid, os_progress, host.state)
|
'in database',
|
||||||
|
hostid
|
||||||
|
)
|
||||||
return
|
return
|
||||||
|
|
||||||
host.state.os_progress = os_progress.progress
|
host.state.percentage = host_progress.progress
|
||||||
"""host.state.os_progress = progress.progress"""
|
host.state.message = host_progress.message
|
||||||
host.state.os_message = os_progress.message
|
if host_progress.severity:
|
||||||
if os_progress.severity:
|
host.state.severity = host_progress.severity
|
||||||
host.state.os_severity = os_progress.severity
|
|
||||||
|
|
||||||
if host.state.os_progress >= 1.0:
|
if host.state.percentage >= 1.0:
|
||||||
host.state.os_state = 'OS_READY'
|
host.state.state = 'SUCCESSFUL'
|
||||||
|
|
||||||
if host.state.os_severity == 'ERROR':
|
|
||||||
host.state.os_state = 'ERROR'
|
|
||||||
|
|
||||||
if host.state.os_state != 'INSTALLING':
|
|
||||||
host.mutable = True
|
|
||||||
|
|
||||||
logging.debug(
|
|
||||||
'update host %s state %s',
|
|
||||||
hostid, host.state)
|
|
||||||
|
|
||||||
@classmethod
|
|
||||||
def _update_host_package_progress(cls, hostid, progress):
|
|
||||||
"""Update host progress to database.
|
|
||||||
|
|
||||||
.. note::
|
|
||||||
The function should be called in database session.
|
|
||||||
"""
|
|
||||||
session = database.current_session()
|
|
||||||
host = session.query(
|
|
||||||
ClusterHost).filter_by(id=hostid).first()
|
|
||||||
|
|
||||||
logging.debug('package progress: %s', progress.progress)
|
|
||||||
logging.debug('package ssssstate: %s', host.state.state)
|
|
||||||
if not host:
|
|
||||||
logging.error(
|
|
||||||
'there is no host for %s in ClusterHost', hostid)
|
|
||||||
return
|
|
||||||
|
|
||||||
if not host.state:
|
|
||||||
logging.error(
|
|
||||||
'there is no related HostState for %s', hostid)
|
|
||||||
return
|
|
||||||
|
|
||||||
if not host.state.state in ['OS_READY', 'INSTALLING']:
|
|
||||||
logging.error(
|
|
||||||
'host %s issssss not in INSTALLING state',
|
|
||||||
hostid)
|
|
||||||
return
|
|
||||||
|
|
||||||
if host.state.progress > progress.progress:
|
|
||||||
logging.error(
|
|
||||||
'host %s progress is not increased '
|
|
||||||
'from %s to %s',
|
|
||||||
hostid, host.state, progress)
|
|
||||||
return
|
|
||||||
|
|
||||||
if (
|
|
||||||
host.state.progress == progress.progress and
|
|
||||||
host.state.message == progress.message
|
|
||||||
):
|
|
||||||
logging.info(
|
|
||||||
'ignore update host %s progress %s to %s',
|
|
||||||
hostid, progress, host.state)
|
|
||||||
return
|
|
||||||
|
|
||||||
host.state.progress = progress.progress
|
|
||||||
host.state.message = progress.message
|
|
||||||
if progress.severity:
|
|
||||||
host.state.severity = progress.severity
|
|
||||||
|
|
||||||
if host.state.progress >= 1.0:
|
|
||||||
host.state.state = 'READY'
|
|
||||||
|
|
||||||
if host.state.severity == 'ERROR':
|
if host.state.severity == 'ERROR':
|
||||||
host.state.state = 'ERROR'
|
host.state.state = 'ERROR'
|
||||||
@ -318,10 +272,81 @@ class AdapterMatcher(object):
|
|||||||
if host.state.state != 'INSTALLING':
|
if host.state.state != 'INSTALLING':
|
||||||
host.mutable = True
|
host.mutable = True
|
||||||
|
|
||||||
|
host_api.update_host_state(
|
||||||
|
session,
|
||||||
|
updater,
|
||||||
|
hostid,
|
||||||
|
state=host.state.state,
|
||||||
|
percentage=host.state.percentage,
|
||||||
|
message=host.state.message,
|
||||||
|
id=hostid
|
||||||
|
)
|
||||||
|
|
||||||
logging.debug(
|
logging.debug(
|
||||||
'update host %s state %s',
|
'update host %s state %s',
|
||||||
hostid, host.state)
|
hostid, host.state)
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def _update_clusterhost_progress(
|
||||||
|
cls,
|
||||||
|
hostid,
|
||||||
|
clusterhost_progress,
|
||||||
|
updater
|
||||||
|
):
|
||||||
|
|
||||||
|
session = database.current_session()
|
||||||
|
clusterhost = session.query(
|
||||||
|
ClusterHost).filter_by(id=hostid).first()
|
||||||
|
|
||||||
|
if not clusterhost.state:
|
||||||
|
logging.error(
|
||||||
|
'ClusterHost state not found for %s',
|
||||||
|
hostid)
|
||||||
|
|
||||||
|
if clusterhost.state.percentage > clusterhost_progress.progress:
|
||||||
|
logging.error(
|
||||||
|
'clusterhost %s state has not been increased'
|
||||||
|
' from %s to %s',
|
||||||
|
hostid, clusterhost.state, clusterhost_progress
|
||||||
|
)
|
||||||
|
return
|
||||||
|
|
||||||
|
if (clusterhost.state.percentage == clusterhost_progress.progress and
|
||||||
|
clusterhost.state.message == clusterhost_progress.message):
|
||||||
|
logging.info(
|
||||||
|
'clusterhost %s update ignored due to same progress'
|
||||||
|
'in database',
|
||||||
|
hostid
|
||||||
|
)
|
||||||
|
return
|
||||||
|
|
||||||
|
clusterhost.state.percentage = clusterhost_progress.progress
|
||||||
|
clusterhost.state.message = clusterhost_progress.message
|
||||||
|
if clusterhost_progress.severity:
|
||||||
|
clusterhost.state.severity = clusterhost_progress.severity
|
||||||
|
|
||||||
|
if clusterhost.state.percentage >= 1.0:
|
||||||
|
clusterhost.state.state = 'SUCCESSFUL'
|
||||||
|
|
||||||
|
if clusterhost.state.severity == 'ERROR':
|
||||||
|
clusterhost.state.state = 'ERROR'
|
||||||
|
|
||||||
|
if clusterhost.state.state != 'INSTALLING':
|
||||||
|
clusterhost.mutable = True
|
||||||
|
|
||||||
|
cluster_api.update_clusterhost_state(
|
||||||
|
session,
|
||||||
|
updater,
|
||||||
|
hostid,
|
||||||
|
state=clusterhost.state.state,
|
||||||
|
percentage=clusterhost.state.percentage,
|
||||||
|
message=clusterhost.state.message
|
||||||
|
)
|
||||||
|
|
||||||
|
logging.debug(
|
||||||
|
'update clusterhost %s state %s',
|
||||||
|
hostid, clusterhost.state)
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def _update_cluster_progress(cls, clusterid):
|
def _update_cluster_progress(cls, clusterid):
|
||||||
"""Update cluster installing progress to database.
|
"""Update cluster installing progress to database.
|
||||||
@ -351,18 +376,41 @@ class AdapterMatcher(object):
|
|||||||
cluster_progress = 0.0
|
cluster_progress = 0.0
|
||||||
cluster_messages = {}
|
cluster_messages = {}
|
||||||
cluster_severities = set([])
|
cluster_severities = set([])
|
||||||
|
cluster_installing_hosts = 0
|
||||||
|
cluster_failed_hosts = 0
|
||||||
hostids = []
|
hostids = []
|
||||||
for host in cluster.hosts:
|
clusterhosts = cluster.clusterhosts
|
||||||
|
hosts = [clusterhost.host for clusterhost in clusterhosts]
|
||||||
|
for host in hosts:
|
||||||
if host.state:
|
if host.state:
|
||||||
hostids.append(host.id)
|
hostids.append(host.id)
|
||||||
cluster_progress += host.state.progress
|
cluster_progress += host.state.percentage
|
||||||
if host.state.message:
|
if host.state.message:
|
||||||
cluster_messages[host.hostname] = host.state.message
|
cluster_messages[host.name] = host.state.message
|
||||||
|
|
||||||
if host.state.severity:
|
if host.state.severity:
|
||||||
cluster_severities.add(host.state.severity)
|
cluster_severities.add(host.state.severity)
|
||||||
|
|
||||||
cluster.state.progress = cluster_progress / len(hostids)
|
for clusterhost in clusterhosts:
|
||||||
|
if clusterhost.state:
|
||||||
|
cluster_progress += clusterhost.state.percentage
|
||||||
|
if clusterhost.state.state == 'INSTALLING':
|
||||||
|
cluster_installing_hosts += 1
|
||||||
|
elif (clusterhost.host.state.state not in
|
||||||
|
['ERROR', 'INITIALIZED'] and
|
||||||
|
clusterhost.state.state != 'ERORR'):
|
||||||
|
cluster_installing_hosts += 1
|
||||||
|
elif (clusterhost.state.state == 'ERROR' or
|
||||||
|
clusterhost.host.state.state == 'ERROR'):
|
||||||
|
cluster_failed_hosts += 1
|
||||||
|
|
||||||
|
if clusterhost.state.message:
|
||||||
|
cluster_messages[host.name] = clusterhost.state.message
|
||||||
|
|
||||||
|
if clusterhost.state.severity:
|
||||||
|
cluster_severities.add(clusterhost.state.severity)
|
||||||
|
|
||||||
|
cluster.state.percentage = cluster_progress / (len(hostids) * 2)
|
||||||
cluster.state.message = '\n'.join(
|
cluster.state.message = '\n'.join(
|
||||||
[
|
[
|
||||||
'%s: %s' % (hostname, message)
|
'%s: %s' % (hostname, message)
|
||||||
@ -374,7 +422,7 @@ class AdapterMatcher(object):
|
|||||||
cluster.state.severity = severity
|
cluster.state.severity = severity
|
||||||
break
|
break
|
||||||
|
|
||||||
if cluster.state.progress >= 1.0:
|
if cluster.state.percentage >= 1.0:
|
||||||
cluster.state.state = 'READY'
|
cluster.state.state = 'READY'
|
||||||
|
|
||||||
if cluster.state.severity == 'ERROR':
|
if cluster.state.severity == 'ERROR':
|
||||||
@ -383,85 +431,92 @@ class AdapterMatcher(object):
|
|||||||
if cluster.state.state != 'INSTALLING':
|
if cluster.state.state != 'INSTALLING':
|
||||||
cluster.mutable = True
|
cluster.mutable = True
|
||||||
|
|
||||||
|
cluster.state.installing_hosts = cluster_installing_hosts
|
||||||
|
cluster.state.total_hosts = len(clusterhosts)
|
||||||
|
cluster.state.failed_hosts = cluster_failed_hosts
|
||||||
|
cluster.state.completed_hosts = cluster.state.total_hosts - \
|
||||||
|
cluster.state.installing_hosts - cluster.state.failed_hosts
|
||||||
|
|
||||||
logging.debug(
|
logging.debug(
|
||||||
'update cluster %s state %s',
|
'update cluster %s state %s',
|
||||||
clusterid, cluster.state)
|
clusterid, cluster.state)
|
||||||
|
|
||||||
def update_progress(self, clusterid, hostids):
|
def update_progress(self, clusterid, hostids):
|
||||||
"""Update cluster progress and hosts progresses.
|
|
||||||
|
|
||||||
:param clusterid: the id of the cluster to update.
|
host_progresses = {}
|
||||||
:type clusterid: int.
|
clusterhost_progresses = {}
|
||||||
:param hostids: the ids of the hosts to update.
|
updater = user_api.get_user_object(
|
||||||
:type hostids: list of int.
|
'admin@abc.com',
|
||||||
"""
|
expire_timestamp=datetime.datetime.now() +
|
||||||
logging.debug('printing os_matcher %s', self.__str__())
|
datetime.timedelta(seconds=10000))
|
||||||
host_os_progresses = {}
|
|
||||||
host_package_progresses = {}
|
|
||||||
with database.session():
|
with database.session():
|
||||||
for hostid in hostids:
|
for hostid in hostids:
|
||||||
host_overall_state = (
|
host_name, host_state, host_progress = \
|
||||||
self._get_host_progress(hostid))
|
self._get_host_progress(hostid)
|
||||||
logging.debug('host overall state: %s', host_overall_state)
|
_, clusterhost_state, clusterhost_progress = \
|
||||||
fullname, host_state, host_os_progress = (
|
self._get_clusterhost_progress(hostid)
|
||||||
host_overall_state['os'])
|
|
||||||
_, _, host_package_progress = host_overall_state['package']
|
if (not host_name or
|
||||||
if (not fullname or
|
not host_progress or
|
||||||
not host_os_progress or
|
not clusterhost_progress):
|
||||||
not host_package_progress):
|
|
||||||
logging.error(
|
logging.error(
|
||||||
'nothing to update host %s',
|
'nothing to update host %s',
|
||||||
fullname)
|
host_name)
|
||||||
continue
|
continue
|
||||||
|
|
||||||
logging.debug('got host %s state %s os_progress %s'
|
logging.debug('got host %s host_state: %s '
|
||||||
'package_progress %s',
|
'host_progress: %s, '
|
||||||
fullname,
|
'clusterhost_state: %s, '
|
||||||
host_state, host_os_progress,
|
'clusterhost_progress: %s ',
|
||||||
host_package_progress)
|
host_name,
|
||||||
|
host_state,
|
||||||
|
host_progress,
|
||||||
|
clusterhost_state,
|
||||||
|
clusterhost_progress)
|
||||||
|
|
||||||
host_os_progresses[hostid] = (
|
host_progresses[hostid] = (
|
||||||
fullname, host_state, host_os_progress)
|
host_name, host_state, host_progress)
|
||||||
host_package_progresses[hostid] = (
|
clusterhost_progresses[hostid] = (
|
||||||
fullname, host_state, host_package_progress)
|
host_name, clusterhost_state, clusterhost_progress)
|
||||||
|
|
||||||
for hostid, host_value in host_os_progresses.items():
|
for hostid, host_value in host_progresses.items():
|
||||||
fullname, host_state, host_os_progress = host_value
|
host_name, host_state, host_progress = host_value
|
||||||
if host_state == 'INSTALLING' and host_os_progress.progress < 1.0:
|
if (host_state == 'INSTALLING' and
|
||||||
self.os_matcher_.update_progress(
|
host_progress.progress < 1.0):
|
||||||
fullname, host_os_progress)
|
self.os_matcher_.update_progress(
|
||||||
else:
|
host_name, host_progress)
|
||||||
logging.error(
|
else:
|
||||||
'there is no need to update host %s '
|
logging.error(
|
||||||
'OS progress: state %s os_progress %s',
|
'there is no need to update host %s '
|
||||||
fullname, host_state, host_os_progress)
|
'progress: state %s progress %s',
|
||||||
|
host_name, host_state, host_progress)
|
||||||
|
|
||||||
for hostid, host_value in host_package_progresses.items():
|
for hostid, clusterhost_value in clusterhost_progresses.items():
|
||||||
fullname, host_state, host_package_progress = host_value
|
host_name, clusterhost_state, clusterhost_progress = \
|
||||||
if (host_state == 'INSTALLING' and
|
clusterhost_value
|
||||||
host_package_progress.progress < 1.0):
|
if (clusterhost_state == 'INSTALLING' and
|
||||||
self.package_matcher_.update_progress(
|
clusterhost_progress.progress < 1.0):
|
||||||
fullname, host_package_progress)
|
self.package_matcher_.update_progress(
|
||||||
else:
|
host_name, clusterhost_progress)
|
||||||
logging.error(
|
else:
|
||||||
'there is no need to update host %s '
|
logging.error(
|
||||||
'Package progress: state %s package_progress %s',
|
'no need to update clusterhost %s'
|
||||||
fullname, host_state, host_package_progress)
|
'progress: state %s progress %s',
|
||||||
|
host_name, clusterhost_state, clusterhost_progress)
|
||||||
|
|
||||||
with database.session():
|
|
||||||
for hostid in hostids:
|
for hostid in hostids:
|
||||||
if hostid not in host_os_progresses:
|
if hostid not in host_progresses:
|
||||||
|
continue
|
||||||
|
if hostid not in clusterhost_progresses:
|
||||||
continue
|
continue
|
||||||
|
|
||||||
if hostid not in host_package_progresses:
|
_, _, host_progress = host_progresses[hostid]
|
||||||
continue
|
_, _, clusterhost_progress = clusterhost_progresses[hostid]
|
||||||
|
self._update_host_progress(hostid, host_progress, updater)
|
||||||
_, _, host_os_progress = host_os_progresses[hostid]
|
self._update_clusterhost_progress(
|
||||||
_, _, host_package_progress = host_package_progresses[hostid]
|
|
||||||
self._update_host_os_progress(hostid, host_os_progress)
|
|
||||||
self._update_host_package_progress(
|
|
||||||
hostid,
|
hostid,
|
||||||
host_package_progress
|
clusterhost_progress,
|
||||||
|
updater
|
||||||
)
|
)
|
||||||
|
|
||||||
self._update_cluster_progress(clusterid)
|
self._update_cluster_progress(clusterid)
|
||||||
|
@ -19,8 +19,8 @@
|
|||||||
import logging
|
import logging
|
||||||
import os.path
|
import os.path
|
||||||
|
|
||||||
from compass.db import database
|
from compass.db.api import database
|
||||||
from compass.db.model import LogProgressingHistory
|
from compass.db.models import LogProgressingHistory
|
||||||
from compass.log_analyzor.line_matcher import Progress
|
from compass.log_analyzor.line_matcher import Progress
|
||||||
from compass.utils import setting_wrapper as setting
|
from compass.utils import setting_wrapper as setting
|
||||||
|
|
||||||
@ -109,24 +109,24 @@ class FileReader(object):
|
|||||||
in the last run, the progress, the message and the
|
in the last run, the progress, the message and the
|
||||||
severity it has got in the last run.
|
severity it has got in the last run.
|
||||||
"""
|
"""
|
||||||
with database.session() as session:
|
session = database.current_session()
|
||||||
history = session.query(
|
history = session.query(
|
||||||
LogProgressingHistory
|
LogProgressingHistory
|
||||||
).filter_by(
|
).filter_by(
|
||||||
pathname=self.pathname_
|
pathname=self.pathname_
|
||||||
).first()
|
).first()
|
||||||
if history:
|
if history:
|
||||||
self.position_ = history.position
|
self.position_ = history.position
|
||||||
self.partial_line_ = history.partial_line
|
self.partial_line_ = history.partial_line
|
||||||
line_matcher_name = history.line_matcher_name
|
line_matcher_name = history.line_matcher_name
|
||||||
progress = Progress(history.progress,
|
progress = Progress(history.percentage,
|
||||||
history.message,
|
history.message,
|
||||||
history.severity)
|
history.severity)
|
||||||
else:
|
else:
|
||||||
line_matcher_name = 'start'
|
line_matcher_name = 'start'
|
||||||
progress = Progress(0.0, '', None)
|
progress = Progress(0.0, '', None)
|
||||||
|
|
||||||
return line_matcher_name, progress
|
return line_matcher_name, progress
|
||||||
|
|
||||||
def update_history(self, line_matcher_name, progress):
|
def update_history(self, line_matcher_name, progress):
|
||||||
"""Update log_progressing_history table.
|
"""Update log_progressing_history table.
|
||||||
@ -138,37 +138,36 @@ class FileReader(object):
|
|||||||
The function should be called out of database session.
|
The function should be called out of database session.
|
||||||
It updates the log_processing_history table.
|
It updates the log_processing_history table.
|
||||||
"""
|
"""
|
||||||
with database.session() as session:
|
session = database.current_session()
|
||||||
history = session.query(LogProgressingHistory).filter_by(
|
history = session.query(LogProgressingHistory).filter_by(
|
||||||
pathname=self.pathname_).first()
|
pathname=self.pathname_).first()
|
||||||
|
if history:
|
||||||
|
if history.position >= self.position_:
|
||||||
|
logging.error(
|
||||||
|
'%s history position %s is ahead of current '
|
||||||
|
'position %s',
|
||||||
|
self.pathname_,
|
||||||
|
history.position,
|
||||||
|
self.position_)
|
||||||
|
return
|
||||||
|
|
||||||
if history:
|
history.position = self.position_
|
||||||
if history.position >= self.position_:
|
history.partial_line = self.partial_line_
|
||||||
logging.error(
|
history.line_matcher_name = line_matcher_name
|
||||||
'%s history position %s is ahead of current '
|
history.progress = progress.progress
|
||||||
'position %s',
|
history.message = progress.message
|
||||||
self.pathname_,
|
history.severity = progress.severity
|
||||||
history.position,
|
else:
|
||||||
self.position_)
|
history = LogProgressingHistory(
|
||||||
return
|
pathname=self.pathname_, position=self.position_,
|
||||||
|
partial_line=self.partial_line_,
|
||||||
history.position = self.position_
|
line_matcher_name=line_matcher_name,
|
||||||
history.partial_line = self.partial_line_
|
percentage=progress.progress,
|
||||||
history.line_matcher_name = line_matcher_name
|
message=progress.message,
|
||||||
history.progress = progress.progress
|
severity=progress.severity)
|
||||||
history.message = progress.message
|
session.merge(history)
|
||||||
history.severity = progress.severity
|
logging.debug('update file %s to history %s',
|
||||||
else:
|
self.pathname_, history)
|
||||||
history = LogProgressingHistory(
|
|
||||||
pathname=self.pathname_, position=self.position_,
|
|
||||||
partial_line=self.partial_line_,
|
|
||||||
line_matcher_name=line_matcher_name,
|
|
||||||
progress=progress.progress,
|
|
||||||
message=progress.message,
|
|
||||||
severity=progress.severity)
|
|
||||||
session.merge(history)
|
|
||||||
logging.debug('update file %s to history %s',
|
|
||||||
self.pathname_, history)
|
|
||||||
|
|
||||||
def readline(self):
|
def readline(self):
|
||||||
"""Generate each line of the log file."""
|
"""Generate each line of the log file."""
|
||||||
|
@ -433,7 +433,7 @@ def _get_package_adapter_matcher(package_installer, target_system):
|
|||||||
else:
|
else:
|
||||||
logging.debug('configuration %s does not match %s and %s',
|
logging.debug('configuration %s does not match %s and %s',
|
||||||
configuration, target_system, package_installer)
|
configuration, target_system, package_installer)
|
||||||
logging.error('No configuration found for os installer %s os %s',
|
logging.error('No configuration found for package installer %s os %s',
|
||||||
package_installer, target_system)
|
package_installer, target_system)
|
||||||
return None
|
return None
|
||||||
|
|
||||||
|
@ -24,6 +24,7 @@ from celery.signals import setup_logging
|
|||||||
from compass.actions import deploy
|
from compass.actions import deploy
|
||||||
from compass.actions import poll_switch
|
from compass.actions import poll_switch
|
||||||
from compass.actions import reinstall
|
from compass.actions import reinstall
|
||||||
|
from compass.actions import update_progress
|
||||||
from compass.db.api import adapter_holder as adapter_api
|
from compass.db.api import adapter_holder as adapter_api
|
||||||
from compass.db.api import database
|
from compass.db.api import database
|
||||||
from compass.db.api import metadata_holder as metadata_api
|
from compass.db.api import metadata_holder as metadata_api
|
||||||
@ -118,3 +119,16 @@ def reset_host(host_id):
|
|||||||
:type cluster_hosts: dict of int to list of int
|
:type cluster_hosts: dict of int to list of int
|
||||||
"""
|
"""
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
@celery.task(name='compass.tasks.update_progress')
|
||||||
|
def update_clusters_progress(cluster_hosts):
|
||||||
|
"""Calculate the installing progress of the given cluster.
|
||||||
|
|
||||||
|
:param cluster_hosts: the cluster and hosts of each cluster to update.
|
||||||
|
:type cluster_hosts: dict of int to list of int
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
update_progress.update_progress(cluster_hosts)
|
||||||
|
except Exception as error:
|
||||||
|
logging.exception(error)
|
||||||
|
@ -190,3 +190,42 @@ def load_configs(
|
|||||||
raise error
|
raise error
|
||||||
configs.append(config_locals)
|
configs.append(config_locals)
|
||||||
return configs
|
return configs
|
||||||
|
|
||||||
|
|
||||||
|
def is_instance(instance, expected_types):
|
||||||
|
"""Check instance type is in one of expected types.
|
||||||
|
|
||||||
|
:param instance: instance to check the type.
|
||||||
|
:param expected_types: types to check if instance type is in them.
|
||||||
|
:type expected_types: list of type
|
||||||
|
|
||||||
|
:returns: True if instance type is in expect_types.
|
||||||
|
"""
|
||||||
|
for expected_type in expected_types:
|
||||||
|
if isinstance(instance, expected_type):
|
||||||
|
return True
|
||||||
|
|
||||||
|
return False
|
||||||
|
|
||||||
|
|
||||||
|
def get_clusters_from_str(clusters_str):
|
||||||
|
"""get clusters from string."""
|
||||||
|
clusters = {}
|
||||||
|
for cluster_and_hosts in clusters_str.split(';'):
|
||||||
|
if not cluster_and_hosts:
|
||||||
|
continue
|
||||||
|
|
||||||
|
if ':' in cluster_and_hosts:
|
||||||
|
cluster_str, hosts_str = cluster_and_hosts.split(
|
||||||
|
':', 1)
|
||||||
|
else:
|
||||||
|
cluster_str = cluster_and_hosts
|
||||||
|
hosts_str = ''
|
||||||
|
|
||||||
|
hosts = [
|
||||||
|
host for host in hosts_str.split(',')
|
||||||
|
if host
|
||||||
|
]
|
||||||
|
clusters[cluster_str] = hosts
|
||||||
|
|
||||||
|
return clusters
|
||||||
|
Loading…
Reference in New Issue
Block a user