实际操作 Svn 迁移到 Git

最近把一个开发了很久的项目从公司的 svn 服务器迁移到 GitLab 上,折腾了几天。我们的 svn 目录结构不是很标准,所以网上的资料没一个能够照搬用的。Anyway,最后成功了,我这里详细记录一下,供参考。

本文地址:https://segmentfault.com/a/1190000007039598

Reference

SVN 迁移到GIT
svn迁移到Git
SVN迁移到Git的过程(+ 一些技巧)
git-svn 使用环境和步骤
git如何恢复本地删除的文件
git alternatives to “svn info” that can be included in a build for traceability?
svn2git
为git配置扩展命令
SVN、GIT生成版本号
git使用规范
Git 添加空文件夹的方法
转:批量查找空文件夹并放置.gitkeep或.gitignore
git add --all 为啥不能添加空文件夹,这样设计的初衷是?
git 为何不能添加空文件夹?
Git 里面的 origin 到底代表啥意思?
新手请教:Git 里面的 origin 到底代表啥意思?
git 查看分支 以及所对应的remote
.gitignore 不起作用
git 放弃本地修改更新
git 提交到本地但未push到远端的代码
Git 分支管理和冲突解决
git cherry-pick合并某个commit
Git错误non-fast-forward后的冲突解决
Git下的冲突解决
Git学习笔记05--git stash

看了好多资料……

Svn 代码库情况

我们的 svn 代码库的 URL 当然是机密啦。不过大致原理上,我需要迁移的目录状况如下:

主分支(在 git 上应成为 master 分支):
http://rdsvn.company.net/some_mid_path/Branches/main

此外打了几个tag:
http://rdsvn.company.net/some_mid_path/Branches/main_tag_25189
http://rdsvn.company.net/some_mid_path/Branches/main_tag_28141
http://rdsvn.company.net/some_mid_path/Branches/main_tag_28576

导出 svn 工程

首先,我们需要git-svn

$ sudo yum install git-svn
or
$ sodu apt-get install git-svn

原理上,就是使用git svn clone ...命令。如果你只是简单的执行这个命令的话:

$ git svn clone http://rdsvn.company.net/some_mid_path/Branches/main

那么我们只是将主分支及其 log 取到了本地,但是其他的分支并没有。如果要把所有的分支都 checkout 出来的话,需要用到--trunk--tags--branches等选项。

需要注意的有三点:

  1. 除了trunk单词是单数之外,其他的选项单词,请注意是复数。前者就是主分支的目录了,而后者表示的则是对应各个分支 / 标签的上层目录。Git-svn 会将该目录下的所有子目录视为一个独立的分支 / 标签进行导出。
  2. 这几个选项指定的都是相对路径,相对于你在git svn clone后面紧跟着的路径的位置。所以请不要在这些选项里面指定诸如 “http://” 开头的绝对地址。
  3. 除了 trunk 之外,其他几个相关选项可以重复指定。如果你的分支 / 标签分别在 svn 目录上的不同文件夹下,可以多次指定。

于是,对应我们的实际情况,git-svn 命令是这样的:

git svn clone http://rdsvn.company.net/some_mid_path/   \  # 注意这里并没有把主分支的路径写完整
--trunk="Branches/main" --tags="Branches"

稍等一会儿,就可以看到整个 svn 的主分支及分支都被导出来了,一颗赛艇!

分支处理

执行git branch -a可以看到相应的分支已经按照我们指定的选项建立起来了。我的实际情况,分支是这样的:

    trunk
    tags/main_tag_25189
    tags/main_tag_28141
    tags/main_tag_28576

这几个分支都被视为 Git 的远程分支,你的本地仓库暂时是空的。可以使用git checkout -b ...来将这几个分支拉到本地。

讲道理,svn 上的分支命名和 git 上的分支命名规则未必是一致的,因此这个时候你会想要重新命名分支。我的建议是这么操作:

  1. 在这个仓库中,首先把这几个虚拟的远程分支 checkout 到本地保存下来,这样可以保存 change log
  2. 向服务器上传主分支,一般分支名就是 master
  3. 将复制出来的其他分支,推到 git 服务器上,按照你希望的规则进行命名

