2008년 2월 5일 화요일

Buildbot과 Subversion으로 빌드와 테스트 자동화하기

글을 두 개로 나눠서 적는 것이 좋겠다. 일단 여기서는 Subversion과 Buildbot을 어떻게 설치하고 사용하는지 설명한다. 두번째 글에서는 Lisp 코드의 컴파일과 테스트를 ASDF와 Buildbot을 이용해서 어떻게 자동화하는지 설명하겠다.

I. 기본 개념

'단위 테스트, 버전 관리, 빌드와 테스트 자동화' 이 세 가지는 꼭 수행해야 하는 실행지침으로 꼽힌다. 이 글에서는 버전 관리를 위해서 Subversion을 사용하고, 빌드와 테스트 자동화를 위해서 Buildbot을 사용한다고 가정한다.

Subversion은 CVS와 같은 버전 관리 툴이고, CVS의 단점을 보완하기 위해서 개발되었다고 한다. 현재는 가장 널리 쓰이고 있지만, darcs라는 툴로 많은 사람이 넘어가고 있는 것 같기도 하다. (같은 일을 할 때 darcs가 Subversion보다 손이 덜 간다고 한다. darcs는 패치 등에 대한 이론적 기반을 가지고 만들어진 같고, 가장 radical한 언어 중 하나라고 평가되는 Haskell로 짜여졌다.)

Buildbot은 Python으로 작성된, 빌드와 테스트를 자동화하기 위한 도구이다. 버전 관리 도구의 저장소를 감시하고 있다가 저장소에 변화가 생기면 코드를 체크아웃해와서 빌드와 테스트를 수행하고, 그 결과를 웹이나 이메일로 보여준다. Buildbot의 구조적인 장점은 build master와 build slave가 분리되어 있어서, 서로 다른 플랫폼에서 수행되는 빌드들을 관리하기가 용이하다는 것이다. build master는 각각의 build slave들이 어떻게 build를 수행하고 그 결과를 보고해야 할지에 대한 모든 정보를 가지고 있다. 실제로 빌드를 수행하는 build slave들은 build master에 접속해서 build master가 지시하는대로 할 뿐이다. build master와 build slave는 서로 다른 머신에 위치해 있어도 상관이 없기 때문에, 예를 들어 서로 다른 빌드를 수행하는 머신 1,2,3을 머신 4에서 총괄하는 것이 가능하다.

II. 설치

환경이 리눅스라면,
sudo apt-get install subversion
sudo apt-get install buildbot

환경이 맥이라면,
MacPorts를 이용해서 설치하면 된다. MacPorts는 DarwinPorts의 새 이름으로 리눅스의 apt-get과 비슷하게 각종 소프트웨어의 설치를 쉽게 해 준다. MacPorts 홈페이지에서 dmg 파일을 받아 설치하면 되는데, 설치 후 port 명령이 매뉴얼대로 되지 않는다면, 경로 설정이 안 되어 있어서 그런 것이다. 홈디렉토리(~)에 .profile 파일을 만들어서 다음과 같이 경로를 추가한다.

export PATH=/opt/local/bin:/opt/local/sbin:$PATH

'port search 프로그램이름'(e.x. port search buildbot)과 같이 하면 port를 통해 설치할 수 있는 프로그램들을 찾아볼 수 있다. 원하는 프로그램 이름을 찾으면 'sudo port install buildbot'과 같이 인스톨하면 된다.

환경이 윈도우라면,
이 경우는 해보질 않아서 모르겠다. 설치는 각각의 홈페이지를 참조해서 어떻게든 알아서;

III. 사용법

Subversion

저장소를 로컬의 /Users/peter/svn-repos 에 만든다고 할 때 각종 명령의 예는 다음과 같다. 로컬이 아니고 웹인 경우에는 'file://'부분을 'http://'로 바꾸면 된다.

새로운 저장소 만들기
svnadmin create /Users/peter/svn-repos

