<?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>Fwolf's Blog &#187; hooks - Fwolf's Blog</title>
	<atom:link href="http://www.fwolf.com/blog/post/tag/hooks/feed" rel="self" type="application/rss+xml" />
	<link>http://www.fwolf.com/blog</link>
	<description>随心·随意·随缘·努力～</description>
	<lastBuildDate>Wed, 07 Jul 2010 07:07:52 +0000</lastBuildDate>
	<language>en</language>
	<sy:updatePeriod>hourly</sy:updatePeriod>
	<sy:updateFrequency>1</sy:updateFrequency>
	<generator>http://wordpress.org/?v=3.0</generator>
		<item>
		<title>[Git]提交后自动发email</title>
		<link>http://www.fwolf.com/blog/post/431</link>
		<comments>http://www.fwolf.com/blog/post/431#comments</comments>
		<pubDate>Thu, 26 Mar 2009 18:37:57 +0000</pubDate>
		<dc:creator>Fwolf</dc:creator>
				<category><![CDATA[Linux]]></category>
		<category><![CDATA[Svn/Git]]></category>
		<category><![CDATA[email]]></category>
		<category><![CDATA[git]]></category>
		<category><![CDATA[gmail]]></category>
		<category><![CDATA[hooks]]></category>
		<category><![CDATA[msmtp]]></category>
		<category><![CDATA[tls]]></category>

		<guid isPermaLink="false">http://www.fwolf.com/blog/post/431</guid>
		<description><![CDATA[当然是通过hooks来实现了，对应post-receive，脚本也是已经随git-core安装就有了的：/usr/share/doc/git-core/contrib/hooks/post-receive-email，不过这个脚本用到了sendmail，我想好多机器上都没配这个东东吧，好在有好心人在这个脚本的基础上进行了完善，可以用msmtp发信了，看来我以前换用msmtp配mutt太正确了，另外msmtp也确实比sendmail小巧多了。 首先把刚才说到的脚本以及依赖文件functions一起下载到某地，加上执行属性（公用），然后ln到仓库的hooks目录下： $ cd test.git/hooks/ $ mv post-receive post-receive.origin $ ln -s ../../gittools/post-receive-email post-receive $ ln -s ../../gittools/functions 然后修改仓库里的config文件，注意不是客户端的： [hooks "post-receive-email"] mailinglist = list1@domain.tld, list2@domain.tld announcelist = envelopesender = mailsender@domain.tld sendmail = /usr/bin/msmtp 其中： mailinglist 默认的收信人，留空就不发信了 announcelist 创建tag时发送从上次创建tag以来的汇总邮件的收信人，比如用版本号作为tag的时候这就生成了每个版本的changelog，留空则用mailinglist的值 envelopesender 发信人/账号，要和msmtp的发信账号对应起来 sendmail sendmail或其它发信程序的路径 还有个环境变量$USER_EMAIL，也是个发信人，不过是显示在邮件From:里的。如果要设置邮件里的From:，需要自己修改post-receive-email或在其之前执行的脚本，设置环境变量： export USER_EMAIL="GIT &#60;mailsender@domain.tld&#62;" 这一版修改中作者还去掉了原来的emailprefix设置参数，固定为用仓库里description文件的内容加上[]替代。 最后，在git（git服务器运行用户）用户的HOME下放一个.msmtprc，配置发信认证信息： defaults tls_trust_file /etc/ssl/certs/ca-certificates.crt account gmail host smtp.gmail.com [...]]]></description>
			<content:encoded><![CDATA[<p>当然是通过hooks来实现了，对应post-receive，脚本也是已经随git-core安装就有了的：<code>/usr/share/doc/git-core/contrib/hooks/post-receive-email</code>，不过这个脚本用到了sendmail，我想好多机器上都没配这个东东吧，好在有好心人在这个脚本的基础上进行了<a href="http://github.com/stephenh/git-central/blob/master/server/post-receive-email">完善</a>，可以用msmtp发信了，看来我以前<a href="379">换用msmtp配mutt</a>太正确了，另外msmtp也确实比sendmail小巧多了。</p>

<p>首先把刚才说到的<a href="http://github.com/stephenh/git-central/blob/master/server/post-receive-email">脚本</a>以及依赖文件<a href="http://github.com/stephenh/git-central/blob/master/server/functions">functions</a>一起下载到某地，加上执行属性（公用），然后ln到仓库的hooks目录下：</p>

<pre><code>$ cd test.git/hooks/
$ mv post-receive post-receive.origin
$ ln -s ../../gittools/post-receive-email post-receive
$ ln -s ../../gittools/functions
</code></pre>

<p>然后修改仓库里的config文件，注意不是客户端的：</p>

<pre><code>[hooks "post-receive-email"]
    mailinglist = list1@domain.tld, list2@domain.tld
    announcelist =
    envelopesender = mailsender@domain.tld
    sendmail = /usr/bin/msmtp
</code></pre>

<p>其中：</p>

<ul>
<li>mailinglist 默认的收信人，留空就不发信了</li>
<li>announcelist 创建tag时发送从上次创建tag以来的汇总邮件的收信人，比如用版本号作为tag的时候这就生成了每个版本的changelog，留空则用mailinglist的值</li>
<li>envelopesender 发信人/账号，要和msmtp的发信账号对应起来</li>
<li>sendmail sendmail或其它发信程序的路径</li>
</ul>

<p>还有个环境变量<code>$USER_EMAIL</code>，也是个发信人，不过是显示在邮件<code>From:</code>里的。如果要设置邮件里的<code>From:</code>，需要自己修改post-receive-email或在其之前执行的脚本，设置环境变量：</p>

<pre><code>export USER_EMAIL="GIT &lt;mailsender@domain.tld&gt;"
</code></pre>

<p>这一版修改中作者还去掉了原来的<code>emailprefix</code>设置参数，固定为用仓库里<code>description</code>文件的内容加上<code>[]</code>替代。</p>

<p>最后，在git（git服务器运行用户）用户的HOME下放一个<code>.msmtprc</code>，配置发信认证信息：</p>

<pre><code>defaults
    tls_trust_file /etc/ssl/certs/ca-certificates.crt
account gmail
    host smtp.gmail.com
    port 465
    auth on
    tls on
    tls_starttls off
account mailsender@domain.tld : gmail
#   from must be same as From: in mail, exclude Name, only mail address
#   Because maybe msmtp select account using from
    from mailsender@domain.tld
    user "mailsender@domain.tld"
    password my_passwd
account default : mailsender@domain.tld
</code></pre>

<p>现在就可以测试push，看自动发信是否正常了。如果提示<code>functions</code>语法错误，比如：</p>

<pre><code>hooks/functions: 213: Syntax error: Bad for loop variable
</code></pre>

<p>可以把这两个脚本第一行的<code>#!/bin/sh</code>改为<code>#!/bin/bash</code>。</p>

<p>注意msmtp调用的是提交动作所属用户的<code>.msmtprc</code>，从本机提交和远程ssh进来可能用的就不是一个<code>.msmtprc</code>了。</p>

<h4>参考</h4>

<ul>
<li><a href="http://developer.berlios.de/docman/display_doc.php?docid=1813&amp;group_id=2#commit-email">Getting email from GIT</a></li>
<li><a href="http://www.rubick.com/blogger/one-entry?entry_id=30408">Setting up git commit emails</a></li>
<li><a href="http://ubuntuforums.org/archive/index.php/t-703467.html">help regarding for loop syntax in ubuntu 7.10</a></li>
</ul>

	Tags: <a href="http://www.fwolf.com/blog/post/tag/email" title="email" rel="tag">email</a>, <a href="http://www.fwolf.com/blog/post/tag/git" title="git" rel="tag">git</a>, <a href="http://www.fwolf.com/blog/post/tag/gmail" title="gmail" rel="tag">gmail</a>, <a href="http://www.fwolf.com/blog/post/tag/hooks" title="hooks" rel="tag">hooks</a>, <a href="http://www.fwolf.com/blog/post/tag/msmtp" title="msmtp" rel="tag">msmtp</a>, <a href="http://www.fwolf.com/blog/post/tag/tls" title="tls" rel="tag">tls</a><br />

	<h4>Related posts</h4>
	<ul class="st-related-posts">
	<li><a href="http://www.fwolf.com/blog/post/379" title="[Mutt]用msmtp替代esmtp作发信代理 (2008-01-12)">[Mutt]用msmtp替代esmtp作发信代理</a> (2)</li>
	<li><a href="http://www.fwolf.com/blog/post/429" title="Git起步 (2009-03-26)">Git起步</a> (6)</li>
	<li><a href="http://www.fwolf.com/blog/post/155" title="终于能够通过phpmailer使用gmail账号发送邮件了 (2006-04-14)">终于能够通过phpmailer使用gmail账号发送邮件了</a> (50)</li>
	<li><a href="http://www.fwolf.com/blog/post/127" title="利用SVN更新网站 (2006-01-19)">利用SVN更新网站</a> (7)</li>
	<li><a href="http://www.fwolf.com/blog/post/441" title="[Git]真正回滚已上传的更新 (2009-05-14)">[Git]真正回滚已上传的更新</a> (0)</li>
</ul>

]]></content:encoded>
			<wfw:commentRss>http://www.fwolf.com/blog/post/431/feed</wfw:commentRss>
		<slash:comments>2</slash:comments>
		</item>
		<item>
		<title>Git起步</title>
		<link>http://www.fwolf.com/blog/post/429</link>
		<comments>http://www.fwolf.com/blog/post/429#comments</comments>
		<pubDate>Wed, 25 Mar 2009 18:18:10 +0000</pubDate>
		<dc:creator>Fwolf</dc:creator>
				<category><![CDATA[Linux]]></category>
		<category><![CDATA[Svn/Git]]></category>
		<category><![CDATA[Apache]]></category>
		<category><![CDATA[cgi]]></category>
		<category><![CDATA[config]]></category>
		<category><![CDATA[git]]></category>
		<category><![CDATA[gitosis]]></category>
		<category><![CDATA[gitweb]]></category>
		<category><![CDATA[hooks]]></category>
		<category><![CDATA[svn]]></category>
		<category><![CDATA[webdav]]></category>

		<guid isPermaLink="false">http://www.fwolf.com/blog/post/429</guid>
		<description><![CDATA[我的学习路径：git basic, http webdav, git-svn, gitosis, gitweb，下面是一些学习记录。 http webdav方式访问 ubuntu下安装非常简单，用apt装上git-core包即可。git仓库的外部访问，首先就是最熟悉的用http协议走apache上的webdav服务，方便啊。git自身的身份认证和权限分配弱了一些，不过可以借助apache实现用户分配，passwd文件直接用svn的也没问题。apache配置示例如下： Alias /git "/big2/git" &#60;Location /git/test.git&#62; DAV on AuthType Basic AuthName "Test git freely" AuthUserFile "/big2/svn/svnpasswd" #Require user Fwolf Require valid-user &#60;/Location&#62; 但这样远不如使用ssh/git协议访问仓库速度快和稳定，但更适合公开/对外发布。另外和svn的webdav方式不同，git webdav是一种dumb transport，不仅慢而且比较傻，只能进行简单的存、取文件操作，所以hooks是不起作用的（很多文章中说的提交后要在hooks/post-update里启用git-update-server-info，指的是git/ssh方式更新后，为webdav对外发布作准备）不会自动进行和git相关的其它处理。 最好用的还是gitosis，后面会提到。有些内容先以webdav方式下来记，使用的时候根据不同的方式转换用户角色即可。 创建服务端的仓库 其实和svn创建仓库也是比较类似的，找一空目录，比如叫test.git，执行： $ git --bare init $ sudo chown www-data:www-data * -R 客户端就可以下载了，但还有些问题： $ git clone https://domain.tld/git/test.git gittest Initialized empty [...]]]></description>
			<content:encoded><![CDATA[<p>我的学习路径：git basic, http webdav, git-svn, gitosis, gitweb，下面是一些学习记录。</p>

<h3>http webdav方式访问</h3>

<p>ubuntu下安装非常简单，用apt装上git-core包即可。git仓库的外部访问，首先就是最熟悉的用http协议走apache上的webdav服务，方便啊。git自身的身份认证和权限分配弱了一些，不过可以借助apache实现用户分配，passwd文件直接用<a href="66">svn的</a>也没问题。apache配置示例如下：</p>

<pre><code>Alias /git "/big2/git"
&lt;Location /git/test.git&gt;
    DAV on
    AuthType Basic
    AuthName "Test git freely"
    AuthUserFile "/big2/svn/svnpasswd"
    #Require user Fwolf
    Require valid-user
&lt;/Location&gt;
</code></pre>

<p>但这样远不如使用ssh/git协议访问仓库速度快和稳定，但更适合公开/对外发布。另外和svn的webdav方式不同，git webdav是一种<code>dumb transport</code>，不仅慢而且比较傻，只能进行简单的存、取文件操作，所以hooks是不起作用的（很多文章中说的提交后要在<code>hooks/post-update</code>里启用<code>git-update-server-info</code>，指的是git/ssh方式更新后，为webdav对外发布作准备）不会自动进行和git相关的其它处理。</p>

<p>最好用的还是gitosis，后面会提到。有些内容先以webdav方式下来记，使用的时候根据不同的方式转换用户角色即可。</p>

<h3>创建服务端的仓库</h3>

<p>其实和<a href="63">svn创建仓库</a>也是比较类似的，找一空目录，比如叫<code>test.git</code>，执行：</p>

<pre><code>$ git --bare init
$ sudo chown www-data:www-data * -R
</code></pre>

<p>客户端就可以下载了，但还有些问题：</p>

<pre><code>$ git clone https://domain.tld/git/test.git gittest
Initialized empty Git repository in /home/fwolf/dev/gittest/.git/

error: server certificate verification failed. CAfile: /etc/ssl/certs/ca-certificates.crt
</code></pre>

<p>首先遇到的是证书错误，git使用curl，所以证书的处理和svn也不一样，用firefox浏览git网站，在Page Info->View Certificate->Details中把网站的根证书(Issued By/root)导出为一个pem文件，然后将这个pem文件的内容添加到<code>/etc/ssl/certs/ca-certificates.crt</code>的末尾就可以了。接着遇到的是401身份验证错误：</p>

<pre><code>error: The requested URL returned error: 401
</code></pre>

<p>git不像svn会自动提示用户名/密码，因为它用的是webdav的原生功能，所以需要在<code>$HOME</code>下创建文件<code>.netrc</code>（记得将权限修改为600）：</p>

<pre><code>machine domain.tld
login Fwolf
password your_passwd
</code></pre>

<p>最后，还是仓库本身不完善的问题：</p>

<pre><code>warning: remote HEAD refers to nonexistent ref, unable to checkout.
</code></pre>

<p>在服务器上仓库目录，执行<code>$ git update-server-info</code>并再次chown到www-data就行了。初次push得指明服务器上的origin对应本地master分支/branch：</p>

<pre><code>$ git push origin master
</code></pre>

<p>而下载之前，还要设置下载后放到那个分支、与哪个分支的现有内容进行merge：</p>

<pre><code>$ git config branch.master.remote origin
$ git config branch.master.merge master
$ git pull  # 注意如果服务器是新建的空repo，得随便push点东西才好pull
</code></pre>

<h3>gitosis</h3>

<p>还是通过apt安装，gitosis使用SSH key来认证用户，但用户不需要在主机上开账号，而是公用主机上的一个受限账号。首先创建这个账号（名称可变）：</p>

<pre><code>sudo adduser \
    --system \
    --shell /bin/sh \
    --gecos 'git user' \
    --group \
    --disabled-password \
    --home /big2/gitosis \
    git
</code></pre>

<p>然后生成自己用户的ssh密钥，可以在git用户的$HOME下操作：</p>

<pre><code>ssh-keygen -t rsa
</code></pre>

<p>不要密码，可以起名叫git_rsa[.pub]，然后用公钥来初始化gitosis的目录：</p>

<pre><code>$ sudo -H -u git gitosis-init &lt; /home/fwolf/.ssh/git_rsa.pub 
Initialized empty Git repository in /big2/gitosis/repositories/gitosis-admin.git/
Reinitialized existing Git repository in /big2/gitosis/repositories/gitosis-admin.git/
</code></pre>

<p>可以看出gitosis自己也在用git仓库的形式存储东西，传递公钥进去是为了配置自动登录，将公钥写入git用户的<code>$HOME/.ssh/authorized_keys</code>。接下来就启用gitosis仓库的post-update hook：</p>

<pre><code>$ sudo chmod 755 repositories/gitosis-admin.git/hooks/post-update
</code></pre>

<p>其实用apt安装的gitosis，这个post-update早已指向其它文件并具有x属性了。</p>

<p>一点提示：把gitosis的repositories和自己原先的git repo目录弄到一起，以后方便，因为gitosis自己的管理也是要通过git来提交的，在客户端先修改<code>$home/.ssh/config</code>：</p>

<pre><code>Host git.domain.tld
Compression yes
IdentityFile ~/.ssh/git_rsa # 注意这个是私钥，不是pub公钥
</code></pre>

<p>然后从客户端clone gitosis的配置文档：</p>

<pre><code>$ git clone git@git.domain.tld:gitosis-admin.git
</code></pre>

<p>注意实际上这个<code>gitosis-admin.git</code>应该是在git用户的$HOME下的<code>repositories</code>目录下，但这么写才能下载。clone得到一个配置文件<code>gitosis.conf</code>和目录<code>keydir/</code>，对了，git配置的修改、上传也是通过git提交，并且由于刚才设置的post-update hook，立即生效。所以现在就能够远程修改、提交配置了。下面是一份比较完整的自说明的<code>gitosis.conf</code>：</p>

<pre><code># example: http://eagain.net/gitweb/?p=gitosis.git;a=blob;f=example.conf
[gitosis]

## Allow gitweb to show all known repositories. If you want gitweb,
## you need either this or a [repo foo] section for each repository
## you want visible in gitweb.
gitweb = no

[group gitosis-admin]
writable = gitosis-admin
# member is key file name, without (.pub)
members = fwolf

# user group
[group all]
# members = jdoe alice bob @anothergroup
members = fwolf
# this user group is writeable to ... repo, eg: test.git.
# writeable = foo bar baz/thud
writeable = test
#readonly = somerepo

[repo test]
## Allow gitweb to show this repository.
gitweb = yes
## Oneline description of the project, mostly for gitweb.
description = Fwolf's test git repository
## Owner of this repository. Used in gitweb list of projects.
owner = Fwolf
## Allow git-daemon to publish this repository.
#daemon = yes
daemon = no
</code></pre>

<p>平时就是为其它用户添加SSH key，然后用<code>git@git.domain.tld:repo.git</code>来连接服务器，gitosis会自动比对服务器上的pub key，判断出用户是谁。</p>

<h3>gitweb</h3>

<p>用apt安装，得到<code>/usr/lib/cgi-bin/gitweb.cgi</code>，由于<code>cgi-bin</code>已经由ubuntu设置好了，apache只要设置一下资源文件路径即可：</p>

<pre><code>Alias /gitweb "/usr/share/gitweb"
</code></pre>

<p>然后修改<code>/etc/gitweb.conf</code>：</p>

<pre><code># path to git projects (&lt;project&gt;.git)
$projectroot = "/big2/git";

$site_name = "Fwolf's Git Repositories";
@git_base_url_list = ("ssh://git\@git.domain.tld");

# directory to use for temp files
$git_temp = "/tmp";

# target of the home link on top of all pages
#$home_link = $my_uri || "/";

# html text to include at home page
$home_text = "indextext.html";

# file with project list; by default, simply scan the projectroot dir.
$projects_list = "/big2/gitosis/gitosis/projects.list";
# Avoid use guess repo's address and edit url manually
$strict_export = 1;

# stylesheet to use
$stylesheet = "/gitweb/gitweb.css";
# logo to use
$logo = "/gitweb/git-logo.png";
# the 'favicon'
$favicon = "/gitweb/git-favicon.png";
</code></pre>

<p>重启apache后就能够用<code>http://git.domain.tld/cgi-bin/gitweb.cgi</code>访问了。</p>

<h3>git-svn混用</h3>

<p>首先还要安装git-svn包，git和svn混用的好处就是既可以用git进行本地提交和灵活的分支，又能够用svn实现集中管理和分发，缺点我想可能就是svn的revision号在git里没有了，另外git有些太灵活，不是很好入门。</p>

<p>首先导入svn的现有项目到一个空目录：</p>

<pre><code>$ git svn clone --username your-name -s https://domain.tld/svn/repo
# older versions of git: replace "-s" with "-Ttrunk -bbranches -ttags"
</code></pre>

<p>就这么简单，很多工作git-svn都代劳了，剩下就当正常的git目录使用即可，比如为他们指定git仓库的位置，并下载文件：</p>

<pre><code>$ git config remote.origin.url https://domain.tld/git/test.git
$ git config remote.origin.fetch +refs/heads/*:refs/remotes/origin/*
$ git config branch.master.remote origin
$ git config branch.master.merge refs/heads/master
$ git checkout master   #回到master分支
$ git pull
</code></pre>

<p>向svn仓库提交：</p>

<pre><code>$ git svn dcommit
</code></pre>

<p>从svn仓库接受更新：</p>

<pre><code>$ git svn rebase
</code></pre>

<p>在git-svn的使用中还发现，如果在机器A上使用git-svn，但是在另外一台机器B上用纯git进行了push，那么B上的commit是不会被git-svn认出来并dcommit的，只有pull下来，然后再假commit一下才能让git-svn看到。</p>

<h3>仓库地址变更</h3>

<p>服务器上直接修改仓库目录名和apache设置，客户端上：</p>

<pre><code>$ git config remote.origin.url https://domain.tld/git/testnew.git
$ git pull  # 这个必须要做，不然不能push
Merge made by recursive.
$ git push
</code></pre>

<h3>git-svn方式下变更svn的地址</h3>

<p>首先要修改所有commit中的地址：</p>

<pre><code>$ git filter-branch --msg-filter 'sed "s/domain_old\/trunk/domain_new\/trunk/g"' $(cat .git/packed-refs | awk '// {print $2}' | grep -v 'pack-refs')
</code></pre>

<p>然后删掉缓存，修改地址，重新下载：</p>

<pre><code>$ rm -rf .git/svn
# 编辑`.git/config`修改`[svn-remote "svn"]`的url。
$ git svn rebase (--all)
</code></pre>

<p>我还曾经遇到过这样的错误：</p>

<pre><code>$ git svn rebase
Unable to determine upstream SVN information from working tree history
</code></pre>

<p>不太好处理，这里有些<a href="http://basementcoders.com/?p=63">参考信息</a>，我是在一个repo里pull另外一个仓库的内容了，照下面方式处理也不管用：</p>

<ul>
<li>从svn里重新clone出一份</li>
<li>git里创建个branch保存一下</li>
<li>git连接上服务器，pull</li>
<li>git checkout 到刚才创建的branch</li>
<li>git merge &#8211;no-ff master</li>
<li>git svn rebase/dcommit</li>
</ul>

<p>效果不好，不仅产生大量的conflict，而且还是只能在新创建的branch上rebase/dcommit。大概是出现了两“棵”树，怎么也捏不到一起，最后只好重建仓库。</p>

<h3>其它使用技巧/tips</h3>

<ul>
<li>回上个版本 <code>$ git reset --hard HEAD^</code></li>
<li>得到一个文件5个版本以前的状态 <code>$ git checkout HEAD~5 foo.c</code></li>
<li>由于git是分布式的scm，所以<code>$Id</code>被彻底消灭掉了。</li>
<li>不像svn，git能够记录文件的执行属性。</li>
<li>为显示加上颜色 <code>$ git config color.(branch|diff|interactive|status) auto</code>，或者<code>$ git config color.ui true</code></li>
<li>hooks不算仓库的内容，每个repo的hooks可以不同。</li>
<li>当心svn的hooks，如果git里已经commit，却因为svn的hooks使git-svn的dcommit失败，很麻烦。</li>
<li><code>.gitignore</code>里可以用<code>config.php</code>忽略文件，并且用<code>!mt/config.php</code>指明个别文件还是要归入scm管理的。</li>
<li>从现有的客户端目录中生成仓库目录 <code>$ git clone --bare dev_dir repo_dir</code> 每个客户端都是一个完整备份，方便。</li>
<li>用于ignore文件，<code>.gitignore</code>是要提交的版本，<code>.git/info/exclude</code>则是只在你本地生效的版本，不提交。</li>
<li>仅给ssh登录的开发者push/pull的访问权限，将其shell设置为<code>/usr/bin/git-shell</code>（未测试）.</li>
<li><code>info/allowed-users</code>可以控制用户对branch/tag的访问权限（未测试）。</li>
<li><code>git --bare init --shared=group</code>可以让仓库被同group的用户共享，不过有gitosis就不用这个了，我没测试。</li>
</ul>

<h4>参考</h4>

<ul>
<li><a href="http://flavio.castelli.name/howto_use_git_with_svn">Howto use Git and svn together(暂时不想向svn提交的用git-stash处理)</a></li>
<li><a href="http://zhwen.org/?p=articles/git">git使用总结</a></li>
<li><a href="http://google-opensource.blogspot.com/2008/05/develop-with-git-on-google-code-project.html">Develop with Git on a Google Code Project</a></li>
<li><a href="http://www.dualface.com/blog/?p=591">git-svn 使用</a></li>
<li><a href="http://blog.csdn.net/sunrock/archive/2008/06/05/2514204.aspx">git使用笔记收藏</a></li>
<li><a href="http://blog.punkid.cn/2008/03/28/howto-setup-git-repo-on-dreamhost/">在Dreamhost主机上搭建Git</a></li>
<li><a href="http://www.kernel.org/pub/software/scm/git/docs/howto/setup-git-server-over-http.txt">Setting up a git repository which can be pushed into and pulled from over HTTP(S)</a></li>
<li><a href="http://blog.ashchan.com/archive/2008/06/30/tags-on-git/">Tags on Git</a></li>
<li><a href="http://www.wretch.cc/blog/michaeloil/22286355">架設 Git Server(透過http &amp; 不透過http)</a></li>
<li><a href="http://blog.wangbin1979.com/2009/Mar/07">尝试使用git post-receive hook来自动更新blog</a></li>
<li><a href="http://git.or.cz/gitwiki/GitFaq#head-bbd07820198a9b8c6ec4fad4932341b64378a5f6">My HTTP repository has updates, which git clone misses. What happened?</a></li>
<li><a href="http://www.kernel.org/pub/software/scm/git/docs/githooks.html">githooks(5) Manual Page </a></li>
<li><a href="http://debuggable.com/posts/git-tip-auto-update-working-tree-via-post-receive-hook:49551efe-6414-4e86-aec6-544f4834cda3">Git Tip: Auto update working tree via post-receive hook</a></li>
<li><a href="http://scie.nti.st/2007/11/14/hosting-git-repositories-the-easy-and-secure-way">Hosting Git repositories, The Easy (and Secure) Way(架设gitosis)</a></li>
<li><a href="http://robbat2.livejournal.com/218418.html">Gitosis+GitWeb on Gentoo</a></li>
<li><a href="http://blog.chunzi.org/2008/11/24/57">gitweb on ubuntu</a></li>
<li><a href="http://www.robinlu.com/blog/archives/194">如何在svn系统中使用git</a></li>
<li><a href="http://s5unty.blogspot.com/2008/03/git-svn.html">git-svn 常用功能示例</a></li>
<li><a href="http://crazycode.javaeye.com/blog/213212">git-svn使用方式及使用中的问题（已解决）</a></li>
<li><a href="http://crazycode.javaeye.com/blog/311564">使用git bisect快速定位引入错误的版本</a></li>
</ul>

<h4>Update @ 2009-03-27</h4>

<p>使用git-svn的时候，有时候会出现git和svn树不太同步的情况，在进行svn rebase的时候总是重新merge文件，这种现象多是由于以前没有正确rebase引起的，而rebase不成功的一种原因就是git的<code>fast forward</code>。简单理解一下，git是按照树形结构排下来的，如果遇到前后两个节点被判定为“重复”的，git就聪明的作了个<code>fast forward</code>，直接使用后者了。这在git里没事，因为都是一颗树，而对于svn就不行了，它还记录着要从前面那个节点开始呢，所以合并的时候就出现了两个源头的情况，自然会产生问题。所以大体上向svn更新的顺序应该是这样的：</p>

<ul>
<li>git push      # git更新</li>
<li>git pull &#8211;no-ff  # 此处的&#8211;no-ff最为重要，平时可以不用，这里要用</li>
<li>git svn rebase    # 准备提交到svn之前先和当前的树合并</li>
<li>git svn dcommit</li>
<li>git pull &#8211;no-ff</li>
<li># 后面就是检查了，前面操作正常的话不会产生新的有效更新了</li>
<li>git push</li>
<li>git svn dcommit</li>
<li>git svn rebase</li>
<li>git pull</li>
<li>git push</li>
</ul>

<p>附一张比较典型的出错后来回折腾的结果图：
<a href="http://www.flickr.com/photos/fwolf/3389188179/" title="git-svn rebase dcommit error by fwolf, on Flickr"><img src="http://farm4.static.flickr.com/3581/3389188179_7302983664.jpg" width="266" height="467" alt="git-svn rebase dcommit error" /></a></p>

<h4>参考 @ 2009-03-29</h4>

<ul>
<li><a href="http://graysky.org/2008/12/git-branch-auto-tracking/">Git Branch Auto-Tracking</a> 从服务器上pull一个branch以后，方便的track起来</li>
<li><a href="http://blog.ericgoodwin.com/2008/4/10/auto-completion-with-git">Git Auto Completion On OS X</a> 命令自动补全</li>
<li><a href="http://www.zorched.net/2008/04/14/start-a-new-branch-on-your-remote-git-repository/">Start a New Branch on your Remote Git Repository</a></li>
</ul>

<p>有了<code>git track</code>以后，新建要和服务器同步的branch：</p>

<pre><code>$ git branch fwolf
Branch fwolf set up to track local branch refs/heads/master.

$ git track fwolf origin
tracking origin/fwolf

$ git checkout fwolf
Switched to branch "fwolf"

$ git push origin fwolf
Total 0 (delta 0), reused 0 (delta 0)
To git@domain.tld:repo.git
 * [new branch]      fwolf -&gt; fwolf
</code></pre>

<p>而从其它客户端下载这个branch就简单多了：</p>

<pre><code>$ git branch --track fwolf origin/fwolf
Branch fwolf set up to track remote branch refs/remotes/origin/fwolf.
</code></pre>

<p>掌握原理，不用<code>git track</code>也一样可以实现的。</p>

<h4>Update @ 2009-04-18</h4>

<p>MediaTemple使用的是CentOS，安装git也很简单：</p>

<pre><code>$ rpm -Uvh http://mirror.centos.org/centos/5/os/i386/CentOS/python-iniparse-0.2.3-4.el5.noarch.rpm
$ rpm -Uvh http://mirror.centos.org/centos/5/os/i386/CentOS/yum-3.2.19-18.el5.centos.noarch.rpm http://mirror.centos.org/centos/5/os/i386/CentOS/yum-fastestmirror-1.1.16-13.el5.centos.noarch.rpm  # 版本号可能会有变化
$ rpm -Uvh http://download.fedora.redhat.com/pub/epel/5/i386/epel-release-5-2.noarch.rpm
$ yum -y update
$ yum install git
</code></pre>

<p>参考：</p>

<ul>
<li><a href="http://www.theonlyjames.com/2008/08/yum-git-dv-3-5/">Installing yum and git on a MediaTemple (dv) 3.5</a></li>
<li><a href="http://uptolife.blogspot.com/2009/03/git.html">装上个git</a></li>
<li><a href="http://gonefish.cn/weblog/easy-install-git-in-centos/">在CentOS上安装Git</a></li>
<li><a href="http://www.themattharris.com/2008/08/06/installing-git-binaries-onto-a-mediatemple-server/">Installing Git binaries onto a MediaTemple server</a></li>
</ul>

<h4>Update @ 2009-04-27</h4>

<p>升级到ubuntu 9.04 jaunty后，客户端push时会遇到错误：</p>

<pre><code>$ git push
/usr/bin/gitosis-serve:5: UserWarning: Unbuilt egg for ClientCookie [unknown version] (/usr/lib/python2.6/dist-packages)
  from pkg_resources import load_entry_point
Traceback (most recent call last):
  File "/usr/bin/gitosis-serve", line 5, in &lt;module&gt;
    from pkg_resources import load_entry_point
  File "/usr/lib/python2.6/dist-packages/pkg_resources.py", line 2562, in &lt;module&gt;  
    working_set.require(__requires__)
  File "/usr/lib/python2.6/dist-packages/pkg_resources.py", line 626, in require
    needed = self.resolve(parse_requirements(requirements))
  File "/usr/lib/python2.6/dist-packages/pkg_resources.py", line 524, in resolve
    raise DistributionNotFound(req)  # XXX put more info here
pkg_resources.DistributionNotFound: gitosis==0.2
fatal: The remote end hung up unexpectedly
</code></pre>

<p>半天查不出原因，只是怀疑和python的版本有关系（2.5/2.6），因为<a href="http://packages.debian.org/lenny/all/gitosis/filelist">gitosis在debian包中的文件</a>，显示应该是对应python 2.5的，不过，试着自己用python（2.6）重新下载安装了一遍gitosis，好了：</p>


<div class="wp_syntax"><div class="code"><pre class="bash" style="font-family:monospace;">	$ git clone git:<span style="color: #000000; font-weight: bold;">//</span>eagain.net<span style="color: #000000; font-weight: bold;">/</span>gitosis
	$ <span style="color: #7a0874; font-weight: bold;">cd</span> gitosis
	$ <span style="color: #c20cb9; font-weight: bold;">sudo</span> python setup.py <span style="color: #c20cb9; font-weight: bold;">install</span></pre></div></div>


<p>不懂Python太吃亏了呀。另外<code>aptitude reinstall gitosis</code>是没用的。</p>

<h4>Update @ 2009-05-15</h4>

<p>从<a href="http://plog.longwin.com.tw/">Tsung</a>那里学来<a href="http://plog.longwin.com.tw/my_note-unix/2009/05/15/git-initial-env-setup-2009">一招</a>，可以更改 Git 默认的信息存放目录 .git ：</p>

<pre><code>修改 Shell 變數 GIT_DIR =&gt; "export GIT_DIR=.test"
</code></pre>

	Tags: <a href="http://www.fwolf.com/blog/post/tag/apache" title="Apache" rel="tag">Apache</a>, <a href="http://www.fwolf.com/blog/post/tag/cgi" title="cgi" rel="tag">cgi</a>, <a href="http://www.fwolf.com/blog/post/tag/config" title="config" rel="tag">config</a>, <a href="http://www.fwolf.com/blog/post/tag/git" title="git" rel="tag">git</a>, <a href="http://www.fwolf.com/blog/post/tag/gitosis" title="gitosis" rel="tag">gitosis</a>, <a href="http://www.fwolf.com/blog/post/tag/gitweb" title="gitweb" rel="tag">gitweb</a>, <a href="http://www.fwolf.com/blog/post/tag/hooks" title="hooks" rel="tag">hooks</a>, <a href="http://www.fwolf.com/blog/post/tag/svn" title="svn" rel="tag">svn</a>, <a href="http://www.fwolf.com/blog/post/tag/webdav" title="webdav" rel="tag">webdav</a><br />

	<h4>Related posts</h4>
	<ul class="st-related-posts">
	<li><a href="http://www.fwolf.com/blog/post/127" title="利用SVN更新网站 (2006-01-19)">利用SVN更新网站</a> (7)</li>
	<li><a href="http://www.fwolf.com/blog/post/411" title="由一个错误学到的一些php安全配置问题 (2008-07-13)">由一个错误学到的一些php安全配置问题</a> (4)</li>
	<li><a href="http://www.fwolf.com/blog/post/431" title="[Git]提交后自动发email (2009-03-27)">[Git]提交后自动发email</a> (2)</li>
	<li><a href="http://www.fwolf.com/blog/post/191" title="Apache的Order Allow Deny心得 (2006-06-26)">Apache的Order Allow Deny心得</a> (6)</li>
	<li><a href="http://www.fwolf.com/blog/post/395" title="防止垃圾评论的另类方法 (2008-04-10)">防止垃圾评论的另类方法</a> (6)</li>
</ul>

]]></content:encoded>
			<wfw:commentRss>http://www.fwolf.com/blog/post/429/feed</wfw:commentRss>
		<slash:comments>6</slash:comments>
		</item>
		<item>
		<title>WordPress 插件编写简例</title>
		<link>http://www.fwolf.com/blog/post/296</link>
		<comments>http://www.fwolf.com/blog/post/296#comments</comments>
		<pubDate>Mon, 26 Feb 2007 06:41:47 +0000</pubDate>
		<dc:creator>Fwolf</dc:creator>
				<category><![CDATA[Blog]]></category>
		<category><![CDATA[PHP]]></category>
		<category><![CDATA[example]]></category>
		<category><![CDATA[fliter]]></category>
		<category><![CDATA[hooks]]></category>
		<category><![CDATA[plugin]]></category>
		<category><![CDATA[WordPress]]></category>

		<guid isPermaLink="false">http://www.fwolf.com/blog/post/296</guid>
		<description><![CDATA[虽然市面上有很多 WordPress 的插件，但未必能够满足我们那些稍微“变态”一些的需求，所以，参照wp官方的插件编写指南，写下这个简单的说明和小例子，希望对大家有所帮助。 插件编写需要php的基础知识，并且本说明的内容大多只适用于wp1.5以后版本。 插件程序建议放置在wp-content/plugins/目录，如果有多个程序文件的话，可以在这个目录下建立子目录，程序所需的资源文件建议也放在这个目录下，便于更新和管理。 wp插件主要分为两种： Action 对wp的行为事件进行修改，比如修改数据库，发送邮件或者修改显示结构等。 Filter 对wp的输入输出内容进行修改。 在本例中，我们将建立一个简单的Filter插件，替换wp本来的文章摘要生成功能，把以前手工修改代码实现的内容，用插件来替代。 wp推荐在插件程序中加入这样的文件头，便于wp辨认插件的信息，就像下面这样： &#60;?php /* Plugin Name: 插件名称 Plugin URI: http://插件网址 Description: 插件说明 Version: 插件版本 Author: 插件作者 Author URI: http://作者网址 */ 插件的代码风格主要是基于过程方式的，所以要尽力避免函数重名，可以在函数名之前加上自己的名称前缀，也可以把自己的类包含在一个类当中，不过小型插件还不至于复杂到要使用类的程度吧？ 函数的编写就是php代码了，注意Filter一定要返回数据，即使没有修改也要返回，不然后续的其他函数就没法干活了。更多的信息可以看官方的插件api。 编写完成之后，函数是不会自动执行的，wp为插件函数的执行提供了一种hook机制，就是在原始的wp程序中，预置了一些hook或者叫挂载点，通过把插件函数挂在这些hook上，就可以被执行，目前wp提供的挂载点一览见Hooks列表。 针对我编写的函数，是要挂在生成文章摘要的地方，查找了一下Hooks列表，发现和摘要（excerpt）相关的有： default_excerpt excerpt_edit_pre excerpt_save_pre get_the_excerpt the_excerpt 从名称可以看出，default_excerpt是文章的默认摘要，excerpt_edit_pre和excerpt_save_pre是编辑文章是对摘要进行的预处理，get_the_excerpt是取摘要，the_excerpt是返回摘要。我编写的是生成摘要的功能，应该挂在get_the_excerpt或者是the_excerpt上，考虑the_excerpt可能更“底层”一些，挂在这里。 挂载的语法也是一个php函数调用，格式如下： add_filter&#40;'hook_name', 'your_filter', &#91;priority&#93;, &#91;accepted_args&#93;&#41;; priority为优先级，默认为10，数字小的会先执行；accepted_args为参数个数，默认值为1，如果插件函数需要多个参数的话需要自行设定。这两个参数在使用默认值的时候都可省略。 基本上有这些就够了，我的插件程序完整代码如下： &#60;?php /* Plugin Name: Excerpt Trimmer Plugin URI: [...]]]></description>
			<content:encoded><![CDATA[<p>虽然市面上有很多 WordPress 的插件，但未必能够满足我们那些稍微“变态”一些的需求，所以，参照wp官方的<a href="http://codex.wordpress.org/Writing_a_Plugin">插件编写指南</a>，写下这个简单的说明和小例子，希望对大家有所帮助。</p>

<p>插件编写需要php的基础知识，并且本说明的内容大多只适用于wp1.5以后版本。</p>

<p>插件程序建议放置在<code>wp-content/plugins/</code>目录，如果有多个程序文件的话，可以在这个目录下建立子目录，程序所需的资源文件建议也放在这个目录下，便于更新和管理。</p>

<p>wp插件主要分为两种：</p>

<ul>
<li>Action 对wp的行为事件进行修改，比如修改数据库，发送邮件或者修改显示结构等。</li>
<li>Filter 对wp的输入输出内容进行修改。</li>
</ul>

<p>在本例中，我们将建立一个简单的Filter插件，替换wp本来的文章摘要生成功能，把以前<a href="102">手工修改代码实现的内容</a>，用插件来替代。</p>

<p>wp推荐在插件程序中加入这样的文件头，便于wp辨认插件的信息，就像下面这样：</p>


<div class="wp_syntax"><div class="code"><pre class="php" style="font-family:monospace;"><span style="color: #000000; font-weight: bold;">&lt;?php</span>
<span style="color: #666666; font-style: italic;">/*
	Plugin Name: 插件名称
	Plugin URI: http://插件网址
	Description: 插件说明
	Version: 插件版本
	Author: 插件作者
	Author URI: http://作者网址
*/</span></pre></div></div>


<p>插件的代码风格主要是基于过程方式的，所以要尽力避免函数重名，可以在函数名之前加上自己的名称前缀，也可以把自己的类包含在一个类当中，不过小型插件还不至于复杂到要使用类的程度吧？</p>

<p>函数的编写就是php代码了，注意Filter一定要返回数据，即使没有修改也要返回，不然后续的其他函数就没法干活了。更多的信息可以看<a href="http://codex.wordpress.org/Plugin_API">官方的插件api</a>。</p>

<p>编写完成之后，函数是不会自动执行的，wp为插件函数的执行提供了一种hook机制，就是在原始的wp程序中，预置了一些hook或者叫挂载点，通过把插件函数挂在这些hook上，就可以被执行，目前wp提供的挂载点一览见<a href="http://codex.wordpress.org/Plugin_API/Hooks">Hooks</a>列表。</p>

<p>针对我编写的函数，是要挂在生成文章摘要的地方，查找了一下<a href="http://codex.wordpress.org/Plugin_API/Hooks">Hooks</a>列表，发现和摘要（excerpt）相关的有：</p>

<pre><code>default_excerpt 
excerpt_edit_pre
excerpt_save_pre 
get_the_excerpt
the_excerpt
</code></pre>

<p>从名称可以看出，default_excerpt是文章的默认摘要，excerpt_edit_pre和excerpt_save_pre是编辑文章是对摘要进行的预处理，get_the_excerpt是取摘要，the_excerpt是返回摘要。我编写的是生成摘要的功能，应该挂在get_the_excerpt或者是the_excerpt上，考虑the_excerpt可能更“底层”一些，挂在这里。</p>

<p>挂载的语法也是一个php函数调用，格式如下：</p>


<div class="wp_syntax"><div class="code"><pre class="php" style="font-family:monospace;">add_filter<span style="color: #009900;">&#40;</span><span style="color: #0000ff;">'hook_name'</span><span style="color: #339933;">,</span> <span style="color: #0000ff;">'your_filter'</span><span style="color: #339933;">,</span> <span style="color: #009900;">&#91;</span>priority<span style="color: #009900;">&#93;</span><span style="color: #339933;">,</span> <span style="color: #009900;">&#91;</span>accepted_args<span style="color: #009900;">&#93;</span><span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span></pre></div></div>


<p>priority为优先级，默认为10，数字小的会先执行；accepted_args为参数个数，默认值为1，如果插件函数需要多个参数的话需要自行设定。这两个参数在使用默认值的时候都可省略。</p>

<p>基本上有这些就够了，我的插件程序完整代码如下：</p>


<div class="wp_syntax"><div class="code"><pre class="php" style="font-family:monospace;"><span style="color: #000000; font-weight: bold;">&lt;?php</span>
<span style="color: #666666; font-style: italic;">/*
	Plugin Name: Excerpt Trimmer
	Plugin URI: http://www.fwolf.com/
	Description: 替换wp本身的“摘要”生成功能，用于在列表或者查询页面显示文章部分内容。
	Version: 1.0
	Author: Fwolf
	Author URI: http://www.fwolf.com/
*/</span>
&nbsp;
<span style="color: #000000; font-weight: bold;">function</span> fw_excerpt<span style="color: #009900;">&#40;</span><span style="color: #000088;">$text</span><span style="color: #009900;">&#41;</span>
<span style="color: #009900;">&#123;</span>
	<span style="color: #000088;">$excerpt_length</span> <span style="color: #339933;">=</span> <span style="color: #cc66cc;">4</span><span style="color: #339933;">;</span>
	<span style="color: #000088;">$words</span> <span style="color: #339933;">=</span> <span style="color: #990000;">explode</span><span style="color: #009900;">&#40;</span><span style="color: #0000ff;">&quot;<span style="color: #000099; font-weight: bold;">\n</span>&quot;</span><span style="color: #339933;">,</span> <span style="color: #000088;">$text</span><span style="color: #339933;">,</span> <span style="color: #000088;">$excerpt_length</span> <span style="color: #339933;">+</span> <span style="color: #cc66cc;">1</span><span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span>
	<span style="color: #b1b100;">if</span> <span style="color: #009900;">&#40;</span><span style="color: #990000;">count</span><span style="color: #009900;">&#40;</span><span style="color: #000088;">$words</span><span style="color: #009900;">&#41;</span> <span style="color: #339933;">&gt;</span> <span style="color: #000088;">$excerpt_length</span><span style="color: #009900;">&#41;</span> <span style="color: #009900;">&#123;</span>
		<span style="color: #990000;">array_pop</span><span style="color: #009900;">&#40;</span><span style="color: #000088;">$words</span><span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span>
		<span style="color: #990000;">array_push</span><span style="color: #009900;">&#40;</span><span style="color: #000088;">$words</span><span style="color: #339933;">,</span> <span style="color: #0000ff;">'&lt;p /&gt;......&lt;a href=&quot;'</span> <span style="color: #339933;">.</span> get_permalink<span style="color: #009900;">&#40;</span><span style="color: #000088;">$post</span><span style="color: #339933;">-&gt;</span><span style="color: #004000;">ID</span><span style="color: #009900;">&#41;</span> <span style="color: #339933;">.</span> <span style="color: #0000ff;">'&quot;&gt;[阅读全文]&lt;/a&gt;'</span><span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span>
		<span style="color: #000088;">$text</span> <span style="color: #339933;">=</span> <span style="color: #990000;">implode</span><span style="color: #009900;">&#40;</span><span style="color: #0000ff;">&quot;<span style="color: #000099; font-weight: bold;">\n</span>&quot;</span><span style="color: #339933;">,</span> <span style="color: #000088;">$words</span><span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span>
	<span style="color: #009900;">&#125;</span>
	<span style="color: #b1b100;">return</span> <span style="color: #000088;">$text</span><span style="color: #339933;">;</span>
<span style="color: #009900;">&#125;</span> <span style="color: #666666; font-style: italic;">// end of func fw_excerpt</span>
&nbsp;
add_filter<span style="color: #009900;">&#40;</span><span style="color: #0000ff;">'the_excerpt'</span><span style="color: #339933;">,</span> <span style="color: #0000ff;">'fw_excerpt'</span><span style="color: #009900;">&#41;</span><span style="color: #339933;">;</span>
<span style="color: #000000; font-weight: bold;">?&gt;</span></pre></div></div>


<p>可以看出插件程序也就是由上面介绍的3部分组成：文件头，函数体，挂载声明。</p>

<p>其实就简单插件来讲，最关键的是找对相应的hook，这就要靠经验和摸索了，wp官方文档好像也没有对每个hook作出详细的解释。</p>

<p>现在，把我这个插件程序保存为<code>wp-contents/plugin/excerpt.php</code>，登录wp，进入plugin设置，就能够看到“Excerpt Trimmer”插件了，启用之就大功告成了。我还存了一份在<a href="http://groups.google.com/group/Fwolf-Tools/t/c133d65b8c812271">这里</a>，方便下载。</p>

	Tags: <a href="http://www.fwolf.com/blog/post/tag/example" title="example" rel="tag">example</a>, <a href="http://www.fwolf.com/blog/post/tag/fliter" title="fliter" rel="tag">fliter</a>, <a href="http://www.fwolf.com/blog/post/tag/hooks" title="hooks" rel="tag">hooks</a>, <a href="http://www.fwolf.com/blog/post/tag/php" title="PHP" rel="tag">PHP</a>, <a href="http://www.fwolf.com/blog/post/tag/plugin" title="plugin" rel="tag">plugin</a>, <a href="http://www.fwolf.com/blog/post/tag/wordpress" title="WordPress" rel="tag">WordPress</a><br />

	<h4>Related posts</h4>
	<ul class="st-related-posts">
	<li><a href="http://www.fwolf.com/blog/post/102" title="更改 WordPress 自动生成摘要的方式 (2005-12-19)">更改 WordPress 自动生成摘要的方式</a> (18)</li>
	<li><a href="http://www.fwolf.com/blog/post/87" title="搞定WordPress的Permalinks (2005-11-07)">搞定WordPress的Permalinks</a> (6)</li>
	<li><a href="http://www.fwolf.com/blog/post/241" title="如何为网站添加sitemap文件 (2006-10-17)">如何为网站添加sitemap文件</a> (3)</li>
	<li><a href="http://www.fwolf.com/blog/post/402" title="升级WP旧模板，让它支持Widgets (2008-05-11)">升级WP旧模板，让它支持Widgets</a> (4)</li>
	<li><a href="http://www.fwolf.com/blog/post/366" title="WordPress升级到2.3.1 (2007-11-02)">WordPress升级到2.3.1</a> (0)</li>
</ul>

]]></content:encoded>
			<wfw:commentRss>http://www.fwolf.com/blog/post/296/feed</wfw:commentRss>
		<slash:comments>17</slash:comments>
		</item>
	</channel>
</rss>
