//
// aegis - project change supervisor
// Copyright (C) 2001-2008, 2012, 2014 Peter Miller
//
// 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 3 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, see
// .
//
#include
#include
#include
#include
#include
#include
#include
#include
void
change_run_patch_diff_command(change::pointer cp, user_ty::pointer up,
string_ty *original, string_ty *input_file_name, string_ty *output_filename,
string_ty *index_name)
{
sub_context_ty *scp;
pconf_ty *pconf_data;
string_ty *the_command;
//
// The difference of 2 files, to send around as a patch.
// (This isn't the same as diff_command, because it's aimed at
// GNU Patch, not at humans.)
//
// All of the substitutions described in aesub(5) are available.
// In addition
//
// ${Original}
// absolute path of original file copied from the baseline
// usually, but not always
//
// ${Input}
// absolute path of current file in the development directory
// usually, but not always
//
// ${Output}
// absolute path of file in which to write the
// patch difference listing usually in the development
// diretcory
//
// ${INDex}
// The project-relative path of file being differenced.
// This is the name to be used to index the file within the
// patch, to ensure that patch(1) applies the patch to the
// correct file. (You don't want it applying the patch to
// any of the absolute paths mentioned above, especially
// as they could well be temporary files created for the
// purpose of generating the patch.)
//
trace(("change_run_patch_diff_command(cp = %p, up = %p, "
"original = \"%s\", input_file_name = \"%s\", "
"output_filename = \"%s\")\n{\n", cp.get(), up.get(),
original->str_text, input_file_name->str_text,
output_filename->str_text));
assert(cp->valid());
pconf_data = change_pconf_get(cp, 1);
string_ty *dd = 0;
switch (cp->cstate_get()->state)
{
case cstate_state_being_developed:
case cstate_state_awaiting_review:
case cstate_state_being_reviewed:
case cstate_state_awaiting_integration:
dd = cp->development_directory_get(false);
break;
case cstate_state_being_integrated:
dd = cp->integration_directory_get(false);
break;
case cstate_state_awaiting_development:
case cstate_state_completed:
#ifndef DEBUG
default:
#endif
dd = os_tmpdir();
break;
}
assert(dd);
scp = sub_context_new();
sub_var_set_string(scp, "ORiginal", original);
sub_var_set_string(scp, "Input", input_file_name);
sub_var_set_string(scp, "Output", output_filename);
sub_var_set_string(scp, "INDex", index_name);
sub_var_optional(scp, "INDex");
the_command = pconf_data->patch_diff_command;
if (!the_command)
{
//
// This one defaults, without user attention.
// It's aimed at GNU Patch, not at a human.
//
the_command =
str_from_c
(
"set +e; "
"$diff "
#ifdef HAVE_GNU_DIFF
"-u "
"--text "
"-L ${quote $index} -L ${quote $index} "
#else
"-c "
#endif
"${quote $original} ${quote $input} > ${quote $output}; "
"test $$? -le 1"
);
}
the_command = substitute(scp, cp, the_command);
sub_context_delete(scp);
trace_string(the_command->str_text);
change_env_set(cp, 0);
user_ty::become scoped(up);
if (os_exists(output_filename))
os_unlink(output_filename);
os_execute(the_command, OS_EXEC_FLAG_NO_INPUT, dd);
str_free(the_command);
trace(("}\n"));
}
// vim: set ts=8 sw=4 et :