2 * util.c : utility functions for the libsvn_client library
4 * ====================================================================
5 * Licensed to the Apache Software Foundation (ASF) under one
6 * or more contributor license agreements. See the NOTICE file
7 * distributed with this work for additional information
8 * regarding copyright ownership. The ASF licenses this file
9 * to you under the Apache License, Version 2.0 (the
10 * "License"); you may not use this file except in compliance
11 * with the License. You may obtain a copy of the License at
13 * http://www.apache.org/licenses/LICENSE-2.0
15 * Unless required by applicable law or agreed to in writing,
16 * software distributed under the License is distributed on an
17 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
18 * KIND, either express or implied. See the License for the
19 * specific language governing permissions and limitations
21 * ====================================================================
24 #include <apr_pools.h>
25 #include <apr_strings.h>
28 #include "svn_pools.h"
29 #include "svn_error.h"
30 #include "svn_types.h"
32 #include "svn_props.h"
35 #include "svn_client.h"
37 #include "private/svn_client_private.h"
38 #include "private/svn_wc_private.h"
39 #include "private/svn_fspath.h"
43 #include "svn_private_config.h"
45 svn_client__pathrev_t *
46 svn_client__pathrev_create(const char *repos_root_url,
47 const char *repos_uuid,
50 apr_pool_t *result_pool)
52 svn_client__pathrev_t *loc = apr_palloc(result_pool, sizeof(*loc));
54 SVN_ERR_ASSERT_NO_RETURN(svn_path_is_url(repos_root_url));
55 SVN_ERR_ASSERT_NO_RETURN(svn_path_is_url(url));
57 loc->repos_root_url = apr_pstrdup(result_pool, repos_root_url);
58 loc->repos_uuid = apr_pstrdup(result_pool, repos_uuid);
60 loc->url = apr_pstrdup(result_pool, url);
64 svn_client__pathrev_t *
65 svn_client__pathrev_create_with_relpath(const char *repos_root_url,
66 const char *repos_uuid,
69 apr_pool_t *result_pool)
71 SVN_ERR_ASSERT_NO_RETURN(svn_relpath_is_canonical(relpath));
73 return svn_client__pathrev_create(
74 repos_root_url, repos_uuid, rev,
75 svn_path_url_add_component2(repos_root_url, relpath, result_pool),
80 svn_client__pathrev_create_with_session(svn_client__pathrev_t **pathrev_p,
81 svn_ra_session_t *ra_session,
84 apr_pool_t *result_pool)
86 svn_client__pathrev_t *pathrev = apr_palloc(result_pool, sizeof(*pathrev));
88 SVN_ERR_ASSERT(svn_path_is_url(url));
90 SVN_ERR(svn_ra_get_repos_root2(ra_session, &pathrev->repos_root_url,
92 SVN_ERR(svn_ra_get_uuid2(ra_session, &pathrev->repos_uuid, result_pool));
94 pathrev->url = apr_pstrdup(result_pool, url);
99 svn_client__pathrev_t *
100 svn_client__pathrev_dup(const svn_client__pathrev_t *pathrev,
101 apr_pool_t *result_pool)
103 return svn_client__pathrev_create(
104 pathrev->repos_root_url, pathrev->repos_uuid,
105 pathrev->rev, pathrev->url, result_pool);
108 svn_client__pathrev_t *
109 svn_client__pathrev_join_relpath(const svn_client__pathrev_t *pathrev,
111 apr_pool_t *result_pool)
113 return svn_client__pathrev_create(
114 pathrev->repos_root_url, pathrev->repos_uuid, pathrev->rev,
115 svn_path_url_add_component2(pathrev->url, relpath, result_pool),
120 svn_client__pathrev_relpath(const svn_client__pathrev_t *pathrev,
121 apr_pool_t *result_pool)
123 return svn_uri_skip_ancestor(pathrev->repos_root_url, pathrev->url,
128 svn_client__pathrev_fspath(const svn_client__pathrev_t *pathrev,
129 apr_pool_t *result_pool)
131 return svn_fspath__canonicalize(svn_uri_skip_ancestor(
132 pathrev->repos_root_url, pathrev->url,
138 svn_client_commit_item3_t *
139 svn_client_commit_item3_create(apr_pool_t *pool)
141 svn_client_commit_item3_t *item = apr_pcalloc(pool, sizeof(*item));
143 item->revision = SVN_INVALID_REVNUM;
144 item->copyfrom_rev = SVN_INVALID_REVNUM;
145 item->kind = svn_node_unknown;
150 svn_client_commit_item3_t *
151 svn_client_commit_item3_dup(const svn_client_commit_item3_t *item,
154 svn_client_commit_item3_t *new_item = apr_palloc(pool, sizeof(*new_item));
159 new_item->path = apr_pstrdup(pool, new_item->path);
162 new_item->url = apr_pstrdup(pool, new_item->url);
164 if (new_item->copyfrom_url)
165 new_item->copyfrom_url = apr_pstrdup(pool, new_item->copyfrom_url);
167 if (new_item->incoming_prop_changes)
168 new_item->incoming_prop_changes =
169 svn_prop_array_dup(new_item->incoming_prop_changes, pool);
171 if (new_item->outgoing_prop_changes)
172 new_item->outgoing_prop_changes =
173 svn_prop_array_dup(new_item->outgoing_prop_changes, pool);
175 if (new_item->session_relpath)
176 new_item->session_relpath = apr_pstrdup(pool, new_item->session_relpath);
178 if (new_item->moved_from_abspath)
179 new_item->moved_from_abspath = apr_pstrdup(pool,
180 new_item->moved_from_abspath);
186 svn_client__wc_node_get_base(svn_client__pathrev_t **base_p,
187 const char *wc_abspath,
188 svn_wc_context_t *wc_ctx,
189 apr_pool_t *result_pool,
190 apr_pool_t *scratch_pool)
194 *base_p = apr_palloc(result_pool, sizeof(**base_p));
196 SVN_ERR(svn_wc__node_get_base(NULL,
199 &(*base_p)->repos_root_url,
200 &(*base_p)->repos_uuid,
203 TRUE /* ignore_enoent */,
204 result_pool, scratch_pool));
205 if ((*base_p)->repos_root_url && relpath)
207 (*base_p)->url = svn_path_url_add_component2(
208 (*base_p)->repos_root_url, relpath, result_pool);
218 svn_client__wc_node_get_origin(svn_client__pathrev_t **origin_p,
219 const char *wc_abspath,
220 svn_client_ctx_t *ctx,
221 apr_pool_t *result_pool,
222 apr_pool_t *scratch_pool)
226 *origin_p = apr_palloc(result_pool, sizeof(**origin_p));
228 SVN_ERR(svn_wc__node_get_origin(NULL /* is_copy */,
231 &(*origin_p)->repos_root_url,
232 &(*origin_p)->repos_uuid,
234 ctx->wc_ctx, wc_abspath,
235 FALSE /* scan_deleted */,
236 result_pool, scratch_pool));
237 if ((*origin_p)->repos_root_url && relpath)
239 (*origin_p)->url = svn_path_url_add_component2(
240 (*origin_p)->repos_root_url, relpath, result_pool);
250 svn_client_get_repos_root(const char **repos_root,
251 const char **repos_uuid,
252 const char *abspath_or_url,
253 svn_client_ctx_t *ctx,
254 apr_pool_t *result_pool,
255 apr_pool_t *scratch_pool)
257 svn_ra_session_t *ra_session;
259 /* If PATH_OR_URL is a local path we can fetch the repos root locally. */
260 if (!svn_path_is_url(abspath_or_url))
263 err = svn_wc__node_get_repos_info(NULL, NULL, repos_root, repos_uuid,
264 ctx->wc_ctx, abspath_or_url,
265 result_pool, scratch_pool);
269 if (err->apr_err != SVN_ERR_WC_PATH_NOT_FOUND
270 && err->apr_err != SVN_ERR_WC_NOT_WORKING_COPY)
271 return svn_error_trace(err);
273 svn_error_clear(err);
282 /* If PATH_OR_URL was a URL, we use the RA layer to look it up. */
283 SVN_ERR(svn_client_open_ra_session2(&ra_session, abspath_or_url, NULL,
284 ctx, scratch_pool, scratch_pool));
287 SVN_ERR(svn_ra_get_repos_root2(ra_session, repos_root, result_pool));
289 SVN_ERR(svn_ra_get_uuid2(ra_session, repos_uuid, result_pool));
294 const svn_opt_revision_t *
295 svn_cl__rev_default_to_head_or_base(const svn_opt_revision_t *revision,
296 const char *path_or_url)
298 static svn_opt_revision_t head_rev = { svn_opt_revision_head, { 0 } };
299 static svn_opt_revision_t base_rev = { svn_opt_revision_base, { 0 } };
301 if (revision->kind == svn_opt_revision_unspecified)
302 return svn_path_is_url(path_or_url) ? &head_rev : &base_rev;
306 const svn_opt_revision_t *
307 svn_cl__rev_default_to_head_or_working(const svn_opt_revision_t *revision,
308 const char *path_or_url)
310 static svn_opt_revision_t head_rev = { svn_opt_revision_head, { 0 } };
311 static svn_opt_revision_t work_rev = { svn_opt_revision_working, { 0 } };
313 if (revision->kind == svn_opt_revision_unspecified)
314 return svn_path_is_url(path_or_url) ? &head_rev : &work_rev;
318 const svn_opt_revision_t *
319 svn_cl__rev_default_to_peg(const svn_opt_revision_t *revision,
320 const svn_opt_revision_t *peg_revision)
322 if (revision->kind == svn_opt_revision_unspecified)
328 svn_client__assert_homogeneous_target_type(const apr_array_header_t *targets)
330 svn_boolean_t wc_present = FALSE, url_present = FALSE;
333 for (i = 0; i < targets->nelts; ++i)
335 const char *target = APR_ARRAY_IDX(targets, i, const char *);
336 if (! svn_path_is_url(target))
340 if (url_present && wc_present)
341 return svn_error_createf(SVN_ERR_ILLEGAL_TARGET, NULL,
342 _("Cannot mix repository and working copy "
349 struct shim_callbacks_baton
351 svn_wc_context_t *wc_ctx;
352 apr_hash_t *relpath_map;
356 fetch_props_func(apr_hash_t **props,
359 svn_revnum_t base_revision,
360 apr_pool_t *result_pool,
361 apr_pool_t *scratch_pool)
363 struct shim_callbacks_baton *scb = baton;
364 const char *local_abspath;
366 local_abspath = svn_hash_gets(scb->relpath_map, path);
369 *props = apr_hash_make(result_pool);
373 /* Reads the pristine properties of WORKING, not those of BASE */
374 SVN_ERR(svn_wc_get_pristine_props(props, scb->wc_ctx, local_abspath,
375 result_pool, scratch_pool));
378 *props = apr_hash_make(result_pool);
384 fetch_kind_func(svn_node_kind_t *kind,
387 svn_revnum_t base_revision,
388 apr_pool_t *scratch_pool)
390 struct shim_callbacks_baton *scb = baton;
391 const char *local_abspath;
393 local_abspath = svn_hash_gets(scb->relpath_map, path);
396 *kind = svn_node_unknown;
399 /* Reads the WORKING kind. Not the BASE kind */
400 SVN_ERR(svn_wc_read_kind2(kind, scb->wc_ctx, local_abspath,
401 TRUE, FALSE, scratch_pool));
407 fetch_base_func(const char **filename,
410 svn_revnum_t base_revision,
411 apr_pool_t *result_pool,
412 apr_pool_t *scratch_pool)
414 struct shim_callbacks_baton *scb = baton;
415 const char *local_abspath;
416 svn_stream_t *pristine_stream;
417 svn_stream_t *temp_stream;
420 local_abspath = svn_hash_gets(scb->relpath_map, path);
427 /* Reads the pristine of WORKING, not of BASE */
428 err = svn_wc_get_pristine_contents2(&pristine_stream, scb->wc_ctx,
429 local_abspath, scratch_pool,
431 if (err && err->apr_err == SVN_ERR_WC_PATH_NOT_FOUND)
433 svn_error_clear(err);
438 return svn_error_trace(err);
440 SVN_ERR(svn_stream_open_unique(&temp_stream, filename, NULL,
441 svn_io_file_del_on_pool_cleanup,
442 result_pool, scratch_pool));
443 SVN_ERR(svn_stream_copy3(pristine_stream, temp_stream, NULL, NULL,
449 svn_delta_shim_callbacks_t *
450 svn_client__get_shim_callbacks(svn_wc_context_t *wc_ctx,
451 apr_hash_t *relpath_map,
452 apr_pool_t *result_pool)
454 svn_delta_shim_callbacks_t *callbacks =
455 svn_delta_shim_callbacks_default(result_pool);
456 struct shim_callbacks_baton *scb = apr_pcalloc(result_pool, sizeof(*scb));
458 scb->wc_ctx = wc_ctx;
460 scb->relpath_map = relpath_map;
462 scb->relpath_map = apr_hash_make(result_pool);
464 callbacks->fetch_props_func = fetch_props_func;
465 callbacks->fetch_kind_func = fetch_kind_func;
466 callbacks->fetch_base_func = fetch_base_func;
467 callbacks->fetch_baton = scb;