# Part of the A-A-P project: Action execution module

# Copyright (C) 2002-2003 Stichting NLnet Labs
# Permission to copy and use this file is specified in the file COPYING.
# If this file is missing you can find it here: http://www.a-a-p.org/COPYING

#
# This module selects commands to be executed for an action on a file 
#
# EXTERNAL INTERFACE:
#
# action_run(recdict, action, dict, fname)
#                               Detects the type of file "fname", selects
#                               the application to do "action" and runs it.
#                               "dict" has attributes for customization.
#
# action_add(rpstack, recdict, args, commands)
#                               Add a connection between action-filetype to
#                               commands
#
# action_get_list()             Return a dictionary that lists all available
#                               actions.
#

import string
import os
import os.path

from Util import *
from Dictlist import listitem2str, str2dictlist, dictlist2str
from ParsePos import ParsePos
from Process import Process, recipe_error
from Filetype import filetype_root, ft_known
from Commands import expand
from Scope import get_build_recdict
import RecPython

# All actions are stored in this dictionary.
# Key is the name of the action.  Value is a list of Action objects.
# When multiple actions match, the last action in the list is preferred over
# earlier actions.
# Use this as:
#    _action_dict[action].append(Action())
_action_dict = {}

class Action:
    """
    Class to store the commands and rpstack for an action/filetype.
    In a derived class get_in_types() and get_out_types() may be implemented
    differently.
    """
    def __init__(self, rpstack, recdict, attributes, commands,
                            outtypes = [], intypes = [],
                            defer_var_names = [],
                            outfilename = ''):
        self.rpstack = rpstack
        self.buildrecdict = recdict
        self.attributes = attributes
        self.commands = commands
        self.outtypes = outtypes
        self.intypes = intypes
        self.primary = attributes.get("primary", 0)
        self.defer_var_names = defer_var_names
        self.outfilename = outfilename

    def get_out_types(self, source = "", intype = None):
        """
        Return list of supported output filetypes.
        When "intype" is given, this is the required output type.
        """
        aname = self.defer_action_name(source = source)
        if aname:
            return get_ftypes(aname, 0, intype = intype)
        if intype and intype not in self.intypes:
            return []
        return self.outtypes

    def get_in_types(self, source = "", outtype = None):
        """
        Return list of supported input filetypes.
        When "outtype" is given, this is the required output type.
        """
        aname = self.defer_action_name(source = source)
        if aname:
            return get_ftypes(aname, 1, outtype = outtype)
        if outtype and outtype not in self.outtypes:
            return []
        return self.intypes

    def defer_action_name(self, source = ""):
        """Return name of action to defer the work to."""
        for n in self.defer_var_names:
            if len(n) > 2 and n[0] == '{' and n[-1] == '}':
                aname = RecPython.get_var_attr(source, n[1:-1])
            else:
                aname = get_var_val(0, Global.globals, "_no", n)
            if aname:
                return aname
        return None

    def __str__(self):
        return self.commands


def action_add(rpstack, recdict, arglist, commands):
    """
    Add "commands" for an action-filetype combination defined by dictlist
    "arglist".
    "rpstack" is used for an error message when executing the commdands.
    """

    if len(arglist) == 2:
        outtypes = ["default"]
        intypes = string.split(arglist[1]["name"], ',')
    elif len(arglist) == 3:
        outtypes = string.split(arglist[1]["name"], ',')
        intypes = string.split(arglist[2]["name"], ',')
    else:
        recipe_error(rpstack, _(':action must have two or three arguments'))

    # Check that the filetype names exist
    for i in intypes + outtypes:
        if i != "default" and not ft_known(i):
            msg_warning(recdict, _('unknown filetype "%s" in :action command; recipe %s line %d') % (i, rpstack[-1].name, rpstack[-1].line_nr))

    act = Action(rpstack, recdict, arglist[0], commands, outtypes, intypes)
    
    for action in string.split(arglist[0]["name"], ','):
        action_add_to_dict(action, act)


def action_add_to_dict(action, act):
    """
    Add action object "act" to the dictionary of actions.
    """
    global _action_dict
    if not _action_dict.has_key(action):
        _action_dict[action] = []
    _action_dict[action].append(act)


