| /* |
| * copy-cmd.c -- Subversion copy command |
| * |
| * ==================================================================== |
| * Licensed to the Apache Software Foundation (ASF) under one |
| * or more contributor license agreements. See the NOTICE file |
| * distributed with this work for additional information |
| * regarding copyright ownership. The ASF licenses this file |
| * to you under the Apache License, Version 2.0 (the |
| * "License"); you may not use this file except in compliance |
| * with the License. You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, |
| * software distributed under the License is distributed on an |
| * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
| * KIND, either express or implied. See the License for the |
| * specific language governing permissions and limitations |
| * under the License. |
| * ==================================================================== |
| */ |
| |
| /* ==================================================================== */ |
| |
| |
| |
| /*** Includes. ***/ |
| |
| #include "svn_client.h" |
| #include "svn_path.h" |
| #include "svn_error.h" |
| #include "cl.h" |
| |
| #include "svn_private_config.h" |
| |
| |
| /*** Code. ***/ |
| |
| /* This implements the `svn_opt_subcommand_t' interface. */ |
| svn_error_t * |
| svn_cl__copy(apr_getopt_t *os, |
| void *baton, |
| apr_pool_t *pool) |
| { |
| svn_cl__opt_state_t *opt_state = ((svn_cl__cmd_baton_t *) baton)->opt_state; |
| svn_client_ctx_t *ctx = ((svn_cl__cmd_baton_t *) baton)->ctx; |
| apr_array_header_t *targets, *sources; |
| const char *src_path, *dst_path; |
| svn_boolean_t srcs_are_urls, dst_is_url; |
| svn_error_t *err; |
| int i; |
| |
| SVN_ERR(svn_cl__args_to_target_array_print_reserved(&targets, os, |
| opt_state->targets, |
| ctx, FALSE, pool)); |
| if (targets->nelts < 2) |
| return svn_error_create(SVN_ERR_CL_INSUFFICIENT_ARGS, 0, NULL); |
| |
| /* Get the src list and associated peg revs */ |
| sources = apr_array_make(pool, targets->nelts - 1, |
| sizeof(svn_client_copy_source_t *)); |
| for (i = 0; i < (targets->nelts - 1); i++) |
| { |
| const char *target = APR_ARRAY_IDX(targets, i, const char *); |
| svn_client_copy_source_t *source = apr_palloc(pool, sizeof(*source)); |
| const char *src; |
| svn_opt_revision_t *peg_revision = apr_palloc(pool, |
| sizeof(*peg_revision)); |
| |
| err = svn_opt_parse_path(peg_revision, &src, target, pool); |
| |
| if (err) |
| { |
| /* Issue #3606: 'svn cp .@HEAD target' gives |
| svn: '@HEAD' is just a peg revision. Maybe try '@HEAD@' instead? |
| |
| This is caused by a first round of canonicalization in |
| svn_cl__args_to_target_array_print_reserved(). Undo that in an |
| attempt to fix this issue without revving many apis. |
| */ |
| if (*target == '@' && err->apr_err == SVN_ERR_BAD_FILENAME) |
| { |
| svn_error_t *err2; |
| |
| err2 = svn_opt_parse_path(peg_revision, &src, |
| apr_pstrcat(pool, ".", target, |
| (const char *)NULL), pool); |
| |
| if (err2) |
| { |
| /* Fix attempt failed; return original error */ |
| svn_error_clear(err2); |
| } |
| else |
| { |
| /* Error resolved. Use path */ |
| svn_error_clear(err); |
| err = NULL; |
| } |
| } |
| |
| if (err) |
| return svn_error_trace(err); |
| } |
| |
| source->path = src; |
| source->revision = &(opt_state->start_revision); |
| source->peg_revision = peg_revision; |
| |
| APR_ARRAY_PUSH(sources, svn_client_copy_source_t *) = source; |
| } |
| |
| /* Get DST_PATH (the target path or URL) and check that no peg revision is |
| * specified for it. */ |
| { |
| const char *tgt = APR_ARRAY_IDX(targets, targets->nelts - 1, const char *); |
| svn_opt_revision_t peg; |
| |
| SVN_ERR(svn_opt_parse_path(&peg, &dst_path, tgt, pool)); |
| if (peg.kind != svn_opt_revision_unspecified) |
| return svn_error_createf(SVN_ERR_ILLEGAL_TARGET, NULL, |
| _("'%s': a peg revision is not allowed here"), |
| tgt); |
| } |
| |
| /* Figure out which type of notification to use. |
| (There is no need to check that the src paths are homogeneous; |
| svn_client_copy6() through its subroutine try_copy() will return an |
| error if they are not.) */ |
| src_path = APR_ARRAY_IDX(targets, 0, const char *); |
| srcs_are_urls = svn_path_is_url(src_path); |
| dst_is_url = svn_path_is_url(dst_path); |
| |
| if ((! srcs_are_urls) && (! dst_is_url)) |
| { |
| /* WC->WC */ |
| } |
| else if ((! srcs_are_urls) && (dst_is_url)) |
| { |
| /* WC->URL : Use notification. */ |
| if (! opt_state->quiet) |
| SVN_ERR(svn_cl__notifier_mark_wc_to_repos_copy(ctx->notify_baton2)); |
| } |
| else if ((srcs_are_urls) && (! dst_is_url)) |
| { |
| /* URL->WC : Use checkout-style notification. */ |
| if (! opt_state->quiet) |
| SVN_ERR(svn_cl__notifier_mark_checkout(ctx->notify_baton2)); |
| } |
| else |
| { |
| /* URL -> URL */ |
| } |
| |
| if (! dst_is_url) |
| { |
| ctx->log_msg_func3 = NULL; |
| if (opt_state->message || opt_state->filedata || opt_state->revprop_table) |
| return svn_error_create |
| (SVN_ERR_CL_UNNECESSARY_LOG_MESSAGE, NULL, |
| _("Local, non-commit operations do not take a log message " |
| "or revision properties")); |
| } |
| |
| if (ctx->log_msg_func3) |
| SVN_ERR(svn_cl__make_log_msg_baton(&(ctx->log_msg_baton3), opt_state, |
| NULL, ctx->config, pool)); |
| |
| err = svn_client_copy7(sources, dst_path, TRUE, |
| opt_state->parents, opt_state->ignore_externals, |
| FALSE /* metadata_only */, |
| opt_state->pin_externals, |
| NULL, /* pin all externals */ |
| opt_state->revprop_table, |
| (opt_state->quiet ? NULL : svn_cl__print_commit_info), |
| NULL, |
| ctx, pool); |
| |
| if (ctx->log_msg_func3) |
| SVN_ERR(svn_cl__cleanup_log_msg(ctx->log_msg_baton3, err, pool)); |
| else if (err) |
| return svn_error_trace(err); |
| |
| return SVN_NO_ERROR; |
| } |