crowbar/crowbar-core

View on GitHub
crowbar_framework/app/helpers/nodes_helper.rb

Summary

Maintainability
F
4 days
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.
#

module NodesHelper
  def all_non_admin_nodes
    Node.find("NOT admin_node:true")
  end

  def upgrading_nodes_count
    Node.find("state:crowbar_upgrade OR state:os-upgrading OR state:os-upgraded").count
  end

  def failed_nodes_count
    Node.find("state:problem").count
  end

  def nodes_ready?
    all_non_admin_nodes.select { |n| n.status != "ready" }.empty?
  end

  def nodes_by_role(role)
    Node.find("roles:#{role}").sort_by(&:alias)
  end

  def piechart_for(group)
    values = piechart_values(group)
    tooltip = piechart_tooltip(values)

    content_tag(
      :span,
      "",
      :title => tooltip,
      "data-piechart" => values.join(","),
      "data-tooltip" => "true"
    )
  end

  def piechart_tooltip(values)
    [].tap do |result|
      result.push content_tag(
        :strong,
        t("total", count: values.sum, scope: "nodes.index.status_pie")
      )

      result.push t("ready", count: values[0], scope: "nodes.index.status_pie") if values[0] > 0
      result.push t("unknown", count: values[1], scope: "nodes.index.status_pie") if values[1] > 0
      result.push t("unready", count: values[2], scope: "nodes.index.status_pie") if values[2] > 0
      result.push t("pending", count: values[3], scope: "nodes.index.status_pie") if values[3] > 0
      result.push t("crowbar_upgrade",
                    count: values[4],
                    scope: "nodes.index.status_pie") if values[4] > 0
    end.join(tag(:br))
  end

  def piechart_values(group)
    [].tap do |result|
      result.push group[:status]["ready"]
      result.push group[:status]["failed"]
      result.push group[:status]["unknown"]

      if group[:status]["building"]
        result.push group[:status]["unready"] + group[:status]["pending"] + group[:status]["building"]
      else
        result.push group[:status]["unready"] + group[:status]["pending"]
      end
      result.push group[:status]["crowbar_upgrade"]
    end
  end

  def format_memory(kbyte)
    sprintf(
      "%#1.2f GB",
      kbyte.to_f / 1024 / 1024
    )
  end

  def node_list_for(value)
    {}.tap do |nodes|
      nodes_hash.each do |k, v|
        nodes[k] = v if value.member? k
      end
    end.sort_by { |k, v| v[:alias] }
  end

  def node_links_for(value)
    node_list_for(value).map do |name, node|
      link_to node[:alias], node_path(node[:handle]), title: node[:title]
    end
  end

  def nodes_hash(group = nil)
    @nodes_hash ||= {}

    @nodes_hash[group] ||= begin
      {}.tap do |nodes|
        Node.all.each do |node|
          nodes[node.name] = {
            handle: node.handle,
            alias: node.alias,
            title: node.description(false, true),
            admin: node.admin?,
            group: node.group,
            platform: node[:platform],
            platform_version: node[:platform_version],
            cluster: false
          } if node.group == group || group.nil?
        end
      end.sort_by{ |k, v| v[:alias] }
    end
  end

  def node_detail_count
    @node_detail_count ||= [
      node_detail_software.count,
      node_detail_hardware.count
    ].max
  end

  def node_detail_software
    @node_detail_software ||= begin
      [].tap do |result|
        result.push [
          t("model.attributes.node.name"),
          dash_or(@node.name)
        ]

        result.push [
          t("model.attributes.node.public_name"),
          dash_or(@node.public_name)
        ]

        result.push [
          t("model.attributes.node.description"),
          dash_or(@node.description)
        ]

        result.push [
          t("model.attributes.node.target_platform"),
          dash_or(@node.pretty_target_platform)
        ]

        if Crowbar::Platform.require_license_key? @node.target_platform
          value = if @node.license_key
            hidetext_tag(@node.license_key, class: "pull-right")
          else
            value_for(nil, t("model.attributes.node.license_key_not_set"))
          end

          result.push [
            t("model.attributes.node.license_key"),
            value
          ]
        end

        result.push [
          t("model.attributes.node.uptime"),
          value_for(@node.uptime, t("model.attributes.node.na"), @node.ready?)
        ]

        result.push [
          t("model.attributes.node.allocated"),
          value_for(t(".active"), t(".inactive"), @node.allocated?)
        ]

        result.push [
          t("model.attributes.node.state"),
          content_tag(
            :span,
            t(@node.state, scope: :state, default: @node.state.titlecase),
            "data-node-state" => @node.handle
          )
        ]
      end
    end
  end

  def node_detail_hardware
    @node_detail_hardware ||= begin
      raid_data = if bios_raid_options[:show].include?(:raid)
        content_tag(
          :em,
          t(@node.raid_set, scope: "raid")
        )
      else
        nil
      end

      [].tap do |result|
        result.push [
          t("model.attributes.node.hardware"),
          dash_or(@node.hardware)
        ]

        result.push [
          t("model.attributes.node.asset_tag"),
          dash_or(@node.asset_tag)
        ]

        result.push [
          t("model.attributes.node.cpu"),
          dash_or(@node.cpu_arch)
        ]

        result.push [
          t("model.attributes.node.memory"),
          format_memory(@node.memory)
        ]

        if Crowbar::Platform.support_software_raid.include? @node.target_platform
          unless bios_raid_options[:show].include?(:raid) || @node.raid_type == "single"
            result.push [
              t("model.attributes.node.raid_type"),
              value_for(t(@node.raid_type, scope: "nodes.form.raid_types"), t("unknown"))
            ]

            result.push [
              t("model.attributes.node.raid_disks"),
              value_for(@node.raid_disks.length.to_s, t("unknown"))
            ]
          end
        end

        result.push [
          t("model.attributes.node.number_of_drives"),
          [
            @node.pretty_drives,
            raid_data
          ].compact.join(" - ")
        ]

        result.push [
          t("model.attributes.node.mac"),
          value_for(@node.mac, t("unknown"))
        ]

        show_name = if @node.switch_name.nil?
          false
        else
          @node.switch_name.to_i != -1
        end

        show_port = if @node.switch_port.nil?
          false
        else
          @node.switch_port.to_i != -1
        end

        show_unit = if @node.switch_unit.nil?
          false
        else
          @node.switch_unit.to_i != -1
        end

        if @node.switch_unit.nil?
          switch_label = [
            value_for(@node.switch_name, t("unknown"), show_name),
            value_for(@node.switch_port, t("unknown"), show_port)
          ].join(" / ").html_safe

          switch_title = t("model.attributes.node.switch_name_port")
        else
          switch_label = [
            value_for(@node.switch_name, t("unknown"), show_name),
            value_for(@node.switch_unit, t("unknown"), show_unit),
            value_for(@node.switch_port, t("unknown"), show_port)
          ].join(" / ").html_safe

          switch_title = t("model.attributes.node.switch_name_unit_port")
        end

        result.push [
          switch_title,
          link_to(switch_label, switch_path(node: @node.handle))
        ]
      end
    end
  end

  def node_ip_list(ips)
    entries = [].tap do |result|
      ips.each do |network, addresses|
        unless network == "~notconnected" and addresses.nil?
          network_list = if ["[not managed]", "[dhcp]"].include? network
            address_list = [*addresses].map do |address|
              content_tag(
                :li,
                address
              )
            end

            [
              content_tag(
                :strong,
                network
              ),
              content_tag(
                :ul,
                address_list.join("\n").html_safe
              )
            ].join("\n").html_safe
          else
            address_list = if addresses.is_a? String
              addresses
            else
              content_tag(
                :ul,
                addresses.keys.collect { |key|
                  content_tag(
                    :li,
                    "#{key}: #{addresses[key]}"
                  )
                }.join("\n").html_safe
              )
            end

            [
              content_tag(
                :strong,
                network
              ),
              address_list
            ].join("\n").html_safe
          end

          result.push content_tag(
            :li,
            network_list
          )
        end
      end
    end

    if entries.empty?
      entries.push content_tag(
        :li,
        t(".no_entry"),
        class: "empty"
      )
    end

    content_tag(
      :ul,
      entries.join("\n").html_safe
    )
  end

  def node_wall_list(node)
    [].tap do |result|
      intended_roles_text = {
        "no_role" => t("nodes.form.no_role"),
        "controller" => t("nodes.form.controller"),
        "compute" => t("nodes.form.compute"),
        "network" => t("nodes.form.network"),
        "storage" => t("nodes.form.storage"),
        "monitoring" => t("nodes.form.monitoring")
      }

      intended_role = node.intended_role
      intended_role = "no_role" if intended_role.blank?

      result.push [
        t("model.attributes.node.intended_role"),
        intended_roles_text[intended_role] || intended_role
      ]

      if have_openstack
        result.push [
          t("model.attributes.node.availability_zone"),
          dash_or(node.availability_zone)
        ]
      end
    end
  end

  def node_barclamps(node)
    return [] if node.roles.nil? || node.roles.empty?

    node.roles.delete_if do |role|
      not role =~ /^.*-config-/
    end
  end

  def node_barclamp_list(node)
    all_proposals = Proposal.all

    list_items = ActiveSupport::OrderedHash.new.tap do |listing|
      node_barclamps(node).map do |role|
        proposal = all_proposals.find { |p| p.key == role_to_proposal_name(role) }

        if proposal.nil?
          listing["Unknown"] ||= []
          listing["Unknown"].push role.gsub("-config-", " ").titleize
        else
          display_name = if proposal.allow_multiple_proposals?
            "#{proposal.display_name} (#{proposal.name.titleize})"
          else
            proposal.display_name
          end

          route = show_proposal_path(controller: proposal.barclamp, id: proposal.name)

          listing[proposal.category] ||= []
          listing[proposal.category].push link_to(display_name, route)
        end
      end
    end

    if list_items.empty?
      list_result = content_tag(
        :li,
        t(".no_entry"),
        class: "empty"
      )
    else
      list_result = [].tap do |result|
        list_items.sort.each do |category, proposals|
          children = proposals.sort { |x, y| strip_links(x) <=> strip_links(y) }.map do |proposal|
            content_tag(
              :li,
              proposal
            )
          end

          result.push content_tag(
            :li,
            [
              category,
              content_tag(
                :ul,
                children.join("\n").html_safe
              )
            ].join("\n").html_safe
          )
        end
      end.join("\n").html_safe
    end

    content_tag(
      :ul,
      list_result,
      class: "barclamps"
    )
  end

  def node_roles(node)
    return [] if node.roles.nil? || node.roles.empty?

    node.roles.delete_if do |role|
      role =~ /^.*-config-/
    end
  end

  def node_role_list(node)
    all_roles     = RoleObject.all
    all_proposals = Proposal.all

    list_items = ActiveSupport::OrderedHash.new.tap do |listing|
      node_roles(node).map do |role|
        object = all_roles.find { |r| r.name == role }

        next if role =~ /^crowbar-.*_#{Crowbar::Settings.domain.tr(".", "_")}/

        if object.nil?
          listing["Unknown"] ||= []
          listing["Unknown"].push role.titleize
        else
          barclamp = node_barclamps(node).select do |barclamp|
            barclamp.include? object.barclamp
          end.first

          unless barclamp.nil?
            proposal = all_proposals.find { |p| p.key == role_to_proposal_name(barclamp) }
          end

          if proposal.nil?
            listing[object.category] ||= []
            listing[object.category].push role.titleize
          else
            route = show_proposal_path(
              controller: proposal.barclamp,
              id: proposal.name
            )

            listing[proposal.category] ||= []
            listing[proposal.category].push link_to(role, route)
          end
        end
      end
    end

    if list_items.empty?
      list_result = content_tag(
        :li,
        t(".no_entry"),
        class: "empty"
      )
    else
      list_result = [].tap do |result|
        list_items.sort.each do |category, roles|
          children = roles.sort { |x, y| strip_links(x) <=> strip_links(y) }.map do |role|
            content_tag(
              :li,
              role
            )
          end

          result.push content_tag(
            :li,
            [
              category,
              content_tag(
                :ul,
                children.join("\n").html_safe
              )
            ].join("\n").html_safe
          )
        end
      end.join("\n").html_safe
    end

    content_tag(
      :ul,
      list_result,
      class: "roles"
    )
  end

  def node_link_list(node)
    link_list = [].tap do |result|
      if node.bmc_configured?
        path = node["crowbar_wall"]["ipmi"]["address"] rescue "none"

        result.push content_tag(
          :li,
          link_to(
            t(".bmc"),
            "https://#{path}"
          )
        )
      end

      unless node["crowbar"]["links"].nil?
        node["crowbar"]["links"].sort_by { |name, link| name }.each do |name, link|
          result.push content_tag(
            :li,
            link_to(
              name,
              link,
              target: "_blank"
            )
          )
        end
      end
    end

    if link_list.empty?
      link_list.push content_tag(
        :li,
        t(".no_entry"),
        class: "empty"
      )
    end

    content_tag(
      :ul,
      link_list.join("\n").html_safe
    )
  end

  def dl_item(term, definition, options={})
    unless definition.blank? && options[:show_if_blank] != true
      html  = "<dt>#{options[:escape_html] != false ? (h term) : (term)}</dt>"
      dd = "<dd" + (options[:class].nil? ? "" : " class=""+options[:class]+""") + (options[:title].nil? ? "" : " title="" + options[:title]+""") + ">"
      html += "#{dd}#{options[:escape_html] != false ? (h definition) : (definition)}</dd>"
      raw html
    end
  end

  def node_title(node)
    tooltip = [
      node.ip,
      node.hardware,
      format_memory(node.memory),
      "#{node.nics} NICs"
    ]

    link_to(
      icon_tag("info-circle", node.handle),
      node_path(node.handle),
      title: tooltip.join("<br/>"),
      data: {
        toggle: "tooltip",
        placement: "top"
      }
    )
  end

  protected

  def role_to_proposal_name(role)
    role_split = role.split("-", 3)
    "#{role_split.first}-#{role_split.last}"
  end

  def node_role_map
    {
      "bmc" => "ipmi"
    }
  end
end