def find_action(action, intype, outtype):
    """
    Find the last action that supports intype and outtype.
    """
    act = None
    for a in _action_dict[action]:
        if intype in a.get_in_types(outtype = outtype) and outtype in a.get_out_types(intype = intype):
            act = a;
    return act


def find_action_route(in_ftype, out_ftype):
    """
    Find a route from "in_ftype" to "out_ftype" with primary actions.
    Returns a route object or None.
    """
    # Use a list of dictionaries.  Each dictionary holds info about one
    # incomplete route:
    #    type       : input for the next action to be found
    #    actions    : list of action lines of the steps found so far
    #    used_types : list of intermediate types used in the steps
    trylist = [ {"type" : in_ftype, "actions" : [], "used_types" : [[in_ftype]]} ]

    # Try up to ten steps deep before giving up.
    depth = 0
    while trylist and depth < 10:
        new_trylist = []
        for ent in trylist:
            # Find actions that uses the type of this entry as input
            for action in _action_dict.keys():
              for act in _action_dict[action]:
                if act.primary and ent["type"] in act.get_in_types():
                    for atype in act.get_out_types():
                        if atype == out_ftype:
                            # Found a route!
                            from Work import Route
                            from RecPos import RecPos

                            actions = ent["actions"] + [action]
                            return Route(Global.globals,
                                    [RecPos("route from primary actions")], 0,
                                    ent["used_types"] + [[out_ftype]],
                                    {},
                                    actions, map(lambda x: 0, actions))

                        # No direct route, may add to list for next try.
                        if atype != "default" and act.outfilename:
                            # Make the line look like what ":route" uses.
                            actions = ent["actions"] + [action + " " + act.outfilename]
                            new_trylist.append({"type" : atype,
                                        "actions" : actions,
                                        "used_types" : ent["used_types"] + [[atype]]})
        depth = depth + 1
        trylist = new_trylist

    return None


def find_primary_action(action):
    """
    Find the prinary action with name "action".
    """
    act = None
    for a in _action_dict[action]:
        if a.primary:
            act = a;
    return act


def get_ftypes(action, in_types, intype = None, outtype = None):
    """
    Return the list of filetypes supported by actions with name "action".
    If "intype" given this input type must be supported.
    If "outtype" given this output type must be supported.
    """
    retval = []
    if _action_dict.has_key(action):
        for act in _action_dict[action]:
            if in_types:
                l = act.get_in_types(outtype = outtype)
            else:
                l = act.get_out_types(intype = intype)
            for i in l:
                if i not in retval:
                    retval.append(i)
    return retval


def action_get_list():
    """
    Return the dictionary that lists all actions.
    This turns _action_dict[] into another kind of dictionary for backwards
    compatibility.
    The dictionary key is the action name, the item is the intype
    dictionary.
    The intype dictionary key is the input file type, the item is the outtype
    dictionary.
    The outtype dictionary key is the output file type, the item is an Action
    object.
    retval[action-name][input-filetype][output-filetype].
    """
    retval = {}
    for action in _action_dict.keys():
        retval[action] = {}
        for act in _action_dict[action]:
            for intype in act.get_in_types():
                if not retval[action].has_key(intype):
                    retval[action][intype] = {}
                for outtype in act.get_out_types():
                    retval[action][intype][outtype] = act

    return retval


def action_ftype(recdict, action, dict):
    """
    Decide what filetype to use for "action" for the file specified with
    dictionary "dict".
    """
    if dict.has_key("filetype"):
        return dict["filetype"]

    from Work import getwork
    fname = dict["name"]
    node = getwork(recdict).find_node(fname)
    if node and node.attributes.has_key("filetype"):
        return node.attributes["filetype"]

    # A ":program", ":produce" etc. adds a "filetypehint" attribute that has a
    # lower priority than a "filetype" attribute the user specifies.
    if dict.has_key("filetypehint"):
        return dict["filetypehint"]

    # For viewing a remote file the filetype doesn't really matter, need to
    # use a browser anyway.
    # TODO: expand list of methods
    i = string.find(fname, "://")
    if (i > 0 and fname[:i] in [ "http", "https", "ftp" ]
                                                     and action == "view"):
        return "html"

    # Detect the filetype
    from Filetype import ft_detect
    return ft_detect(fname, 1, recdict)


