yast/yast-yast2

View on GitHub
library/general/src/modules/MailAliases.rb

Summary

Maintainability
A
1 hr
Test Coverage
# ***************************************************************************
#
# Copyright (c) 2002 - 2012 Novell, Inc.
# All Rights Reserved.
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of version 2 of the GNU General Public License as
# published by the Free Software Foundation.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.   See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, contact Novell, Inc.
#
# To contact Novell about this file by physical or electronic mail,
# you may find current contact information at www.novell.com
#
# ***************************************************************************
# File:
#   modules/Mail.ycp
#
# Package:
#   Configuration of mail aliases
#
# Summary:
#   Data for configuration of mail aliases, input and output functions.
#
# Authors:
#   Martin Vidner <mvidner@suse.cz>
#
# $Id: MailAliases.ycp 35035 2007-01-02 14:35:30Z mvidner $
#
# Representation of the configuration of mail aliases.
# Input and output routines.
# Separated from Mail.ycp because yast2-users need us.
# Virtusertable/virtual users are not included, arbitrarily.
#
require "yast"

module Yast
  class MailAliasesClass < Module
    def main
      # no translatable strings, no textdomain.
      Yast.import "MailTable"

      # ----------------------------------------------------------------

      # List of maps: $[comment:, alias:, destinations:] (all are strings)
      # Except root.
      @aliases = []
      # Separated/joined with aliases by read/write/set/export
      @root_alias = ""
      # Separated/joined with aliases by read/write/set/export
      @root_alias_comment = ""
    end

    # Useful for autoinstall: the provided aliases will be (with
    # higher priority) merged with existing ones (presumably system defaults).
    #    global boolean merge_aliases = false;

    # ----------------------------------------------------------------

    # Separates aliases into aliases, root_alias and root_alias_comment
    def FilterRootAlias
      @root_alias = ""
      @root_alias_comment = ""
      @aliases = Builtins.filter(@aliases) do |e|
        if Ops.get_string(e, "alias", "") == "root"
          @root_alias = Ops.get_string(e, "destinations", "")
          @root_alias_comment = Ops.get_string(e, "comment", "")
          next false
        end
        true
      end

      nil
    end

    # Read the aliases table (and separate the root alias)
    # @return success?
    def ReadAliases
      a_raw = MailTable.Read("aliases")
      @aliases = Builtins.maplist(a_raw) do |e|
        {
          "comment"      => Ops.get_string(e, "comment", ""),
          "alias"        => Ops.get_string(e, "key", ""),
          "destinations" => Ops.get_string(e, "value", "")
        }
      end
      FilterRootAlias()
      true
    end

    # @param [Array<Hash>] aliases an alias table
    # @return prepend root alias data to aliases, if set
    def MergeRootAlias(aliases)
      aliases = deep_copy(aliases)
      ret = deep_copy(aliases)
      if @root_alias != ""
        ret = Builtins.prepend(
          ret,
          "alias"        => "root",
          "destinations" => @root_alias,
          "comment"      => @root_alias_comment
        )
      end
      deep_copy(ret)
    end

    # ----------------------------------------------------------------

    # Merges mail tables, which are order-preserving maps.
    # First are the entries of the old map, with values updated
    # from the new one, then the rest of the new map.
    # @param [Array<Hash>] new  new table
    # @param [Array<Hash>] old  old table
    # @return    merged table
    def mergeTables(new, old)
      new = deep_copy(new)
      old = deep_copy(old)
      # make a true map
      new_m = Builtins.listmap(new) do |e|
        {
          Ops.get_string(e, "key", "") => [
            Ops.get_string(e, "comment", ""),
            Ops.get_string(e, "value", "")
          ]
        }
      end
      # update old values
      replaced = Builtins.maplist(old) do |e|
        k = Ops.get_string(e, "key", "")
        cv = Ops.get_list(new_m, k, [])
        if Ops.greater_than(Builtins.size(cv), 0)
          Ops.set(new_m, k, []) # ok, newly constructed
          next {
            "key"     => k,
            "comment" => Ops.get_string(cv, 0, ""),
            "value"   => Ops.get_string(cv, 1, "")
          }
        else
          next {
            "key"     => k,
            "comment" => Ops.get_string(e, "comment", ""),
            "value"   => Ops.get_string(e, "value", "")
          }
        end
      end
      # remove already updated values
      filtered = Builtins.filter(new) do |e|
        Ops.greater_than(
          Builtins.size(Ops.get_list(new_m, Ops.get_string(e, "key", ""), [])),
          0
        )
      end
      Builtins.flatten([replaced, filtered])
    end

    # Part of Write.
    # @return success
    # @see #SetRootAlias
    def WriteAliases
      a_raw = Builtins.maplist(MergeRootAlias(@aliases)) do |e|
        {
          "comment" => Ops.get_string(e, "comment", ""),
          "key"     => Ops.get_string(e, "alias", ""),
          "value"   => Ops.get_string(e, "destinations", "")
        }
      end
      MailTable.Write("aliases", a_raw)
      true
    end

    # ----------------------------------------------------------------

    # For use by the Users package.
    # Does not rely on the internal state, first calls the agent.
    # @return eg. "joe, \\root", "" if not defined
    def GetRootAlias
      return "" if !ReadAliases()

      @root_alias
    end

    # For use by the Users package.
    # Does not use the internal state, just calls the agent.
    # SuSEconfig or newaliases is NOT called!
    # (TODO: what if it is called while the main module is running?)
    # Errors are reported via Report::Error.
    # @param [String] destinations The new alias. If "", it is removed.
    # @return true on success
    def SetRootAlias(destinations)
      return false if !ReadAliases()

      @root_alias = destinations
      @root_alias_comment = "" # TODO: "created by the ... yast2 module"?

      return false if !WriteAliases()

      return false if !MailTable.Flush("aliases")

      true
    end

    publish variable: :aliases, type: "list <map>"
    publish variable: :root_alias, type: "string"
    publish function: :FilterRootAlias, type: "void ()"
    publish function: :ReadAliases, type: "boolean ()"
    publish function: :MergeRootAlias, type: "list <map> (list <map>)"
    publish function: :WriteAliases, type: "boolean ()"
    publish function: :GetRootAlias, type: "string ()"
    publish function: :SetRootAlias, type: "boolean (string)"
  end

  MailAliases = MailAliasesClass.new
  MailAliases.main
end