Logo Search packages:      
Sourcecode: aegis version File versions

aemt.c

/*
 *    aegis - project change supervisor
 *    Copyright (C) 2002 Peter Miller;
 *    All rights reserved.
 *
 *    This program is free software; you can redistribute it and/or modify
 *    it under the terms of the GNU General Public License as published by
 *    the Free Software Foundation; either version 2 of the License, or
 *    (at your option) any later version.
 *
 *    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, write to the Free Software
 *    Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111, USA.
 *
 * MANIFEST: functions to manipulate aemts
 */

#include <ac/stdio.h>

#include <ael/project/files.h>
#include <aemt.h>
#include <arglex2.h>
#include <change.h>
#include <change/branch.h>
#include <change/file.h>
#include <commit.h>
#include <error.h>
#include <file.h>
#include <help.h>
#include <lock.h>
#include <log.h>
#include <os.h>
#include <progname.h>
#include <project.h>
#include <project/file.h>
#include <str_list.h>
#include <sub.h>
#include <trace.h>
#include <user.h>


/*
 * NAME
 *    make_transparent_usage
 *
 * SYNOPSIS
 *    void make_transparent_usage(void);
 *
 * DESCRIPTION
 *    The make_transparent_usage function is used to
 *    tell the user how to use the 'aegis -Make_Transparent' command.
 */

static void
make_transparent_usage(void)
{
    char        *progname;

    progname = progname_get();
    fprintf
    (
      stderr,
      "usage: %s -Make_Transparent <filename>... [ <option>... ]\n",
      progname
    );
    fprintf
    (
      stderr,
      "       %s -Make_Transparent -List [ <option>... ]\n",
      progname
    );
    fprintf(stderr, "       %s -Make_Transparent -Help\n", progname);
    quit(1);
}


/*
 * NAME
 *    make_transparent_help
 *
 * SYNOPSIS
 *    void make_transparent_help(void);
 *
 * DESCRIPTION
 *    The make_transparent_help function is used to
 *    describe the 'aegis -Make_Transparent' command to the user.
 */

static void
make_transparent_help(void)
{
    help("aemt", make_transparent_usage);
}


/*
 * NAME
 *    make_transparent_list
 *
 * SYNOPSIS
 *    void make_transparent_list(void);
 *
 * DESCRIPTION
 *    The make_transparent_list function is used to
 *    list the file the user may wish to add to the change
 *    as a transparency.  All project files are listed.
 */

static void
make_transparent_list(void)
{
    string_ty         *project_name;
    long        change_number;

    trace(("make_transparent_list()\n{\n"));
    project_name = 0;
    change_number = 0;
    arglex();
    while (arglex_token != arglex_token_eoln)
    {
      switch (arglex_token)
      {
      default:
          generic_argument(make_transparent_usage);
          continue;

      case arglex_token_change:
          if (arglex() != arglex_token_number)
          {
            option_needs_number
            (
                arglex_token_change,
                make_transparent_usage
            );
          }
          /* fall through... */

      case arglex_token_number:
          if (change_number)
          {
            duplicate_option_by_name
            (
                arglex_token_change,
                make_transparent_usage
            );
          }
          change_number = arglex_value.alv_number;
          if (change_number == 0)
            change_number = MAGIC_ZERO;
          else if (change_number < 1)
          {
            sub_context_ty    *scp;

            scp = sub_context_new();
            sub_var_set_long(scp, "Number", change_number);
            fatal_intl(scp, i18n("change $number out of range"));
            /* NOTREACHED */
            sub_context_delete(scp);
          }
          break;

      case arglex_token_project:
          if (arglex() != arglex_token_string)
            option_needs_name(arglex_token_project, make_transparent_usage);
          if (project_name)
          {
            duplicate_option_by_name
            (
                arglex_token_project,
                make_transparent_usage
            );
          }
          project_name = str_from_c(arglex_value.alv_string);
          break;
      }
      arglex();
    }
    list_project_files(project_name, change_number);
    if (project_name)
      str_free(project_name);
    trace(("}\n"));
}