def has_action(recdict, action, dict):
    """
    Return non-zero if "action" is defined for the file specified with "dict".
    Does not use the default.
    """
    if not _action_dict.has_key(action):
        return 0
    intype = action_ftype(recdict, action, dict)
    for act in _action_dict[action]:
        if intype in act.get_in_types():
            return 1
    return 0


def find_depend_action(ftype):
    """Find the depend action for a source file with type "ftype"."""
    if not _action_dict.has_key("depend"):
        return None

    found_act = None
    for intype in [ftype, filetype_root(ftype), "default"]:
        for act in _action_dict["depend"]:
            if intype in act.get_in_types():
                found_act = act
        if found_act:
            break

    # TODO: Should we check for the "default" outtype in the loop above?
    if found_act and "default" in found_act.get_out_types():
        return found_act

    return None


def find_out_ftype(recdict, actlist, action, in_ftype, out_ftype, msg):
    """
    Find the output filetype to be used for "action" and "in_ftype".
    Use only the actions in the list "actlist".
    Use the first one of "out_ftype", its root or "default" that is defined.
    Return the output filetype and non-zero if found, otherwise anything and
    zero.
    """
    if out_ftype:
        # Find an action that supports "out_ftype".
        for act in actlist:
            if out_ftype in act.get_out_types():
                return out_ftype, 1

        # Find an action that supports the root of "out_ftype".
        root = filetype_root(out_ftype)
        for act in actlist:
            if root in act.get_out_types():
                return root, 1

    # Find an action that supports "default".
    for act in actlist:
        if "default" in act.get_out_types():
            if msg:
                msg_extra(recdict, _('Using default for %s %s from %s')
                                               % (action, out_ftype, in_ftype))
            return "default", 1

    return out_ftype, 0

# For pychecker
find_type = None


def action_find(recdict, action, in_ftype, out_ftype, msg):
    """
    Find the action to be used for "action", with detected input filetype
    "in_ftype" and detected output filetype "out_ftype".
    When "msg" is non-zero give a message about using a default type.
    """
    # Try three input filetypes and three output filetypes for each of them.
    # If no commands defined for the detected filetype, use the root filetype.
    # If still no commands defined, use the default action.
    use_in_ftype = in_ftype
    use_out_ftype = out_ftype
    found = 0

    # Find_type must be global for this to work in Python 1.5.
    global find_type
    find_type = in_ftype
    actlist = filter(lambda act: find_type in act.get_in_types(),
                                                          _action_dict[action])
    if actlist:
        use_out_ftype, found = find_out_ftype(recdict, actlist,
                                              action, in_ftype, out_ftype, msg)

    if not found:
        find_type = filetype_root(in_ftype)
        actlist = filter(lambda act: find_type in act.get_in_types(),
                                                          _action_dict[action])
        if actlist:
            use_in_ftype = find_type
            use_out_ftype, found = find_out_ftype(recdict, actlist,
                                             action, find_type, out_ftype, msg)

    if not found:
        actlist = filter(lambda act: "default" in act.get_in_types(),
                                                          _action_dict[action])
        if actlist:
            use_in_ftype = "default"
            use_out_ftype, found = find_out_ftype(recdict, actlist, 
                                          action, use_in_ftype, out_ftype, msg)
            if msg and found and use_out_ftype != "default":
                msg_extra(recdict, _('Using default for %s from %s')
                                                          % (action, in_ftype))

    if not found:
        return None, None
    return use_in_ftype, use_out_ftype


def get_vars_from_attr(fromdict, todict, savedict = None):
    """
    Set variables from attributes:
    - Take all the entries in "fromdict" that start with "var_" and copy the
      value to "todict".
    - Take all the entries in "fromdict" that start with "add_" and append the
      value to a variable in "todict".
    If "savedict" is given, save the old value in it.
    """
    for k in fromdict.keys():
        if len(k) > 4 and (k[:4] == "var_" or k[:4] == "add_"):
            varname = k[4:]
            val = fromdict[k]
            if k[0] == 'a':
                oldval = get_var_val(0, todict, "_no", varname)
                if oldval:
                    # If the value already appears, don't append it again.
                    if string.find(' ' + oldval + ' ', ' ' + val + ' ') >= 0:
                        val = oldval
                    else:
                        val = oldval + ' ' + val
            if savedict:
                savedict[varname] = todict.get(varname)
            todict[varname] = val


