diff options
Diffstat (limited to 'subversion/svn/checkout-cmd.c')
-rw-r--r-- | subversion/svn/checkout-cmd.c | 173 |
1 files changed, 173 insertions, 0 deletions
diff --git a/subversion/svn/checkout-cmd.c b/subversion/svn/checkout-cmd.c new file mode 100644 index 0000000..6c192a0 --- /dev/null +++ b/subversion/svn/checkout-cmd.c @@ -0,0 +1,173 @@ +/* + * checkout-cmd.c -- Subversion checkout 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_dirent_uri.h" +#include "svn_path.h" +#include "svn_error.h" +#include "svn_pools.h" +#include "cl.h" + +#include "svn_private_config.h" + + +/*** Code. ***/ + +/* + This is what it does + + - case 1: one URL + $ svn co http://host/repos/module + checkout into ./module/ + + - case 2: one URL and explicit path + $ svn co http://host/repos/module path + checkout into ./path/ + + - case 3: multiple URLs + $ svn co http://host1/repos1/module1 http://host2/repos2/module2 + checkout into ./module1/ and ./module2/ + + - case 4: multiple URLs and explicit path + $ svn co http://host1/repos1/module1 http://host2/repos2/module2 path + checkout into ./path/module1/ and ./path/module2/ + + Is this the same as CVS? Does it matter if it is not? +*/ + + +/* This implements the `svn_opt_subcommand_t' interface. */ +svn_error_t * +svn_cl__checkout(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_pool_t *subpool; + apr_array_header_t *targets; + const char *last_target, *local_dir; + int i; + + SVN_ERR(svn_cl__args_to_target_array_print_reserved(&targets, os, + opt_state->targets, + ctx, FALSE, pool)); + + if (! targets->nelts) + return svn_error_create(SVN_ERR_CL_INSUFFICIENT_ARGS, NULL, NULL); + + /* Determine LOCAL_DIR (case 1: URL basename; 2,4: specified; 3: "") + * and leave TARGETS holding just the source URLs. */ + last_target = APR_ARRAY_IDX(targets, targets->nelts - 1, const char *); + if (svn_path_is_url(last_target)) + { + if (targets->nelts == 1) + { + svn_opt_revision_t pegrev; + + /* Use the URL basename, discarding any peg revision. */ + SVN_ERR(svn_opt_parse_path(&pegrev, &local_dir, last_target, pool)); + local_dir = svn_uri_basename(local_dir, pool); + } + else + { + local_dir = ""; + } + } + else + { + if (targets->nelts == 1) + /* What? They gave us one target, and it wasn't a URL. */ + return svn_error_create(SVN_ERR_CL_ARG_PARSING_ERROR, NULL, NULL); + + apr_array_pop(targets); + local_dir = last_target; + } + + if (! opt_state->quiet) + SVN_ERR(svn_cl__notifier_mark_checkout(ctx->notify_baton2)); + + subpool = svn_pool_create(pool); + for (i = 0; i < targets->nelts; ++i) + { + const char *repos_url = APR_ARRAY_IDX(targets, i, const char *); + const char *target_dir; + const char *true_url; + svn_opt_revision_t revision = opt_state->start_revision; + svn_opt_revision_t peg_revision; + + svn_pool_clear(subpool); + + SVN_ERR(svn_cl__check_cancel(ctx->cancel_baton)); + + /* Validate the REPOS_URL */ + if (! svn_path_is_url(repos_url)) + return svn_error_createf + (SVN_ERR_BAD_URL, NULL, + _("'%s' does not appear to be a URL"), repos_url); + + /* Get a possible peg revision. */ + SVN_ERR(svn_opt_parse_path(&peg_revision, &true_url, repos_url, + subpool)); + + /* Use sub-directory of destination if checking-out multiple URLs */ + if (targets->nelts == 1) + { + target_dir = local_dir; + } + else + { + target_dir = svn_dirent_join(local_dir, + svn_uri_basename(true_url, subpool), + subpool); + } + + /* Checkout doesn't accept an unspecified revision, so default to + the peg revision, or to HEAD if there wasn't a peg. */ + if (revision.kind == svn_opt_revision_unspecified) + { + if (peg_revision.kind != svn_opt_revision_unspecified) + revision = peg_revision; + else + revision.kind = svn_opt_revision_head; + } + + SVN_ERR(svn_client_checkout3 + (NULL, true_url, target_dir, + &peg_revision, + &revision, + opt_state->depth, + opt_state->ignore_externals, + opt_state->force, + ctx, subpool)); + } + svn_pool_destroy(subpool); + + return SVN_NO_ERROR; +} |