/*
 * NAME
 *    make_transparent_main
 *
 * SYNOPSIS
 *    void make_transparent_main(void);
 *
 * DESCRIPTION
 *    The make_transparent_main function is used to add a file to a
 *    change as a transparency (like aecpu for a branch).  The file
 *    will be elided from the branch on successful integration of
 *    the change.
 *
 *    The names of the relevant files are gleaned from the command line.
 */

static void
make_transparent_main(void)
{
    string_list_ty  wl;
    string_list_ty  wl2;
    string_list_ty  wl_in;
    string_ty         *s1;
    string_ty         *s2;
    cstate      cstate_data;
    fstate_src        c_src_data;
    fstate_src        b_src_data;
    size_t      j;
    size_t      k;
    string_ty         *project_name;
    project_ty        *pp;
    long        change_number;
    change_ty         *cp;
    change_ty         *bcp;
    log_style_ty    log_style;
    user_ty     *up;
    string_ty         *dd;
    int               number_of_errors;
    string_list_ty  search_path;
    int               based;
    string_ty         *base;
    sub_context_ty  *scp;

    trace(("make_transparent_main()\n{\n"));
    arglex();
    project_name = 0;
    change_number = 0;
    string_list_constructor(&wl);
    log_style = log_style_append_default;
    while (arglex_token != arglex_token_eoln)
    {
      switch (arglex_token)
      {
      default:
          generic_argument(make_transparent_usage);
          continue;

      case arglex_token_file:
      case arglex_token_directory:
          if (arglex() != arglex_token_string)
            make_transparent_usage();
          /* fall through... */

      case arglex_token_string:
          s2 = str_from_c(arglex_value.alv_string);
          string_list_append(&wl, s2);
          str_free(s2);
          break;

      case arglex_token_change:
          if (arglex() != arglex_token_number)
          {
            option_needs_number
            (
                arglex_token_change,
                make_transparent_usage
            );
          }
          /* fall through... */

      case arglex_token_number:
          if (change_number)
          {
            duplicate_option_by_name
            (
                arglex_token_change,
                make_transparent_usage
            );
          }
          change_number = arglex_value.alv_number;
          if (change_number == 0)
            change_number = MAGIC_ZERO;
          else if (change_number < 1)
          {
            scp = sub_context_new();
            sub_var_set_long(scp, "Name", change_number);
            fatal_intl(scp, i18n("change $number out of range"));
            /* NOTREACHED */
            sub_context_delete(scp);
          }
          break;

      case arglex_token_project:
          if (arglex() != arglex_token_string)
            option_needs_name(arglex_token_project, make_transparent_usage);
          if (project_name)
          {
            duplicate_option_by_name
            (
                arglex_token_project,
                make_transparent_usage
            );
          }
          project_name = str_from_c(arglex_value.alv_string);
          break;

      case arglex_token_nolog:
          if (log_style == log_style_none)
            duplicate_option(make_transparent_usage);
          log_style = log_style_none;
          break;

      case arglex_token_wait:
      case arglex_token_wait_not:
          user_lock_wait_argument(make_transparent_usage);
          break;

      case arglex_token_whiteout:
      case arglex_token_whiteout_not:
          user_whiteout_argument(make_transparent_usage);
          break;

      case arglex_token_base_relative:
      case arglex_token_current_relative:
          user_relative_filename_preference_argument(make_transparent_usage);
          break;
      }
      arglex();
    }
    if (!wl.nstrings)
    {
      error_intl(0, i18n("no file names"));
      make_transparent_usage();
    }

    /*
     * locate project data
     */
    if (!project_name)
      project_name = user_default_project();
    pp = project_alloc(project_name);
    str_free(project_name);
    project_bind_existing(pp);

    /*
     * locate user data
     */
    up = user_executing(pp);

    /*
     * locate change data
     */
    if (!change_number)
      change_number = user_default_change(up);
    cp = change_alloc(pp, change_number);
    change_bind_existing(cp);

    /*
     * take the locks and read the change state
     */
    change_cstate_lock_prepare(cp);
    lock_take();
    cstate_data = change_cstate_get(cp);

    log_open(change_logfile_get(cp), up, log_style);

    /*
     * It is an error if the change is not in the in_development state.
     * It is an error if the change is not assigned to the current user.
     */
    if (cstate_data->state != cstate_state_being_developed)
      change_fatal(cp, 0, i18n("make transparent bad state"));
    if (change_is_a_branch(cp))
      change_fatal(cp, 0, i18n("bad nf branch"));
    if (!str_equal(change_developer_name(cp), user_name(up)))
      change_fatal(cp, 0, i18n("not developer"));

    /*
     * Where to search to resolve file names.
     */
    change_search_path_get(cp, &search_path, 1);

    /*
     * Find the base for relative filenames.
     */
    based =
      (
          search_path.nstrings >= 1
      &&
          (
            user_relative_filename_preference
            (
                up,
                uconf_relative_filename_preference_current
            )
          ==
            uconf_relative_filename_preference_base
          )
      );
    if (based)
      base = search_path.string[0];
    else
    {
      os_become_orig();
      base = os_curdir();
      os_become_undo();
    }

    /*
     * resolve the path of each file
     * 1.   the absolute path of the file name is obtained
     * 2.   if the file is inside the development directory, ok
     * 3.   if the file is inside the baseline, ok
     * 4.   if neither, error
     */
    bcp = project_change_get(pp);
    number_of_errors = 0;
    string_list_constructor(&wl2);
    for (j = 0; j < wl.nstrings; ++j)
    {
      s1 = wl.string[j];
      if (s1->str_text[0] == '/')
          s2 = str_copy(s1);
      else
          s2 = str_format("%S/%S", base, s1);
      user_become(up);
      s1 = os_pathname(s2, 1);
      user_become_undo();
      str_free(s2);
      s2 = 0;
      for (k = 0; k < search_path.nstrings; ++k)
      {
          s2 = os_below_dir(search_path.string[k], s1);
          if (s2)
            break;
      }
      str_free(s1);
      if (!s2)
      {
          scp = sub_context_new();
          sub_var_set_string(scp, "File_Name", wl.string[j]);
          change_error(cp, scp, i18n("$filename unrelated"));
          sub_context_delete(scp);
          ++number_of_errors;
          continue;
      }
      change_file_directory_query(bcp, s2, &wl_in, 0);
      if (wl_in.nstrings)
      {
          int               used;

          /*
           * If the user named a directory, add all of the
           * source files in that directory, provided they
           * are not already in the change.
           */
          used = 0;
          for (k = 0; k < wl_in.nstrings; ++k)
          {
            string_ty   *s3;

            s3 = wl_in.string[k];
            if (!change_file_find(cp, s3))
            {
                if (string_list_member(&wl2, s3))
                {
                  scp = sub_context_new();
                  sub_var_set_string(scp, "File_Name", s3);
                  change_error(cp, scp, i18n("too many $filename"));
                  sub_context_delete(scp);
                  ++number_of_errors;
                }
                else
                  string_list_append(&wl2, s3);
                ++used;
            }
          }
          if (!used)
          {
            scp = sub_context_new();
            if (s2->str_length)
                sub_var_set_string(scp, "File_Name", s2);
            else
                sub_var_set_charstar(scp, "File_Name", ".");
            sub_var_set_long(scp, "Number", (long)wl_in.nstrings);
            sub_var_optional(scp, "Number");
            change_error
            (
                cp,
                scp,
                i18n("directory $filename contains no relevant files")
            );
            sub_context_delete(scp);
            ++number_of_errors;
          }
      }
      else
      {
          if (string_list_member(&wl2, s2))
          {
            scp = sub_context_new();
            sub_var_set_string(scp, "File_Name", s2);
            change_error(cp, scp, i18n("too many $filename"));
            sub_context_delete(scp);
            ++number_of_errors;
          }
          else
            string_list_append(&wl2, s2);
      }
      string_list_destructor(&wl_in);
      str_free(s2);
    }
    string_list_destructor(&wl);
    wl = wl2;
    string_list_destructor(&search_path);

    /*
     * ensure that each file
     * 1. is not already part of the change
     * 2. is being modified or removed by the branch
     * add it to the change
     */
    for (j = 0; j < wl.nstrings; ++j)
    {
      fstate_src      pp_src_data;

      s1 = wl.string[j];
      if (change_file_find(cp, s1))
      {
          scp = sub_context_new();
          sub_var_set_string(scp, "File_Name", s1);
          change_error(cp, scp, i18n("file $filename dup"));
          sub_context_delete(scp);
          ++number_of_errors;
          continue;
      }
      b_src_data = change_file_find(bcp, s1);
      if (!b_src_data)
      {
          b_src_data = change_file_find_fuzzy(bcp, s1);
          if (b_src_data)
          {
            scp = sub_context_new();
            sub_var_set_string(scp, "File_Name", s1);
            sub_var_set_string(scp, "Guess", b_src_data->file_name);
            project_error(pp, scp, i18n("no $filename, closest is $guess"));
            sub_context_delete(scp);
            ++number_of_errors;
          }
      }

      /*
       * This is here to cope with cases where the users upgrade
       * from 4.9 to 4.10 with changes between being reviewed and
       * being integrated.
       */
      if
      (
          b_src_data
      &&
          (
                  b_src_data->about_to_be_created_by
          ||
                  b_src_data->about_to_be_copied_by
          )
      )
          b_src_data->action = file_action_transparent;

      if (!b_src_data || b_src_data->action == file_action_transparent)
      {
          scp = sub_context_new();
          sub_var_set_string(scp, "File_Name", s1);
          if
          (
            pp->parent
          &&
            project_file_find(pp->parent, s1, view_path_simple)
          )
          {
            project_error
            (
                pp->parent,
                scp,
                i18n("make $filename transparent, fail, too deep")
            );
          }
          else
          {
            change_error(bcp, scp, i18n("no $filename"));
          }
          sub_context_delete(scp);
          ++number_of_errors;
          continue;
      }

      pp_src_data = 0;
      switch (b_src_data->action)
      {
      case file_action_create:
      case file_action_modify:
      case file_action_remove:
          if (!pp->parent)
          {
            /*
             * You may not make a trunk file transparent.
             * What would you see through it?
             */
            scp = sub_context_new();
            sub_var_set_string(scp, "File_Name", s1);
            change_error
            (
                cp,
                scp,
                i18n("make $filename transparent, fail, trunk")
            );
            sub_context_delete(scp);
            ++number_of_errors;
            continue;
          }

          pp_src_data =
            project_file_find(pp->parent, s1, view_path_simple);
          if (!pp_src_data)
          {
            scp = sub_context_new();
            sub_var_set_string(scp, "File_Name", s1);
            project_error
            (
                pp->parent,
                scp,
                i18n("make $filename transparent, fail, too shallow")
            );
            sub_context_delete(scp);
            ++number_of_errors;
            continue;
          }
          break;

      case file_action_transparent:
      case file_action_insulate:
          /* not possible */
          assert(0);
          continue;
      }
      assert(pp_src_data);
      c_src_data = change_file_new(cp, s1);
      c_src_data->action = file_action_transparent;
      c_src_data->usage = pp_src_data->usage;

      /*
       * b_src_data->edit_number
       *    The head revision of the branch.
       * b_src_data->edit_number_origin
       *    The version originally copied.
       *
       * c_src_data->edit_number
       *    Not meaningful until after integrate pass.
       * c_src_data->edit_number_origin
       *    The version originally copied.
       * c_src_data->edit_number_origin_new
       *    Updates branch edit_number_origin on
       *    integrate pass.
       */
      assert(b_src_data->edit);
      assert(b_src_data->edit->revision);
      c_src_data->edit_origin = history_version_copy(b_src_data->edit);
    }
    if (number_of_errors)
    {
      scp = sub_context_new();
      sub_var_set_long(scp, "Number", number_of_errors);
      sub_var_optional(scp, "Number");
      change_fatal(cp, scp, i18n("remove file fail"));
      /* NOTREACHED */
      sub_context_delete(scp);
    }

    /*
     * Create each file in the development directory.  If the file exists
     * in the grandparent project, copy it from there, otherwise put
     * a whiteout file there.  Create any necessary directories along
     * the way.
     */
    dd = change_development_directory_get(cp, 0);
    for (j = 0; j < wl.nstrings; ++j)
    {
      fstate_src      pp_src_data;

      s1 = wl.string[j];

      c_src_data = change_file_find(cp, s1);
      assert(c_src_data);
      assert(pp->parent);
      pp_src_data = project_file_find(pp->parent, s1, view_path_extreme);
      if (pp_src_data)
      {
          string_ty       *from;
          string_ty       *to;
          int             mode;

          from = project_file_path(pp->parent, s1);
          assert(from);
          to = str_format("%S/%S", dd, s1);

          /*
           * copy the file
           */
          user_become(up);
          os_mkdir_between(dd, s1, 02755);
          if (os_exists(to))
            os_unlink(to);
          copy_whole_file(from, to, 0);

          /*
           * set the file mode
           */
          mode = 0444;
          if (pp_src_data->executable)
            mode |= 0111;
          mode &= ~change_umask(cp);
          os_chmod(to, mode);
          user_become_undo();

          /*
           * clean up afterwards
           */
          str_free(from);
          str_free(to);
      }
      else
      {
          /*
           * Remove any existing file (this cleans up junk, and
           * breaks the link if we are using symlink trees).
           */
          change_file_whiteout_write(cp, s1, up);
      }
    }

    /*
     * the number of files changed, or the version did,
     * so stomp on the validation fields.
     */
    change_build_times_clear(cp);

    /*
     * run the change file command
     * and the project file command if necessary
     */
#if 0
    change_run_make_transparent_command(cp, &wl, up);
#endif
    change_run_project_file_command(cp, up);

    /*
     * write the data and release the lock
     */
    change_cstate_write(cp);
    commit();
    lock_release();

    /*
     * verbose success message
     */
    scp = sub_context_new();
    sub_var_set_long(scp, "Number", (long)wl.nstrings);
    sub_var_optional(scp, "Number");
    change_verbose(cp, scp, i18n("make transparent complete"));
    sub_context_delete(scp);

    string_list_destructor(&wl);
    change_free(cp);
    project_free(pp);
    user_free(up);
    trace(("}\n"));
}


/*
 * NAME
 *    make_transparent
 *
 * SYNOPSIS
 *    void make_transparent(void);
 *
 * DESCRIPTION
 *    The make_transparent function is used to
 *    dispatch the 'aegis -Make_Transparent' command to the relevant
 *    function to do it's work.
 */

void
make_transparent(void)
{
    static arglex_dispatch_ty dispatch[] =
    {
      {arglex_token_help, make_transparent_help, },
      {arglex_token_list, make_transparent_list, },
    };

    trace(("make_transparent()\n{\n"));
    arglex_dispatch(dispatch, SIZEOF(dispatch), make_transparent_main);
    trace(("}\n"));
}

Generated by  Doxygen 1.6.0   Back to index