def action_run(recdict, args):
    """Run the associated program for action "args[0]" on a list of files
       "args[1:]".  "args" is a dictlist.
       Use the attributes in "args[0]" to customize the action.
       When the "filetype" attribute isn't specified, detect it from args[1].
       Returns None for success, an error message for failure."""
    action = args[0]["name"]
    if not _action_dict.has_key(action):
        return _("Unknown action: %s") % action

    in_ftype = action_ftype(recdict, action, args[1])
    if not in_ftype:
        msg_note(recdict,
              _('Warning; Filetype not recognized for "%s", using "default"')
                                                             % args[1]["name"])
        in_ftype = "default"

    out_ftype = args[0].get("filetype")
    if not out_ftype:
        if args[0].has_key("target"):
            t = args[0]["target"]
        elif recdict.has_key("target"):
            t = recdict["target"]
        else:
            t = None
        if t:
            try:
                out_ftype = action_ftype(recdict, action,
                                                        str2dictlist([], t)[0])
            except UserError, e:
                return _('Error parsing target attribute: ') + str(e)

    use_in_ftype, use_out_ftype = action_find(recdict, action, in_ftype,
                                                                  out_ftype, 1)

    if not use_in_ftype:
        return (_("No commands defined for %s %s from %s")
                                               % (action, out_ftype, in_ftype))
        
    msg_extra(recdict, _('Do %s %s -> %s')
                                       % (action, use_in_ftype, use_out_ftype))
    act = find_action(action, use_in_ftype, use_out_ftype)
    if not act:
        msg_error(recdict, _("Internal error: action_run() didn't find action"))

    # Make a new scope for the the command block.
    new_recdict = get_build_recdict(recdict, act.buildrecdict,
                                 keep_current_scope = 1, rpstack = act.rpstack,
                                 xscope = args[0].get("scope"))

    # Take care of local and build command variables.

    # $source is the whole list of filenames.  Need to use quotes around items
    # with white space.
    # $fname is the first file name
    new_recdict["source"] = dictlist2str(args[1:])
    new_recdict["fname"] = listitem2str(args[1]["name"])

    # $filetype is the detected input filetype
    # $targettype is the detected output filetype
    # $action is the name of the action
    new_recdict["filetype"] = in_ftype
    new_recdict["targettype"] = out_ftype
    new_recdict["action"] = action

    # If the action is to be deferred to another action, set $DEFER_ACTION_NAME
    # to the name of that action.
    new_recdict["DEFER_ACTION_NAME"] = act.defer_action_name(source = new_recdict["source"])

    # Turn the attributes on the action into variables.
    # Both the full name and "var_" names.
    # Skip the "scope" attribute, it's handled above.
    for k in args[0].keys():
        if len(k) <= 4 or (k[:4] != "var_" and k[:4] != "add_"
                                 and k not in ["scope", "filetype", "remove"]):
            new_recdict[k] = args[0][k]
    get_vars_from_attr(args[0], new_recdict)

    from Work import getwork

    # First use "var_" and "add_" attributes of the node.
    # Turn the "var_" and "add_" attributes of the sources into variables.
    for s in args[1:]:
        node = getwork(recdict).find_node(s["name"])
        if node:
            get_vars_from_attr(node.attributes, new_recdict)
        get_vars_from_attr(s, new_recdict)

    # Also use "var_" and "add_" attributes from the target.
    if new_recdict.has_key("target"):
        for d in str2dictlist([], new_recdict["target"]):
            get_vars_from_attr(d, new_recdict)

    # Create a ParsePos object to contain the parse position in the string.
    fp = ParsePos(act.rpstack, string = act.commands)

    #
    # Parse and execute the commands.
    #
    try:
        Process(fp, new_recdict, 0)
    except UserError, e:
        return ((_('Error executing commands for %s %s: ')
                                            % (action, use_in_ftype)) + str(e))

    return None


