zoukankan      html  css  js  c++  java
  • SVN部署

    centos下部署SVN

    大多数情况下,我们日常工作中用的版本控制系统都会选择分布式的Git,它相比于集中式的SVN有很多优势。但是有些项目软件基于自身限制,可能只支持SVN做工程同步。废话就不多说了,下面记录下SVN的部署和使用过程:

    安装SVN

    [root@svn_server ~]# rpm -qa | grep subversion
    subversion-libs-1.7.14-10.el7.x86_64
    subversion-1.7.14-10.el7.x86_64
    [root@svn_server ~]# yum remove subversion
    [root@svn_server ~]# yum -y install subversion
    [root@svn_server ~]# svnversion --version
    svnversion, version 1.7.14 (r1542130)
       compiled Apr 11 2018, 02:40:28
    
    Copyright (C) 2013 The Apache Software Foundation.
    This software consists of contributions made by many people; see the NOTICE
    file for more information.
    Subversion is open source software, see http://subversion.apache.org/
    
    [root@svn_server ~]# 
    
    #启动svn,启动时要指定svn的仓库目录
    [root@svn_server ~]# mkdir -p /data/svn
    [root@svn_server ~]# svnserve -d -r /data/svn/
    参数:
    -d:守护程序模式
    -r:指定服务的根目录
    [root@svn_server ~]# ps -ef |grep svn
    root       1568      1  0 21:24 ?        00:00:00 svnserve -d -r /data/svn/
    #svn默认端口是3690
    [root@svn_server ~]# netstat -lntup |grep svn
    tcp        0      0 0.0.0.0:3690            0.0.0.0:*               LISTEN      1568/svnserve       
    
    #特别注意:
    svnserver的启动命令要使用上面的"/usr/bin/svnserve -d -r /data/svn"
    不要使用"service svnserve start"命令来启动,否则会造成svn下载时报错:svn: No repository found in 'svn://*.*.*.*/*'
    
    #设置开机启动
    [root@svn_server ~]# echo '/usr/bin/svnserve -d -r /data/svn/' >>/etc/rc.local 
    [root@svn_server ~]# chmod +x /etc/rc.local
    
    #svn停止
    [root@svn_server ~]# yum provides |grep killall
    [root@svn_server ~]# killall svnserve 
    
    #svn启动
    [root@svn_server ~]# svnserve -d -r /data/svn/
    
    #如果已经有svn在运行,可以换一个端口运行
    [root@svn_server ~]# svnserve -d -r /data/svn --listen-port 3691
    

    代码库创建及配置

    #下面创建两个代码库,库名为kevin和grace
    [root@svn_server ~]# svnadmin create /data/svn/kevin
    [root@svn_server ~]# svnadmin create /data/svn/grace
    [root@svn_server ~]# ls /data/svn/kevin/
    conf  db  format  hooks  locks  README.txt
    [root@svn_server ~]# ls /data/svn/grace/
    conf  db  format  hooks  locks  README.txt
    [root@svn_server ~]# 
    
    #配置代码库,这里以kevin代码库为例进行说明
    [root@svn_server ~]# cd /data/svn/kevin/conf/
    [root@svn_server conf]# ll
    total 12
    -rw-r--r-- 1 root root 1080 Jan  5 21:38 authz			#权限控制文件
    -rw-r--r-- 1 root root  309 Jan  5 21:38 passwd			#账号密码文件
    -rw-r--r-- 1 root root 3090 Jan  5 21:38 svnserve.conf	 #svn服务配置文件
    [root@svn_server conf]# 
    
    #设置该代码库的登录帐号和密码(由于是svn自己启动的,没有借助于apache启动,所以这里的密码是明文)
    [root@svn_server conf]# vim passwd 
    [users]
    # harry = harryssecret
    # sally = sallyssecret
    
    dev=123456
    test=123456
    
    #设置该代码库的操作权限
    权限主体可以是别名,用户组、用户或*;别名在前面加&;用户组在前面加@;*表示全部用户;
    权限可以是w、r、wr和空,空表示没有任何权限。
    [root@svn_server conf]# egrep -v '^#|^$' authz
    [aliases]			#设置别名
    opesn=dev,test
    [groups]			#设置组
    admin=dev,test
    [/]					#根目录权限设置,用户对kevin代码库根目录的读写权限
    dev=rw
    test=r
    [root@svn_server conf]# 
    
    #修改svnserve.conf文件(在[general]区域添加下面四行内容)
    [root@svn_server conf]# egrep -v '^#|^$' svnserve.conf
    [general]
    anon-access = none			#匿名用户禁止读写
    auth-access = write			#授权用户允许完整访问
    password-db = passwd		#使用哪个文件作为账号文件。由于同在一个目录路径下,所以这里不用全路径
    authz-db = authz			#使用哪个文件作为权限文件
    realm = /data/svn/kevin		#认证命名空间,subversion会在认证提示里显示,并且作为凭证缓存的关键字
    [sasl]
    [root@svn_server conf]# 
    
    #重启svn
    [root@svn_server conf]# killall svnserve 
    [root@svn_server conf]# svnserve -d -r /data/svn/
    
    

    SVN客户端常规操作命令

    #客户机要安装svn,确保有svn相关操作命令
    [root@m01 ~]# yum -y install subversion
    #查看svn的相关操作命令
    [root@m01 ~]# svn --help
    
    #客户机下载svn代码库文件(10.0.1.80是上面svn服务端地址。即下面kevin代码库)
    即从版本库中导出
    [root@m01 ~]# svn checkout svn://10.0.1.80/kevin
    Authentication realm: <svn://10.0.1.80:3690> /data/svn/kevin
    Password for 'root': 		#首次需要输入本机root密码
    Authentication realm: <svn://10.0.1.80:3690> /data/svn/kevin
    Username: dev				#输入svn设置的用户名,这里选择dev
    Password for 'dev': 		#输入dev密码
    
    -----------------------------------------------------------------------
    ATTENTION!  Your password for authentication realm:
    
       <svn://10.0.1.80:3690> /data/svn/kevin
    
    can only be stored to disk unencrypted!  You are advised to configure
    your system so that Subversion can store passwords encrypted, if
    possible.  See the documentation for details.
    
    You can avoid future appearances of this warning by setting the value
    of the 'store-plaintext-passwords' option to either 'yes' or 'no' in
    '/root/.subversion/servers'.
    -----------------------------------------------------------------------
    Store password unencrypted (yes/no)? yes
    Checked out revision 0.
    [root@m01 ~]# 
    
    #需要注意:也可以使用带用户名和密码的访问(svn co 等同于svn checkout):
    [root@m01 svndata]# svn co --username dev --password 123456 svn://10.0.1.80/kevin
    Checked out revision 0.
    [root@m01 svndata]# 
    [root@m01 kevin]# ll -a
    total 0
    drwxr-xr-x 3 root root 18 Jan  5 21:57 .
    drwxr-xr-x 3 root root 19 Jan  5 21:57 ..
    drwxr-xr-x 4 root root 75 Jan  5 21:57 .svn
    [root@m01 kevin]# 
    
    #温馨提示:
    svn checkout(即svn co)表示检出。这样下载到的svn代码库里包括.svn
    # svn co http://路径(目录或文件的全路径) [本地目录全路径] --username 用户名 --password 密码
    # svn co svn://路径(目录或文件的全路径) [本地目录全路径] --username 用户名 --password 密码
    # svn checkout http://路径(目录或文件的全路径) [本地目录全路径] --username 用户名
    # svn checkout svn://路径(目录或文件的全路径) [本地目录全路径] --username 用户名
    
    #注意:如果不带--password 参数传输密码的话,会提示输入密码,建议不要用明文的--password 选项。
      其中 username 与 password前是两个短线,不是一个。
      不指定本地目录全路径,则检出到当前目录下。
    例子:
    svn co svn://192.168.10.202/kevin /data/svndata --username dev --password 123456
    svn co http://192.168.10.202/kevin --username dev --password 123456
    svn checkout svn://192.168.10.202/kevin /data/svndata --username dev --password 123456
    svn checkout http://192.168.10.202/kevin --username dev --password 123456
    
    #svn导出(导出一个干净的不带.svn文件夹的目录树)
    svn export [-r 版本号] http://路径(目录或文件的全路径) [本地目录全路径] --username 用户名
    svn export [-r 版本号] svn://路径(目录或文件的全路径) [本地目录全路径] --username 用户名
    svn export 本地检出的(即带有.svn文件夹的)目录全路径 要导出的本地目录全路径
    注意:
    第一种从版本库导出干净工作目录树的形式是指定URL,如果指定了修订版本号,会导出相应的版本,如果没有指定修订版本,则会导出最新的,导出到指定位置。
    如果省略 本地目录全路径,URL的最后一部分会作为本地目录的名字。
    第二种形式是指定 本地检出的目录全路径 到 要导出的本地目录全路径,所有的本地修改将会保留,但是不在版本控制下(即没提交的新文件,因为.svn文件夹
    里没有与之相关的信息记录)的文件不会拷贝。
    
    例子:
    注意/opt/svndata目录不能提前创建,下面导出命令执行后会自动创建该目录
    即把kevin版本库里的所有文件都导出到本地的/op/svndata目录下了,不包括.svn
    [root@m01 ~]# svn export svn://10.0.1.80/kevin /opt/svndata/ --username dev --password 123456
    A    /opt/svndata
    Exported revision 0.
    [root@m01 ~]# 
    [root@m01 ~]# ll /opt/svndata/				#如上,kevin版本库里还没有任何文件
    total 0
    [root@m01 ~]# ll -a /opt/svndata/			#查看,发现导出后没有带.svn
    total 0
    drwxr-xr-x  2 root root  6 Jan  5 22:02 .
    drwxr-xr-x. 3 root root 21 Jan  5 22:02 ..
    [root@m01 ~]# 
    
    #添加新文件(svn add)
    注:告诉SVN服务器要添加文件了,还要用svn commint -m真实的上传上去!
    svn add test.php #添加test.php
    svn commit -m "test.php" test.php  #提交新加的文件到svn服务器里
    svn add *.php #添加当前目录下所有的php文件
    svn commit -m "添加我的测试用全部php文件" *.php
    
    [root@m01 kevin]# svn add test.php 
    [root@m01 kevin]# svn commit -m 'test'
    Adding         test.php
    Transmitting file data .
    Committed revision 1.
    [root@m01 kevin]# 
    [root@m01 kevin]# mkdir haha
    
    #添加目录
    [root@m01 kevin]# svn add haha
    A         haha
    [root@m01 kevin]# svn commit -m 'add haha'
    Adding         haha
    
    Committed revision 2.
    [root@m01 kevin]# 
    [root@m01 kevin]# ll
    total 4
    drwxr-xr-x 2 root root 6 Jan  5 22:08 haha
    -rw-r--r-- 1 root root 4 Jan  5 22:06 test.php
    [root@m01 kevin]# 
    
    #svn 提交
    svn commit -m "提交备注信息文本" [-N] [--no-unlock] 文件名
    svn ci -m "提交备注信息文本" [-N] [--no-unlock] 文件名
    必须带上-m参数,参数可以为空,但是必须写上-m
     
    例子:
    svn commit -m "提交当前目录下的全部在版本控制下的文件" *    #注意这个*表示全部文件
    svn commit -m "提交我的测试用test.php" test.php
    svn commit -m "提交我的测试用test.php" -N --no-unlock test.php    #保持锁就用–no-unlock开关
    svn ci -m "提交当前目录下的全部在版本控制下的文件" *    #注意这个*表示全部文件
    svn ci -m "提交我的测试用test.php" test.php
    svn ci -m "提交我的测试用test.php" -N --no-unlock test.php    #保持锁就用–no-unlock开关
    
    #svn更新操作。即把svn服务器上最新的版本更新下来
    [root@m01 kevin]# svn update 		#或者"svn up"
    Updating '.':
    At revision 2.
    [root@m01 kevin]# 
    
    #svn查看
    [root@m01 kevin]# svn update 
    Updating '.':
    At revision 2.
    [root@m01 kevin]# svn info
    Path: .
    Working Copy Root Path: /svndata/kevin
    URL: svn://10.0.1.80/kevin
    Repository Root: svn://10.0.1.80/kevin
    Repository UUID: 647b3bfb-7257-458f-bb1d-9c403895ccbc
    Revision: 2
    Node Kind: directory
    Schedule: normal
    Last Changed Author: dev
    Last Changed Rev: 2
    Last Changed Date: 2020-01-05 22:08:44 +0800 (Sun, 05 Jan 2020)
    
    [root@m01 kevin]# 
    
    #svn删除文件(简写svn del)
    svn delete svn://路径(目录或文件的全路径) -m "删除备注信息文本"
    推荐如下操作:
    # svn delete 文件名
    # svn ci -m "删除备注信息文本"
    
    [root@m01 kevin]# svn delete haha				#或者svn del haha
    D         haha
    [root@m01 kevin]# svn commit -m "del haha"		 #或者svn ci -m "del haha"
    Deleting       haha
    
    Committed revision 3.
    [root@m01 kevin]# 
    
    [root@m01 kevin]# svn delete svn://10.0.1.80/kevin/test.php -m "del test.php"
    
    Committed revision 4.
    [root@m01 kevin]# 
    #注意:svn的删除使用delete,而不是rm
    
    #svn查看日志
    [root@m01 kevin]# svn log		#显示所有文件的所有修改记录
    ------------------------------------------------------------------------
    r2 | dev | 2020-01-05 22:08:44 +0800 (Sun, 05 Jan 2020) | 1 line
    
    add haha
    ------------------------------------------------------------------------
    r1 | dev | 2020-01-05 22:07:19 +0800 (Sun, 05 Jan 2020) | 1 line
    
    test
    ------------------------------------------------------------------------
    
    [root@m01 kevin]# svn log test.php		#显示test.html这个文件的所有修改记录,及其版本号的变化
    ------------------------------------------------------------------------
    r1 | dev | 2020-01-05 22:07:19 +0800 (Sun, 05 Jan 2020) | 1 line
    
    test
    ------------------------------------------------------------------------
    [root@m01 kevin]# 
    
    #版本库下的文件和目录列表
    [root@m01 kevin]# svn ls
    haha/
    test.php
    [root@m01 kevin]# 
    
    #恢复本地修改
    svn revert: 恢复原始未改变的工作副本文件 (恢复大部份的本地修改)。
    revert: 用法: revert PATH...
    注意: 本子命令不会存取网络,并且会解除冲突的状况。但是它不会恢复被删除的目录;也不会恢复已经经过svn commit提交过的文件
    [root@m01 kevin]# cat test.php
    123
    [root@m01 kevin]# echo '555' >>test.php 
    [root@m01 kevin]# cat test.php
    123
    555
    [root@m01 kevin]# svn revert 
    .svn/     test.php  
    [root@m01 kevin]# svn revert test.php 
    Reverted 'test.php'
    [root@m01 kevin]# cat test.php
    123
    [root@m01 kevin]# 
    
    #加锁/解锁
    svn lock -m “加锁备注信息文本“ [--force] 文件名
    svn unlock 文件名
    例子:
    # svn lock -m “锁信测试用test.php文件“ test.php
    # svn unlock test.php
    
    #比较差异
    svn diff 文件名
    svn diff -r 修正版本号m:修正版本号n 文件名
    例子:
    # svn diff test.php<- 将修改的文件与基础版本比较
    # svn diff -r 200:201 test.php<- 对 修正版本号200 和 修正版本号201 比较差异
    
    #查看文件或目录状态(简称svn st)
    # svn st 目录路径/名
    # svn status 目录路径/名  
    目录下的文件和子目录的状态,正常状态不显示
    ?:不在svn的控制中; M:内容被修改;C:发生冲突;A:预定加入到版本库;K:被锁定
    
    # svn -v 目录路径/名
    # svn status -v 目录路径/名
    显示文件和子目录状态
    第一列保持相同,第二列显示工作版本号,
    第三和第四列显示最后一次修改的版本号和修改人
    注意:
    svn status、svn diff和 svn revert这三条命令在没有网络的情况下也可以执行的,原因是svn在本地的.svn中保留了本地版本的原始拷贝。
    [root@m01 kevin]# svn status -v test.php 
                     5        5 dev          test.php
    [root@m01 kevin]# 
    
    #解决冲突
    # svn resolved [本地目录全路径]
     
    例子:
    # svn update
    C foo.c
    Updated to revision 31.
    如果你在更新时得到冲突,你的工作拷贝会产生三个新的文件:
    # ls
    foo.c
    foo.c.mine
    foo.c.r30
    foo.c.r31
    当你解决了foo.c的冲突,并且准备提交,运行svn resolved让你的工作拷贝知道你已经完成了所有事情。
    你可以仅仅删除冲突的文件并且提交,但是svn resolved除外
    
    #新建一个分支copy
    # svn copy branchA branchB -m "make B branch"
    表示从branchA拷贝出一个新分支branchB
    
    合并内容到分支merge
    # svn merge branchA branchB
    把对branchA的修改合并到分支branchB
    

    备份

    #全量备份
    把/data/svn/kevin/仓库的内容备份到bak/bak.txt文件中。
    [root@m01 ~]# svnadmin dump /data/svn/kevin/ > bak/bak.txt
    * Dumped revision 0.
    * Dumped revision 1.
    * Dumped revision 2.
    
    #增量备份
    [root@m01 ~]# svnlook youngest /data/svn/kevin/
    2	-- 得知svn的版本号为2。
    
    #第一次增量备份
    [root@m01 ~]# svnadmin dump /data/svn/kevin/ -r 0:2 --incremental >/backup/kevin0-2
    * Dumped revision 0.
    * Dumped revision 1.
    * Dumped revision 2.
    -r参数::,上面例子是0:1,指定备份的开始与结束版本号。
    --incremental参数:表示增量备份。
    
    #第二次增量备份
    [root@m01 kevin]# svnlook youngest /data/svn/kevin/
    3
    [root@m01 kevin]# svnadmin dump /data/svn/kevin/ -r 3:3 --incremental >/backup/kevin3-3
    * Dumped revision 3.
    
    

    恢复

    #灾难过后,新建svn仓库,如:/data/svn/repo/。
    #全量恢复
    [root@m01 ~]# svnadmin load /data/svn/repo/ < bak/bak.txt 
    <<< Started new transaction, based on original revision 1
         * adding path : test.php ... done.
    
    ------- Committed revision 1 >>>
    
    <<< Started new transaction, based on original revision 2
         * adding path : 01.txt ... done.
         * adding path : 02.txt ... done.
         * adding path : 03.txt ... done.
         * adding path : 04.txt ... done.
         * adding path : 05.txt ... done.
         * adding path : 06.txt ... done.
         * adding path : 07.txt ... done.
         * adding path : 08.txt ... done.
         * adding path : 09.txt ... done.
         * adding path : 10.txt ... done.
    
    ------- Committed revision 2 >>>
    
    [root@m01 ~]# 
    
    #增量恢复
    [root@m01 ~]# svnadmin load /data/svn/repo/ < /backup/kevin3-3 
    <<< Started new transaction, based on original revision 3
         * adding path : index.html ... done.
    
    ------- Committed revision 3 >>>
    
    [root@m01 ~]# 
    
    #恢复后,设置好密码、权限等配置,启动svnserve服务,就算是恢复完成了。
    
  • 相关阅读:
    gradle添加阿里云maven库
    来谈谈MySQL的临时表,到底是个什么东西,以及怎么样产生的
    MySQL优化相关参数--先做个记录,以后可能用得到
    对于join操作,MySQL它是咋做的?
    Linux-常用命令记录
    有时候我们自认为有用的索引却并没有被MySQL选择使用?
    C#趟坑: Wait()线程结束时,会忽略子线程
    初次使用Windbg检查C#程序内存
    性能优化之三:将Dottrace过程加入持续集成
    性能优化之二:结构体类型的性能优化
  • 原文地址:https://www.cnblogs.com/opesn/p/12994192.html
Copyright © 2011-2022 走看看