저장소에 프로젝트 만들기(기존 소스 코드를 저장소 안의 한 디렉토리로 임포트하는 경우)
app> svn import -m "application initial import" file:///Users/peter/svn-repos/app
이 경우는 저장소 아래에 app 디렉토리를 만들고 현재 디렉토리 아래의 모든 디렉토리와 파일들을 집어넣은 것이다. 하지만 저장소의 디렉토리를 살펴봐도 app 디렉토리는 눈에 보이지 않는다. 저장소에 저장된 프로젝트의 디렉토리 위치를 보고 싶다면 어떻게 해야 할까?
1. 체크아웃 해 온 작업본 디렉토리에서 svn info 또는,
2. 아무 곳에서나 svnlook tree /Users/peter/svn-repos

저장소로부터 파일 체크아웃하기
svn co file:///Users/peter/svn-repos/app app

작업본 디렉토리의 상태 알아보기
svn status
파일 앞의 M 표시는 그 파일이 변경되었다는 것을, ? 표시는 그 파일이 버전 관리 하에 놓여있지 않다는 것을 나타낸다. ? 표시된 파일을 버전 관리 대상으로 추가하기 위해서는 svn add를 사용

작업본 디렉토리의 변경을 저장소에 저장
app> svn commit -m "message"

리비전 내역 전체를 보려면
app> svn log

작업본 디렉토리의 내용을 최근 저장소의 리비전으로 갱신하려면
svn update

작업본 디렉토리에서 수행한 변경을 체크아웃 해 온 시점으로 되돌리려면
svn revert .

리비전 10과 리비전 11 사이의 변경 내역을 보고 싶다면
app> svn diff -r 10:11

작업본 디렉토리의 리비전이 11인데 리비전 9로 되돌리고 싶다면
app> svn merge -r 11:9

기타 알아두면 좋은 것

바이너리 파일이나, 백업 파일들은 저장소에 저장될 필요가 없다. Subversion이 이런 파일들을 무시하게 만들기 위해서는 ~/.subversion/config 파일의 global-ignores 부분을 수정하면 된다. 예를 들어, *.fasl 파일들과 *~ 파일들을 무시하고 싶다면 다음과 같이 수정하면 된다.

global-ignores = *.fasl *~

그리고 에디터가 필요한 Subversion 명령을 사용하게 될 경우가 있다. 예를 들어, emacs를 사용하고 싶다고 하면, ~/.profile 에 다음과 같이 추가한다.

SVN_EDITOR=/usr/bin/emacs
export SVN_EDITOR


Buildbot

1. build master를 만들자
/Users/peter/buildbot/master/our-master 에 마스터를 만든다고 가정하자.

buildbot create-master /Users/peter/buildbot/master/our-master

2. build master를 설정한다.
our-master 디렉토리에 보면 master.cfg.sample 파일이 있을 것이다. 이 파일을 편집해서 master.cfg 파일로 저장한다. 예를 보는 게 쉬울 것이다.

# 슬레이브 이름과 패스워드를 설정
from buildbot.buildslave import BuildSlave
c['slaves'] = [BuildSlave("slave-name", "slave-passwd")]

# 슬레이브가 접속할 포트 설정
c['slavePortnum'] = 9989

# 저장소의 변화를 어떻게 감지할지 설정한다. Subversion이라면 SVNPoller를 사용
from buildbot.changes.svnpoller import SVNPoller
c['change_source'] = SVNPoller("file:///Users/peter/svn-repos/app")

# treeStableTimer와 빌더 이름을 설정해 주면 된다.
from buildbot.scheduler import Scheduler
c['schedulers'] = []
c['schedulers'].append(Scheduler(name="all", branch=None,
treeStableTimer=2*60,
builderNames=["our-builder"]))

