-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathGit-Book_10.html
1615 lines (1464 loc) · 339 KB
/
Git-Book_10.html
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
<!DOCTYPE html><html lang="en"><head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<meta name="generator" content="Asciidoctor 2.0.16">
<meta name="author" content="Valentin Haenel, Julius Plenz">
<title>Git Book — 9. Interacting with Other Version Control Systems</title>
<link rel="stylesheet" href="https://fonts.googleapis.com/css?family=Open+Sans:300,300italic,400,400italic,600,600italic%7CNoto+Serif:400,400italic,700,700italic%7CDroid+Sans+Mono:400,700">
<link rel="stylesheet" href="style0.css" type="text/css">
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/4.7.0/css/font-awesome.min.css">
<link rel="stylesheet" href="style1.css" type="text/css">
<link rel="stylesheet" href="asciidoctor-chunker.css" type="text/css"></head>
<body class="book toc2 toc-left">
<div id="header">
<h1>Git: Distributed Version Control for Code and Documents</h1>
<div class="details">
<span id="author" class="author">Valentin Haenel</span><br>
<span id="author2" class="author">Julius Plenz</span><br>
<span id="revnumber">version 3.0</span>
<br><span id="revremark">Beta Preview</span>
</div>
<div id="toc" class="toc2">
<div id="toctitle">Table of Contents</div>
<ul class="sectlevel1">
<li><a href="index.html">Title Page</a></li><li><a href="Git-Book_1.html">Preface</a>
<ul class="sectlevel2">
<li><a href="Git-Book_1.html#sec.reader">Who Is This Book Intended For?</a></li>
<li><a href="Git-Book_1.html#sec.structure">How to Read the Book?</a></li>
<li><a href="Git-Book_1.html#sec.conventions">Conventions</a></li>
<li><a href="Git-Book_1.html#sec.install-git-repo">Installation and “The Git-Repository”</a></li>
<li><a href="Git-Book_1.html#sec.docs">Documentation and Help</a></li>
<li><a href="Git-Book_1.html#sec.contact">Downloads and Contacts</a></li>
<li><a href="Git-Book_1.html#sec.acknowledgements">Acknowledgements</a></li>
<li><a href="Git-Book_1.html#sec.preface-2nd-edition">Preface to the 2nd Edition</a></li>
<li><a href="Git-Book_1.html#sec.preface-cc-edition">Preface to the Creative Commons Edition</a></li>
</ul>
</li>
<li><a href="Git-Book_2.html">1. Introduction and First Steps</a>
<ul class="sectlevel2">
<li><a href="Git-Book_2.html#sec.terminology">1.1. Basic Terminology</a></li>
<li><a href="Git-Book_2.html#sec.first-steps">1.2. First Steps with Git</a></li>
<li><a href="Git-Book_2.html#chap.git-config">1.3. Configuring Git</a></li>
</ul>
</li>
<li><a href="Git-Book_3.html">2. The Basics</a>
<ul class="sectlevel2">
<li><a href="Git-Book_3.html#sec.basics">2.1. Git Commands</a></li>
<li><a href="Git-Book_3.html#sec.object-model">2.2. The Object Model</a></li>
</ul>
</li>
<li><a href="Git-Book_4.html">3. Practical Version Control</a>
<ul class="sectlevel2">
<li><a href="Git-Book_4.html#sec.branches">3.1. References: Branches and Tags</a></li>
<li><a href="Git-Book_4.html#sec.undo">3.2. Restoring Versions</a></li>
<li><a href="Git-Book_4.html#sec.merge">3.3. Merging Branches</a></li>
<li><a href="Git-Book_4.html#sec.merge-conflicts">3.4. Resolving Merge Conflicts</a></li>
<li><a href="Git-Book_4.html#sec.cherry-pick">3.5. Taking over Individual Commits: Cherry Picking</a></li>
<li><a href="Git-Book_4.html#sec.visualization">3.6. Visualizing Repositories</a></li>
<li><a href="Git-Book_4.html#sec.reflog">3.7. Reflog</a></li>
</ul>
</li>
<li><a href="Git-Book_5.html">4. Advanced Concepts</a>
<ul class="sectlevel2">
<li><a href="Git-Book_5.html#sec.rebase">4.1. Moving commits — Rebase</a></li>
<li><a href="Git-Book_5.html#sec.rebase-i">4.2. Rewriting History — Interactive Rebase</a></li>
<li><a href="Git-Book_5.html#sec.blame">4.3. Who Made These Changes? — Git Blame</a></li>
<li><a href="Git-Book_5.html#sec.ignore">4.4. Ignoring Files</a></li>
<li><a href="Git-Book_5.html#sec.stash">4.5. Outsourcing Changes — Git Stash</a></li>
<li><a href="Git-Book_5.html#sec.notes">4.6. Annotating Commits — Git Notes</a></li>
<li><a href="Git-Book_5.html#sec.multi-root">4.7. Multiple Root Commits</a></li>
<li><a href="Git-Book_5.html#sec.bisect">4.8. Finding Regressions — Git Bisect</a></li>
</ul>
</li>
<li><a href="Git-Book_6.html">5. Distributed Git</a>
<ul class="sectlevel2">
<li><a href="Git-Book_6.html#sec.distributed-systems">5.1. How Does Distributed Version Control Work?</a></li>
<li><a href="Git-Book_6.html#sec.clone">5.2. Cloning Repositories</a></li>
<li><a href="Git-Book_6.html#sec.downloading-commits">5.3. Downloading Commits</a></li>
<li><a href="Git-Book_6.html#sec.uploading-commits">5.4. Uploading Commits: git push</a></li>
<li><a href="Git-Book_6.html#sec.remotes-check">5.5. Examining Remotes</a></li>
<li><a href="Git-Book_6.html#sec.multi-remote">5.6. Distributed Workflow with Multiple Remotes</a></li>
<li><a href="Git-Book_6.html#sec.managing-remotes">5.7. Managing Remotes</a></li>
<li><a href="Git-Book_6.html#sec.remote-tags">5.8. Exchanging Tags</a></li>
<li><a href="Git-Book_6.html#sec.patch-queue">5.9. Patches via E-mail</a></li>
<li><a href="Git-Book_6.html#sec.dictator">5.10. A Distributed, Hierarchical Workflow</a></li>
<li><a href="Git-Book_6.html#sec.subprojects">5.11. Managing Subprojects</a></li>
</ul>
</li>
<li><a href="Git-Book_7.html">6. Workflows</a>
<ul class="sectlevel2">
<li><a href="Git-Book_7.html#sec.workflows-user">6.1. User</a></li>
<li><a href="Git-Book_7.html#sec.branching-model">6.2. A Branching Model</a></li>
<li><a href="Git-Book_7.html#sec.releases-management">6.3. Release Management</a></li>
</ul>
</li>
<li><a href="Git-Book_8.html">7. Git Servers</a>
<ul class="sectlevel2">
<li><a href="Git-Book_8.html#sec.server">7.1. Hosting a Git Server</a></li>
<li><a href="Git-Book_8.html#sec.gitolite">7.2. Gitolite: Simple Git Hosting</a></li>
<li><a href="Git-Book_8.html#sec.git-daemon">7.3. Git Daemon: Anonymous Read-Only Access</a></li>
<li><a href="Git-Book_8.html#sec.gitweb">7.4. Gitweb: The Integrated Web Frontend</a></li>
<li><a href="Git-Book_8.html#sec.cgit">7.5. CGit — CGI for Git</a></li>
</ul>
</li>
<li><a href="Git-Book_9.html">8. Git Automation</a>
<ul class="sectlevel2">
<li><a href="Git-Book_9.html#sec.attributes">8.1. Git Attributes — Treating Files Separately</a></li>
<li><a href="Git-Book_9.html#sec.hooks">8.2. Hooks</a></li>
<li><a href="Git-Book_9.html#sec.scripting">8.3. Writing Your Own Git Commands</a></li>
<li><a href="Git-Book_9.html#sec.filter-branch">8.4. Rewriting Version History</a></li>
</ul>
</li>
<li class="current"><a href="Git-Book_10.html">9. Interacting with Other Version Control Systems</a>
<ul class="sectlevel2">
<li class="current"><a href="Git-Book_10.html#sec.subversion">9.1. Subversion</a></li>
<li class="current"><a href="Git-Book_10.html#sec.fast-import">9.2. Custom Importers</a></li>
</ul>
</li>
<li><a href="Git-Book_11.html">10. Shell Integration</a>
<ul class="sectlevel2">
<li><a href="Git-Book_11.html#sec.bash-integration">10.1. Git and the Bash</a></li>
<li><a href="Git-Book_11.html#sec.zsh-integration">10.2. Git and the Z-Shell</a></li>
</ul>
</li>
<li><a href="Git-Book_12.html">11. GitHub</a></li>
<li><a href="Git-Book_13.html">Appendix A: Installation</a>
<ul class="sectlevel2">
<li><a href="Git-Book_13.html#linux">A.1. Linux</a></li>
<li><a href="Git-Book_13.html#sec.osx">A.2. Mac OS X</a></li>
<li><a href="Git-Book_13.html#sec.windows">A.3. Windows</a></li>
</ul>
</li>
<li><a href="Git-Book_14.html">Appendix B: Repository Structure</a>
<ul class="sectlevel2">
<li><a href="Git-Book_14.html#sec.gc">B.1. Cleaning Up</a></li>
<li><a href="Git-Book_14.html#sec.gc-performance">B.2. Performance</a></li>
</ul>
</li>
</ul>
</div>
</div>
<div id="content"><div class="sect1">
<h2 id="sec.git-svn-cvs"><a class="anchor" href="Git-Book_10.html"></a>9. Interacting with Other Version Control Systems</h2>
<div class="sectionbody">
<div class="paragraph">
<p>Git has interfaces to other version control systems, which are important for two basic use cases:</p>
</div>
<div class="dlist">
<dl>
<dt class="hdlist1"><em>Bidirectional communication</em> </dt>
<dd>
<p>You want to develop locally in a Git repository, but also transfer the changes to an external repository or import changes from there to Git.</p>
</dd>
<dt class="hdlist1"><em>Migration</em> </dt>
<dd>
<p>You want to import the version history stored in an existing repository of another system into Git.</p>
</dd>
</dl>
</div>
<div class="paragraph">
<p>Git offers the following interfaces — all of which allow two-way communication and complete conversion:</p>
</div>
<div class="dlist">
<dl>
<dt class="hdlist1">Subversion (<code>svn</code>) </dt>
<dd>
<p>The <code>git-svn</code> tool provides all the essential subcommands for dealing with Subversion repositories and is discussed in detail in this chapter.
The program is implemented in Perl and uses the Perl bindings for Git and Subversion.
It is managed together with the Git sources in the <code>git.git</code> repository (stored as <code>git-svn.perl</code>).
Note: The tool is called <code>git-svn</code>, but is called as usual with <code>git svn <command></code>.
The technical documentation is available in the <code>git-svn(1)</code> man page.</p>
</dd>
<dt class="hdlist1">Concurrent Versioning System (<code>cvs</code>) </dt>
<dd>
<p>The <code>git cvsimport</code> command imports and synchronizes a CVS repository — its counterpart is <code>git cvsexportcommit</code>.</p>
</dd>
<dt class="hdlist1">Perforce (<code>p4</code>) </dt>
<dd>
<p>With <code>git p4</code> you address repositories of the proprietary Perforce system.</p>
</dd>
</dl>
</div>
<div class="paragraph">
<p>For the interaction with other VCS there are also a lot of additional tools and scripts that improve, extend and partly replace the mentioned commands.
But also interfaces to other version control systems, such as Mercurial, are offered.
If the commands and recipes described in this chapter are not sufficient, an internet research is worthwhile.
As a first starting point we recommend the Git-Wiki.<sup class="footnote">[<a id="_footnoteref_117" class="footnote" href="#_footnotedef_117" title="View footnote.">117</a>]</sup></p>
</div>
<div class="paragraph">
<p>In addition to its immediate communication capabilities with other systems, Git has its own simple plain-text protocol that lets you translate the version history from any system in such a way that Git creates a repository from it.
For a detailed description including an example, see <a href="Git-Book_10.html#sec.fast-import">Sec. 9.2, “Custom Importers”</a> about <em>Fast Import</em>.</p>
</div>
<div class="sect2">
<h3 id="sec.subversion"><a class="anchor" href="Git-Book_10.html#sec.subversion"></a>9.1. Subversion</h3>
<div class="paragraph">
<p>The following is about how to use <code>git-svn</code>.
We’ll show you how to convert Subversion repositories and how to use it to exchange changes between a Subversion repository and Git.</p>
</div>
<div class="sect3">
<h4 id="sec.git-svn-convert"><a class="anchor" href="Git-Book_10.html#sec.git-svn-convert"></a>9.1.1. Conversion</h4>
<div class="paragraph">
<p>The goal is to transfer the version history from a Subversion repository to a Git repository.
Before you start, you will need to make preparations that may take some time, depending on the size of your project.
However, good preparation helps you to avoid mistakes from the start.</p>
</div>
<div class="sect4">
<h5 id="sec.git-svn-preparation"><a class="anchor" href="Git-Book_10.html#sec.git-svn-preparation"></a>9.1.1.1. Preparation</h5>
<div class="paragraph">
<p>You should have the following information at hand:</p>
</div>
<div class="olist arabic">
<ol class="arabic">
<li>
<p>Who are the authors? What are their e-mail addresses?</p>
</li>
<li>
<p>How is the repository structured? Are there branches and tags?</p>
</li>
<li>
<p>Should metadata about the Subversion revision be stored in the git commits?</p>
</li>
</ol>
</div>
<div class="paragraph">
<p>Later, you will run the command <code>git svn clone</code>.
The answers to the above questions will determine which options and arguments you use to do this.</p>
</div>
<div class="admonitionblock tip">
<table>
<tbody><tr>
<td class="icon">
<i class="fa icon-tip" title="Tip"></i>
</td>
<td class="content">
<div class="paragraph">
<p>Our experience has shown that rarely a single conversion attempt is sufficient.
If the Subversion repository is not already local, it’s definitely worth making a local copy of it — so you don’t have to download the revisions over the network on a second attempt.
You can use <code>rsvndump</code>, for example, to do this.<sup class="footnote">[<a id="_footnoteref_118" class="footnote" href="#_footnotedef_118" title="View footnote.">118</a>]</sup></p>
</div>
</td>
</tr>
</tbody></table>
</div>
<div class="paragraph">
<p>Subversion uses less extensive author metadata than Git; revisions are simply marked with a Subversion username, and there is no difference between the author and committer of a revision.
In order for <code>git-svn</code> to convert Subversion usernames to the full names with email addresses typical of Git, a so-called <em>authors file</em> is required:</p>
</div>
<div class="listingblock">
<div class="content">
<pre>jplenz = Julius Plenz <[email protected]>
vhaenel = Valentin Haenel <[email protected]></pre>
</div>
</div>
<div class="paragraph">
<p>The file, e.g. <code>authors.txt</code>, is later passed to <code>git-svn</code> via <code>--authors-file=</code> or <code>-A</code>.</p>
</div>
<div class="paragraph">
<p>The following one-liner determines all Subversion usernames and helps you to create the file:</p>
</div>
<div class="listingblock">
<div class="content">
<pre>$ <strong>svn log --xml | grep author | sed 's_^.*>\(.*\)<.*$_\1_' | \</strong>
<strong>sort --unique</strong></pre>
</div>
</div>
<div class="paragraph">
<p>If you do not specify an authors file when converting (or if an author is missing), <code>git-svn</code> will use the Subversion username as the author.
The e-mail address is composed of the Subversion username and the UUID of the Subversion repository.</p>
</div>
<div class="paragraph">
<p>Find out how the repository is structured in the next step.
The following questions will help:</p>
</div>
<div class="olist arabic">
<ol class="arabic">
<li>
<p>Does the repository have a so-called <em>trunk</em> (main development thread), branches and tags?</p>
<div class="olist loweralpha">
<ol class="loweralpha" type="a">
<li>
<p>If so, is the default Subversion layout (<code>trunk/</code>, <code>branches/</code>, <code>tags/</code>) used?</p>
</li>
<li>
<p>If not, in which directories are trunk, branches and tags then?</p>
</li>
</ol>
</div>
</li>
<li>
<p>Are only a single or multiple projects managed in the repository?</p>
</li>
</ol>
</div>
<div class="paragraph">
<p>If the project follows the Subversion standard layout (<a href="Git-Book_10.html#fig.svn-stdlayout">Figure 47, “Standard Subversion layout”</a>), use the argument <code>--stdlayout</code> or <code>-s</code> for short.</p>
</div>
<div id="fig.svn-stdlayout" class="imageblock text-center">
<div class="content">
<img src="" alt="svn stdlayout crop" width="17%">
</div>
<div class="title">Figure 47. Standard Subversion layout</div>
</div>
</div>
<div class="sect4">
<h5 id="sec.git-svn-metadata"><a class="anchor" href="Git-Book_10.html#sec.git-svn-metadata"></a>9.1.1.2. SVN Metadata</h5>
<div class="paragraph">
<p>The <code>--no-metadata</code> argument prevents additional metadata from being included in the commit message.
To what extent this makes sense for your use case is up to you to decide.
From a technical standpoint, metadata is only necessary if you want to continue to interact with the Subversion repository.
However, it may also be helpful to preserve the metadata, for example if you use the Subversion revision number in your bug tracking system.</p>
</div>
<div class="paragraph">
<p>The SVN metadata appears in the last line of each commit message and takes the following form:</p>
</div>
<div class="listingblock">
<div class="content">
<pre>git-svn-id: <URL>@<Revision> <UUID></pre>
</div>
</div>
<div class="paragraph">
<p><code><URL></code> ist die URL des Subversion-Repositorys, <code><Revision></code> die Subversion-Revision und <code><UUID></code> (<em>Universally Unique Identifier</em>) eine Art “Fingerabdruck” des Subversion-Repositorys.
Zum Beispiel:</p>
</div>
<div class="paragraph">
<p><code><URL></code> is the URL of the Subversion repository, <code><Revision></code> is the Subversion revision, and <code><UUID></code> (<em>Universally Unique Identifier</em>) is a sort of “fingerprint” of the Subversion repository.
For example:</p>
</div>
<div class="listingblock">
<div class="content">
<pre>git-svn-id: file:///demo/trunk@8 2423f1c7-8de6-44f9-ab07-c0d4e8840b78</pre>
</div>
</div>
</div>
<div class="sect4">
<h5 id="sec.git-svn-username"><a class="anchor" href="Git-Book_10.html#sec.git-svn-username"></a>9.1.1.3. Specifying a Username</h5>
<div class="paragraph">
<p>How you specify the user name depends on the transport protocol.
For those where Subversion handles authentication (e.g. <code>http</code>, <code>https</code>, and <code>svn</code>), use the <code>--username</code> option.
For others (<code>svn+ssh</code>), you must specify the username as part of the URL, for example, <code>svn+ssh://[email protected]</code>.</p>
</div>
</div>
<div class="sect4">
<h5 id="sec.git-svn-convert-command"><a class="anchor" href="Git-Book_10.html#sec.git-svn-convert-command"></a>9.1.1.4. Converting Standard Layouts</h5>
<div class="paragraph">
<p>You can convert an SVN repository in standard layout with the following call (after you have created an Authors file):</p>
</div>
<div class="listingblock">
<div class="content">
<pre>$ <strong>git svn clone <http://svn.example.com/> -s -A <authors.txt> \</strong>
<strong>--no-metadata <projekt-konvertiert></strong></pre>
</div>
</div>
</div>
<div class="sect4">
<h5 id="sec.git-svn-nonstdlayout"><a class="anchor" href="Git-Book_10.html#sec.git-svn-nonstdlayout"></a>9.1.1.5. Non-Standard Layouts</h5>
<div class="paragraph">
<p>If the repository is not laid out according to the Subversion standard layout, adjust the call to <code>git svn</code> accordingly: Instead of <code>--stdlayout</code>, explicitly specify the trunk with <code>--trunk</code> or <code>-T</code>, the branches with <code>--branches</code> or <code>-b</code>, and the tags with <code>--tags</code> or <code>-t</code> — if, for example, several projects are managed in one Subversion repository (<a href="Git-Book_10.html#fig.svn-nonstdlayout">Figure 48, “Non-Standard Layout”</a>).</p>
</div>
<div id="fig.svn-nonstdlayout" class="imageblock text-center">
<div class="content">
<img src="" alt="svn nonstdlayout crop" width="20%">
</div>
<div class="title">Figure 48. Non-Standard Layout</div>
</div>
<div class="paragraph">
<p>To convert <code>project1</code>, the call would be as follows:<sup class="footnote">[<a id="_footnoteref_119" class="footnote" href="#_footnotedef_119" title="View footnote.">119</a>]</sup></p>
</div>
<div class="listingblock">
<div class="content">
<pre>$ <strong>git svn clone <http://svn.example.com/> -T trunk/projekt1 \</strong>
<strong>-b branches/projekt1 -t tags/projekt1 \</strong>
<strong>-A <authors.txt> <projekt1-konvertiert></strong></pre>
</div>
</div>
<div class="paragraph">
<p>An SVN repository without branches or tags can simply be cloned by using the URL of the project directory and omit <code>--stdlayout</code> entirely:</p>
</div>
<div class="listingblock">
<div class="content">
<pre>$ <strong>git svn clone <http://svn.example.com/projekt> -A authors.txt \</strong>
<strong>--no-metadata <projekt-konvertiert></strong></pre>
</div>
</div>
<div class="paragraph">
<p>If several independent projects are managed in one repository, we recommend that you create a separate Git repository for each project.
Unlike Subversion, Git is not suitable for managing multiple projects in one repository.
The object model means that the development histories (commit graphs) would become inextricably linked.
How to “link” projects from different Git repositories is described in <a href="Git-Book_6.html#sec.subprojects">Sec. 5.11, “Managing Subprojects”</a>.</p>
</div>
</div>
<div class="sect4">
<h5 id="sec.git-svn-postprocessing"><a class="anchor" href="Git-Book_10.html#sec.git-svn-postprocessing"></a>9.1.1.6. Postprocessing</h5>
<div class="paragraph">
<p>Once <code>git svn clone</code> has run, you’ll usually need to do a bit of rework on the repository.</p>
</div>
<div class="admonitionblock tip">
<table>
<tbody><tr>
<td class="icon">
<i class="fa icon-tip" title="Tip"></i>
</td>
<td class="content">
<div class="paragraph">
<p>During conversion, <code>git-svn</code> ignores all Subversion properties except <code>svn:execute</code>.
If the Subversion repository uses the <code>svn:ignore</code> properties to exclude files, you can translate them into one (or recursively for multiple) <code>.gitignore</code> file(s):</p>
</div>
<div class="listingblock">
<div class="content">
<pre>$ <strong>git svn create-ignore</strong></pre>
</div>
</div>
<div class="paragraph">
<p>The <code>.gitignore</code> files are only created and added to the index — you still have to check them in.</p>
</div>
</td>
</tr>
</tbody></table>
</div>
<div class="paragraph">
<p>Git creates special git branches under <code>remotes/origin</code> for the Subversion trunk and the Subversion branches and tags.
They are very similar to the remote tracking branches, in that they reflect the state of the Subversion repository-that is, they are <em>Subversion tracking branches</em>, so to speak.
They are mainly used for bidirectional communication and are updated when synchronized with the Subversion repository.
However, if you only want to convert the repository, these branches are of no use anymore and should be rewritten to “real” Git repositories (see below).</p>
</div>
<div class="paragraph">
<p>A Subversion tracking branch is created for the trunk and for each Subversion branch,<sup class="footnote">[<a id="_footnoteref_120" class="footnote" href="#_footnotedef_120" title="View footnote.">120</a>]</sup> and for each Subversion tag a Subversion tracking branch is also created (<em>no</em> git tag, see below), but under <code>remotes/origin/tags</code>.</p>
</div>
<div class="paragraph">
<p>Assume that the Subversion repository has the following Subversion branches and tags:</p>
</div>
<div id="fig.svn-branches" class="imageblock text-center">
<div class="content">
<img src="" alt="svn branches crop" width="20%">
</div>
<div class="title">Figure 49. Example Subversion branches and tags</div>
</div>
<div class="paragraph">
<p>In this case <code>git svn</code> creates the following git branches:</p>
</div>
<div id="fig.git-converted-branches" class="imageblock text-center">
<div class="content">
<img src="" alt="git branches crop" width="35%">
</div>
<div class="title">Figure 50. Converted Git Branches</div>
</div>
<div class="paragraph">
<p>You can adjust the prefix with the option <code>--prefix=</code>.
For example, with the <code>--prefix=svn/</code> statement, all converted references are stored under <code>remotes/svn/</code> instead of <code>remotes/origin</code>.</p>
</div>
<div class="paragraph">
<p>As already mentioned, <code>git-svn</code> <em>does not</em> create git tags for Subversion tags.
This is because from a technical point of view, Subversion tags are hardly different from Subversion branches.
They are also created with <code>git svn copy</code> and — unlike git tags — can be changed afterwards.
To be able to track such updates, Subversion tags are therefore also displayed as Subversion tracking branches.
Like the Subversion branches, they are of no use (but rather cause confusion) in a converted repository, and should be converted to real Git tags.</p>
</div>
<div class="paragraph">
<p>If you want to keep the Subversion branches and tags, you should translate the Subversion tracking branches into local Git branches or lightweight Git tags.
The following shell script git-convert-refs will help you in the first step:<sup class="footnote">[<a id="_footnoteref_121" class="footnote" href="#_footnotedef_121" title="View footnote.">121</a>]</sup></p>
</div>
<div id="code.git-convert-refs" class="listingblock">
<div class="content">
<pre class="rouge highlight"><code data-lang="shell"><span class="c">#!/bin/sh</span>
<span class="nb">.</span> <span class="si">$(</span>git <span class="nt">--exec-path</span><span class="si">)</span>/git-sh-setup
<span class="nv">svn_prefix</span><span class="o">=</span><span class="s1">'svn/'</span>
convert_ref<span class="o">(){</span>
<span class="nb">echo</span> <span class="nt">-n</span> <span class="s2">"converting: </span><span class="nv">$1</span><span class="s2"> to: </span><span class="nv">$2</span><span class="s2"> ..."</span>
git update-ref <span class="nv">$2</span> <span class="nv">$1</span>
git update-ref <span class="nt">-d</span> <span class="nv">$1</span>
<span class="nb">echo</span> <span class="s2">"done"</span>
<span class="o">}</span>
get_refs<span class="o">(){</span>
git <span class="k">for</span><span class="nt">-each-ref</span> <span class="nv">$1</span> <span class="nt">--format</span><span class="o">=</span><span class="s1">'%(refname)'</span>
<span class="o">}</span>
<span class="nb">echo</span> <span class="s1">'Converting svn tags'</span>
get_refs refs/remotes/<span class="k">${</span><span class="nv">svn_prefix</span><span class="k">}</span>tags | <span class="k">while </span><span class="nb">read </span>svn_tag
<span class="k">do
</span><span class="nv">new_ref</span><span class="o">=</span><span class="si">$(</span><span class="nb">echo</span> <span class="nv">$svn_tag</span> | <span class="nb">sed</span> <span class="nt">-e</span> <span class="s2">"s|remotes/</span><span class="nv">$svn_prefix</span><span class="s2">||"</span><span class="si">)</span>
convert_ref <span class="nv">$svn_tag</span> <span class="nv">$new_ref</span>
<span class="k">done
</span><span class="nb">echo</span> <span class="s2">"Converting svn branches"</span>
get_refs refs/remotes/<span class="k">${</span><span class="nv">svn_prefix</span><span class="k">}</span> | <span class="k">while </span><span class="nb">read </span>svn_branch
<span class="k">do
</span><span class="nv">new_ref</span><span class="o">=</span><span class="si">$(</span><span class="nb">echo</span> <span class="nv">$svn_branch</span> | <span class="nb">sed</span> <span class="nt">-e</span> <span class="s2">"s|remotes/</span><span class="nv">$svn_prefix</span><span class="s2">|heads/|"</span><span class="si">)</span>
convert_ref <span class="nv">$svn_branch</span> <span class="nv">$new_ref</span>
<span class="k">done</span></code></pre>
</div>
</div>
<div class="paragraph">
<p>The script assumes that the repository was converted with the <code>--prefix=svn/</code> option.
The two <code>while</code> loops do the following:</p>
</div>
<div class="ulist">
<ul>
<li>
<p>A git tag is created for each Subversion tracking branch that corresponds to a Subversion <em>tag</em> (e.g. <code>refs/remotes/svn/tags/v1.0</code> → <code>refs/tags/v1.0</code>).</p>
</li>
<li>
<p>For each Subversion tracking branch that corresponds to a Subversion <em>branch</em>, a “real” local Git branch is created (e.g. <code>refs/remotes/svn/bugfix</code> → <code>refs/heads/bugfix</code>)</p>
</li>
</ul>
</div>
<div class="paragraph">
<p>The script uses the plumbing commands <code>git for-each-ref</code>, which prints references matching the given expression line by line, and <code>git update-ref</code>, which rewrites and deletes references.<sup class="footnote">[<a id="_footnoteref_122" class="footnote" href="#_footnotedef_122" title="View footnote.">122</a>]</sup></p>
</div>
<div class="paragraph">
<p>See <a href="Git-Book_10.html#fig.git-convert-refs-before">Figure 51, “Converted branches and tags before translation”</a> and <a href="Git-Book_10.html#fig.git-convert-refs-after">Figure 52, “Converted branches and tags after translation”</a> to see how the script works.
In the Subversion repository there is a trunk, a branch feature and the <code>v1.0</code> tag.
<code>git-svn</code> creates three branches under <code>remotes/svn</code> during the conversion process, as described above.
The script <code>git-convert-refs</code> finally translates <code>remotes/svn/trunk</code> → <code>trunk</code>, <code>remotes/svn/feature</code> → <code>feature</code> and <code>remotes/svn/tags/v1.0</code> becomes a lightweight tag.</p>
</div>
<div id="fig.git-convert-refs-before" class="imageblock text-center">
<div class="content">
<img src="" alt="git convert refs before" width="60%">
</div>
<div class="title">Figure 51. Converted branches and tags before translation</div>
</div>
<div id="fig.git-convert-refs-after" class="imageblock text-center">
<div class="content">
<img src="" alt="git convert refs after" width="60%">
</div>
<div class="title">Figure 52. Converted branches and tags after translation</div>
</div>
<div class="paragraph">
<p>After rewriting Subversion branches and tags, you will notice that all Git tags “sit” on very short branches (see tag <code>v1.0</code> in <a href="Git-Book_10.html#fig.git-convert-refs-after">Figure 52, “Converted branches and tags after translation”</a> and <a href="Git-Book_10.html#fig.git-svn-tag-fix-before">Figure 53, “Converted Git tags on branches”</a>).
This is because each Subversion tag is stored with a Subversion commit.
So the conversion behavior of <code>git-svn</code> is correct in principle, because one Git commit is created per Subversion revision — but a bit unwieldy for a Git repository: you cannot use <code>git describe --tags</code>, for example.</p>
</div>
<div class="paragraph">
<p>However, unless the Subversion tag has been modified afterwards, the tagged commit references the same tree as its ancestor, so you can move the tags to the ancestors.
The following shell script <code>git-fix-tags</code><sup class="footnote">[<a id="_footnoteref_123" class="footnote" href="#_footnotedef_123" title="View footnote.">123</a>]</sup> will help here:</p>
</div>
<div class="listingblock">
<div class="content">
<pre class="rouge highlight"><code data-lang="shell"><span class="c">#!/bin/sh</span>
<span class="nb">.</span> <span class="si">$(</span>git <span class="nt">--exec-path</span><span class="si">)</span>/git-sh-setup
get_tree<span class="o">(){</span> git rev-parse <span class="nv">$1</span>^<span class="o">{</span>tree<span class="o">}</span><span class="p">;</span> <span class="o">}</span>
git <span class="k">for</span><span class="nt">-each-ref</span> refs/tags <span class="nt">--format</span><span class="o">=</span><span class="s1">'%(refname)'</span> <span class="se">\</span>
| <span class="k">while </span><span class="nb">read </span>tag
<span class="k">do
</span><span class="nv">sha1</span><span class="o">=</span><span class="si">$(</span>git rev-parse <span class="nv">$tag</span><span class="si">)</span>
<span class="nv">tree</span><span class="o">=</span><span class="si">$(</span>get_tree <span class="nv">$tag</span> <span class="si">)</span>
<span class="nv">new</span><span class="o">=</span><span class="nv">$sha1</span>
<span class="k">while </span><span class="nb">true
</span><span class="k">do
</span><span class="nv">parent</span><span class="o">=</span><span class="si">$(</span>git rev-parse <span class="nv">$new</span>^<span class="si">)</span>
git rev-parse <span class="nv">$new</span>^2 <span class="o">></span> /dev/null 2>&1 <span class="o">&&</span> <span class="nb">break
</span><span class="nv">parent_tree</span><span class="o">=</span><span class="si">$(</span>get_tree <span class="nv">$parent</span><span class="si">)</span>
<span class="o">[</span> <span class="s2">"</span><span class="nv">$parent_tree</span><span class="s2">"</span> <span class="o">!=</span> <span class="s2">"</span><span class="nv">$tree</span><span class="s2">"</span> <span class="o">]</span> <span class="o">&&</span> <span class="nb">break
</span><span class="nv">new</span><span class="o">=</span><span class="nv">$parent</span>
<span class="k">done</span>
<span class="o">[</span> <span class="s2">"</span><span class="nv">$sha1</span><span class="s2">"</span> <span class="o">=</span> <span class="s2">"</span><span class="nv">$new</span><span class="s2">"</span> <span class="o">]</span> <span class="o">&&</span> <span class="nb">break
echo</span> <span class="nt">-n</span> <span class="s2">"Found new commit for tag </span><span class="k">${</span><span class="nv">tag</span><span class="p">#refs/tags/</span><span class="k">}</span><span class="s2">: "</span> <span class="se">\</span>
<span class="si">$(</span>git rev-parse <span class="nt">--short</span> <span class="nv">$new</span><span class="si">)</span><span class="s2">", resetting..."</span>
git update-ref <span class="nv">$tag</span> <span class="nv">$new</span>
<span class="nb">echo</span> <span class="s1">'done'</span>
<span class="k">done</span></code></pre>
</div>
</div>
<div class="paragraph">
<p>The script examines every tagged commit.
If there is a commit among the ancestors that references the same tree, the tag is renewed.
If the commit or one of its ancestors itself has multiple ancestors (after a merge), the search is aborted.
In <a href="Git-Book_10.html#fig.git-svn-tag-fix-before">Figure 53, “Converted Git tags on branches”</a>, you can see two tags that come into consideration: <code>v1.0</code> and <code>v2.0</code>.
The <code>v1.0</code> tag was created from commit <code>C1</code> and does not contain any subsequent changes.
The <code>v2.0</code> tag, on the other hand, was modified again after it was created from Commit <code>C2</code>.</p>
</div>
<div id="fig.git-svn-tag-fix-before" class="imageblock text-center">
<div class="content">
<img src="" alt="git svn tag fix before" width="50%">
</div>
<div class="title">Figure 53. Converted Git tags on branches</div>
</div>
<div class="paragraph">
<p>In <a href="Git-Book_10.html#fig.git-svn-tag-fix-after">Figure 54, “Tag <code>v1.0</code> was rewritten”</a> you can see how tag <code>v1.0</code> was moved from the above script to the ancestor (because the trees are the same).
However, tag <code>v2.0</code> remains in place (because the trees are different due to subsequent changes).</p>
</div>
<div id="fig.git-svn-tag-fix-after" class="imageblock text-center">
<div class="content">
<img src="" alt="git svn tag fix after" width="50%">
</div>
<div class="title">Figure 54. Tag <code>v1.0</code> was rewritten</div>
</div>
<div class="admonitionblock tip">
<table>
<tbody><tr>
<td class="icon">
<i class="fa icon-tip" title="Tip"></i>
</td>
<td class="content">
<div class="paragraph">
<p>The tool <code>git-svn-abandon</code><sup class="footnote">[<a id="_footnoteref_124" class="footnote" href="#_footnotedef_124" title="View footnote.">124</a>]</sup> takes a similar approach to the two scripts presented, i.e. it converts Subversion tracking branches and moves tags.
Instead of lightweight tags, however, it creates annotated tags and does some additional cleanup work, similar to the ones we’ll cover next.
Another alternative for moving tags is the script <code>git-move-tags-up</code>.<sup class="footnote">[<a id="_footnoteref_125" class="footnote" href="#_footnotedef_125" title="View footnote.">125</a>]</sup></p>
</div>
</td>
</tr>
</tbody></table>
</div>
<div class="paragraph">
<p>You should still decide how to handle the trunk reference (<code>trunk</code> or <code>git-svn</code>).
After conversion, it will point to the same commit as <code>master</code>, so you can actually delete it:</p>
</div>
<div class="listingblock">
<div class="content">
<pre>$ <strong>git branch -d trunk</strong></pre>
</div>
</div>
<div class="paragraph">
<p>There may still be Git branches in the repository after the conversion that have already been merged into <code>master</code>.
Remove them with the following command:</p>
</div>
<div class="listingblock">
<div class="content">
<pre>$ <strong>git checkout master</strong>
$ <strong>git branch --merged | grep -v '^*' | xargs git branch -d</strong></pre>
</div>
</div>
<div class="paragraph">
<p>You can also dispose of the remaining legacy files that are both in the repository configuration and in <code>.git/</code>:</p>
</div>
<div class="listingblock">
<div class="content">
<pre>$ <strong>rm -r .git/svn</strong>
$ <strong>git config --remove-section svn</strong>
$ <strong>git config --remove-section svn-remote.svn</strong></pre>
</div>
</div>
<div class="paragraph">
<p>You are then ready to upload the converted history to a remote repository to share it with other developers.</p>
</div>
<div class="listingblock">
<div class="content">
<pre>$ <strong>git remote add <example> <[email protected]:projekt1.git></strong>
$ <strong>git push <example> --mirror</strong></pre>
</div>
</div>
</div>
<div class="sect4">
<h5 id="sec.git-svn-merges"><a class="anchor" href="Git-Book_10.html#sec.git-svn-merges"></a>9.1.1.7. Subversion Merges</h5>
<div class="paragraph">
<p>Subversion merges are detected by <code>git-svn</code> using the <code>svn:mergeinfo</code> properties and translated as git merges — although not always.
It depends on which Subversion revisions were merged and how.
If all revisions affecting a branch have been merged (<code>svn merge -r <N:M></code>), this is represented by a Git merge commit.
However, if only individual revisions have been merged (via <code>svn merge -c <N></code>), then they are simply committed with <code>git cherry-pick</code> instead.</p>
</div>
<div class="paragraph">
<p>For the following example, we have created a Subversion repository with a branch feature that is merged twice: once as a Subversion merge, which is considered a Git merge commit, and once as a Subversion merge, which is translated as cherry-pick.
The result converted with <code>git-svn</code> is shown below.</p>
</div>
<div id="fig.git-svn-merge-demo" class="imageblock text-center">
<div class="content">
<img src="" alt="git svn merge demo" width="60%">
</div>
<div class="title">Figure 55. Converted Subversion repository</div>
</div>
<div class="paragraph">
<p>The commits in the Subversion repository were made in the following order:</p>
</div>
<div class="olist arabic">
<ol class="arabic">
<li>
<p>Standardlayout</p>
</li>
<li>
<p><code>C1</code> on <code>trunk</code></p>
</li>
<li>
<p>Branch <code>feature</code></p>
</li>
<li>
<p><code>C1</code> on <code>feature</code></p>
</li>
<li>
<p><code>C2</code> on <code>feature</code></p>
</li>
<li>
<p><code>C2</code> on <code>trunk</code></p>
</li>
<li>
<p><code>svn merge branches/feature trunk -c 5</code> (commit <code>C2</code> on <code>feature</code>)</p>
</li>
<li>
<p><code>svn merge branches/feature trunk -r 3:5</code> (commit <code>C1</code>&`C2` on <code>feature</code>)</p>
</li>
</ol>
</div>
<div class="paragraph">
<p>Finally, it should be mentioned that <code>git-svn</code> is by far not the only tool for conversion.
<code>git-svn</code> often suffers from speed problems with very large repositories.
In this context, two tools are mentioned very often that work faster: on the one hand <code>svn2git</code><sup class="footnote">[<a id="_footnoteref_126" class="footnote" href="#_footnotedef_126" title="View footnote.">126</a>]</sup> and also <code>svn-fe</code><sup class="footnote">[<a id="_footnoteref_127" class="footnote" href="#_footnotedef_127" title="View footnote.">127</a>]</sup> (svn-fast-export).
If you encounter problems during the conversion (e.g. if the conversion has been running for several days and there is no end in sight), it is worth taking a look at the alternatives.</p>
</div>
</div>
</div>
<div class="sect3">
<h4 id="sec.git-svn-conduit"><a class="anchor" href="Git-Book_10.html#sec.git-svn-conduit"></a>9.1.2. Bidirectional Communication</h4>
<div class="paragraph">
<p>The <code>git-svn</code> tool can not only convert a Subversion repository, it is also a better Subversion client.
This means you have all the benefits of Git locally (easy and flexible branching, local commits and history) — but you can upload your Git commits from your local Git repository as Subversion commits to a Subversion repository.
Additionally, <code>git-svn</code> allows you to download new commits from other developers in the Subversion repository to your local Git repository.
You should use <code>git-svn</code> if a complete conversion to Git is not feasible, but you’d like to take advantage of the local benefits of Git.
Note that <code>git-svn</code> is a somewhat limited version of Subversion, and not all features are fully available.
There are some subtleties to consider, especially when uploading.</p>
</div>
<div class="paragraph">
<p>First, a summary of the most important <code>git-svn</code> commands:</p>
</div>
<div class="hdlist">
<table>
<tbody><tr>
<td class="hdlist1">
<code>git svn init</code>
</td>
<td class="hdlist2">
<p>Create a Git repository to track a Subversion repository.</p>
</td>
</tr>
<tr>
<td class="hdlist1">
<code>git svn fetch</code>
</td>
<td class="hdlist2">
<p>Download new revisions from the Subversion repository.</p>
</td>
</tr>
<tr>
<td class="hdlist1">
<code>git svn clone</code>
</td>
<td class="hdlist2">
<p>Combination of 'git svn init` and <code>git svn fetch</code>.</p>
</td>
</tr>
<tr>
<td class="hdlist1">
<code>git svn dcommit</code>
</td>
<td class="hdlist2">
<p>Upload Git commits as Subversion revisions to the Subversion repository (<em>diff commit</em>)</p>
</td>
</tr>
<tr>
<td class="hdlist1">
<code>git svn rebase</code>
</td>
<td class="hdlist2">
<p>Combination of <code>git svn fetch</code> and <code>git rebase</code>, usually executed before a <code>git svn dcommit</code>.</p>
</td>
</tr>
</tbody></table>
</div>
<div class="sect4">
<h5 id="sec.git-svn-clone"><a class="anchor" href="Git-Book_10.html#sec.git-svn-clone"></a>9.1.2.1. Cloning a Subversion Repository</h5>
<div class="paragraph">
<p>To retrieve the repository, first follow the same procedure as in the Subversion conversion section — create an authors file and determine the repository layout.
Then you can use <code>git svn clone</code> to clone the Subversion repository, for example:</p>
</div>
<div class="listingblock">
<div class="content">
<pre>$ <strong>git svn clone http://svn.example.com/ -s \</strong>
<strong>-A <authors.txt> <projekt-git></strong></pre>
</div>
</div>
<div class="paragraph">
<p>The call downloads all Subversion revisions and creates a Git repository from the history under <code><project-git></code>.</p>
</div>
<div class="admonitionblock tip">
<table>
<tbody><tr>
<td class="icon">
<i class="fa icon-tip" title="Tip"></i>
</td>
<td class="content">
<div class="paragraph">
<p>Cloning an entire Subversion history can be extremely time consuming under certain circumstances.
From a Subversion point of view, a long history is not a problem because the <code>svn checkout</code> command usually only downloads the current revision.
Something similar can be done with <code>git-svn</code>.
To do this, you first have to initialize the local Git repository and then only download the current revision (<code>HEAD</code>) from the trunk or branch.
The advantage here is certainly the speed, the disadvantage is that there is no local history:</p>
</div>
<div class="listingblock">
<div class="content">
<pre>$ <strong>git svn init http://svn.example.com/trunk projekt-git</strong>
$ <strong>cd projekt-git</strong>
$ <strong>git svn fetch -r HEAD</strong></pre>
</div>
</div>
<div class="paragraph">
<p>As an alternative to <code>HEAD</code>, you could specify any revision and then use <code>git svn fetch</code> to download the missing revisions up to <code>HEAD</code>, thus cloning only part of the history.</p>
</div>
</td>
</tr>
</tbody></table>
</div>
<div class="paragraph">
<p>As part of the conversion, we described how to post-process the repository.
Since you want to continue interacting with the Subversion repository in the future, this is not necessary here.
Also, the <code>--no-metadata</code> option must not be used, because otherwise the metadata of the form <code>git-svn-id:</code> will disappear from the commit message, and Git will no longer be able to map the commits and revisions.</p>
</div>
<div class="paragraph">
<p>The call to <code>git-svn</code> creates several entries in the configuration file <code>.git/config</code>.
First, an entry <code>svn-remote.svn</code>, which, similar to a remote entry for a Git repository, contains information about the URL and the Subversion branches and tags to track.
For example, if you cloned a repository with a standard layout, it might look like this:</p>
</div>
<div class="listingblock">
<div class="content">
<pre>[svn-remote "svn"]
url = http://svn.example.com/
fetch = trunk:refs/remotes/origin/trunk
branches = branches/*:refs/remotes/origin/*
tags = tags/*:refs/remotes/origin/tags/*</pre>
</div>
</div>
<div class="paragraph">
<p>In contrast to a regular <code>remote</code> entry this one additionally contains the values <code>branches</code> and <code>tags</code>.
These in turn each contain a refspec describing how Subversion branches and tags are stored locally as Subversion tracking branches.
The <code>fetch</code> entry only handles the Subversion trunk and must not contain any glob expressions.</p>
</div>
<div class="paragraph">
<p>If you do not have any Subversion branches and tags, the corresponding entries are omitted:</p>
</div>
<div class="listingblock">
<div class="content">
<pre>[svn-remote "svn"]
url = http://svn.example.com/
fetch = :refs/remotes/git-svn</pre>
</div>
</div>
<div class="paragraph">
<p>If you clone the repository with the prefix option, for example with <code>--prefix=svn/</code>, <code>git svn</code> will adjust the refspecs:</p>
</div>
<div class="listingblock">
<div class="content">
<pre>[svn-remote "svn"]
url = http://svn.example.com/
fetch = trunk:refs/remotes/svn/trunk
branches = branches/*:refs/remotes/svn/*
tags = tags/*:refs/remotes/svn/tags/*</pre>
</div>
</div>
<div class="paragraph">
<p>If you specify an authors file, a separate entry is created for it.
The file will still be needed in the future when you download new commits from the Subversion repository.</p>
</div>
<div class="listingblock">
<div class="content">
<pre>[svn]
authorsfile = /home/valentin/svn-testing/authors.txt</pre>
</div>
</div>
<div class="admonitionblock tip">
<table>
<tbody><tr>
<td class="icon">
<i class="fa icon-tip" title="Tip"></i>
</td>
<td class="content">
<div class="paragraph">
<p>In the section on conversion we described how to use <code>create-ignore</code> to create <code>.gitignore</code> files.
However, if you want to continue working with the Subversion repository, there is little point in checking in the <code>.gitignore</code> files there.
They have no effect on Subversion and only confuse other developers who continue to work with the native Subversion client (<code>svn</code>).
Instead, there is an option to store the patterns to ignore in the <code>.git/info/excludes</code> file (see <a href="Git-Book_5.html#sec.ignore">Sec. 4.4, “Ignoring Files”</a>), which is not part of the repository.
The <code>git svn show-ignore</code> command, which searches for and outputs all <code>svn-ignore</code> properties, can help here:</p>
</div>
<div class="listingblock">
<div class="content">
<pre>$ <strong>git svn show-ignore > .git/info/excludes</strong></pre>
</div>
</div>
</td>
</tr>
</tbody></table>
</div>
</div>
<div class="sect4">
<h5 id="sec.git-svn-examine"><a class="anchor" href="Git-Book_10.html#sec.git-svn-examine"></a>9.1.2.2. Examining a Repository</h5>
<div class="paragraph">
<p>In addition, <code>git-svn</code> provides some commands for examining the history and other properties of the repository:</p>
</div>
<div class="hdlist">
<table>
<tbody><tr>
<td class="hdlist1">
<code>git svn log</code>
</td>
<td class="hdlist2">
<p>A hybrid of <code>svn log</code> and <code>git log</code>.
The subcommand produces output similar to <code>svn log</code>, but uses the local repository to create it.
Several options of <code>git svn</code> have been recreated, such as <code>-r <N>:<M></code>.
Unknown options, e.g. <code>-p</code>, are passed directly to <code>git log</code> so that options from both commands can be mixed:</p>
<div class="listingblock">
<div class="content">
<pre>$ <strong>git svn log -r 3:16 -p</strong></pre>
</div>
</div>
<div class="paragraph">
<p>It would now show the revisions 3–16, including a patch of the changes.</p>
</div>
</td>
</tr>
<tr>
<td class="hdlist1">
<code>git svn blame</code>
</td>
<td class="hdlist2">
<p>Similar to <code>svn blame</code>.
With the <code>--git-format</code> option, the output has the same format as <code>git blame</code>, but with Subversion revisions instead of the SHA-1 IDs.</p>
</td>
</tr>
<tr>
<td class="hdlist1">
<code>git svn find-rev</code>
</td>
<td class="hdlist2">
<p>Shows the SHA-1 ID of the Git commit, which is the changeset of a particular Subversion revision.
The revision is passed with the syntax <code>r<N></code>, where <code><N></code> is the revision number:</p>
<div class="listingblock">
<div class="content">
<pre>$ <strong>git svn find-rev r6</strong>
c56506a535f9d41b64850a757a9f6b15480b2c07</pre>
</div>
</div>
</td>
</tr>
<tr>
<td class="hdlist1">
<code>git svn info</code>
</td>
<td class="hdlist2">
<p>Like <code>svn info</code>.
Returns various information about the Subversion repository.</p>
</td>
</tr>
<tr>
<td class="hdlist1">
<code>git svn proplist</code>
</td>
<td class="hdlist2">
<p>Like <code>svn proplist</code>, prints a list of existing Subversion properties.</p>
</td>
</tr>
<tr>
<td class="hdlist1">
<code>git svn propget</code>
</td>
<td class="hdlist2">
<p>Like <code>svn propget</code>, outputs the value of a single Subversion property.</p>
</td>
</tr>
</tbody></table>
</div>
<div class="paragraph">
<p>Unfortunately, currently <code>git-svn</code> can only query Subversion properties, but cannot create, modify or delete them.</p>
</div>
</div>
<div class="sect4">
<h5 id="sec.git-svn-exchange"><a class="anchor" href="Git-Book_10.html#sec.git-svn-exchange"></a>9.1.2.3. Exchanging Commits</h5>
<div class="paragraph">
<p>Similar to <code>git fetch</code>, <code>git svn fetch</code> downloads new commits from the Subversion repository.
In the process, <code>git-svn</code> fetches all new Subversion revisions, translates them into Git commits, and finally updates the Subversion tracking branches.
The output is a list of downloaded Subversion revisions, the files changed by the revision, the SHA-1 sum, and the Subversion tracking branch of the resulting Git commit, e.g:</p>
</div>
<div class="listingblock">
<div class="content">
<pre>$ <strong>git svn fetch</strong>
A COPYING
M README
r21 = 8d707316e1854afbc1b728af9f834e6954273425 (refs/remotes/trunk)</pre>
</div>
</div>
<div class="paragraph">
<p>You can work locally in the Git repository as usual, but there is an important restriction when uploading commits to the Subversion repository: While <code>git-svn</code> is capable of rendering Subversion merges to some degree (see above), it can’t map local Git merges to Subversion merges, so only linear histories should be uploaded via <code>git svn dcommit</code>.</p>
</div>
<div class="paragraph">
<p>To make this linearization easier, there is the command <code>git svn rebase</code>.
It first downloads all new commits from the Subversion repository and then rebuilds the current Git branch to the appropriate Subversion tracking branch via <code>git rebase</code>.</p>
</div>
<div class="paragraph">
<p>Essentially, the workflow consists of the following commands:</p>
</div>
<div class="listingblock">
<div class="content">
<pre>$ <strong>git add/commit ...</strong>
$ <strong>git svn rebase</strong>
$ <strong>git svn dcommit</strong></pre>
</div>
</div>
<div class="paragraph">
<p><a href="Git-Book_10.html#fig.git-svn-rebase">Figure 56</a> shows what <code>git svn rebase</code> does.
First, new revisions are downloaded from the Subversion repository, in this case <code>C</code>.
Then the <code>remote/origin/trunk</code> tracking branch is “advanced” so to speak, and then corresponds to the current status in the Subversion repository.
Finally, the current branch (in this case <code>master</code>) is rebuilt using <code>git rebase</code>.