def action_expand_do(recdict, commands, targetlist, sourcelist):
    """Expand ":do" commands in "commands" to the build commands they stand
       for.  Used for computing the checksum, not for executing the commands!.
       The filetype is sometimes guessed."""
    # Return quickly when there is no ":do" command.
    if string.find(commands, ":do") < 0:
        return commands

    # Remember the end of an expanded action.  It is not allowed to expand
    # again before this position, it would mean an action invokes itself and
    # causes an endless loop.
    exp_action_end = {}

    # Locate each ":do" command.  When we can find the commands of the action,
    # replace the ":do" command with them.
    idx = 0
    while 1:
        # Find the next ":do" command.
        do = string.find(commands, ":do", idx)
        if do < 0:
            break

        # Get the arguments of the ":do" command; advance "idx" to the end.
        i = skip_white(commands, do + 3)
        idx = string.find(commands, '\n', i)
        args = str2dictlist([], commands[i:idx])
        if len(args) >= 2:
            # The action is the first argument.
            action = args[0]["name"]

            if _action_dict.has_key(action):
                # Guess the input filetype to be used:
                # 1. an explicitly defined filetype after the action.
                # 2. if the first filename doesn't contain a $, get the
                #    filetype from it.
                # 3. use the filetype from the first source item
                if args[1].has_key("filetype"):
                    in_ftype = args[1]["filetype"]
                elif not '$' in args[1]["name"]:
                    in_ftype = action_ftype(recdict, action, args[1])
                else:
                    in_ftype = None
                if not in_ftype and sourcelist:
                    # Get the filetype of the first source item.
                    in_ftype = action_ftype(recdict, action, sourcelist[0])
                if not in_ftype:
                    in_ftype = "default"

                # Guess the output filetype to be used:
                # 1. an explicitly defined filetype after the target.
                # 2. if the first filename doesn't contain a $, get the
                #    filetype from it.
                # 3. use the filetype from the first source item
                tt = recdict.get("target")
                try:
                    recdict["target"] = targetlist[0]["name"]
                    target = str2dictlist([], expand(0, recdict,
                            args[0]["target"], Expand(1, Expand.quote_aap)))[0]
                except:
                    target = targetlist[0]
                if recdict.has_key("target"):
                    if tt is None:
                        del recdict["target"]
                    else:
                        recdict["target"] = tt
                if target.has_key("filetype"):
                    out_ftype = target["filetype"]
                elif not '$' in target["name"]:
                    out_ftype = action_ftype(recdict, action, target)
                else:
                    out_ftype = None
                if not out_ftype:
                    out_ftype = "default"

                # Find the action to be used for these filetypes, falling back
                # to "default" when necessary.
                in_ftype, out_ftype = action_find(recdict, action,
                                                        in_ftype, out_ftype, 0)
                                        
                if in_ftype:
                    # Use the buildcheck attribute of the Action if present,
                    # use the commands otherwise.
                    act = find_action(action, in_ftype, out_ftype)
                    cmd = act.attributes.get("buildcheck")
                    if not cmd:
                        cmd = act.commands
                    # Check if this action isn't used recursively.
                    key = action + '@' + in_ftype + '@' + out_ftype
                    if (exp_action_end.has_key(key)
                                             and exp_action_end[key] > do):
                        # :do command starts before expanded commands.
                        act = None

                    if act:
                        # Correct the end for the already expanded actions.
                        for k in exp_action_end.keys():
                            if exp_action_end[k] > idx:
                                exp_action_end[k] = (exp_action_end[k]
                                                   + len(cmd) - (idx - do))

                        # Set the end of the currently expanded action.
                        exp_action_end[key] = do + len(cmd)

                        # Replace the ":do" command with the commands of
                        # the action
                        commands = commands[:do] + cmd + commands[idx:]

                        # Continue at the start of the replaced commands,
                        # so that it works recursively.
                        idx = do

    return commands


# vim: set sw=4 et sts=4 tw=79 fo+=l:


syntax highlighted by Code2HTML, v. 0.9.1