aboutsummaryrefslogtreecommitdiff
path: root/subversion/libsvn_client/patch.c
diff options
context:
space:
mode:
Diffstat (limited to 'subversion/libsvn_client/patch.c')
-rw-r--r--subversion/libsvn_client/patch.c1580
1 files changed, 1045 insertions, 535 deletions
diff --git a/subversion/libsvn_client/patch.c b/subversion/libsvn_client/patch.c
index 6d8d0a4632c8..1b2d86b1da94 100644
--- a/subversion/libsvn_client/patch.c
+++ b/subversion/libsvn_client/patch.c
@@ -46,6 +46,7 @@
#include "private/svn_eol_private.h"
#include "private/svn_wc_private.h"
#include "private/svn_dep_compat.h"
+#include "private/svn_diff_private.h"
#include "private/svn_string_private.h"
#include "private/svn_subr_private.h"
#include "private/svn_sorts_private.h"
@@ -66,7 +67,10 @@ typedef struct hunk_info_t {
/* The fuzz factor used when matching this hunk, i.e. how many
* lines of leading and trailing context to ignore during matching. */
- svn_linenum_t fuzz;
+ svn_linenum_t match_fuzz;
+
+ /* match_fuzz + the penalty caused by bad patch files */
+ svn_linenum_t report_fuzz;
} hunk_info_t;
/* A struct carrying information related to the patched and unpatched
@@ -152,6 +156,9 @@ typedef struct prop_patch_target_t {
* ### Should we use flags instead since we're not using all enum values? */
svn_diff_operation_kind_t operation;
+ /* When true the property change won't be applied */
+ svn_boolean_t skipped;
+
/* ### Here we'll add flags telling if the prop was added, deleted,
* ### had_rejects, had_local_mods prior to patching and so on. */
} prop_patch_target_t;
@@ -190,8 +197,8 @@ typedef struct patch_target_t {
/* Path to the patched file. */
const char *patched_path;
- /* Hunks that are rejected will be written to this file. */
- apr_file_t *reject_file;
+ /* Hunks that are rejected will be written to this stream. */
+ svn_stream_t *reject_stream;
/* Path to the reject file. */
const char *reject_path;
@@ -209,15 +216,23 @@ typedef struct patch_target_t {
/* True if the target had to be skipped for some reason. */
svn_boolean_t skipped;
+ /* True if the reason for skipping is a local obstruction */
+ svn_boolean_t obstructed;
+
/* True if at least one hunk was rejected. */
svn_boolean_t had_rejects;
/* True if at least one property hunk was rejected. */
svn_boolean_t had_prop_rejects;
- /* True if the target file had local modifications before the
- * patch was applied to it. */
- svn_boolean_t local_mods;
+ /* True if at least one hunk was handled as already applied */
+ svn_boolean_t had_already_applied;
+
+ /* True if at least one property hunk was handled as already applied */
+ svn_boolean_t had_prop_already_applied;
+
+ /* The operation on the target as set in the patch file */
+ svn_diff_operation_kind_t operation;
/* True if the target was added by the patch, which means that it did
* not exist on disk before patching and has content after patching. */
@@ -226,10 +241,6 @@ typedef struct patch_target_t {
/* True if the target ended up being deleted by the patch. */
svn_boolean_t deleted;
- /* True if the target ended up being replaced by the patch
- * (i.e. a new file was added on top locally deleted node). */
- svn_boolean_t replaced;
-
/* Set if the target is supposed to be moved by the patch.
* This applies to --git diffs which carry "rename from/to" headers. */
const char *move_target_abspath;
@@ -252,6 +263,10 @@ typedef struct patch_target_t {
/* A hash table of prop_patch_target_t objects keyed by property names. */
apr_hash_t *prop_targets;
+ /* When TRUE, this patch uses the raw git symlink format instead of the
+ Subversion internal style format where links start with 'link '. */
+ svn_boolean_t git_symlink_format;
+
} patch_target_t;
@@ -261,8 +276,58 @@ typedef struct patch_target_t {
typedef struct patch_target_info_t {
const char *local_abspath;
svn_boolean_t deleted;
+ svn_boolean_t added;
} patch_target_info_t;
+/* Check if LOCAL_ABSPATH is recorded as added in TARGETS_INFO */
+static svn_boolean_t
+target_is_added(const apr_array_header_t *targets_info,
+ const char *local_abspath,
+ apr_pool_t *scratch_pool)
+{
+ int i;
+
+ for (i = targets_info->nelts - 1; i >= 0; i--)
+ {
+ const patch_target_info_t *target_info =
+ APR_ARRAY_IDX(targets_info, i, const patch_target_info_t *);
+
+ const char *info = svn_dirent_skip_ancestor(target_info->local_abspath,
+ local_abspath);
+
+ if (info && !*info)
+ return target_info->added;
+ else if (info)
+ return FALSE;
+ }
+
+ return FALSE;
+}
+
+/* Check if LOCAL_ABSPATH or an ancestor is recorded as deleted in
+ TARGETS_INFO */
+static svn_boolean_t
+target_is_deleted(const apr_array_header_t *targets_info,
+ const char *local_abspath,
+ apr_pool_t *scratch_pool)
+{
+ int i;
+
+ for (i = targets_info->nelts - 1; i >= 0; i--)
+ {
+ const patch_target_info_t *target_info =
+ APR_ARRAY_IDX(targets_info, i, const patch_target_info_t *);
+
+ const char *info = svn_dirent_skip_ancestor(target_info->local_abspath,
+ local_abspath);
+
+ if (info)
+ return target_info->deleted;
+ }
+
+ return FALSE;
+}
+
/* Strip STRIP_COUNT components from the front of PATH, returning
* the result in *RESULT, allocated in RESULT_POOL.
@@ -368,18 +433,20 @@ obtain_eol_and_keywords_for_file(apr_hash_t **keywords,
* Indicate in TARGET->SKIPPED whether the target should be skipped.
* STRIP_COUNT specifies the number of leading path components
* which should be stripped from target paths in the patch.
- * PROP_CHANGES_ONLY specifies whether the target path is allowed to have
- * only property changes, and no content changes (in which case the target
- * must be a directory).
+ * HAS_TEXT_CHANGES specifies whether the target path will have some text
+ * changes applied, implying that the target should be a file and not a
+ * directory.
* Use RESULT_POOL for allocations of fields in TARGET.
* Use SCRATCH_POOL for all other allocations. */
static svn_error_t *
resolve_target_path(patch_target_t *target,
const char *path_from_patchfile,
- const char *wcroot_abspath,
+ const char *root_abspath,
int strip_count,
- svn_boolean_t prop_changes_only,
+ svn_boolean_t has_text_changes,
+ svn_boolean_t follow_moves,
svn_wc_context_t *wc_ctx,
+ const apr_array_header_t *targets_info,
apr_pool_t *result_pool,
apr_pool_t *scratch_pool)
{
@@ -391,8 +458,8 @@ resolve_target_path(patch_target_t *target,
target->canon_path_from_patchfile = svn_dirent_internal_style(
path_from_patchfile, result_pool);
- /* We allow properties to be set on the wc root dir. */
- if (! prop_changes_only && target->canon_path_from_patchfile[0] == '\0')
+ /* We can't handle text changes on the patch root dir. */
+ if (has_text_changes && target->canon_path_from_patchfile[0] == '\0')
{
/* An empty patch target path? What gives? Skip this. */
target->skipped = TRUE;
@@ -409,14 +476,14 @@ resolve_target_path(patch_target_t *target,
if (svn_dirent_is_absolute(stripped_path))
{
- target->local_relpath = svn_dirent_is_child(wcroot_abspath,
+ target->local_relpath = svn_dirent_is_child(root_abspath,
stripped_path,
result_pool);
if (! target->local_relpath)
{
/* The target path is either outside of the working copy
- * or it is the working copy itself. Skip it. */
+ * or it is the patch root itself. Skip it. */
target->skipped = TRUE;
target->local_abspath = NULL;
target->local_relpath = stripped_path;
@@ -429,9 +496,9 @@ resolve_target_path(patch_target_t *target,
}
/* Make sure the path is secure to use. We want the target to be inside
- * of the working copy and not be fooled by symlinks it might contain. */
+ * the locked tree and not be fooled by symlinks it might contain. */
SVN_ERR(svn_dirent_is_under_root(&under_root,
- &target->local_abspath, wcroot_abspath,
+ &target->local_abspath, root_abspath,
target->local_relpath, result_pool));
if (! under_root)
@@ -442,6 +509,13 @@ resolve_target_path(patch_target_t *target,
return SVN_NO_ERROR;
}
+ if (target_is_deleted(targets_info, target->local_abspath, scratch_pool))
+ {
+ target->locally_deleted = TRUE;
+ target->db_kind = svn_node_none;
+ return SVN_NO_ERROR;
+ }
+
/* Skip things we should not be messing with. */
err = svn_wc_status3(&status, wc_ctx, target->local_abspath,
result_pool, scratch_pool);
@@ -463,6 +537,7 @@ resolve_target_path(patch_target_t *target,
status->conflicted)
{
target->skipped = TRUE;
+ target->obstructed = TRUE;
return SVN_NO_ERROR;
}
else if (status->node_status == svn_wc_status_deleted)
@@ -481,20 +556,29 @@ resolve_target_path(patch_target_t *target,
if (target->locally_deleted)
{
- const char *moved_to_abspath;
+ const char *moved_to_abspath = NULL;
+
+ if (follow_moves
+ && !target_is_added(targets_info, target->local_abspath,
+ scratch_pool))
+ {
+ SVN_ERR(svn_wc__node_was_moved_away(&moved_to_abspath, NULL,
+ wc_ctx, target->local_abspath,
+ result_pool, scratch_pool));
+ }
- SVN_ERR(svn_wc__node_was_moved_away(&moved_to_abspath, NULL,
- wc_ctx, target->local_abspath,
- result_pool, scratch_pool));
- /* ### BUG: moved_to_abspath contains the target where the op-root was
- ### moved to... not the target itself! */
if (moved_to_abspath)
{
target->local_abspath = moved_to_abspath;
- target->local_relpath = svn_dirent_skip_ancestor(wcroot_abspath,
- moved_to_abspath);
- SVN_ERR_ASSERT(target->local_relpath &&
- target->local_relpath[0] != '\0');
+ target->local_relpath = svn_dirent_skip_ancestor(root_abspath,
+ moved_to_abspath);
+
+ if (!target->local_relpath || target->local_relpath[0] == '\0')
+ {
+ /* The target path is outside of the patch area. Skip it. */
+ target->skipped = TRUE;
+ return SVN_NO_ERROR;
+ }
/* As far as we are concerned this target is not locally deleted. */
target->locally_deleted = FALSE;
@@ -511,6 +595,22 @@ resolve_target_path(patch_target_t *target,
}
}
+#ifndef HAVE_SYMLINK
+ if (target->kind_on_disk == svn_node_file
+ && !target->is_symlink
+ && !target->locally_deleted
+ && status->prop_status != svn_wc_status_none)
+ {
+ const svn_string_t *value;
+
+ SVN_ERR(svn_wc_prop_get2(&value, wc_ctx, target->local_abspath,
+ SVN_PROP_SPECIAL, scratch_pool, scratch_pool));
+
+ if (value)
+ target->is_symlink = TRUE;
+ }
+#endif
+
return SVN_NO_ERROR;
}
@@ -540,7 +640,7 @@ readline_prop(void *baton, svn_stringbuf_t **line, const char **eol_str,
svn_boolean_t *eof, apr_pool_t *result_pool,
apr_pool_t *scratch_pool)
{
- prop_read_baton_t *b = (prop_read_baton_t *)baton;
+ prop_read_baton_t *b = baton;
svn_stringbuf_t *str = NULL;
const char *c;
svn_boolean_t found_eof;
@@ -593,7 +693,7 @@ readline_prop(void *baton, svn_stringbuf_t **line, const char **eol_str,
while (c < b->value->data + b->value->len);
if (eof)
- *eof = found_eof;
+ *eof = found_eof && !(str && str->len > 0);
*line = str;
return SVN_NO_ERROR;
@@ -605,7 +705,8 @@ readline_prop(void *baton, svn_stringbuf_t **line, const char **eol_str,
static svn_error_t *
tell_prop(void *baton, apr_off_t *offset, apr_pool_t *scratch_pool)
{
- prop_read_baton_t *b = (prop_read_baton_t *)baton;
+ prop_read_baton_t *b = baton;
+
*offset = b->offset;
return SVN_NO_ERROR;
}
@@ -615,7 +716,8 @@ tell_prop(void *baton, apr_off_t *offset, apr_pool_t *scratch_pool)
static svn_error_t *
seek_prop(void *baton, apr_off_t offset, apr_pool_t *scratch_pool)
{
- prop_read_baton_t *b = (prop_read_baton_t *)baton;
+ prop_read_baton_t *b = baton;
+
b->offset = offset;
return SVN_NO_ERROR;
}
@@ -626,7 +728,8 @@ static svn_error_t *
write_prop(void *baton, const char *buf, apr_size_t len,
apr_pool_t *scratch_pool)
{
- svn_stringbuf_t *patched_value = (svn_stringbuf_t *)baton;
+ svn_stringbuf_t *patched_value = baton;
+
svn_stringbuf_appendbytes(patched_value, buf, len);
return SVN_NO_ERROR;
}
@@ -638,6 +741,7 @@ write_prop(void *baton, const char *buf, apr_size_t len,
* Use SCRATCH_POOL for temporary allocations. */
static svn_error_t *
init_prop_target(prop_patch_target_t **prop_target,
+ const patch_target_t *target,
const char *prop_name,
svn_diff_operation_kind_t operation,
svn_wc_context_t *wc_ctx,
@@ -647,7 +751,6 @@ init_prop_target(prop_patch_target_t **prop_target,
prop_patch_target_t *new_prop_target;
target_content_t *content;
const svn_string_t *value;
- svn_error_t *err;
prop_read_baton_t *prop_read_baton;
content = apr_pcalloc(result_pool, sizeof(*content));
@@ -664,18 +767,12 @@ init_prop_target(prop_patch_target_t **prop_target,
new_prop_target->operation = operation;
new_prop_target->content = content;
- err = svn_wc_prop_get2(&value, wc_ctx, local_abspath, prop_name,
- result_pool, scratch_pool);
- if (err)
- {
- if (err->apr_err == SVN_ERR_WC_PATH_NOT_FOUND)
- {
- svn_error_clear(err);
- value = NULL;
- }
- else
- return svn_error_trace(err);
- }
+ if (!(target->deleted || target->db_kind == svn_node_none))
+ SVN_ERR(svn_wc_prop_get2(&value, wc_ctx, local_abspath, prop_name,
+ result_pool, scratch_pool));
+ else
+ value = NULL;
+
content->existed = (value != NULL);
new_prop_target->value = value;
new_prop_target->patched_value = svn_stringbuf_create_empty(result_pool);
@@ -716,71 +813,15 @@ readline_file(void *baton, svn_stringbuf_t **line, const char **eol_str,
svn_boolean_t *eof, apr_pool_t *result_pool,
apr_pool_t *scratch_pool)
{
- apr_file_t *file = (apr_file_t *)baton;
- svn_stringbuf_t *str = NULL;
- apr_size_t numbytes;
- char c;
- svn_boolean_t found_eof;
-
- /* Read bytes into STR up to and including, but not storing,
- * the next EOL sequence. */
- *eol_str = NULL;
- numbytes = 1;
- found_eof = FALSE;
- while (!found_eof)
- {
- SVN_ERR(svn_io_file_read_full2(file, &c, sizeof(c), &numbytes,
- &found_eof, scratch_pool));
- if (numbytes != 1)
- {
- found_eof = TRUE;
- break;
- }
-
- if (c == '\n')
- {
- *eol_str = "\n";
- }
- else if (c == '\r')
- {
- *eol_str = "\r";
-
- if (!found_eof)
- {
- apr_off_t pos;
-
- /* Check for "\r\n" by peeking at the next byte. */
- pos = 0;
- SVN_ERR(svn_io_file_seek(file, APR_CUR, &pos, scratch_pool));
- SVN_ERR(svn_io_file_read_full2(file, &c, sizeof(c), &numbytes,
- &found_eof, scratch_pool));
- if (numbytes == 1 && c == '\n')
- {
- *eol_str = "\r\n";
- }
- else
- {
- /* Pretend we never peeked. */
- SVN_ERR(svn_io_file_seek(file, APR_SET, &pos, scratch_pool));
- found_eof = FALSE;
- numbytes = 1;
- }
- }
- }
- else
- {
- if (str == NULL)
- str = svn_stringbuf_create_ensure(80, result_pool);
- svn_stringbuf_appendbyte(str, c);
- }
+ apr_file_t *file = baton;
- if (*eol_str)
- break;
- }
+ SVN_ERR(svn_io_file_readline(file, line, eol_str, eof, APR_SIZE_MAX,
+ result_pool, scratch_pool));
- if (eof)
- *eof = found_eof;
- *line = str;
+ if (!(*line)->len)
+ *line = NULL;
+ else
+ *eof = FALSE;
return SVN_NO_ERROR;
}
@@ -791,9 +832,9 @@ readline_file(void *baton, svn_stringbuf_t **line, const char **eol_str,
static svn_error_t *
tell_file(void *baton, apr_off_t *offset, apr_pool_t *scratch_pool)
{
- apr_file_t *file = (apr_file_t *)baton;
- *offset = 0;
- SVN_ERR(svn_io_file_seek(file, APR_CUR, offset, scratch_pool));
+ apr_file_t *file = baton;
+
+ SVN_ERR(svn_io_file_get_offset(offset, file, scratch_pool));
return SVN_NO_ERROR;
}
@@ -802,7 +843,8 @@ tell_file(void *baton, apr_off_t *offset, apr_pool_t *scratch_pool)
static svn_error_t *
seek_file(void *baton, apr_off_t offset, apr_pool_t *scratch_pool)
{
- apr_file_t *file = (apr_file_t *)baton;
+ apr_file_t *file = baton;
+
SVN_ERR(svn_io_file_seek(file, APR_SET, &offset, scratch_pool));
return SVN_NO_ERROR;
}
@@ -813,27 +855,15 @@ static svn_error_t *
write_file(void *baton, const char *buf, apr_size_t len,
apr_pool_t *scratch_pool)
{
- apr_file_t *file = (apr_file_t *)baton;
+ apr_file_t *file = baton;
+
SVN_ERR(svn_io_file_write_full(file, buf, len, &len, scratch_pool));
return SVN_NO_ERROR;
}
-/* Handling symbolic links:
- *
- * In Subversion, symlinks can be represented on disk in two distinct ways.
- * On systems which support symlinks, a symlink is created on disk.
- * On systems which do not support symlink, a file is created on disk
- * which contains the "normal form" of the symlink, which looks like:
- * link TARGET
- * where TARGET is the file the symlink points to.
- *
- * When reading symlinks (i.e. the link itself, not the file the symlink
- * is pointing to) through the svn_subst_create_specialfile() function
- * into a buffer, the buffer always contains the "normal form" of the symlink.
- * Due to this representation symlinks always contain a single line of text.
- *
- * The functions below are needed to deal with the case where a patch
- * wants to change the TARGET that a symlink points to.
+/* Symlinks appear in patches in their repository normal form, abstracted by
+ * the svn_subst_* module. The functions below enable patches to change the
+ * targets of symlinks.
*/
/* Baton for the (readline|tell|seek|write)_symlink functions. */
@@ -869,16 +899,36 @@ readline_symlink(void *baton, svn_stringbuf_t **line, const char **eol_str,
}
else
{
- svn_string_t *dest;
+ svn_stream_t *stream;
+ const apr_size_t len_hint = 64; /* arbitrary */
- SVN_ERR(svn_io_read_link(&dest, sb->local_abspath, scratch_pool));
- *line = svn_stringbuf_createf(result_pool, "link %s", dest->data);
+ SVN_ERR(svn_subst_read_specialfile(&stream, sb->local_abspath,
+ scratch_pool, scratch_pool));
+ SVN_ERR(svn_stringbuf_from_stream(line, stream, len_hint, result_pool));
+ *eof = FALSE;
sb->at_eof = TRUE;
}
return SVN_NO_ERROR;
}
+/* Identical to readline_symlink(), but returns symlink in raw format to
+ * allow patching links in git-style.
+ */
+static svn_error_t *
+readline_symlink_git(void *baton, svn_stringbuf_t **line, const char **eol_str,
+ svn_boolean_t *eof, apr_pool_t *result_pool,
+ apr_pool_t *scratch_pool)
+{
+ SVN_ERR(readline_symlink(baton, line, eol_str, eof,
+ result_pool, scratch_pool));
+
+ if (*line && (*line)->len > 5 && !strncmp((*line)->data, "link ", 5))
+ svn_stringbuf_remove(*line, 0, 5); /* Skip "link " */
+
+ return SVN_NO_ERROR;
+}
+
/* Set *OFFSET to 1 or 0 depending on whether the "normal form" of
* the symlink has already been read. */
static svn_error_t *
@@ -901,35 +951,6 @@ seek_symlink(void *baton, apr_off_t offset, apr_pool_t *scratch_pool)
return SVN_NO_ERROR;
}
-
-/* Set the target of the symlink accessed via BATON.
- * The contents of BUF must be a valid "normal form" of a symlink. */
-static svn_error_t *
-write_symlink(void *baton, const char *buf, apr_size_t len,
- apr_pool_t *scratch_pool)
-{
- const char *target_abspath = baton;
- const char *new_name;
- const char *link = apr_pstrndup(scratch_pool, buf, len);
-
- if (strncmp(link, "link ", 5) != 0)
- return svn_error_create(SVN_ERR_IO_WRITE_ERROR, NULL,
- _("Invalid link representation"));
-
- link += 5; /* Skip "link " */
-
- /* We assume the entire symlink is written at once, as the patch
- format is line based */
-
- SVN_ERR(svn_io_create_unique_link(&new_name, target_abspath, link,
- ".tmp", scratch_pool));
-
- SVN_ERR(svn_io_file_rename(new_name, target_abspath, scratch_pool));
-
- return SVN_NO_ERROR;
-}
-
-
/* Return a suitable filename for the target of PATCH.
* Examine the ``old'' and ``new'' file names, and choose the file name
* with the fewest path components, the shortest basename, and the shortest
@@ -987,32 +1008,19 @@ choose_target_filename(const svn_patch_t *patch)
static svn_error_t *
init_patch_target(patch_target_t **patch_target,
const svn_patch_t *patch,
- const char *wcroot_abspath,
+ const char *root_abspath,
svn_wc_context_t *wc_ctx, int strip_count,
svn_boolean_t remove_tempfiles,
+ const apr_array_header_t *targets_info,
apr_pool_t *result_pool, apr_pool_t *scratch_pool)
{
patch_target_t *target;
target_content_t *content;
- svn_boolean_t has_prop_changes = FALSE;
- svn_boolean_t prop_changes_only = FALSE;
-
- {
- apr_hash_index_t *hi;
-
- for (hi = apr_hash_first(scratch_pool, patch->prop_patches);
- hi;
- hi = apr_hash_next(hi))
- {
- svn_prop_patch_t *prop_patch = apr_hash_this_val(hi);
- if (! has_prop_changes)
- has_prop_changes = prop_patch->hunks->nelts > 0;
- else
- break;
- }
- }
+ svn_boolean_t has_text_changes = FALSE;
+ svn_boolean_t follow_moves;
- prop_changes_only = has_prop_changes && patch->hunks->nelts == 0;
+ has_text_changes = ((patch->hunks && patch->hunks->nelts > 0)
+ || patch->binary_patch);
content = apr_pcalloc(result_pool, sizeof(*content));
@@ -1030,57 +1038,35 @@ init_patch_target(patch_target_t **patch_target,
target->kind_on_disk = svn_node_none;
target->content = content;
target->prop_targets = apr_hash_make(result_pool);
+ target->operation = patch->operation;
+
+ if (patch->operation == svn_diff_op_added /* Allow replacing */
+ || patch->operation == svn_diff_op_moved)
+ {
+ follow_moves = FALSE;
+ }
+ else if (patch->operation == svn_diff_op_unchanged
+ && patch->hunks && patch->hunks->nelts == 1)
+ {
+ svn_diff_hunk_t *hunk = APR_ARRAY_IDX(patch->hunks, 0,
+ svn_diff_hunk_t *);
+
+ follow_moves = (svn_diff_hunk_get_original_start(hunk) != 0);
+ }
+ else
+ follow_moves = TRUE;
SVN_ERR(resolve_target_path(target, choose_target_filename(patch),
- wcroot_abspath, strip_count, prop_changes_only,
- wc_ctx, result_pool, scratch_pool));
+ root_abspath, strip_count, has_text_changes,
+ follow_moves, wc_ctx, targets_info,
+ result_pool, scratch_pool));
*patch_target = target;
if (! target->skipped)
{
- const char *diff_header;
- apr_size_t len;
-
- /* Create a temporary file to write the patched result to.
- * Also grab various bits of information about the file. */
- if (target->is_symlink)
- {
- struct symlink_baton_t *sb = apr_pcalloc(result_pool, sizeof(*sb));
- content->existed = TRUE;
-
- sb->local_abspath = target->local_abspath;
-
- /* Wire up the read callbacks. */
- content->read_baton = sb;
-
- content->readline = readline_symlink;
- content->seek = seek_symlink;
- content->tell = tell_symlink;
- }
- else if (target->kind_on_disk == svn_node_file)
+ if (patch->old_symlink_bit == svn_tristate_true
+ || patch->new_symlink_bit == svn_tristate_true)
{
- SVN_ERR(svn_io_file_open(&target->file, target->local_abspath,
- APR_READ | APR_BUFFERED,
- APR_OS_DEFAULT, result_pool));
- SVN_ERR(svn_wc_text_modified_p2(&target->local_mods, wc_ctx,
- target->local_abspath, FALSE,
- scratch_pool));
- SVN_ERR(svn_io_is_file_executable(&target->executable,
- target->local_abspath,
- scratch_pool));
- SVN_ERR(obtain_eol_and_keywords_for_file(&content->keywords,
- &content->eol_style,
- &content->eol_str,
- wc_ctx,
- target->local_abspath,
- result_pool,
- scratch_pool));
- content->existed = TRUE;
-
- /* Wire up the read callbacks. */
- content->readline = readline_file;
- content->seek = seek_file;
- content->tell = tell_file;
- content->read_baton = target->file;
+ target->git_symlink_format = TRUE;
}
/* ### Is it ok to set the operation of the target already here? Isn't
@@ -1098,6 +1084,7 @@ init_patch_target(patch_target_t **patch_target,
const char *move_target_path;
const char *move_target_relpath;
svn_boolean_t under_root;
+ svn_boolean_t is_special;
svn_node_kind_t kind_on_disk;
svn_node_kind_t wc_kind;
@@ -1110,7 +1097,7 @@ init_patch_target(patch_target_t **patch_target,
if (svn_dirent_is_absolute(move_target_path))
{
- move_target_relpath = svn_dirent_is_child(wcroot_abspath,
+ move_target_relpath = svn_dirent_is_child(root_abspath,
move_target_path,
scratch_pool);
if (! move_target_relpath)
@@ -1118,7 +1105,6 @@ init_patch_target(patch_target_t **patch_target,
/* The move target path is either outside of the working
* copy or it is the working copy itself. Skip it. */
target->skipped = TRUE;
- target->local_abspath = NULL;
return SVN_NO_ERROR;
}
}
@@ -1128,76 +1114,133 @@ init_patch_target(patch_target_t **patch_target,
/* Make sure the move target path is secure to use. */
SVN_ERR(svn_dirent_is_under_root(&under_root,
&target->move_target_abspath,
- wcroot_abspath,
+ root_abspath,
move_target_relpath, result_pool));
if (! under_root)
{
/* The target path is outside of the working copy. Skip it. */
target->skipped = TRUE;
- target->local_abspath = NULL;
+ target->move_target_abspath = NULL;
return SVN_NO_ERROR;
}
- SVN_ERR(svn_io_check_path(target->move_target_abspath,
- &kind_on_disk, scratch_pool));
+ SVN_ERR(svn_io_check_special_path(target->move_target_abspath,
+ &kind_on_disk, &is_special,
+ scratch_pool));
SVN_ERR(svn_wc_read_kind2(&wc_kind, wc_ctx,
target->move_target_abspath,
FALSE, FALSE, scratch_pool));
- if (kind_on_disk != svn_node_none || wc_kind != svn_node_none)
+ if (wc_kind == svn_node_file || wc_kind == svn_node_dir)
{
- /* The move target path already exists on disk. Skip target. */
- target->skipped = TRUE;
- target->move_target_abspath = NULL;
- return SVN_NO_ERROR;
+ /* The move target path already exists on disk. */
+ svn_error_t *err;
+ const char *moved_from_abspath;
+
+ err = svn_wc__node_was_moved_here(&moved_from_abspath, NULL,
+ wc_ctx,
+ target->move_target_abspath,
+ scratch_pool, scratch_pool);
+
+ if (err && err->apr_err == SVN_ERR_WC_PATH_NOT_FOUND)
+ {
+ svn_error_clear(err);
+ err = NULL;
+ moved_from_abspath = NULL;
+ }
+ else
+ SVN_ERR(err);
+
+ if (moved_from_abspath && (strcmp(moved_from_abspath,
+ target->local_abspath) == 0))
+ {
+ target->local_abspath = target->move_target_abspath;
+ target->move_target_abspath = NULL;
+ target->operation = svn_diff_op_modified;
+ target->locally_deleted = FALSE;
+ target->db_kind = wc_kind;
+ target->kind_on_disk = kind_on_disk;
+ target->is_special = is_special;
+
+ target->had_already_applied = TRUE; /* Make sure we notify */
+ }
+ else
+ {
+ target->skipped = TRUE;
+ target->move_target_abspath = NULL;
+ return SVN_NO_ERROR;
+ }
+
+ }
+ else if (kind_on_disk != svn_node_none
+ || target_is_added(targets_info, target->move_target_abspath,
+ scratch_pool))
+ {
+ target->skipped = TRUE;
+ target->move_target_abspath = NULL;
+ return SVN_NO_ERROR;
}
}
- if (! target->is_symlink)
+ /* Create a temporary file to write the patched result to.
+ * Also grab various bits of information about the file. */
+ if (target->is_symlink)
{
- /* Open a temporary file to write the patched result to. */
- SVN_ERR(svn_io_open_unique_file3(&target->patched_file,
- &target->patched_path, NULL,
- remove_tempfiles ?
- svn_io_file_del_on_pool_cleanup :
- svn_io_file_del_none,
- result_pool, scratch_pool));
+ struct symlink_baton_t *sb = apr_pcalloc(result_pool, sizeof(*sb));
+ content->existed = TRUE;
+
+ sb->local_abspath = target->local_abspath;
- /* Put the write callback in place. */
- content->write = write_file;
- content->write_baton = target->patched_file;
+ /* Wire up the read callbacks. */
+ content->read_baton = sb;
+
+ content->readline = target->git_symlink_format ? readline_symlink_git
+ : readline_symlink;
+ content->seek = seek_symlink;
+ content->tell = tell_symlink;
}
- else
+ else if (target->kind_on_disk == svn_node_file)
{
- /* Put the write callback in place. */
- SVN_ERR(svn_io_open_unique_file3(NULL,
- &target->patched_path, NULL,
- remove_tempfiles ?
- svn_io_file_del_on_pool_cleanup :
- svn_io_file_del_none,
- result_pool, scratch_pool));
-
- content->write_baton = (void*)target->patched_path;
+ SVN_ERR(svn_io_file_open(&target->file, target->local_abspath,
+ APR_READ | APR_BUFFERED,
+ APR_OS_DEFAULT, result_pool));
+ SVN_ERR(svn_io_is_file_executable(&target->executable,
+ target->local_abspath,
+ scratch_pool));
+ SVN_ERR(obtain_eol_and_keywords_for_file(&content->keywords,
+ &content->eol_style,
+ &content->eol_str,
+ wc_ctx,
+ target->local_abspath,
+ result_pool,
+ scratch_pool));
+ content->existed = TRUE;
- content->write = write_symlink;
+ /* Wire up the read callbacks. */
+ content->readline = readline_file;
+ content->seek = seek_file;
+ content->tell = tell_file;
+ content->read_baton = target->file;
}
- /* Open a temporary file to write rejected hunks to. */
- SVN_ERR(svn_io_open_unique_file3(&target->reject_file,
- &target->reject_path, NULL,
+ /* Open a temporary file to write the patched result to. */
+ SVN_ERR(svn_io_open_unique_file3(&target->patched_file,
+ &target->patched_path, NULL,
remove_tempfiles ?
svn_io_file_del_on_pool_cleanup :
svn_io_file_del_none,
result_pool, scratch_pool));
- /* The reject file needs a diff header. */
- diff_header = apr_psprintf(scratch_pool, "--- %s%s+++ %s%s",
- target->canon_path_from_patchfile,
- APR_EOL_STR,
- target->canon_path_from_patchfile,
- APR_EOL_STR);
- len = strlen(diff_header);
- SVN_ERR(svn_io_file_write_full(target->reject_file, diff_header, len,
- &len, scratch_pool));
+ /* Put the write callback in place. */
+ content->write = write_file;
+ content->write_baton = target->patched_file;
+
+ /* Open a temporary stream to write rejected hunks to. */
+ SVN_ERR(svn_stream_open_unique(&target->reject_stream,
+ &target->reject_path, NULL,
+ remove_tempfiles ?
+ svn_io_file_del_on_pool_cleanup :
+ svn_io_file_del_none,
+ result_pool, scratch_pool));
/* Handle properties. */
if (! target->skipped)
@@ -1213,13 +1256,185 @@ init_patch_target(patch_target_t **patch_target,
prop_patch_target_t *prop_target;
SVN_ERR(init_prop_target(&prop_target,
- prop_name,
+ target, prop_name,
prop_patch->operation,
wc_ctx, target->local_abspath,
result_pool, scratch_pool));
svn_hash_sets(target->prop_targets, prop_name, prop_target);
}
+
+ /* Now, check for out-of-band mode changes and convert these in
+ their Subversion equivalent properties. */
+ if (patch->new_executable_bit != svn_tristate_unknown
+ && patch->new_executable_bit != patch->old_executable_bit)
+ {
+ svn_diff_operation_kind_t operation;
+
+ if (patch->new_executable_bit == svn_tristate_true)
+ operation = svn_diff_op_added;
+ else if (patch->new_executable_bit == svn_tristate_false)
+ {
+ /* Made non-executable. */
+ if (patch->old_executable_bit == svn_tristate_true)
+ operation = svn_diff_op_deleted;
+ else
+ operation = svn_diff_op_unchanged;
+ }
+ else
+ operation = svn_diff_op_unchanged;
+
+ if (operation != svn_diff_op_unchanged)
+ {
+ prop_patch_target_t *prop_target;
+
+ prop_target = svn_hash_gets(target->prop_targets,
+ SVN_PROP_EXECUTABLE);
+
+ if (prop_target && operation != prop_target->operation)
+ {
+ return svn_error_createf(SVN_ERR_INVALID_INPUT, NULL,
+ _("Invalid patch: specifies "
+ "contradicting mode changes and "
+ "%s changes (for '%s')"),
+ SVN_PROP_EXECUTABLE,
+ target->local_abspath);
+ }
+ else if (!prop_target)
+ {
+ SVN_ERR(init_prop_target(&prop_target,
+ target, SVN_PROP_EXECUTABLE,
+ operation,
+ wc_ctx, target->local_abspath,
+ result_pool, scratch_pool));
+ svn_hash_sets(target->prop_targets, SVN_PROP_EXECUTABLE,
+ prop_target);
+ }
+ }
+ }
+
+ if (patch->new_symlink_bit != svn_tristate_unknown
+ && patch->new_symlink_bit != patch->old_symlink_bit)
+ {
+ svn_diff_operation_kind_t operation;
+
+ if (patch->new_symlink_bit == svn_tristate_true)
+ operation = svn_diff_op_added;
+ else if (patch->new_symlink_bit == svn_tristate_false)
+ {
+ /* Made non-symlink. */
+ if (patch->old_symlink_bit == svn_tristate_true)
+ operation = svn_diff_op_deleted;
+ else
+ operation = svn_diff_op_unchanged;
+ }
+ else
+ operation = svn_diff_op_unchanged;
+
+ if (operation != svn_diff_op_unchanged)
+ {
+ prop_patch_target_t *prop_target;
+ prop_target = svn_hash_gets(target->prop_targets,
+ SVN_PROP_SPECIAL);
+
+ if (prop_target && operation != prop_target->operation)
+ {
+ return svn_error_createf(SVN_ERR_INVALID_INPUT, NULL,
+ _("Invalid patch: specifies "
+ "contradicting mode changes and "
+ "%s changes (for '%s')"),
+ SVN_PROP_SPECIAL,
+ target->local_abspath);
+ }
+ else if (!prop_target)
+ {
+ SVN_ERR(init_prop_target(&prop_target,
+ target, SVN_PROP_SPECIAL,
+ operation,
+ wc_ctx, target->local_abspath,
+ result_pool, scratch_pool));
+ svn_hash_sets(target->prop_targets, SVN_PROP_SPECIAL,
+ prop_target);
+ }
+ }
+ }
+ }
+ }
+
+ if ((target->locally_deleted || target->db_kind == svn_node_none)
+ && !target->added
+ && target->operation == svn_diff_op_unchanged)
+ {
+ svn_boolean_t maybe_add = FALSE;
+
+ if (patch->hunks && patch->hunks->nelts == 1)
+ {
+ svn_diff_hunk_t *hunk = APR_ARRAY_IDX(patch->hunks, 0,
+ svn_diff_hunk_t *);
+
+ if (svn_diff_hunk_get_original_start(hunk) == 0)
+ maybe_add = TRUE;
+ }
+ else if (patch->prop_patches && apr_hash_count(patch->prop_patches))
+ {
+ apr_hash_index_t *hi;
+ svn_boolean_t all_add = TRUE;
+
+ for (hi = apr_hash_first(result_pool, patch->prop_patches);
+ hi;
+ hi = apr_hash_next(hi))
+ {
+ svn_prop_patch_t *prop_patch = apr_hash_this_val(hi);
+
+ if (prop_patch->operation != svn_diff_op_added)
+ {
+ all_add = FALSE;
+ break;
+ }
+ }
+
+ maybe_add = all_add;
+ }
+ /* Other implied types */
+
+ if (maybe_add)
+ target->added = TRUE;
+ }
+ else if (!target->deleted && !target->added
+ && target->operation == svn_diff_op_unchanged)
+ {
+ svn_boolean_t maybe_delete = FALSE;
+
+ if (patch->hunks && patch->hunks->nelts == 1)
+ {
+ svn_diff_hunk_t *hunk = APR_ARRAY_IDX(patch->hunks, 0,
+ svn_diff_hunk_t *);
+
+ if (svn_diff_hunk_get_modified_start(hunk) == 0)
+ maybe_delete = TRUE;
}
+
+ /* Other implied types */
+
+ if (maybe_delete)
+ target->deleted = TRUE;
+ }
+
+ if (target->reject_stream != NULL)
+ {
+ /* The reject file needs a diff header. */
+ const char *left_src = target->canon_path_from_patchfile;
+ const char *right_src = target->canon_path_from_patchfile;
+
+ /* Handle moves specifically? */
+ if (target->added)
+ left_src = "/dev/null";
+ if (target->deleted)
+ right_src = "/dev/null";
+
+ SVN_ERR(svn_stream_printf(target->reject_stream, scratch_pool,
+ "--- %s" APR_EOL_STR
+ "+++ %s" APR_EOL_STR,
+ left_src, right_src));
}
return SVN_NO_ERROR;
@@ -1354,12 +1569,20 @@ match_hunk(svn_boolean_t *matched, target_content_t *content,
svn_linenum_t hunk_length;
svn_linenum_t leading_context;
svn_linenum_t trailing_context;
+ svn_linenum_t fuzz_penalty;
*matched = FALSE;
if (content->eof)
return SVN_NO_ERROR;
+ fuzz_penalty = svn_diff_hunk__get_fuzz_penalty(hunk);
+
+ if (fuzz_penalty > fuzz)
+ return SVN_NO_ERROR;
+ else
+ fuzz -= fuzz_penalty;
+
saved_line = content->current_line;
lines_read = 0;
lines_matched = FALSE;
@@ -1568,14 +1791,14 @@ match_existing_target(svn_boolean_t *match,
*match = FALSE;
return SVN_NO_ERROR;
}
- }
- while (lines_matched && ! content->eof && ! hunk_eof);
- svn_pool_destroy(iterpool);
+ }
+ while (lines_matched && ! content->eof && ! hunk_eof);
+ svn_pool_destroy(iterpool);
- *match = (lines_matched && content->eof == hunk_eof);
- SVN_ERR(seek_to_line(content, saved_line, scratch_pool));
+ *match = (lines_matched && content->eof == hunk_eof);
+ SVN_ERR(seek_to_line(content, saved_line, scratch_pool));
- return SVN_NO_ERROR;
+ return SVN_NO_ERROR;
}
/* Determine the line at which a HUNK applies to CONTENT of the TARGET
@@ -1636,8 +1859,6 @@ get_hunk_info(hunk_info_t **hi, patch_target_t *target,
{
svn_boolean_t file_matches;
- /* ### I can't reproduce anything but a no-match here.
- The content is already at eof, so any hunk fails */
SVN_ERR(match_existing_target(&file_matches, content, hunk,
scratch_pool));
if (file_matches)
@@ -1682,8 +1903,11 @@ get_hunk_info(hunk_info_t **hi, patch_target_t *target,
}
else if (original_start > 0 && content->existed)
{
+ svn_linenum_t modified_start;
svn_linenum_t saved_line = content->current_line;
+ modified_start = svn_diff_hunk_get_modified_start(hunk);
+
/* Scan for a match at the line where the hunk thinks it
* should be going. */
SVN_ERR(seek_to_line(content, original_start, scratch_pool));
@@ -1707,20 +1931,24 @@ get_hunk_info(hunk_info_t **hi, patch_target_t *target,
* check would be ambiguous. */
if (fuzz == 0)
{
- svn_linenum_t modified_start;
-
- modified_start = svn_diff_hunk_get_modified_start(hunk);
- if (modified_start == 0)
+ if (modified_start == 0
+ && (target->operation == svn_diff_op_unchanged
+ || target->operation == svn_diff_op_deleted))
{
- /* Patch wants to delete the file.
+ /* Patch wants to delete the file. */
- ### locally_deleted is always false here? */
already_applied = target->locally_deleted;
}
else
{
- SVN_ERR(seek_to_line(content, modified_start,
- scratch_pool));
+ svn_linenum_t seek_to;
+
+ if (modified_start == 0)
+ seek_to = 1; /* Empty file case */
+ else
+ seek_to = modified_start;
+
+ SVN_ERR(seek_to_line(content, seek_to, scratch_pool));
SVN_ERR(scan_for_match(&matched_line, content,
hunk, TRUE,
modified_start + 1,
@@ -1817,12 +2045,42 @@ get_hunk_info(hunk_info_t **hi, patch_target_t *target,
}
}
}
+ else if (matched_line > 0
+ && fuzz == 0
+ && (svn_diff_hunk_get_leading_context(hunk) == 0
+ || svn_diff_hunk_get_trailing_context(hunk) == 0)
+ && (svn_diff_hunk_get_modified_length(hunk) >
+ svn_diff_hunk_get_original_length(hunk)))
+ {
+ /* Check that we are not applying the same change that just adds some
+ lines again, when we don't have enough context to see the
+ difference */
+ svn_linenum_t reverse_matched_line;
+
+ SVN_ERR(seek_to_line(content, modified_start, scratch_pool));
+ SVN_ERR(scan_for_match(&reverse_matched_line, content,
+ hunk, TRUE,
+ modified_start + 1,
+ fuzz, ignore_whitespace, TRUE,
+ cancel_func, cancel_baton,
+ scratch_pool));
+
+ /* We might want to check that we are actually at the start or the
+ end of the file. Having no context implies that we should be. */
+ already_applied = (reverse_matched_line == modified_start);
+ }
SVN_ERR(seek_to_line(content, saved_line, scratch_pool));
}
+ else if (!content->existed && svn_diff_hunk_get_modified_start(hunk) == 0)
+ {
+ /* The hunk wants to delete a file or property which doesn't exist. */
+ matched_line = 0;
+ already_applied = TRUE;
+ }
else
{
- /* The hunk wants to modify a file which doesn't exist. */
+ /* The hunk wants to modify a file or property which doesn't exist. */
matched_line = 0;
}
@@ -1831,7 +2089,8 @@ get_hunk_info(hunk_info_t **hi, patch_target_t *target,
(*hi)->matched_line = matched_line;
(*hi)->rejected = (matched_line == 0);
(*hi)->already_applied = already_applied;
- (*hi)->fuzz = fuzz;
+ (*hi)->report_fuzz = fuzz;
+ (*hi)->match_fuzz = fuzz - svn_diff_hunk__get_fuzz_penalty(hunk);
return SVN_NO_ERROR;
}
@@ -1877,8 +2136,6 @@ reject_hunk(patch_target_t *target, target_content_t *content,
svn_diff_hunk_t *hunk, const char *prop_name,
apr_pool_t *pool)
{
- const char *hunk_header;
- apr_size_t len;
svn_boolean_t eof;
static const char * const text_atat = "@@";
static const char * const prop_atat = "##";
@@ -1887,14 +2144,9 @@ reject_hunk(patch_target_t *target, target_content_t *content,
if (prop_name)
{
- const char *prop_header;
-
- /* ### Print 'Added', 'Deleted' or 'Modified' instead of 'Property'.
- */
- prop_header = apr_psprintf(pool, "Property: %s\n", prop_name);
- len = strlen(prop_header);
- SVN_ERR(svn_io_file_write_full(target->reject_file, prop_header,
- len, &len, pool));
+ /* ### Print 'Added', 'Deleted' or 'Modified' instead of 'Property'. */
+ svn_stream_printf(target->reject_stream,
+ pool, "Property: %s" APR_EOL_STR, prop_name);
atat = prop_atat;
}
else
@@ -1902,17 +2154,14 @@ reject_hunk(patch_target_t *target, target_content_t *content,
atat = text_atat;
}
- hunk_header = apr_psprintf(pool, "%s -%lu,%lu +%lu,%lu %s%s",
- atat,
- svn_diff_hunk_get_original_start(hunk),
- svn_diff_hunk_get_original_length(hunk),
- svn_diff_hunk_get_modified_start(hunk),
- svn_diff_hunk_get_modified_length(hunk),
- atat,
- APR_EOL_STR);
- len = strlen(hunk_header);
- SVN_ERR(svn_io_file_write_full(target->reject_file, hunk_header, len,
- &len, pool));
+ SVN_ERR(svn_stream_printf(target->reject_stream, pool,
+ "%s -%lu,%lu +%lu,%lu %s" APR_EOL_STR,
+ atat,
+ svn_diff_hunk_get_original_start(hunk),
+ svn_diff_hunk_get_original_length(hunk),
+ svn_diff_hunk_get_modified_start(hunk),
+ svn_diff_hunk_get_modified_length(hunk),
+ atat));
iterpool = svn_pool_create(pool);
do
@@ -1928,17 +2177,15 @@ reject_hunk(patch_target_t *target, target_content_t *content,
{
if (hunk_line->len >= 1)
{
- len = hunk_line->len;
- SVN_ERR(svn_io_file_write_full(target->reject_file,
- hunk_line->data, len, &len,
- iterpool));
+ apr_size_t len = hunk_line->len;
+
+ SVN_ERR(svn_stream_write(target->reject_stream,
+ hunk_line->data, &len));
}
if (eol_str)
{
- len = strlen(eol_str);
- SVN_ERR(svn_io_file_write_full(target->reject_file, eol_str,
- len, &len, iterpool));
+ SVN_ERR(svn_stream_puts(target->reject_stream, eol_str));
}
}
}
@@ -1965,6 +2212,7 @@ apply_hunk(patch_target_t *target, target_content_t *content,
svn_linenum_t lines_read;
svn_boolean_t eof;
apr_pool_t *iterpool;
+ svn_linenum_t fuzz = hi->match_fuzz;
/* ### Is there a cleaner way to describe if we have an existing target?
*/
@@ -1976,13 +2224,13 @@ apply_hunk(patch_target_t *target, target_content_t *content,
* Also copy leading lines of context which matched with fuzz.
* The target has changed on the fuzzy-matched lines,
* so we should retain the target's version of those lines. */
- SVN_ERR(copy_lines_to_target(content, hi->matched_line + hi->fuzz,
+ SVN_ERR(copy_lines_to_target(content, hi->matched_line + fuzz,
pool));
/* Skip the target's version of the hunk.
* Don't skip trailing lines which matched with fuzz. */
line = content->current_line +
- svn_diff_hunk_get_original_length(hi->hunk) - (2 * hi->fuzz);
+ svn_diff_hunk_get_original_length(hi->hunk) - (2 * fuzz);
SVN_ERR(seek_to_line(content, line, pool));
if (content->current_line != line && ! content->eof)
{
@@ -2009,8 +2257,8 @@ apply_hunk(patch_target_t *target, target_content_t *content,
&eol_str, &eof,
iterpool, iterpool));
lines_read++;
- if (lines_read > hi->fuzz &&
- lines_read <= svn_diff_hunk_get_modified_length(hi->hunk) - hi->fuzz)
+ if (lines_read > fuzz &&
+ lines_read <= svn_diff_hunk_get_modified_length(hi->hunk) - fuzz)
{
apr_size_t len;
@@ -2079,7 +2327,7 @@ send_hunk_notification(const hunk_info_t *hi,
notify->hunk_modified_length =
svn_diff_hunk_get_modified_length(hi->hunk);
notify->hunk_matched_line = hi->matched_line;
- notify->hunk_fuzz = hi->fuzz;
+ notify->hunk_fuzz = hi->report_fuzz;
notify->prop_name = prop_name;
ctx->notify_func2(ctx->notify_baton2, notify, pool);
@@ -2092,7 +2340,7 @@ send_hunk_notification(const hunk_info_t *hi,
static svn_error_t *
send_patch_notification(const patch_target_t *target,
const svn_client_ctx_t *ctx,
- apr_pool_t *pool)
+ apr_pool_t *scratch_pool)
{
svn_wc_notify_t *notify;
svn_wc_notify_action_t action;
@@ -2105,7 +2353,7 @@ send_patch_notification(const patch_target_t *target,
action = svn_wc_notify_skip;
else if (target->deleted)
action = svn_wc_notify_delete;
- else if (target->added || target->replaced || target->move_target_abspath)
+ else if (target->added || target->move_target_abspath)
action = svn_wc_notify_add;
else
action = svn_wc_notify_patch;
@@ -2116,16 +2364,17 @@ send_patch_notification(const patch_target_t *target,
notify_path = target->local_abspath ? target->local_abspath
: target->local_relpath;
- notify = svn_wc_create_notify(notify_path, action, pool);
- notify->kind = svn_node_file;
+ notify = svn_wc_create_notify(notify_path, action, scratch_pool);
+ notify->kind = (target->db_kind == svn_node_dir) ? svn_node_dir
+ : svn_node_file;
if (action == svn_wc_notify_skip)
{
- if (target->db_kind == svn_node_none ||
- target->db_kind == svn_node_unknown)
- notify->content_state = svn_wc_notify_state_missing;
- else if (target->db_kind == svn_node_dir)
+ if (target->obstructed)
notify->content_state = svn_wc_notify_state_obstructed;
+ else if (target->db_kind == svn_node_none ||
+ target->db_kind == svn_node_unknown)
+ notify->content_state = svn_wc_notify_state_missing;
else
notify->content_state = svn_wc_notify_state_unknown;
}
@@ -2133,26 +2382,32 @@ send_patch_notification(const patch_target_t *target,
{
if (target->had_rejects)
notify->content_state = svn_wc_notify_state_conflicted;
- else if (target->local_mods)
- notify->content_state = svn_wc_notify_state_merged;
else if (target->has_text_changes)
notify->content_state = svn_wc_notify_state_changed;
+ else if (target->had_already_applied)
+ notify->content_state = svn_wc_notify_state_merged;
+ else
+ notify->content_state = svn_wc_notify_state_unchanged;
if (target->had_prop_rejects)
notify->prop_state = svn_wc_notify_state_conflicted;
else if (target->has_prop_changes)
notify->prop_state = svn_wc_notify_state_changed;
+ else if (target->had_prop_already_applied)
+ notify->prop_state = svn_wc_notify_state_merged;
+ else
+ notify->prop_state = svn_wc_notify_state_unchanged;
}
- ctx->notify_func2(ctx->notify_baton2, notify, pool);
+ ctx->notify_func2(ctx->notify_baton2, notify, scratch_pool);
if (action == svn_wc_notify_patch)
{
int i;
apr_pool_t *iterpool;
- apr_hash_index_t *hash_index;
+ apr_array_header_t *prop_targets;
- iterpool = svn_pool_create(pool);
+ iterpool = svn_pool_create(scratch_pool);
for (i = 0; i < target->content->hunks->nelts; i++)
{
const hunk_info_t *hi;
@@ -2165,26 +2420,30 @@ send_patch_notification(const patch_target_t *target,
ctx, iterpool));
}
- for (hash_index = apr_hash_first(pool, target->prop_targets);
- hash_index;
- hash_index = apr_hash_next(hash_index))
+ prop_targets = svn_sort__hash(target->prop_targets,
+ svn_sort_compare_items_lexically,
+ scratch_pool);
+ for (i = 0; i < prop_targets->nelts; i++)
{
- prop_patch_target_t *prop_target;
+ int j;
+ svn_sort__item_t item = APR_ARRAY_IDX(prop_targets, i,
+ svn_sort__item_t);
- prop_target = apr_hash_this_val(hash_index);
+ prop_patch_target_t *prop_target = item.value;
- for (i = 0; i < prop_target->content->hunks->nelts; i++)
+ for (j = 0; j < prop_target->content->hunks->nelts; j++)
{
const hunk_info_t *hi;
svn_pool_clear(iterpool);
- hi = APR_ARRAY_IDX(prop_target->content->hunks, i,
+ hi = APR_ARRAY_IDX(prop_target->content->hunks, j,
hunk_info_t *);
/* Don't notify on the hunk level for added or deleted props. */
- if (prop_target->operation != svn_diff_op_added &&
+ if ((prop_target->operation != svn_diff_op_added &&
prop_target->operation != svn_diff_op_deleted)
+ || hi->rejected || hi->already_applied)
SVN_ERR(send_hunk_notification(hi, target, prop_target->name,
ctx, iterpool));
}
@@ -2192,13 +2451,13 @@ send_patch_notification(const patch_target_t *target,
svn_pool_destroy(iterpool);
}
- if (target->move_target_abspath)
+ if (!target->skipped && target->move_target_abspath)
{
/* Notify about deletion of move source. */
notify = svn_wc_create_notify(target->local_abspath,
- svn_wc_notify_delete, pool);
+ svn_wc_notify_delete, scratch_pool);
notify->kind = svn_node_file;
- ctx->notify_func2(ctx->notify_baton2, notify, pool);
+ ctx->notify_func2(ctx->notify_baton2, notify, scratch_pool);
}
return SVN_NO_ERROR;
@@ -2252,7 +2511,8 @@ sort_matched_hunks(const void *a, const void *b)
* in RESULT_POOL. Use WC_CTX as the working copy context.
* STRIP_COUNT specifies the number of leading path components
* which should be stripped from target paths in the patch.
- * REMOVE_TEMPFILES, PATCH_FUNC, and PATCH_BATON as in svn_client_patch().
+ * REMOVE_TEMPFILES is as in svn_client_patch().
+ * TARGETS_INFO is for preserving info across calls.
* IGNORE_WHITESPACE tells whether whitespace should be considered when
* doing the matching.
* Call cancel CANCEL_FUNC with baton CANCEL_BATON to trigger cancellation.
@@ -2263,6 +2523,7 @@ apply_one_patch(patch_target_t **patch_target, svn_patch_t *patch,
int strip_count,
svn_boolean_t ignore_whitespace,
svn_boolean_t remove_tempfiles,
+ const apr_array_header_t *targets_info,
svn_cancel_func_t cancel_func,
void *cancel_baton,
apr_pool_t *result_pool, apr_pool_t *scratch_pool)
@@ -2273,9 +2534,11 @@ apply_one_patch(patch_target_t **patch_target, svn_patch_t *patch,
static const svn_linenum_t MAX_FUZZ = 2;
apr_hash_index_t *hash_index;
svn_linenum_t previous_offset = 0;
+ apr_array_header_t *prop_targets;
SVN_ERR(init_patch_target(&target, patch, abs_wc_path, wc_ctx, strip_count,
- remove_tempfiles, result_pool, scratch_pool));
+ remove_tempfiles, targets_info,
+ result_pool, scratch_pool));
if (target->skipped)
{
*patch_target = target;
@@ -2283,79 +2546,196 @@ apply_one_patch(patch_target_t **patch_target, svn_patch_t *patch,
}
iterpool = svn_pool_create(scratch_pool);
- /* Match hunks. */
- for (i = 0; i < patch->hunks->nelts; i++)
+
+ if (patch->hunks && patch->hunks->nelts)
{
- svn_diff_hunk_t *hunk;
- hunk_info_t *hi;
- svn_linenum_t fuzz = 0;
+ /* Match hunks. */
+ for (i = 0; i < patch->hunks->nelts; i++)
+ {
+ svn_diff_hunk_t *hunk;
+ hunk_info_t *hi;
+ svn_linenum_t fuzz = 0;
- svn_pool_clear(iterpool);
+ svn_pool_clear(iterpool);
- if (cancel_func)
- SVN_ERR(cancel_func(cancel_baton));
+ if (cancel_func)
+ SVN_ERR(cancel_func(cancel_baton));
- hunk = APR_ARRAY_IDX(patch->hunks, i, svn_diff_hunk_t *);
+ hunk = APR_ARRAY_IDX(patch->hunks, i, svn_diff_hunk_t *);
- /* Determine the line the hunk should be applied at.
- * If no match is found initially, try with fuzz. */
- do
- {
- SVN_ERR(get_hunk_info(&hi, target, target->content, hunk, fuzz,
- previous_offset,
- ignore_whitespace,
- FALSE /* is_prop_hunk */,
- cancel_func, cancel_baton,
- result_pool, iterpool));
- fuzz++;
+ /* Determine the line the hunk should be applied at.
+ * If no match is found initially, try with fuzz. */
+ do
+ {
+ SVN_ERR(get_hunk_info(&hi, target, target->content, hunk, fuzz,
+ previous_offset,
+ ignore_whitespace,
+ FALSE /* is_prop_hunk */,
+ cancel_func, cancel_baton,
+ result_pool, iterpool));
+ fuzz++;
+ }
+ while (hi->rejected && fuzz <= MAX_FUZZ && ! hi->already_applied);
+
+ if (hi->matched_line)
+ previous_offset
+ = hi->matched_line - svn_diff_hunk_get_original_start(hunk);
+
+ APR_ARRAY_PUSH(target->content->hunks, hunk_info_t *) = hi;
}
- while (hi->rejected && fuzz <= MAX_FUZZ && ! hi->already_applied);
- if (hi->matched_line)
- previous_offset
- = hi->matched_line - svn_diff_hunk_get_original_start(hunk);
+ /* Hunks are applied in the order determined by the matched line and
+ this may be different from the order of the original lines. */
+ svn_sort__array(target->content->hunks, sort_matched_hunks);
- APR_ARRAY_PUSH(target->content->hunks, hunk_info_t *) = hi;
- }
+ /* Apply or reject hunks. */
+ for (i = 0; i < target->content->hunks->nelts; i++)
+ {
+ hunk_info_t *hi;
+
+ svn_pool_clear(iterpool);
+
+ if (cancel_func)
+ SVN_ERR(cancel_func(cancel_baton));
- /* Hunks are applied in the order determined by the matched line and
- this may be different from the order of the original lines. */
- svn_sort__array(target->content->hunks, sort_matched_hunks);
+ hi = APR_ARRAY_IDX(target->content->hunks, i, hunk_info_t *);
+ if (hi->already_applied)
+ {
+ target->had_already_applied = TRUE;
+ continue;
+ }
+ else if (hi->rejected)
+ SVN_ERR(reject_hunk(target, target->content, hi->hunk,
+ NULL /* prop_name */,
+ iterpool));
+ else
+ SVN_ERR(apply_hunk(target, target->content, hi,
+ NULL /* prop_name */, iterpool));
+ }
- /* Apply or reject hunks. */
- for (i = 0; i < target->content->hunks->nelts; i++)
+ if (target->kind_on_disk == svn_node_file)
+ {
+ /* Copy any remaining lines to target. */
+ SVN_ERR(copy_lines_to_target(target->content, 0, scratch_pool));
+ if (! target->content->eof)
+ {
+ /* We could not copy the entire target file to the temporary
+ * file, and would truncate the target if we copied the
+ * temporary file on top of it. Skip this target. */
+ target->skipped = TRUE;
+ }
+ }
+ }
+ else if (patch->binary_patch)
{
- hunk_info_t *hi;
+ svn_stream_t *orig_stream;
+ svn_boolean_t same;
+
+ if (target->file)
+ orig_stream = svn_stream_from_aprfile2(target->file, TRUE, iterpool);
+ else
+ orig_stream = svn_stream_empty(iterpool);
+ SVN_ERR(svn_stream_contents_same2(
+ &same, orig_stream,
+ svn_diff_get_binary_diff_original_stream(patch->binary_patch,
+ iterpool),
+ iterpool));
svn_pool_clear(iterpool);
- if (cancel_func)
- SVN_ERR(cancel_func(cancel_baton));
+ if (same)
+ {
+ /* The file in the working copy is identical to the one expected by
+ the patch... So we can write the result stream; no fuzz,
+ just a 100% match */
- hi = APR_ARRAY_IDX(target->content->hunks, i, hunk_info_t *);
- if (hi->already_applied)
- continue;
- else if (hi->rejected)
- SVN_ERR(reject_hunk(target, target->content, hi->hunk,
- NULL /* prop_name */,
- iterpool));
+ target->has_text_changes = TRUE;
+ }
else
- SVN_ERR(apply_hunk(target, target->content, hi,
- NULL /* prop_name */, iterpool));
- }
+ {
+ /* Perhaps the file is identical to the resulting version, implying
+ that the patch has already been applied */
+ if (target->file)
+ {
+ apr_off_t start = 0;
- if (target->kind_on_disk == svn_node_file)
- {
- /* Copy any remaining lines to target. */
- SVN_ERR(copy_lines_to_target(target->content, 0, scratch_pool));
- if (! target->content->eof)
+ SVN_ERR(svn_io_file_seek(target->file, APR_SET, &start, iterpool));
+
+ orig_stream = svn_stream_from_aprfile2(target->file, TRUE, iterpool);
+ }
+ else
+ orig_stream = svn_stream_empty(iterpool);
+
+ SVN_ERR(svn_stream_contents_same2(
+ &same, orig_stream,
+ svn_diff_get_binary_diff_result_stream(patch->binary_patch,
+ iterpool),
+ iterpool));
+ svn_pool_clear(iterpool);
+
+ if (same)
+ target->had_already_applied = TRUE;
+ }
+
+ if (same)
{
- /* We could not copy the entire target file to the temporary file,
- * and would truncate the target if we copied the temporary file
- * on top of it. Skip this target. */
+ SVN_ERR(svn_stream_copy3(
+ svn_diff_get_binary_diff_result_stream(patch->binary_patch,
+ iterpool),
+ svn_stream_from_aprfile2(target->patched_file, TRUE,
+ iterpool),
+ cancel_func, cancel_baton,
+ iterpool));
+ }
+ else
+ {
+ /* ### TODO: Implement a proper reject of a binary patch
+
+ This should at least setup things for a proper notification,
+ and perhaps install a normal text conflict. Unlike normal unified
+ diff based patches we have all the versions we would need for
+ that in a much easier format than can be obtained from the patch
+ file. */
target->skipped = TRUE;
}
}
+ else if (target->move_target_abspath)
+ {
+ /* ### Why do we do this?
+ BH: I don't know, but if we don't do this some tests
+ on git style patches break.
+
+ ### It would be much better to really move the actual file instead
+ of copying to a temporary file; move that to target and then
+ delete the original file
+
+ ### BH: I have absolutely no idea if moving directories would work.
+ */
+ if (target->kind_on_disk == svn_node_file)
+ {
+ /* Copy any remaining lines to target. (read: all lines) */
+ SVN_ERR(copy_lines_to_target(target->content, 0, scratch_pool));
+ if (!target->content->eof)
+ {
+ /* We could not copy the entire target file to the temporary
+ * file, and would truncate the target if we copied the
+ * temporary file on top of it. Skip this target. */
+ target->skipped = TRUE;
+ }
+ }
+ }
+
+ if (target->had_rejects || target->locally_deleted)
+ target->deleted = FALSE;
+
+ if (target->added
+ && !(target->locally_deleted || target->db_kind == svn_node_none))
+ {
+ target->added = FALSE;
+ }
+
+ /* Assume nothing changed. Will be updated via property hunks */
+ target->is_special = target->is_symlink;
/* Match property hunks. */
for (hash_index = apr_hash_first(scratch_pool, patch->prop_patches);
@@ -2369,8 +2749,8 @@ apply_one_patch(patch_target_t **patch_target, svn_patch_t *patch,
prop_name = apr_hash_this_key(hash_index);
prop_patch = apr_hash_this_val(hash_index);
- if (! strcmp(prop_name, SVN_PROP_SPECIAL))
- target->is_special = TRUE;
+ if (!strcmp(prop_name, SVN_PROP_SPECIAL))
+ target->is_special = (prop_patch->operation != svn_diff_op_deleted);
/* We'll store matched hunks in prop_content. */
prop_target = svn_hash_gets(target->prop_targets, prop_name);
@@ -2406,48 +2786,174 @@ apply_one_patch(patch_target_t **patch_target, svn_patch_t *patch,
}
}
- /* Apply or reject property hunks. */
- for (hash_index = apr_hash_first(scratch_pool, target->prop_targets);
- hash_index;
- hash_index = apr_hash_next(hash_index))
+ /* Match implied property hunks. */
+ if (patch->new_executable_bit != svn_tristate_unknown
+ && patch->new_executable_bit != patch->old_executable_bit
+ && svn_hash_gets(target->prop_targets, SVN_PROP_EXECUTABLE)
+ && !svn_hash_gets(patch->prop_patches, SVN_PROP_EXECUTABLE))
{
- prop_patch_target_t *prop_target;
+ hunk_info_t *hi;
+ svn_diff_hunk_t *hunk;
+ prop_patch_target_t *prop_target = svn_hash_gets(target->prop_targets,
+ SVN_PROP_EXECUTABLE);
+
+ if (patch->new_executable_bit == svn_tristate_true)
+ SVN_ERR(svn_diff_hunk__create_adds_single_line(
+ &hunk,
+ SVN_PROP_EXECUTABLE_VALUE,
+ patch,
+ result_pool,
+ iterpool));
+ else
+ SVN_ERR(svn_diff_hunk__create_deletes_single_line(
+ &hunk,
+ SVN_PROP_EXECUTABLE_VALUE,
+ patch,
+ result_pool,
+ iterpool));
+
+ /* Derive a hunk_info from hunk. */
+ SVN_ERR(get_hunk_info(&hi, target, prop_target->content,
+ hunk, 0 /* fuzz */, 0 /* previous_offset */,
+ ignore_whitespace,
+ TRUE /* is_prop_hunk */,
+ cancel_func, cancel_baton,
+ result_pool, iterpool));
+ APR_ARRAY_PUSH(prop_target->content->hunks, hunk_info_t *) = hi;
+ }
+
+ if (patch->new_symlink_bit != svn_tristate_unknown
+ && patch->new_symlink_bit != patch->old_symlink_bit
+ && svn_hash_gets(target->prop_targets, SVN_PROP_SPECIAL)
+ && !svn_hash_gets(patch->prop_patches, SVN_PROP_SPECIAL))
+ {
+ hunk_info_t *hi;
+ svn_diff_hunk_t *hunk;
+
+ prop_patch_target_t *prop_target = svn_hash_gets(target->prop_targets,
+ SVN_PROP_SPECIAL);
+
+ if (patch->new_symlink_bit == svn_tristate_true)
+ {
+ SVN_ERR(svn_diff_hunk__create_adds_single_line(
+ &hunk,
+ SVN_PROP_SPECIAL_VALUE,
+ patch,
+ result_pool,
+ iterpool));
+ target->is_special = TRUE;
+ }
+ else
+ {
+ SVN_ERR(svn_diff_hunk__create_deletes_single_line(
+ &hunk,
+ SVN_PROP_SPECIAL_VALUE,
+ patch,
+ result_pool,
+ iterpool));
+ target->is_special = FALSE;
+ }
+
+ /* Derive a hunk_info from hunk. */
+ SVN_ERR(get_hunk_info(&hi, target, prop_target->content,
+ hunk, 0 /* fuzz */, 0 /* previous_offset */,
+ ignore_whitespace,
+ TRUE /* is_prop_hunk */,
+ cancel_func, cancel_baton,
+ result_pool, iterpool));
+ APR_ARRAY_PUSH(prop_target->content->hunks, hunk_info_t *) = hi;
+ }
+
+ /* When the node is deleted or does not exist after the patch is applied
+ we should reject a few more property hunks that can't be applied even
+ though the source matched */
+ if (target->deleted
+ || (!target->added &&
+ (target->locally_deleted || target->db_kind == svn_node_none)))
+ {
+ for (hash_index = apr_hash_first(scratch_pool, target->prop_targets);
+ hash_index;
+ hash_index = apr_hash_next(hash_index))
+ {
+ prop_patch_target_t *prop_target = apr_hash_this_val(hash_index);
+
+ if (prop_target->operation == svn_diff_op_deleted)
+ continue;
+
+ for (i = 0; i < prop_target->content->hunks->nelts; i++)
+ {
+ hunk_info_t *hi;
+
+ hi = APR_ARRAY_IDX(prop_target->content->hunks, i, hunk_info_t*);
+
+ if (hi->already_applied || hi->rejected)
+ continue;
+ else
+ {
+ hi->rejected = TRUE;
+ prop_target->skipped = TRUE;
+
+ if (!target->deleted && !target->added)
+ target->skipped = TRUE;
+ }
+ }
+ }
+ }
- prop_target = apr_hash_this_val(hash_index);
+ /* Apply or reject property hunks. */
- for (i = 0; i < prop_target->content->hunks->nelts; i++)
+ prop_targets = svn_sort__hash(target->prop_targets,
+ svn_sort_compare_items_lexically,
+ scratch_pool);
+ for (i = 0; i < prop_targets->nelts; i++)
+ {
+ svn_sort__item_t item = APR_ARRAY_IDX(prop_targets, i, svn_sort__item_t);
+ prop_patch_target_t *prop_target = item.value;
+ svn_boolean_t applied_one = FALSE;
+ int j;
+
+ for (j = 0; j < prop_target->content->hunks->nelts; j++)
{
hunk_info_t *hi;
svn_pool_clear(iterpool);
- hi = APR_ARRAY_IDX(prop_target->content->hunks, i,
+ hi = APR_ARRAY_IDX(prop_target->content->hunks, j,
hunk_info_t *);
if (hi->already_applied)
- continue;
+ {
+ target->had_prop_already_applied = TRUE;
+ continue;
+ }
else if (hi->rejected)
SVN_ERR(reject_hunk(target, prop_target->content, hi->hunk,
prop_target->name,
iterpool));
else
- SVN_ERR(apply_hunk(target, prop_target->content, hi,
- prop_target->name,
- iterpool));
+ {
+ SVN_ERR(apply_hunk(target, prop_target->content, hi,
+ prop_target->name,
+ iterpool));
+ applied_one = TRUE;
+ }
}
- if (prop_target->content->existed)
- {
- /* Copy any remaining lines to target. */
- SVN_ERR(copy_lines_to_target(prop_target->content, 0,
- scratch_pool));
- if (! prop_target->content->eof)
- {
- /* We could not copy the entire target property to the
- * temporary file, and would truncate the target if we
- * copied the temporary file on top of it. Skip this target. */
- target->skipped = TRUE;
- }
- }
+ if (!applied_one)
+ prop_target->skipped = TRUE;
+
+ if (applied_one && prop_target->content->existed)
+ {
+ /* Copy any remaining lines to target. */
+ SVN_ERR(copy_lines_to_target(prop_target->content, 0,
+ scratch_pool));
+ if (! prop_target->content->eof)
+ {
+ /* We could not copy the entire target property to the
+ * temporary stream, and would truncate the target if we
+ * copied the temporary stream on top of it. Skip this target. */
+ prop_target->skipped = TRUE;
+ }
+ }
}
svn_pool_destroy(iterpool);
@@ -2460,57 +2966,9 @@ apply_one_patch(patch_target_t **patch_target, svn_patch_t *patch,
* will be closed later in write_out_rejected_hunks(). */
if (target->kind_on_disk == svn_node_file)
SVN_ERR(svn_io_file_close(target->file, scratch_pool));
-
- SVN_ERR(svn_io_file_close(target->patched_file, scratch_pool));
}
- if (! target->skipped)
- {
- apr_finfo_t working_file;
- apr_finfo_t patched_file;
-
- /* Get sizes of the patched temporary file and the working file.
- * We'll need those to figure out whether we should delete the
- * patched file. */
- SVN_ERR(svn_io_stat(&patched_file, target->patched_path,
- APR_FINFO_SIZE | APR_FINFO_LINK, scratch_pool));
- if (target->kind_on_disk == svn_node_file)
- SVN_ERR(svn_io_stat(&working_file, target->local_abspath,
- APR_FINFO_SIZE | APR_FINFO_LINK, scratch_pool));
- else
- working_file.size = 0;
-
- if (patched_file.size == 0 && working_file.size > 0)
- {
- /* If a unidiff removes all lines from a file, that usually
- * means deletion, so we can confidently schedule the target
- * for deletion. In the rare case where the unidiff was really
- * meant to replace a file with an empty one, this may not
- * be desirable. But the deletion can easily be reverted and
- * creating an empty file manually is not exactly hard either. */
- target->deleted = (target->db_kind == svn_node_file);
- }
- else if (patched_file.size == 0 && working_file.size == 0)
- {
- /* The target was empty or non-existent to begin with
- * and no content was changed by patching.
- * Report this as skipped if it didn't exist, unless in the special
- * case of adding an empty file which has properties set on it or
- * adding an empty file with a 'git diff' */
- if (target->kind_on_disk == svn_node_none
- && ! target->has_prop_changes
- && ! target->added)
- target->skipped = TRUE;
- }
- else if (patched_file.size > 0 && working_file.size == 0)
- {
- /* The patch has created a file. */
- if (target->locally_deleted)
- target->replaced = TRUE;
- else if (target->db_kind == svn_node_none)
- target->added = TRUE;
- }
- }
+ SVN_ERR(svn_io_file_close(target->patched_file, scratch_pool));
*patch_target = target;
@@ -2520,6 +2978,9 @@ apply_one_patch(patch_target_t **patch_target, svn_patch_t *patch,
/* Try to create missing parent directories for TARGET in the working copy
* rooted at ABS_WC_PATH, and add the parents to version control.
* If the parents cannot be created, mark the target as skipped.
+ *
+ * In dry run mode record missing parents in ALREADY_ADDED
+ *
* Use client context CTX. If DRY_RUN is true, do not create missing
* parents but issue notifications only.
* Use SCRATCH_POOL for temporary allocations. */
@@ -2528,6 +2989,7 @@ create_missing_parents(patch_target_t *target,
const char *abs_wc_path,
svn_client_ctx_t *ctx,
svn_boolean_t dry_run,
+ apr_array_header_t *targets_info,
apr_pool_t *scratch_pool)
{
const char *local_abspath;
@@ -2608,12 +3070,28 @@ create_missing_parents(patch_target_t *target,
for (i = present_components; i < components->nelts - 1; i++)
{
const char *component;
+ patch_target_info_t *pti;
svn_pool_clear(iterpool);
+ if (ctx->cancel_func)
+ SVN_ERR(ctx->cancel_func(ctx->cancel_baton));
+
component = APR_ARRAY_IDX(components, i, const char *);
local_abspath = svn_dirent_join(local_abspath, component,
scratch_pool);
+
+ if (target_is_added(targets_info, local_abspath, iterpool))
+ continue;
+
+ pti = apr_pcalloc(targets_info->pool, sizeof(*pti));
+
+ pti->local_abspath = apr_pstrdup(targets_info->pool,
+ local_abspath);
+ pti->added = TRUE;
+
+ APR_ARRAY_PUSH(targets_info, patch_target_info_t *) = pti;
+
if (dry_run)
{
if (ctx->notify_func2)
@@ -2634,10 +3112,6 @@ create_missing_parents(patch_target_t *target,
* to version control. Allow cancellation since we
* have not modified the working copy yet for this
* target. */
-
- if (ctx->cancel_func)
- SVN_ERR(ctx->cancel_func(ctx->cancel_baton));
-
SVN_ERR(svn_wc_add_from_disk3(ctx->wc_ctx, local_abspath,
NULL /*props*/,
FALSE /* skip checks */,
@@ -2653,11 +3127,17 @@ create_missing_parents(patch_target_t *target,
/* Install a patched TARGET into the working copy at ABS_WC_PATH.
* Use client context CTX to retrieve WC_CTX, and possibly doing
- * notifications. If DRY_RUN is TRUE, don't modify the working copy.
+ * notifications.
+ *
+ * Pass on ALREADY_ADDED to allow recording already added ancestors
+ * in dry-run mode.
+ *
+ * If DRY_RUN is TRUE, don't modify the working copy.
* Do temporary allocations in POOL. */
static svn_error_t *
install_patched_target(patch_target_t *target, const char *abs_wc_path,
svn_client_ctx_t *ctx, svn_boolean_t dry_run,
+ apr_array_header_t *targets_info,
apr_pool_t *pool)
{
if (target->deleted)
@@ -2671,13 +3151,15 @@ install_patched_target(patch_target_t *target, const char *abs_wc_path,
* notify about what we did before aborting. */
SVN_ERR(svn_wc_delete4(ctx->wc_ctx, target->local_abspath,
FALSE /* keep_local */, FALSE,
- NULL, NULL, NULL, NULL, pool));
+ ctx->cancel_func, ctx->cancel_baton,
+ NULL, NULL /* notify */,
+ pool));
}
}
else
{
svn_node_kind_t parent_db_kind;
- if (target->added || target->replaced)
+ if (target->added)
{
const char *parent_abspath;
@@ -2707,7 +3189,7 @@ install_patched_target(patch_target_t *target, const char *abs_wc_path,
}
else
SVN_ERR(create_missing_parents(target, abs_wc_path, ctx,
- dry_run, pool));
+ dry_run, targets_info, pool));
}
else
@@ -2723,6 +3205,8 @@ install_patched_target(patch_target_t *target, const char *abs_wc_path,
|| wc_kind != target->kind_on_disk)
{
target->skipped = TRUE;
+ if (wc_kind != target->kind_on_disk)
+ target->obstructed = TRUE;
}
}
@@ -2739,6 +3223,8 @@ install_patched_target(patch_target_t *target, const char *abs_wc_path,
SVN_ERR(svn_subst_create_specialfile(&stream,
target->local_abspath,
pool, pool));
+ if (target->git_symlink_format)
+ SVN_ERR(svn_stream_puts(stream, "link "));
SVN_ERR(svn_stream_copy3(patched_stream, stream,
ctx->cancel_func, ctx->cancel_baton,
pool));
@@ -2766,7 +3252,7 @@ install_patched_target(patch_target_t *target, const char *abs_wc_path,
ctx->cancel_func, ctx->cancel_baton, pool));
}
- if (target->added || target->replaced)
+ if (target->added)
{
/* The target file didn't exist previously,
* so add it to version control.
@@ -2800,7 +3286,8 @@ install_patched_target(patch_target_t *target, const char *abs_wc_path,
target->move_target_abspath,
TRUE, /* metadata_only */
FALSE, /* allow_mixed_revisions */
- NULL, NULL, NULL, NULL,
+ ctx->cancel_func, ctx->cancel_baton,
+ NULL, NULL,
pool));
/* Delete the patch target's old location from disk. */
@@ -2818,20 +3305,54 @@ install_patched_target(patch_target_t *target, const char *abs_wc_path,
*/
static svn_error_t *
write_out_rejected_hunks(patch_target_t *target,
+ const char *root_abspath,
svn_boolean_t dry_run,
- apr_pool_t *pool)
+ apr_pool_t *scratch_pool)
{
- SVN_ERR(svn_io_file_close(target->reject_file, pool));
-
if (! dry_run && (target->had_rejects || target->had_prop_rejects))
{
/* Write out rejected hunks, if any. */
- SVN_ERR(svn_io_copy_file(target->reject_path,
- apr_psprintf(pool, "%s.svnpatch.rej",
- target->local_abspath),
- FALSE, pool));
+ apr_file_t *reject_file;
+ svn_error_t *err;
+
+ err = svn_io_open_uniquely_named(&reject_file, NULL,
+ svn_dirent_dirname(target->local_abspath,
+ scratch_pool),
+ svn_dirent_basename(
+ target->local_abspath,
+ NULL),
+ ".svnpatch.rej",
+ svn_io_file_del_none,
+ scratch_pool, scratch_pool);
+ if (err && APR_STATUS_IS_ENOENT(err->apr_err))
+ {
+ /* The hunk applies to a file in a directory which does not exist.
+ * Put the reject file into the working copy root instead. */
+ svn_error_clear(err);
+ SVN_ERR(svn_io_open_uniquely_named(&reject_file, NULL,
+ root_abspath,
+ svn_dirent_basename(
+ target->local_abspath,
+ NULL),
+ ".svnpatch.rej",
+ svn_io_file_del_none,
+ scratch_pool, scratch_pool));
+ }
+ else
+ SVN_ERR(err);
+
+ SVN_ERR(svn_stream_reset(target->reject_stream));
+
+ /* svn_stream_copy3() closes the files for us */
+ SVN_ERR(svn_stream_copy3(target->reject_stream,
+ svn_stream_from_aprfile2(reject_file, FALSE,
+ scratch_pool),
+ NULL, NULL, scratch_pool));
/* ### TODO mark file as conflicted. */
}
+ else
+ SVN_ERR(svn_stream_close(target->reject_stream));
+
return SVN_NO_ERROR;
}
@@ -2845,6 +3366,13 @@ install_patched_prop_targets(patch_target_t *target,
{
apr_hash_index_t *hi;
apr_pool_t *iterpool;
+ const char *local_abspath;
+
+ /* Apply properties to a move target if there is one */
+ if (target->move_target_abspath)
+ local_abspath = target->move_target_abspath;
+ else
+ local_abspath = target->local_abspath;
iterpool = svn_pool_create(scratch_pool);
@@ -2861,11 +3389,14 @@ install_patched_prop_targets(patch_target_t *target,
if (ctx->cancel_func)
SVN_ERR(ctx->cancel_func(ctx->cancel_baton));
+ if (prop_target->skipped)
+ continue;
+
/* For a deleted prop we only set the value to NULL. */
if (prop_target->operation == svn_diff_op_deleted)
{
if (! dry_run)
- SVN_ERR(svn_wc_prop_set4(ctx->wc_ctx, target->local_abspath,
+ SVN_ERR(svn_wc_prop_set4(ctx->wc_ctx, local_abspath,
prop_target->name, NULL, svn_depth_empty,
TRUE /* skip_checks */,
NULL /* changelist_filter */,
@@ -2875,31 +3406,6 @@ install_patched_prop_targets(patch_target_t *target,
continue;
}
- /* If the patch target doesn't exist yet, the patch wants to add an
- * empty file with properties set on it. So create an empty file and
- * add it to version control. But if the patch was in the 'git format'
- * then the file has already been added.
- *
- * ### How can we tell whether the patch really wanted to create
- * ### an empty directory? */
- if (! target->has_text_changes
- && target->kind_on_disk == svn_node_none
- && ! target->added)
- {
- if (! dry_run)
- {
- SVN_ERR(svn_io_file_create_empty(target->local_abspath,
- scratch_pool));
- SVN_ERR(svn_wc_add_from_disk3(ctx->wc_ctx, target->local_abspath,
- NULL /*props*/,
- FALSE /* skip checks */,
- /* suppress notification */
- NULL, NULL,
- iterpool));
- }
- target->added = TRUE;
- }
-
/* Attempt to set the property, and reject all hunks if this
fails. If the property had a non-empty value, but now has
an empty one, we'll just delete the property altogether. */
@@ -2915,7 +3421,7 @@ install_patched_prop_targets(patch_target_t *target,
err = svn_wc_canonicalize_svn_prop(&canon_propval,
prop_target->name,
- prop_val, target->local_abspath,
+ prop_val, local_abspath,
target->db_kind,
TRUE, /* ### Skipping checks */
NULL, NULL,
@@ -2923,7 +3429,7 @@ install_patched_prop_targets(patch_target_t *target,
}
else
{
- err = svn_wc_prop_set4(ctx->wc_ctx, target->local_abspath,
+ err = svn_wc_prop_set4(ctx->wc_ctx, local_abspath,
prop_target->name, prop_val, svn_depth_empty,
TRUE /* skip_checks */,
NULL /* changelist_filter */,
@@ -3105,7 +3611,7 @@ static svn_error_t *
apply_patches(/* The path to the patch file. */
const char *patch_abspath,
/* The abspath to the working copy the patch should be applied to. */
- const char *abs_wc_path,
+ const char *root_abspath,
/* Indicates whether we're doing a dry run. */
svn_boolean_t dry_run,
/* Number of leading components to strip from patch target paths. */
@@ -3150,9 +3656,10 @@ apply_patches(/* The path to the patch file. */
patch_target_t *target;
svn_boolean_t filtered = FALSE;
- SVN_ERR(apply_one_patch(&target, patch, abs_wc_path,
+ SVN_ERR(apply_one_patch(&target, patch, root_abspath,
ctx->wc_ctx, strip_count,
ignore_whitespace, remove_tempfiles,
+ targets_info,
ctx->cancel_func, ctx->cancel_baton,
iterpool, iterpool));
@@ -3172,31 +3679,34 @@ apply_patches(/* The path to the patch file. */
target_info->local_abspath = apr_pstrdup(scratch_pool,
target->local_abspath);
target_info->deleted = target->deleted;
+ target_info->added = target->added;
if (! target->skipped)
{
- APR_ARRAY_PUSH(targets_info,
- patch_target_info_t *) = target_info;
-
if (target->has_text_changes
|| target->added
|| target->move_target_abspath
|| target->deleted)
- SVN_ERR(install_patched_target(target, abs_wc_path,
- ctx, dry_run, iterpool));
+ SVN_ERR(install_patched_target(target, root_abspath,
+ ctx, dry_run,
+ targets_info, iterpool));
if (target->has_prop_changes && (!target->deleted))
SVN_ERR(install_patched_prop_targets(target, ctx,
dry_run, iterpool));
- SVN_ERR(write_out_rejected_hunks(target, dry_run, iterpool));
- }
+ SVN_ERR(write_out_rejected_hunks(target, root_abspath,
+ dry_run, iterpool));
+
+ APR_ARRAY_PUSH(targets_info,
+ patch_target_info_t *) = target_info;
+ }
SVN_ERR(send_patch_notification(target, ctx, iterpool));
if (target->deleted && !target->skipped)
{
SVN_ERR(check_ancestor_delete(target_info->local_abspath,
- targets_info, abs_wc_path,
+ targets_info, root_abspath,
dry_run, ctx,
scratch_pool, iterpool));
}