# 빌드 절차를 팩토리로 만드는 부분. 체크아웃->빌드 및 테스트 순서로 이루어져야 할 것이다.
# svnurl 에 빌드를 위해 체크아웃 해 와야 할 저장소의 디렉토리를 넣는다.
# ShellCommand 로 쉘에서 수행할 명령을 지정할 수 있다.
# 이 경우에는 build-and-test.lisp 파일에 빌드와 테스트 방법을 넣어놓고 로드하게 한 것이다.
from buildbot.process import factory
from buildbot.steps import source, shell
f1 = factory.BuildFactory()
f1.addStep(source.SVN(svnurl="file:///Users/peter/svn-repos/app"))
f1.addStep(shell.ShellCommand(command=["sbcl", "--load", "build-and-test.lisp"]))

# 빌더 이름, 빌더가 사용할 슬레이브 이름과 팩토리 이름을 설정해준다.
# builddir은 슬레이브 디렉토리 아래의 어떤 디렉토리에서 빌드를 수행할지를 설정하는 것이다.
# 이 예에서는 슬레이브가 full이라는 디렉토리를 만들어 빌드를 수행할 것이다.
b1 = {'name': "our-builder",
'slavename': "slave-name",
'builddir': "full",
'factory': f1,
}
c['builders'] = [b1]

c['status'] = []

# 웹과 메일로 빌드의 상태를 볼 수 있다.
from buildbot.status import html
c['status'].append(html.WebStatus(http_port=8010))

# sendToInterestedUsers를 False로 설정하면 extraRecipients 로만 메일이 간다고 한다.
# 빌드가 실패할 때만 메일이 가도록 mode='failing'으로 설정할 수도 있다.
# 하지만 무슨 이유에선지 메일은 제대로 보내지질 않는 것 같다.
from buildbot.status import mail
c['status'].append(mail.MailNotifier(fromaddr="buildbot@localhost", extraRecipients=['peter@matrix'], sendToInterestedUsers=False)

c['buildbotURL'] = "http://localhost:8010/"

3. build slave를 만들자
/Users/peter/buildbot/slave/our-slave 에 슬레이브를 만들고, 위에서 한 설정을 사용한다고 가정하자.

buildbot create-slave /Users/peter/buildbot/slave/our-slave localhost:9989 slave-name slave-passwd

4. build master와 build slave를 실행시키자
buildbot start /Users/peter/buildbot/master/our-master
buildbot start /Users/peter/buildbot/slave/our-slave

build master가 master.cfg 파일을 다시 읽어들이게 하려면
buildbot reconfig /Users/peter/buildbot/master/our-master

master와 slave를 멈추려면
buildbot stop /Users/peter/buildbot/slave/our-slave
buildbot stop /Users/peter/buildbot/master/our-master

이제 localhost:8010 을 살펴보자. 모든 것이 제대로 되었다면 BuildSlave를 클릭했을 때 Slave가 연결된 상태라고 나올 것이다. 저장소에 커밋이 일어나면 treeStableTimer 만큼 기다렸다가 체크아웃과 빌드가 수행된다. 쉘에서 수행된 커맨드의 exit code가 0이 아닌 값이거나 아무 stdout이 없이 20분이 지나면 빌드는 실패한 것으로 간주된다. 그 외의 경우는 성공한 것으로 처리되고, 어떤 경우든 stdout이 로그로 남는다. 웹의 Waterfall Display에서 시간대별로 모든 체크아웃과 빌드의 로그를 볼 수 있다.

여기까지 Subversion과 Buildbot을 살펴보았다. 빌드 과정은 빌드해야 하는 프로그램의 언어가 무엇이냐에 따라서 달라질 수 있다. Lisp 프로그램을 빌드하고 테스트하는 것에 대해서는 다음 글에서 다루도록 하겠다.

댓글 1개:

Unknown :

안녕 나 동훈~ 오랜만이야~
http://build.chromium.org/buildbot/waterfall/waterfall
최근에 본 buildbot chrome waterfall 인데, 이것 저것 손대놓고 css 도 이쁘게 만들어 놓은듯. 회사에서는 css 가져다 쓰는데 이뻐서 만족 ^^;