<?xml version="1.0" encoding="UTF-8"?><rss version="2.0"
	xmlns:content="http://purl.org/rss/1.0/modules/content/"
	xmlns:wfw="http://wellformedweb.org/CommentAPI/"
	xmlns:dc="http://purl.org/dc/elements/1.1/"
	xmlns:atom="http://www.w3.org/2005/Atom"
	xmlns:sy="http://purl.org/rss/1.0/modules/syndication/"
	xmlns:slash="http://purl.org/rss/1.0/modules/slash/"
	>

<channel>
	<title>Bernie Yu &#187; linux</title>
	<atom:link href="http://bernieyu.com/tag/linux/feed/" rel="self" type="application/rss+xml" />
	<link>http://bernieyu.com</link>
	<description>复刻生活</description>
	<lastBuildDate>Tue, 17 May 2016 09:03:15 +0000</lastBuildDate>
	<language>zh-CN</language>
	<sy:updatePeriod>hourly</sy:updatePeriod>
	<sy:updateFrequency>1</sy:updateFrequency>
	<generator>https://wordpress.org/?v=4.2.38</generator>
	<item>
		<title>[转] lsyncd实时同步搭建指南——取代rsync+inotify</title>
		<link>http://bernieyu.com/2015/10/lsyncd-synchronize-realtime/</link>
		<comments>http://bernieyu.com/2015/10/lsyncd-synchronize-realtime/#comments</comments>
		<pubDate>Thu, 22 Oct 2015 00:59:40 +0000</pubDate>
		<dc:creator><![CDATA[Bernie Yu]]></dc:creator>
				<category><![CDATA[Linux]]></category>
		<category><![CDATA[linux]]></category>
		<category><![CDATA[lsyncd]]></category>
		<category><![CDATA[同步]]></category>

		<guid isPermaLink="false">http://bernieyu.com/?p=261</guid>
		<description><![CDATA[转自http://seanlook.com/2015/05/06/lsyncd-synchronize-rea ... <a class="more-link" href="http://bernieyu.com/2015/10/lsyncd-synchronize-realtime/">　　>>阅读全文&#60;&#60;</a>]]></description>
				<content:encoded><![CDATA[<p>转自http://seanlook.com/2015/05/06/lsyncd-synchronize-realtime/</p>
<h1 id="1-_几大实时同步工具比较">1. 几大实时同步工具比较</h1>
<h2 id="1-1_inotify_+_rsync">1.1 inotify + rsync</h2>
<p>最近一直在寻求生产服务服务器上的同步替代方案，原先使用的是<code>inotify + rsync</code>，但随着文件数量的增大到100W+，目录下的文件列表就达20M，在网络状况不佳或者限速的情况下，变更的文件可能10来个才几M，却因此要发送的文件列表就达20M，严重减低的带宽的使用效率以及同步效率；更为要紧的是，加入inotifywait在5s内监控到10个小文件发生变化，便会触发10个rsync同步操作，结果就是真正需要传输的才2-3M的文件，比对的文件列表就达200M。使用这两个组合的好处在于，它们都是最基本的软件，可以通过不同选项做到很精确的控制，比如排除同步的目录，同步多个模块或同步到多个主机。</p>
<p>搭建过程参考 <a href="http://seanlook.com/2014/12/12/rsync_inotify_setup/" target="_blank" rel="external">Linux下同步工具inotify+rsync使用详解</a> 。</p>
<h2 id="1-2_sersync">1.2 sersync</h2>
<p>后来听同事说 <a href="http://www.ccvita.com/422.html" target="_blank" rel="external">sersync</a> 这么个工具可以提高同步的性能，也解决了同步大文件时出现异常的问题，所以就尝试了一下。sersync是国内的一个开发者开源出来的，使用c++编写，采用多线程的方式进行同步，失败后还有重传机制，对临时文件过滤，自带crontab定时同步功能。网上看到有人说性能还不错，说一下我的观点：</p>
<ul>
<li>国产开源，文档不是很全，在2011年之后就没更新了（googlecode都要快关闭了，其实可以转交其他人维护），网上关于它的使用和讨论都止于10年了</li>
<li>采用xml配置文件的方式，可读性比较好，但是有些原生的有些功能没有实现就没法使用了</li>
<li>无法实现多目录同步，只能通过多个配置文件启动多个进程</li>
<li>文件排除功能太弱。这个要看需求，不是每个人都需要排除子目录。而对于我的环境中，这个功能很重要，而且排除的规则较多</li>
<li>虽然提供插件的功能，但很鸡肋，因为软件本身没有持续更新，也没有看到贡献有其它插件出现（可能是我知识面不够，还用不到里面的refreshCDN plugin）。</li>
</ul>
<p>虽然不懂c++，但大致看了下源码 <a href="http://code.google.com/p/sersync/source/browse/FileSynchronize.cpp" target="_blank" rel="external">FileSynchronize</a>，拼接rsync命令大概在273行左右，最后一个函数就是排除选项，简单一点可以将<code>--exclude=</code>改成<code>--eclude-from</code>来灵活控制。有机会再改吧。</p>
<p>另外，在作者的文章 <a href="http://hi.baidu.com/johntech/item/a4a2060ecf3053c6905718e1" target="_blank" rel="external">Sersync服务器同步程序 项目简介与设计框架</a> 评论中，说能解决上面 <code>rsync + inotify</code>中所描述的问题。阅读了下源码，这个应该是没有解决，因为在拼接rsync命令时，后面的目的地址始终是针对module的，只要执行rsync命令，就会对整个目录进行遍历，发送要比对的文件列表，然后再发送变化的文件。sersync只是减少了监听的事件，减少了rsync的次数——这已经是很大的改进，但每次rsync没办法改变。（如有其它看法可与我讨论）</p>
<p>其实我们也不能要求每一个软件功能都十分健全，关键是看能否满足我们当下的特定的需求。所谓好的架构不是设计出来的，而是进化来的。目前使用<code>sersync2</code>没什么问题，而且看了它的设计思路应该是比较科学的，特别是过滤队列的设计。双向同步看起来也是可以实现。</p>
<h2 id="1-3_lsyncd">1.3 lsyncd</h2>
<p>废话说这么多，本文就是介绍它了。有些博客说lsyncd是谷歌开源的，实际不是了，只是托管在了googlecode上而已，幸运的是已经迁移到github了：<a href="https://github.com/axkibe/lsyncd" target="_blank" rel="external">https://github.com/axkibe/lsyncd</a> 。</p>
<p>Lysncd 实际上是lua语言封装了 inotify 和 rsync 工具，采用了 Linux 内核（2.6.13 及以后）里的 inotify 触发机制，然后通过rsync去差异同步，达到实时的效果。我认为它最令人称道的特性是，完美解决了 <code>inotify + rsync</code>海量文件同步带来的文件频繁发送文件列表的问题 —— 通过时间延迟或累计触发事件次数实现。另外，它的配置方式很简单，lua本身就是一种配置语言，可读性非常强。lsyncd也有多种工作模式可以选择，本地目录cp，本地目录rsync，远程目录rsyncssh。</p>
<p>实现简单高效的本地目录同步备份（网络存储挂载也当作本地目录），一个命令搞定。</p>
<h1 id="2-_使用_lsyncd_本地目录实时备份">2. 使用 lsyncd 本地目录实时备份</h1>
<p>这一节实现的功能是，本地目录source实时同步到另一个目录target，而在source下有大量的文件，并且有部分目录和临时文件不需要同步。</p>
<h2 id="2-1_安装lsyncd">2.1 安装lsyncd</h2>
<p>安装<code>lsyncd</code>极为简单，已经收录在ubuntu的官方镜像源里，直接通过<code>apt-get install lsyncd</code>就可以。<br />
在Redhat系（我的环境是CentOS 6.2 x86_64 ），可以手动去下载 <a href="ftp://195.220.108.108/linux/fedora/linux/updates/21/x86_64/l/lsyncd-2.1.5-6.fc21.x86_64.rpm" target="_blank" rel="external">lsyncd-2.1.5-6.fc21.x86_64.rpm</a>，但首先你得安装两个依赖<code>yum install lua lua-devel</code>。也可以通过在线安装，需要<code>epel-release</code>扩展包：</p>
<figure class="highlight">
<table>
<tbody>
<tr>
<td class="gutter">
<div class="line">1</div>
<div class="line">2</div>
</td>
<td class="code">
<div class="line"><span class="preprocessor"># rpm -ivh http://dl.fedoraproject.org/pub/epel/6/x86_64/epel-release-6-8.noarch.rpm</span></div>
<div class="line"><span class="preprocessor"># yum install lsyncd</span></div>
</td>
</tr>
</tbody>
</table>
</figure>
<p><strong>源码编译安装</strong><br />
从源码编译安装可以使用最新版的lsyncd程序，但必须要相应的依赖库文件和编译工具：<code>yum install lua lua-devel asciidoc cmake</code>。</p>
<p>从 <a href="http://code.google.com/p/lsyncd/downloads/list" target="_blank" rel="external">googlecode lsyncd</a> 上下载的<code>lsyncd-2.1.5.tar.gz</code>，直接<code>./configure</code>、<code>make &amp;&amp; make install</code>就可以了。</p>
<p>从github上下载<a href="https://github.com/axkibe/lsyncd/archive/master.zip" target="_blank" rel="external">lsyncd-master.zip</a> 的2.1.5版本使用的是 cmake 编译工具，无法<code>./configure</code>：</p>
<figure class="highlight">
<table>
<tbody>
<tr>
<td class="gutter">
<div class="line">1</div>
<div class="line">2</div>
<div class="line">3</div>
<div class="line">4</div>
</td>
<td class="code">
<div class="line"><span class="preprocessor"># uzip lsyncd-master.zip</span></div>
<div class="line"><span class="preprocessor"># cd lsyncd-master</span></div>
<div class="line"><span class="preprocessor"># cmake -DCMAKE_INSTALL_PREFIX=/usr/local/lsyncd-2.1.5</span></div>
<div class="line"><span class="preprocessor"># make &amp;&amp; make install</span></div>
</td>
</tr>
</tbody>
</table>
</figure>
<p>我这个版本编译时有个小bug，如果按照<code>INSTALL</code>在<code>build</code>目录中make，会提示：</p>
<figure class="highlight">
<table>
<tbody>
<tr>
<td class="gutter">
<div class="line">1</div>
<div class="line">2</div>
<div class="line">3</div>
<div class="line">4</div>
<div class="line">5</div>
<div class="line">6</div>
</td>
<td class="code">
<div class="line">[100%] Generating doc/lsyncd.1</div>
<div class="line">Updating the manpage</div>
<div class="line">a2x: failed: source file not found: doc/lsyncd.1.txt</div>
<div class="line">make[2]: <span class="keyword">*</span><span class="keyword">*</span><span class="keyword">*</span> [doc/lsyncd.1] Error 1</div>
<div class="line">make[1]: <span class="keyword">*</span><span class="keyword">*</span><span class="keyword">*</span> [CMakeFiles/manpage.dir/all] Error 2</div>
<div class="line">make: <span class="keyword">*</span><span class="keyword">*</span><span class="keyword">*</span> [all] Error 2</div>
</td>
</tr>
</tbody>
</table>
</figure>
<p>解决办法是要么直接在解压目录下cmake，不要<code>mkdir build</code>，要么在<code>CMakeList.txt</code>中搜索<code>doc</code>字符串，在前面加上<code>${PROJECT_SOURCE_DIR}</code>。</p>
<h2 id="2-2_lsyncd-conf">2.2 lsyncd.conf</h2>
<p>下面都是在编译安装的情况下操作。</p>
<h3 id="2-2-1_lsyncd同步配置">2.2.1 lsyncd同步配置</h3>
<figure class="highlight">
<table>
<tbody>
<tr>
<td class="gutter">
<div class="line">1</div>
<div class="line">2</div>
<div class="line">3</div>
<div class="line">4</div>
<div class="line">5</div>
<div class="line">6</div>
<div class="line">7</div>
<div class="line">8</div>
<div class="line">9</div>
<div class="line">10</div>
<div class="line">11</div>
<div class="line">12</div>
<div class="line">13</div>
<div class="line">14</div>
<div class="line">15</div>
<div class="line">16</div>
<div class="line">17</div>
<div class="line">18</div>
<div class="line">19</div>
<div class="line">20</div>
<div class="line">21</div>
<div class="line">22</div>
<div class="line">23</div>
</td>
<td class="code">
<div class="line"><span class="comment"># cd /usr/local/lsyncd-2.1.5</span></div>
<div class="line"><span class="comment"># mkdir etc var</span></div>
<div class="line"><span class="comment"># vi etc/lsyncd.conf</span></div>
<div class="line">settings {</div>
<div class="line"><span class="variable">logfile =</span><span class="string">&#8220;/usr/local/lsyncd-2.1.5/var/lsyncd.log&#8221;</span>,</div>
<div class="line"><span class="variable">statusFile =</span><span class="string">&#8220;/usr/local/lsyncd-2.1.5/var/lsyncd.status&#8221;</span>,</div>
<div class="line"><span class="variable">inotifyMode =</span> <span class="string">&#8220;CloseWrite&#8221;</span>,</div>
<div class="line"><span class="variable">maxProcesses =</span> <span class="number">7</span>,</div>
<div class="line">&#8212; <span class="variable">nodaemon =</span><span class="constant">true</span>,</div>
<div class="line">}</div>
<div class="line"></div>
<div class="line">sync {</div>
<div class="line">default.rsync,</div>
<div class="line"><span class="variable">source =</span> <span class="string">&#8220;/tmp/src&#8221;</span>,</div>
<div class="line"><span class="variable">target =</span> <span class="string">&#8220;/tmp/dest&#8221;</span>,</div>
<div class="line">&#8212; <span class="variable">excludeFrom =</span> <span class="string">&#8220;/etc/rsyncd.d/rsync_exclude.lst&#8221;</span>,</div>
<div class="line"><span class="variable">rsync =</span> {</div>
<div class="line"><span class="variable">binary =</span> <span class="string">&#8220;/usr/bin/rsync&#8221;</span>,</div>
<div class="line"><span class="variable">archive =</span> <span class="constant">true</span>,</div>
<div class="line"><span class="variable">compress =</span> <span class="constant">true</span>,</div>
<div class="line"><span class="variable">verbose =</span> <span class="constant">true</span></div>
<div class="line">}</div>
<div class="line">}</div>
</td>
</tr>
</tbody>
</table>
</figure>
<p>到这启动 lsycnd 就可以完成实时同步了，默认的许多参数可以满足绝大部分需求，非常简单。</p>
<h3 id="2-2-2_lsyncd-conf配置选项说明">2.2.2 lsyncd.conf配置选项说明</h3>
<p><strong>settings</strong><br />
里面是全局设置，<code>--</code>开头表示注释，下面是几个常用选项说明：</p>
<ul>
<li><code>logfile</code> 定义日志文件</li>
<li><code>stausFile</code> 定义状态文件</li>
<li><code>nodaemon=true</code> 表示不启用守护模式，默认</li>
<li><code>statusInterval</code> 将lsyncd的状态写入上面的statusFile的间隔，默认10秒</li>
<li><code>inotifyMode</code> 指定inotify监控的事件，默认是<code>CloseWrite</code>，还可以是<code>Modify</code>或<code>CloseWrite or Modify</code></li>
<li><code>maxProcesses</code> 同步进程的最大个数。假如同时有20个文件需要同步，而<code>maxProcesses = 8</code>，则最大能看到有8个rysnc进程</li>
<li><code>maxDelays</code> 累计到多少所监控的事件激活一次同步，即使后面的<code>delay</code>延迟时间还未到</li>
</ul>
<p><strong>sync</strong><br />
里面是定义同步参数，可以继续使用<code>maxDelays</code>来重写settings的全局变量。一般第一个参数指定<code>lsyncd</code>以什么模式运行：<code>rsync</code>、<code>rsyncssh</code>、<code>direct</code>三种模式：</p>
<ul>
<li><code>default.rsync</code> ：本地目录间同步，使用rsync，也可以达到使用ssh形式的远程rsync效果，或daemon方式连接远程rsyncd进程；<br />
<code>default.direct</code> ：本地目录间同步，使用<code>cp</code>、<code>rm</code>等命令完成差异文件备份；<br />
<code>default.rsyncssh</code> ：同步到远程主机目录，rsync的ssh模式，需要使用key来认证</li>
<li><code>source</code> 同步的源目录，使用绝对路径。</li>
<li><code>target</code> 定义目的地址.对应不同的模式有几种写法：<br />
<code>/tmp/dest</code> ：本地目录同步，可用于<code>direct</code>和<code>rsync</code>模式<br />
<code>172.29.88.223:/tmp/dest</code> ：同步到远程服务器目录，可用于<code>rsync</code>和<code>rsyncssh</code>模式，拼接的命令类似于<code>/usr/bin/rsync -ltsd --delete --include-from=- --exclude=* SOURCE TARGET</code>，剩下的就是rsync的内容了，比如指定username，免密码同步<br />
<code>172.29.88.223::module</code> ：同步到远程服务器目录，用于<code>rsync</code>模式<br />
三种模式的示例会在后面给出。</li>
<li><code>init</code> 这是一个优化选项，当<code>init = false</code>，只同步进程启动以后发生改动事件的文件，原有的目录即使有差异也不会同步。默认是<code>true</code></li>
<li><code>delay</code> 累计事件，等待rsync同步延时时间，默认15秒（最大累计到1000个不可合并的事件）。也就是15s内监控目录下发生的改动，会累积到一次rsync同步，避免过于频繁的同步。（可合并的意思是，15s内两次修改了同一文件，最后只同步最新的文件）</li>
<li><code>excludeFrom</code> 排除选项，后面指定排除的列表文件，如<code>excludeFrom = "/etc/lsyncd.exclude"</code>，如果是简单的排除，可以使用<code>exclude = LIST</code>。<br />
这里的排除规则写法与原生rsync有点不同，更为简单：</p>
<ul>
<li>监控路径里的任何部分匹配到一个文本，都会被排除，例如<code>/bin/foo/bar</code>可以匹配规则<code>foo</code></li>
<li>如果规则以斜线<code>/</code>开头，则从头开始要匹配全部</li>
<li>如果规则以<code>/</code>结尾，则要匹配监控路径的末尾</li>
<li><code>?</code>匹配任何字符，但不包括<code>/</code></li>
<li><code>*</code>匹配0或多个字符，但不包括<code>/</code></li>
<li><code>**</code>匹配0或多个字符，可以是<code>/</code></li>
</ul>
</li>
</ul>
<ul>
<li><code>delete</code> 为了保持target与souce完全同步，Lsyncd默认会<code>delete = true</code>来允许同步删除。它除了<code>false</code>，还有<code>startup</code>、<code>running</code>值，请参考 <a href="https://github.com/axkibe/lsyncd/wiki/Lsyncd%202.1.x%20%E2%80%96%20Layer%204%20Config%20%E2%80%96%20Default%20Behavior" target="_blank" rel="external">Lsyncd 2.1.x ‖ Layer 4 Config ‖ Default Behavior</a>。</li>
</ul>
<p><strong>rsync</strong><br />
（提示一下，<code>delete</code>和<code>exclude</code>本来都是<strong>rsync</strong>的选项，上面是配置在<strong>sync</strong>中的，我想这样做的原因是为了减少rsync的开销）</p>
<ul>
<li><code>bwlimit</code> 限速，单位kb/s，与rsync相同（这么重要的选项在文档里竟然没有标出）</li>
<li><code>compress</code> 压缩传输默认为<code>true</code>。在带宽与cpu负载之间权衡，本地目录同步可以考虑把它设为<code>false</code></li>
<li><code>perms</code> 默认保留文件权限。</li>
<li>其它rsync的选项</li>
</ul>
<p>其它还有rsyncssh模式独有的配置项，如<code>host</code>、<code>targetdir</code>、<code>rsync_path</code>、<code>password_file</code>，见后文示例。<code>rsyncOps={"-avz","--delete"}</code>这样的写法在2.1.*版本已经不支持。</p>
<p><code>lsyncd.conf</code>可以有多个<code>sync</code>，各自的source，各自的target，各自的模式，互不影响。</p>
<h2 id="2-3_启动lsyncd">2.3 启动lsyncd</h2>
<p>使用命令加载配置文件，启动守护进程，自动同步目录操作。</p>
<pre><code>lsyncd -<span class="built_in">log</span> Exec /usr/local/lsyncd-<span class="number">2.1</span>.<span class="number">5</span>/etc/lsyncd.<span class="keyword">conf</span>
</code></pre>
<h2 id="2-4_lsyncd-conf其它模式示例">2.4 lsyncd.conf其它模式示例</h2>
<p>以下配置本人都已经过验证可行，必须根据实际需要裁剪配置：</p>
<figure class="highlight">
<table>
<tbody>
<tr>
<td class="gutter">
<div class="line">1</div>
<div class="line">2</div>
<div class="line">3</div>
<div class="line">4</div>
<div class="line">5</div>
<div class="line">6</div>
<div class="line">7</div>
<div class="line">8</div>
<div class="line">9</div>
<div class="line">10</div>
<div class="line">11</div>
<div class="line">12</div>
<div class="line">13</div>
<div class="line">14</div>
<div class="line">15</div>
<div class="line">16</div>
<div class="line">17</div>
<div class="line">18</div>
<div class="line">19</div>
<div class="line">20</div>
<div class="line">21</div>
<div class="line">22</div>
<div class="line">23</div>
<div class="line">24</div>
<div class="line">25</div>
<div class="line">26</div>
<div class="line">27</div>
<div class="line">28</div>
<div class="line">29</div>
<div class="line">30</div>
<div class="line">31</div>
<div class="line">32</div>
<div class="line">33</div>
<div class="line">34</div>
<div class="line">35</div>
<div class="line">36</div>
<div class="line">37</div>
<div class="line">38</div>
<div class="line">39</div>
<div class="line">40</div>
<div class="line">41</div>
<div class="line">42</div>
<div class="line">43</div>
<div class="line">44</div>
<div class="line">45</div>
<div class="line">46</div>
<div class="line">47</div>
<div class="line">48</div>
<div class="line">49</div>
<div class="line">50</div>
<div class="line">51</div>
<div class="line">52</div>
<div class="line">53</div>
<div class="line">54</div>
<div class="line">55</div>
<div class="line">56</div>
<div class="line">57</div>
<div class="line">58</div>
<div class="line">59</div>
<div class="line">60</div>
<div class="line">61</div>
<div class="line">62</div>
<div class="line">63</div>
<div class="line">64</div>
<div class="line">65</div>
<div class="line">66</div>
<div class="line">67</div>
<div class="line">68</div>
<div class="line">69</div>
<div class="line">70</div>
<div class="line">71</div>
<div class="line">72</div>
<div class="line">73</div>
<div class="line">74</div>
<div class="line">75</div>
<div class="line">76</div>
<div class="line">77</div>
<div class="line">78</div>
<div class="line">79</div>
<div class="line">80</div>
<div class="line">81</div>
<div class="line">82</div>
<div class="line">83</div>
<div class="line">84</div>
<div class="line">85</div>
<div class="line">86</div>
<div class="line">87</div>
<div class="line">88</div>
<div class="line">89</div>
<div class="line">90</div>
<div class="line">91</div>
</td>
<td class="code">
<div class="line">settings {</div>
<div class="line"><span class="variable">logfile =</span><span class="string">&#8220;/usr/local/lsyncd-2.1.5/var/lsyncd.log&#8221;</span>,</div>
<div class="line"><span class="variable">statusFile =</span><span class="string">&#8220;/usr/local/lsyncd-2.1.5/var/lsyncd.status&#8221;</span>,</div>
<div class="line"><span class="variable">inotifyMode =</span> <span class="string">&#8220;CloseWrite&#8221;</span>,</div>
<div class="line"><span class="variable">maxProcesses =</span> <span class="number">8</span>,</div>
<div class="line">}</div>
<div class="line"></div>
<div class="line"></div>
<div class="line">&#8212; I. 本地目录同步，direct：cp/rm/mv。 适用：<span class="number">500</span>+万文件，变动不大</div>
<div class="line">sync {</div>
<div class="line">default.direct,</div>
<div class="line"><span class="variable">source =</span> <span class="string">&#8220;/tmp/src&#8221;</span>,</div>
<div class="line"><span class="variable">target =</span> <span class="string">&#8220;/tmp/dest&#8221;</span>,</div>
<div class="line"><span class="variable">delay =</span> <span class="number">1</span></div>
<div class="line"><span class="variable">maxProcesses =</span> <span class="number">1</span></div>
<div class="line">}</div>
<div class="line"></div>
<div class="line">&#8212; II. 本地目录同步，rsync模式：rsync</div>
<div class="line">sync {</div>
<div class="line">default.rsync,</div>
<div class="line"><span class="variable">source =</span> <span class="string">&#8220;/tmp/src&#8221;</span>,</div>
<div class="line"><span class="variable">target =</span> <span class="string">&#8220;/tmp/dest1&#8243;</span>,</div>
<div class="line"><span class="variable">excludeFrom =</span> <span class="string">&#8220;/etc/rsyncd.d/rsync_exclude.lst&#8221;</span>,</div>
<div class="line"><span class="variable">rsync =</span> {</div>
<div class="line"><span class="variable">binary =</span> <span class="string">&#8220;/usr/bin/rsync&#8221;</span>,</div>
<div class="line"><span class="variable">archive =</span> <span class="constant">true</span>,</div>
<div class="line"><span class="variable">compress =</span> <span class="constant">true</span>,</div>
<div class="line"><span class="variable">bwlimit =</span> <span class="number">2000</span></div>
<div class="line">}</div>
<div class="line">}</div>
<div class="line"></div>
<div class="line">&#8212; III. 远程目录同步，rsync模式 + rsyncd daemon</div>
<div class="line">sync {</div>
<div class="line">default.rsync,</div>
<div class="line"><span class="variable">source =</span> <span class="string">&#8220;/tmp/src&#8221;</span>,</div>
<div class="line"><span class="variable">target =</span> <span class="string">&#8220;syncuser@172.29.88.223::module1&#8243;</span>,</div>
<div class="line"><span class="variable">delete=</span><span class="string">&#8220;running&#8221;</span>,</div>
<div class="line"><span class="variable">exclude =</span> { <span class="string">&#8220;.*&#8221;</span>, <span class="string">&#8220;.tmp&#8221;</span> },</div>
<div class="line"><span class="variable">delay =</span> <span class="number">30</span>,</div>
<div class="line"><span class="variable">init =</span> <span class="constant">false</span>,</div>
<div class="line"><span class="variable">rsync =</span> {</div>
<div class="line"><span class="variable">binary =</span> <span class="string">&#8220;/usr/bin/rsync&#8221;</span>,</div>
<div class="line"><span class="variable">archive =</span> <span class="constant">true</span>,</div>
<div class="line"><span class="variable">compress =</span> <span class="constant">true</span>,</div>
<div class="line"><span class="variable">verbose =</span> <span class="constant">true</span>,</div>
<div class="line"><span class="variable">password_file =</span> <span class="string">&#8220;/etc/rsyncd.d/rsync.pwd&#8221;</span>,</div>
<div class="line"><span class="variable">_extra =</span> {<span class="string">&#8220;&#8211;bwlimit=200&#8243;</span>}</div>
<div class="line">}</div>
<div class="line">}</div>
<div class="line"></div>
<div class="line">&#8212; IV. 远程目录同步，rsync模式 + ssh shell</div>
<div class="line">sync {</div>
<div class="line">default.rsync,</div>
<div class="line"><span class="variable">source =</span> <span class="string">&#8220;/tmp/src&#8221;</span>,</div>
<div class="line"><span class="variable">target =</span> <span class="string">&#8220;172.29.88.223:/tmp/dest&#8221;</span>,</div>
<div class="line">&#8212; <span class="variable">target =</span> <span class="string">&#8220;root@172.29.88.223:/remote/dest&#8221;</span>,</div>
<div class="line">&#8212; 上面target，注意如果是普通用户，必须拥有写权限</div>
<div class="line"><span class="variable">maxDelays =</span> <span class="number">5</span>,</div>
<div class="line"><span class="variable">delay =</span> <span class="number">30</span>,</div>
<div class="line">&#8212; <span class="variable">init =</span> <span class="constant">true</span>,</div>
<div class="line"><span class="variable">rsync =</span> {</div>
<div class="line"><span class="variable">binary =</span> <span class="string">&#8220;/usr/bin/rsync&#8221;</span>,</div>
<div class="line"><span class="variable">archive =</span> <span class="constant">true</span>,</div>
<div class="line"><span class="variable">compress =</span> <span class="constant">true</span>,</div>
<div class="line"><span class="variable">bwlimit =</span> <span class="number">2000</span></div>
<div class="line">&#8212; <span class="variable">rsh =</span> <span class="string">&#8220;/usr/bin/ssh -p 22 -o StrictHostKeyChecking=no&#8221;</span></div>
<div class="line">&#8212; 如果要指定其它端口，请用上面的rsh</div>
<div class="line">}</div>
<div class="line">}</div>
<div class="line"></div>
<div class="line">&#8212; V. 远程目录同步，rsync模式 + rsyncssh，效果与上面相同</div>
<div class="line">sync {</div>
<div class="line">default.rsyncssh,</div>
<div class="line"><span class="variable">source =</span> <span class="string">&#8220;/tmp/src2&#8243;</span>,</div>
<div class="line"><span class="variable">host =</span> <span class="string">&#8220;172.29.88.223&#8221;</span>,</div>
<div class="line"><span class="variable">targetdir =</span> <span class="string">&#8220;/remote/dir&#8221;</span>,</div>
<div class="line"><span class="variable">excludeFrom =</span> <span class="string">&#8220;/etc/rsyncd.d/rsync_exclude.lst&#8221;</span>,</div>
<div class="line">&#8212; <span class="variable">maxDelays =</span> <span class="number">5</span>,</div>
<div class="line"><span class="variable">delay =</span> <span class="number">0</span>,</div>
<div class="line">&#8212; <span class="variable">init =</span> <span class="constant">false</span>,</div>
<div class="line"><span class="variable">rsync =</span> {</div>
<div class="line"><span class="variable">binary =</span> <span class="string">&#8220;/usr/bin/rsync&#8221;</span>,</div>
<div class="line"><span class="variable">archive =</span> <span class="constant">true</span>,</div>
<div class="line"><span class="variable">compress =</span> <span class="constant">true</span>,</div>
<div class="line"><span class="variable">verbose =</span> <span class="constant">true</span>,</div>
<div class="line"><span class="variable">_extra =</span> {<span class="string">&#8220;&#8211;bwlimit=2000&#8243;</span>},</div>
<div class="line">},</div>
<div class="line"><span class="variable">ssh =</span> {</div>
<div class="line"><span class="variable">port =</span> <span class="number">1234</span></div>
<div class="line">}</div>
<div class="line">}</div>
</td>
</tr>
</tbody>
</table>
</figure>
<p>上面的内容几乎涵盖了所有同步的模式，其中第<code>III</code>个要求像rsync一样配置rsyncd服务端，见本文开头。第<code>IV</code>、<code>V</code>配置ssh方式同步，达到的效果相同，但实际同步时你会发现每次同步都会提示输入ssh的密码，可以通过以下方法解决：</p>
<p>在远端被同步的服务器上开启ssh无密码登录，请注意用户身份：</p>
<figure class="highlight">
<table>
<tbody>
<tr>
<td class="gutter">
<div class="line">1</div>
<div class="line">2</div>
<div class="line">3</div>
<div class="line">4</div>
</td>
<td class="code">
<div class="line">user$ ssh-keygen -t rsa</div>
<div class="line">一路回车<span class="keyword">&#8230;</span></div>
<div class="line">user$ cd ~/.ssh</div>
<div class="line">user$ cat id_rsa.pub &gt;&gt; authorized_keys</div>
</td>
</tr>
</tbody>
</table>
</figure>
<p>把<code>id_rsa</code>私钥拷贝到执行lsyncd的机器上</p>
<figure class="highlight">
<table>
<tbody>
<tr>
<td class="gutter">
<div class="line">1</div>
<div class="line">2</div>
<div class="line">3</div>
</td>
<td class="code">
<div class="line">user<span class="variable">$ </span>chmod <span class="number">600</span> ~<span class="regexp">/.ssh/id</span>_rsa</div>
<div class="line">测试能否无密码登录</div>
<div class="line">user<span class="variable">$ </span>ssh user<span class="variable">@172</span>.<span class="number">29.88</span>.<span class="number">223</span></div>
</td>
</tr>
</tbody>
</table>
</figure>
<h1 id="3-_lsyncd的其它功能">3. lsyncd的其它功能</h1>
<p><code>lsyncd</code>的功能不仅仅是同步，官方手册 <a href="https://github.com/axkibe/lsyncd/wiki/Lsyncd%202.1.x%20%E2%80%96%20Layer%202%20Config%20%E2%80%96%20Advanced%20onAction" target="_blank" rel="external">Lsyncd 2.1.x ‖ Layer 2 Config ‖ Advanced onAction</a> 高级功能提到，还可以监控某个目录下的文件，根据触发的事件自己定义要执行的命令，example是监控某个某个目录，只要是有jpg、gif、png格式的文件参数，就把它们转成pdf，然后同步到另一个目录。正好在我运维的一个项目中有这个需求，现在都是在java代码里转换，还容易出现异常，通过lsyncd可以代替这样的功能。但，门槛在于要会一点点lua语言（根据官方example还是可以写出来）。</p>
<p>另外偶然想到个问题，同时设置了<code>maxDelays</code>和<code>delay</code>，当监控目录一直没有文件变化了，也会发生同步操作，虽然没有可rsync的文件。</p>
<p>TO-DO：</p>
<ul>
<li>其它同步工具：<a href="http://www.chinaxing.org/articles/linux/2013/04/18/2013-04-18-incrond-and-csync2.html" target="_blank" rel="external">csync2</a>，clsync，btsync，drdb 。</li>
<li>lsyncd双向同步：<a href="https://github.com/axkibe/lsyncd/wiki/Faq" target="_blank" rel="external">GlusterFS</a></li>
</ul>
]]></content:encoded>
			<wfw:commentRss>http://bernieyu.com/2015/10/lsyncd-synchronize-realtime/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>VPS搭建LNMP过程全记录</title>
		<link>http://bernieyu.com/2015/06/build-nginx-mysql-php-full-record/</link>
		<comments>http://bernieyu.com/2015/06/build-nginx-mysql-php-full-record/#comments</comments>
		<pubDate>Tue, 09 Jun 2015 08:56:47 +0000</pubDate>
		<dc:creator><![CDATA[Bernie Yu]]></dc:creator>
				<category><![CDATA[IT技术]]></category>
		<category><![CDATA[Linux]]></category>
		<category><![CDATA[linux]]></category>
		<category><![CDATA[mysql]]></category>
		<category><![CDATA[nginx]]></category>
		<category><![CDATA[php]]></category>
		<category><![CDATA[vps]]></category>
		<category><![CDATA[安装]]></category>

		<guid isPermaLink="false">http://bernieyu.com/?p=227</guid>
		<description><![CDATA[最近有朋友新买一个VPS，让帮忙搭建服务器（Linux + Nginx + MySQL + PHP）。 本来觉 ... <a class="more-link" href="http://bernieyu.com/2015/06/build-nginx-mysql-php-full-record/">　　>>阅读全文&#60;&#60;</a>]]></description>
				<content:encoded><![CDATA[<p>最近有朋友新买一个VPS，让帮忙搭建服务器（Linux + Nginx + MySQL + PHP）。<br />
本来觉得手到擒来的事，可操作起来却发现有些东西记忆已经模糊了。帮朋友弄好后，赶紧把过程记下来，以备后查。</p>
<ol>
<li>更新系统
<ul>
<li>
<pre>apt-get update
apt-get upgrade</pre>
</li>
</ul>
</li>
<li>安装MySQL
<ul>
<li>
<pre>apt-get install mysql-server mysql-client</pre>
<p>MySQL 安全设置</p>
<pre>mysql_secure_installation</pre>
<p>过程中会询问是否更改 root密码，是否移除匿名用户，是否禁止root远程登录等。</li>
</ul>
</li>
<li>安装Nginx
<ul>
<li>
<pre>apt-get install nginx
service nginx start</pre>
<p>如果没有service命令，则使用</p>
<pre>/etc/init.d/nginx start</pre>
<p>直接访问IP地址，验证nginx安装是否成功<br />
<img class="alignnone size-full wp-image-229" src="http://bernieyu.com/wp-content/uploads/2015/06/20150609154658.png" alt="20150609154658" width="390" height="181" /></li>
</ul>
</li>
<li>安装PHP
<ul>
<li>
<pre>apt-get install php5-fpm php5-gd php5-mysql php5-curl</pre>
<p>安装完毕之后，查看PHP的版本，验证安装是否成功 ：</p>
<pre>root@vultr:/www# php5-fpm -v
PHP 5.3.10-1ubuntu3.18 (fpm-fcgi) (built: Apr 17 2015 15:08:33)
Copyright (c) 1997-2009 The PHP Group
Zend Engine v2.3.0, Copyright (c) 1998-2012 Zend Technologies</pre>
</li>
</ul>
</li>
<li>配置PHP-FPM
<ul>
<li>编辑/etc/php5/fpm/php.ini
<pre>vi /etc/php5/fpm/php.ini</pre>
<pre>cgi.fix_pathinfo=0

error_reporting =E_ALL | E_STRICT

display_errors = On

log_errors = On
error_log = /var/log/php_error.log</pre>
</li>
<li>编辑 /etc/php5/fpm/pool.d/www.conf
<pre>vi /etc/php5/fpm/pool.d/www.conf</pre>
<pre>user = www-data
group = www-data

;listen = 127.0.0.1:9000
listen = /var/run/php5-fpm.sock
listen.owner = www-data
listen.group = www-data
listen.mode = 0660

pm.status_path = /fpm_status</pre>
</li>
<li>重启php5-fpm
<pre>service php5-fpm restart</pre>
</li>
</ul>
</li>
<li>配置Nginx
<ul>
<li>编辑/etc/nginx/nginx.conf
<pre>user  www-data;
worker_processes  4;

error_log  /var/log/nginx/error.log warn;
pid        /var/run/nginx.pid;


events {
    worker_connections  1024;
}


http {
    include       /etc/nginx/mime.types;
    default_type  application/octet-stream;

    log_format  main  '$remote_addr - $remote_user [$time_local] "$request" '
                      '$status $body_bytes_sent "$http_referer" '
                      '"$http_user_agent" "$http_x_forwarded_for"';

    access_log  /var/log/nginx/access.log  main;

    sendfile        on;
    #tcp_nopush     on;

    keepalive_timeout  65;

    server_tokens off;

    gzip  on;
    gzip_disable "msie6";
    gzip_buffers 4 16K;
    gzip_comp_level 5;
    gzip_types text/plain application/x-javascript text/css application/xml text/javascript application/x-httpd-php image/jpeg image/gif image/png application/javascript;
    gzip_vary on;
    include /etc/nginx/sites-available/*.conf;
}</pre>
</li>
</ul>
</li>
<li>配置网站
<ul>
<li>新建/www/test目录
<pre>mkdir /www
mkdir /www/test</pre>
</li>
<li>新建index.php
<pre>vi /www/test/index.php</pre>
<pre>&lt;?php

phpinfo();

?&gt;</pre>
</li>
<li>更改目录权限
<pre>chown -R www-data:www-data /www/test</pre>
</li>
<li>新建网站配置
<pre>vi /etc/nginx/conf.d/www.test.com.conf</pre>
<pre>server {
    listen       80;
    server_name  www.test.com;

    access_log /var/log/nginx/www.test.com.access.log main;
    error_log  /var/log/nginx/www.test.com.error.log warn;

    error_page  404              /404.html;
    error_page  500 502 503 504  /50x.html;
    location = /50x.html {
        root   /usr/share/nginx/html;
    }

    location / {
        root /www/test;
        index  index.html index.htm index.php;
        #WordPress伪静态,不需要可以删除
        if (-f $request_filename/index.html){
                rewrite (.*) $1/index.html break;
        }
        if (-f $request_filename/index.php){
                rewrite (.*) $1/index.php;
        }
        if (!-f $request_filename){
                rewrite (.*) /index.php;
        }
    }

    #fpm_status,可通过www.test.com/fpm_status查看fpm状态，添加?full参数可查看详细状态。
    #正式上线时需要注释掉此段
    location ~ ^/(fpm_status)$ {
     access_log off;
     include fastcgi_params;
     fastcgi_pass unix:/var/run/php5-fpm.sock;
     fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name;
    }

    location ~ \.php$ {
        proxy_buffers 8 16k;
        proxy_buffer_size 32k;
        root   /www/test;
        # With php5-fpm:
        fastcgi_pass unix:/var/run/php5-fpm.sock;
        fastcgi_index index.php;
        fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name;
        include fastcgi_params;
    }
}
</pre>
</li>
<li>链接到sites-available
<pre>ln -s /etc/nginx/conf.d/www.test.com.conf /etc/nginx/sites-available/www.test.com.conf</pre>
</li>
<li>测试nginx配置
<pre>nginx -t</pre>
<p>如果配置文件正常，则显示：</p>
<pre>nginx: the configuration file /etc/nginx/nginx.conf syntax is ok
nginx: configuration file /etc/nginx/nginx.conf test is successful</pre>
<p>如果报错，则检查配置文件</li>
<li>待一切正常后，重启nginx
<pre>service nginx restart</pre>
</li>
<li>此时访问www.test.com，会出现PHP运行信息<br />
<img class="alignnone size-medium wp-image-230" src="http://bernieyu.com/wp-content/uploads/2015/06/20150609164645-300x236.png" alt="20150609164645" width="300" height="236" /></li>
<li>访问 www.test.com/fpm_status 或者 www.test.com/fpm_status?full 可以查看fpm状态</li>
</ul>
</li>
</ol>
]]></content:encoded>
			<wfw:commentRss>http://bernieyu.com/2015/06/build-nginx-mysql-php-full-record/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>Linux创建和修改用户及密码</title>
		<link>http://bernieyu.com/2015/06/linux-create-and-modify-user-and-password/</link>
		<comments>http://bernieyu.com/2015/06/linux-create-and-modify-user-and-password/#comments</comments>
		<pubDate>Mon, 01 Jun 2015 04:23:08 +0000</pubDate>
		<dc:creator><![CDATA[Bernie Yu]]></dc:creator>
				<category><![CDATA[IT技术]]></category>
		<category><![CDATA[Linux]]></category>
		<category><![CDATA[linux]]></category>

		<guid isPermaLink="false">http://bernieyu.com/?p=219</guid>
		<description><![CDATA[linux下创建用户 1、添加ftp用户 useradd ftpname -d /home/ftp passw ... <a class="more-link" href="http://bernieyu.com/2015/06/linux-create-and-modify-user-and-password/">　　>>阅读全文&#60;&#60;</a>]]></description>
				<content:encoded><![CDATA[<p>linux下创建用户<br />
1、添加ftp用户</p>
<pre>useradd ftpname -d /home/ftp
passwd ftppwd</pre>
<p>以下操作都以root权限进行：</p>
<pre>service vsftpd start 启动Linux自带的FTP服务</pre>
<p>2. 添加普通用户</p>
<pre> useradd nagios –d /usr/local/nagios
chown nagios.nagios /usr/local/nagios</pre>
<p>将用户切换到nagios 用户，添加nagios的访问用户：</p>
<pre>su nagios
htpasswd –c /usr/local/nagios/etc/htpasswd.users nagiosadmin</pre>
<p>系统会提示输入两次密码，例如两次输入nagios。<br />
注: 这是第一次创建用户，需要“-c”选项，表示创建一个存放用户名和密码的文件，当再次添加用户或修改密码时不用“-c”选项。</p>
<p>添加新的用户账号使用useradd命令，其语法如下：</p>
<pre>　　useradd 选项 用户名</pre>
<p>其中各选项含义如下：</p>
<pre>　　-c comment 指定一段注释性描述。

　　-d 目录 指定用户主目录，如果此目录不存在，则同时使用-m选项，可以创建主目录。

　　-g 用户组 指定用户所属的用户组。

　　-G 用户组，用户组 指定用户所属的附加组。

　　-s Shell文件 指定用户的登录Shell。

　　-u 用户号 指定用户的用户号，如果同时有-o选项，则可以重复使用其他用户的标识号。</pre>
<p>用户名 指定新账号的登录名。</p>
<p>例如：</p>
<pre>　　# useradd –d /usr/sam -m sam</pre>
<p>此命令创建了一个用户sam，其中-d和-m选项用来为登录名sam产生一个主目录/usr/sam（/usr为默认的用户主目录所在的父目录）。</p>
<pre>　　# useradd -s /bin/sh -g group –G adm,root gem</pre>
<p>此命令新建了一个用户gem，该用户的登录Shell是/bin/sh，它属于group用户组，同时又属于adm和root用户组，其中group用户组是其主组。</p>
<p>增加用户账号就是在/etc/passwd文件中为新用户增加一条记录，同时更新其他系统文件如/etc/shadow, /etc/group等。这几个文件的内容在后面会做详细介绍。</p>
<p>Linux提供了集成的系统管理工具userconf，它可以用来对用户账号进行统一管理。</p>
<p>如果一个用户的账号不再使用，可以从系统中删除。删除用户账号就是要将/etc/passwd等系统文件中的该用户记录删除，必要时还删除用户的主目录。删除一个已有的用户账号使用userdel命令，其格式如下：</p>
<p>userdel 选项 用户名</p>
<p>常用的选项是-r，它的作用是把用户的主目录一起删除。</p>
<p>例如：</p>
<pre>　　# userdel sam</pre>
<p>此命令删除用户sam在系统文件中（主要是/etc/passwd, /etc/shadow, /etc/group等）的记录，同时删除用户的主目录。</p>
<p>修改用户账号就是根据实际情况更改用户的有关属性，如用户号、主目录、用户组、登录Shell等。</p>
<p>修改已有用户的信息使用usermod命令，其格式如下：</p>
<pre>　　usermod 选项 用户名</pre>
<p>常用的选项包括-c, -d, -m, -g, -G, -s, -u以及-o等，这些选项的意义与useradd命令中的选项一样，可以为用户指定新的资源值。另外，有些系统可以使用如下选项：</p>
<p>-l 新用户名</p>
<p>这个选项指定一个新的账号，即将原来的用户名改为新的用户名。</p>
<p>例如：</p>
<pre>　　# usermod -s /bin/ksh -d /home/z –g developer sam</pre>
<p>此命令将用户sam的登录Shell修改为ksh，主目录改为/home/z，用户组改为developer。</p>
<p>用户管理的一项重要内容是用户口令的管理。用户账号刚创建时没有口令，但是被系统锁定，无法使用，必须为其指定口令后才可以使用，即使是指定空口令。</p>
<p>指定和修改用户口令的Shell命令是passwd。超级用户可以为自己和其他用户指定口令，普通用户只能用它修改自己的口令。命令的格式为：</p>
<pre>　　passwd 选项 用户名</pre>
<p>可使用的选项：</p>
<pre>　　-l 锁定口令，即禁用账号。

　　-u 口令解锁。

　　-d 使账号无口令。

　　-f 强迫用户下次登录时修改口令。</pre>
<p>如果默认用户名，则修改当前用户的口令。</p>
<p>例如，假设当前用户是sam，则下面的命令修改该用户自己的口令：</p>
<pre>　　$ passwd

　　Old password:******

　　New password:*******

　　Re-enter new password:*******</pre>
<p>如果是超级用户，可以用下列形式指定任何用户的口令：</p>
<pre>　　# passwd sam

　　New password:*******

　　Re-enter new password:*******</pre>
<p>普通用户修改自己的口令时，passwd命令会先询问原口令，验证后再要求用户输入两遍新口令，如果两次输入的口令一致，则将这个口令指定给用户；而超级用户为用户指定口令时，就不需要知道原口令。</p>
<p>为了系统安全起见，用户应该选择比较复杂的口令，例如最好使用8位长的口令，口令中包含有大写、小写字母和数字，并且应该与姓名、生日等不相同。</p>
<p>为用户指定空口令时，执行下列形式的命令：</p>
<pre>　　# passwd -d sam</pre>
<p>此命令将用户sam的口令删除，这样用户sam下一次登录时，系统就不再询问口令。</p>
<p>passwd命令还可以用-l(lock)选项锁定某一用户，使其不能登录，例如：</p>
<pre>　　# passwd -l sam</pre>
]]></content:encoded>
			<wfw:commentRss>http://bernieyu.com/2015/06/linux-create-and-modify-user-and-password/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
	</channel>
</rss>
