crowbar/crowbar-core

View on GitHub
crowbar_framework/app/models/role_object.rb

Summary

Maintainability
D
1 day
Test Coverage
#
# Copyright 2011-2013, Dell
# Copyright 2013-2014, SUSE LINUX Products GmbH
#
# 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.
#

require "ostruct"

class RoleObject < ChefObject
  self.chef_type = "role"

  def cluster_roles(roles = RoleObject.all)
    @cluster_roles ||= begin
      roles.select do |role|
        role.elements.values.flatten.compact.uniq.include?("cluster:#{inst}")
      end
    end
  end

  def cluster_remote_roles(roles = RoleObject.all)
    @cluster_remote_roles ||= begin
      roles.select do |role|
        role.elements.values.flatten.compact.uniq.include?("remotes:#{inst}")
      end
    end
  end

  def cluster_nodes(nodes = Node.all)
    @cluster_nodes ||= begin
      proposal_nodes(nodes, nil, ["pacemaker-remote"]).values.flatten.uniq
    end
  end

  def cluster_remote_nodes(nodes = Node.all)
    @cluster_remote_nodes ||= begin
      proposal_nodes(nodes, ["pacemaker-remote"], nil).values.flatten.uniq
    end
  end

  def proposal_nodes(nodes = Node.all, include_roles = nil, exclude_roles = nil)
    assigned_nodes = {}
    elements.each do |role_name, node_names|
      next if !include_roles.nil? && !include_roles.include?(role_name)
      next if !exclude_roles.nil? && exclude_roles.include?(role_name)
      assigned_nodes[role_name] = node_names.map do |node_name|
        nodes.find { |n| n.name == node_name }
      end.compact
    end
    assigned_nodes
  end

  # Returns all proposal roles where this role is mentioned
  def proposal_roles(roles = self.class.all)
    if proposal?
      []
    else
      roles.select do |role|
        role.proposal? && role.elements.keys.include?(name)
      end
    end
  end

  def proposal(proposals = nil)
    @associated_proposal ||= begin
      if proposals
        proposals.find { |p| p.barclamp == barclamp && p.name == inst }
      else
        Proposal.where(barclamp: barclamp, name: inst).first
      end
    end
  end

  def self.assigned(roles = self.all)
    assigned_roles = []
    roles.select(&:proposal?).map(&:elements).each do |element|
      element.each do |role_name, node_names|
        assigned_roles.push(roles.find { |r| r.name == role_name })
      end
    end
    assigned_roles.compact.uniq
  end

  # Returns all nodes which have this role applied
  def element_nodes(roles = self.class.all, nodes = Node.all)
    assigned_nodes = []

    # Get all nodes from proposal roles mentioning this one
    proposal_roles(roles).each do |prop|
      prop.elements[self.name].each do |node_name|
        # Resolve clusters. We do not use the Pacemaker helper as that means
        # another call to Chef. Obvious FIXME is to patch that method to accept
        # cache param and fallback to proposal look up.
        if ServiceObject.is_cluster?(node_name) || ServiceObject.is_remotes?(node_name)
          cluster = roles.find { |r| r.name == "pacemaker-config-#{ServiceObject.cluster_name(node_name)}" }
          if ServiceObject.is_remotes?(node_name)
            include_roles = ["pacemaker-remote"]
            exclude_roles = nil
          else
            include_roles = nil
            exclude_roles = ["pacemaker-remote"]
          end
          cluster_roles_nodes = cluster.proposal_nodes(nodes, include_roles, exclude_roles)

          obj = OpenStruct.new(
            cluster: true,
            nodes: cluster_roles_nodes.map { |role, nodes| nodes }.flatten.uniq,
            node: nil,
            name: ServiceObject.cluster_name(node_name)
          )
        else
          node = nodes.find { |n| n.name == node_name }
          obj  = !node ? nil : OpenStruct.new(
            cluster: false,
            nodes: [node],
            node: node,
            name: node.name
          )
        end
        assigned_nodes.push(obj) if obj
      end
    end
    assigned_nodes
  end

  def core_role?
    core_barclamps = BarclampCatalog.members("crowbar").keys
    core_barclamps.any? { |core_barclamp| core_barclamp == barclamp }
  end

  def ha?
    barclamp == "pacemaker"
  end

  def proposal?
    @role.name.to_s =~ /.*\-config\-.*/
  end

  def self.all
    self.find_roles_by_search(nil)
  end

  def self.all_dependencies
    active_roles = RoleObject.find_roles_by_name("*-config-*")
    dependencies = {}
    active_roles.map do |role|
      service = ServiceObject.get_service(role.barclamp).new
      dependencies[role.name] = service.proposal_dependencies(role).map do |dep|
        "#{dep["barclamp"]}-config-#{dep["inst"]}"
      end
    end
    dependencies
  end

  def self.dependencies(service)
    all_dependencies[service]
  end

  def self.reverse_dependencies(service)
    Hash[all_dependencies.select { |s, r| r.include?(service) }].keys
  end

  def self.active(barclamp = nil, inst = nil)
    full = if barclamp.nil?
      RoleObject.find_roles_by_name "*-config-*"
    else
      RoleObject.find_roles_by_name "#{barclamp}-config-#{inst || "*"}"
    end
    full.map { |x| "#{x.barclamp}_#{x.inst}" }
  end

  def self.find_roles_by_name(name)
    roles = []
    #TODO this call could be moved to fild_roles_by_search
    arr = ChefObject.query_chef.search "role", "name:#{chef_escape(name)}"
    if arr[2] != 0
      roles = arr[0].map { |x| RoleObject.new x }
      roles.delete_if { |x| x.nil? or x.role.nil? }
    end
    roles
  end

  def self.find_roles_by_search(search)
    roles = []
    arr = if search.nil?
      ChefObject.fetch_roles_from_cdb
    else
      ChefObject.query_chef.search "role", search
    end
    if arr[2] != 0
      roles = arr[0].map { |x| RoleObject.new x }
      roles.delete_if { |x| x.nil? or x.role.nil? }
    end
    roles
  end

  def self.find_role_by_name(name)
    begin
      # get role directly from CouchDB, skip Chef API
      return RoleObject.new Chef::Role.cdb_load(name)
    rescue Errno::ECONNREFUSED => e
      raise Crowbar::Error::ChefOffline.new
    rescue Chef::Exceptions::CouchDBNotFound
      return nil
    end
  end

  def barclamp
    # FIXME: this obviously shouldn't need to exist; we need a proper registry
    # to avoid barclamp-crowbar having to know about other barclamps...
    name = @role.name.split("-")[0]

    case name
    when "switch_config"
      "network"
    when "bmc"
      "ipmi"
    when "nfs"
      "nfs_client"
    when "hawk"
      "pacemaker"
    else
      name
    end
  end

  def category
    @category ||= BarclampCatalog.category(barclamp)
  end

  def inst
    @role.name.gsub("#{self.barclamp}-config-", "")
  end

  def prop
    [barclamp, inst].join("_")
  end

  def display_name
    @display_name ||= BarclampCatalog.display_name(barclamp)
  end

  def allow_multiple_proposals?
    ServiceObject.get_service(barclamp).allow_multiple_proposals?
  end

  def role
    @role
  end

  def name
    @role.name
  end

  def name=(value)
    @role.name value
  end

  def description
    @role.description
  end

  def description=(value)
    @role.description value
  end

  def default_attributes
    @role.default_attributes
  end

  def default_attributes=(value)
    @role.default_attributes value
  end

  def override_attributes
    @role.override_attributes
  end

  def override_attributes=(value)
    @role.override_attributes value
  end

  def initialize(x)
    @role = x
  end

  def crowbar_revision
    override_attributes[barclamp] ? override_attributes[barclamp]["crowbar-revision"].to_i : 0
  end

  def increment_crowbar_revision!
    override_attributes[barclamp] ||= {}
    if override_attributes[barclamp]["crowbar-revision"].nil?
      override_attributes[barclamp]["crowbar-revision"] = 0
    else
      override_attributes[barclamp]["crowbar-revision"] += 1
    end
  end

  def save
    Rails.logger.debug("Saving role: #{@role.name} - #{crowbar_revision}")
    Crowbar::Lock::LocalBlocking.with_lock(name: "role:#{@role.name}") do
      upstream_role = RoleObject.find_role_by_name(@role.name)
      ### We assume that if we can not find the role, it has just
      # been created. TODO: If it was actually deleted meanwhile,
      # this might not work as expected.
      if upstream_role
        upstream_rev = upstream_role.crowbar_revision
        new_rev = crowbar_revision
        if upstream_rev && upstream_rev > new_rev
          Rails.logger.warn("WARNING: revision race for role #{@role.name} (previous revision #{upstream_rev})")
        end
        if block_given?
          @role = upstream_role.role
        end
      end
      if block_given?
        yield(@role)
      end
      increment_crowbar_revision!
      @role.save
    end
    Rails.logger.debug("Done saving role: #{@role.name} - #{crowbar_revision}")
  end

  def destroy
    Rails.logger.debug("Destroying role: #{@role.name} - #{crowbar_revision}")
    Crowbar::Lock::LocalBlocking.with_lock(name: "role:#{@role.name}") do
      @role.destroy
    end
    Rails.logger.debug("Done removing role: #{@role.name} - #{crowbar_revision}")
  end

  def elements
    (@role.override_attributes[self.barclamp]["elements"] || {}) rescue {}
  end

  def run_list
    @role.run_list
  end
end