上传到 git 服务器

首先要在 git 服务器上创建一个空的工程,比如 “git@rdgit.company.net/some_mid_path/some_group/firmware.git

然后在本地仓库中执行:

git remote add origin git@rdgit.company.net/some_mid_path/some_group/firmware.git

然后将主分支上传:

git push origin trunk:master

其他的分支也是利用这个语句创建新分支上传。

完成之后,在 git 服务器上就可以看到完整的 change log 以及分支变化情况啦。

关于空目录

其实迁移的时候还需要注意一个问题,那就是 svn 支持空目录的版本控制,但是 git 不支持。

导出 svn 之后,最好对比一下原始的 svn 工程内是否有空目录。如果有,那么你需要手工(或者写脚本)在导出的 git 工程中创建这些目录,并且在目录中建立一个 “.gitkeep” 文件(这是约定俗成的做法),并且将这些文件git add .gitkeep,以加入 git 的版本控制。这样就以迂回的方式通过 git 版本控制了目录。


后台 / 嵌入式全栈之路
曾经的嵌入式 / 后台开发一枚,现在开始走向架构。本专栏没有高深技术,只讲基础组件、工具,请放心食用

电子和互联网深耕多年,拥有丰富的嵌入式和服务器开发经验。现负责腾讯心悦俱乐部后台开发

889 声望
204 粉丝
0 条评论
推荐阅读
Go http.Transport 主要参数说明
在 Go 中创建一个 http client 时,一般会使用 Go http 包的 Transport 类型。本文主要说明 http.Transport 需要关注的主要参数。

amc

封面图
git 常用命令 记录
有错误和改进的地方望留言,谢谢 : )顺便推荐一下,git入门教程Git 教程图解 GitPro Git 2(中文版)Git 使用Git 使用规范流程Git 工作流程Commit message 和 Change log 编写指南全局设置git config --system //...

Luff阅读 3.2k

关于 Ubuntu 20.04 LTS 的网络(dchp/dns/route/PPPoE)
前言旧时代网络管理软件 net-tools、NetworkManager新时代网络管理软件 iproute2、resolvectl新时代旧时代大致可按 2010 年划分检查网卡是否已插入网线直接查看文件 {代码...} ip addr 或 ip link 命令查看网卡状...

qbit2阅读 15k

git/pip/easy_install/apt代理/源设置
在公司内网环境中,访问公网往往需要经过公司的代理,对于浏览器、IDE等开发工具,都提供了设置代理的配置,而git、pip、easy_install等CLI工具,则需要通过命令或配置文件进行代理设置;对于Python、Ubuntu、Mav...

乘着风1阅读 5.5k

Laravel入门及实践,快速上手ThinkSNS+二次开发
【摘要】自从ThinkSNS+不使用ThinkPHP框架而使用Laravel框架之后,很多人都说技术门槛抬高了,其实你与TS+的距离仅仅只是学习一个新框架而已,所以,我们今天来说说Laravel的入门。

ThinkSNS1阅读 2.4k

confluence7.2.1的部署与迁移---呕心沥血版
Confluence是一个专业的企业知识管理与协同软件,也可以用于构建企业wiki。使用简单,但它强大的编辑和站点管理特征能够帮助团队成员之间共享信息、文档协作、集体讨论,信息推送。 到官网下载最新版本,截止目前...

暗涌阅读 7.8k

C++编译器和链接器的完全指南
C++是一种强类型语言,它的编译和链接是程序开发过程中不可或缺的两个环节。编译器和链接器是两个非常重要的概念。本文将详细介绍C++中的编译器和链接器以及它们的工作原理和使用方法。

小万哥2阅读 1k

封面图

电子和互联网深耕多年,拥有丰富的嵌入式和服务器开发经验。现负责腾讯心悦俱乐部后台开发

889 声望
204 粉丝
宣传栏