Fedora 11을 Dell Studio 15에 설치를 했다. 다시 코드를 손보고, 실험을 할 필요가 있기에...
문제는 Fedora 11에서 Dell Studio 15의 Wireless Lan card를 잡아내지 못한다는 거다.
한동안 이 문제로 며칠을 끙끙거리다가 결국에 찾아낸 방법...

Dell Wireless 1510 Wireless-N WLAN Mini-Card라는 이름으로 윈도우에서 잡히는 이 디바이스는
실제는 Broadcom BCM4322라는 놈이고, 이것은 LKM(Loadable Kernel Module)로 잡혀서 부팅시 올라오지 않고 있다는 거다. (대충 보니 Dell과 HP의 일부 laptop이 이 chipset을 쓰는 듯 하다. Intel wireless lan은 초기 LKM이 있어 설치시 바로 잡힌다고..)
그래서, 한참을 낑낑거리고 여기저기 돌아다니며 얻어낸 Solution은:

Fedora 11:
http://mirrors.cat.pdx.edu/rpmfusion/nonfree/fedora/releases/11/Everything/i386/os/ 에서
다음의 rpm 패키지를 다운로드한다.
  1. kmod-wl-2.6.29.4-167.fc11.x86_64-5.10.79.10-2.fc11.9.x86_64.rpm
  2. kmod-wl-5.10.79.10-2.fc11.9.x86_64.rpm
  3. broadcom-wl-5.10.79.10-2.fc11.noarch.rpm
Fedora 10:
http://mirrors.cat.pdx.edu/rpmfusion/nonfree/fedora/releases/10/Everything/i386/os/ 에서
다음의 rpm 패키지를 다운로드한다.
  1. kmod-wl-2.6.27.5-117.fc10.i686-5.10.27.6-5.fc10.5.i686.rpm
  2. kmod-wl-5.10.27.6-5.fc10.5.i686.rpm
  3. broadcom-wl-5.10.27.6-3.fc10.noarch.rpm

다음으로, rpm -ivh *rpm으로 rpm 패키지 설치.
그 다음  modprobe wl 명령을 수행하거나 재부팅 시키면, 신기하게도 그리고 안잡혔던 wireless Lan이 잡힌다!!

'코딩 관련 > 개발 도구/유틸리티' 카테고리의 다른 글

CVS를 이용한 프로젝트 관리  (2) 2007.12.02
Posted by Bart
코딩 관련/팁2008. 12. 7. 08:21

Fedora project에서 Fedora 10을 공개하였다. 언제부터인지 LiveCD라는 개념을 Fedora에서도  도입을 해왔더라. LiveCD는 OS를 CD로부터 boot 시킴으로써 HDD에 OS 코드들을 저장하지 않아도, OS를 이용할 수 있도록 해준다. 이게 왜 편리하냐면, 굳이 HDD를 파티션 분할하고 Linux를 설치해 보지 않아도, 해당 리눅스의 기능들을 바로 테스트 해볼 수 있게 해주기 때문이다. 그리고 사용해보다가 맘에 들면, 그때 추가 패키지들을 선택해서 리눅스 코어 패키지들과 함께 HDD에 설치하면 된다.
 
나는 Windows Vista를 laptop에 설치해서 이용해 오고 있었는데, 알고리즘 구현 때문에 Linux 환경에서 C++로 개발을 해야만 했다. 데스크탑에 이미 Fedora9이 설치되어 있었으나,  데스크탑의 성능이 좋지 않은 관계로(Pentium IV 1.7GHz, 512MB 메모리 ㅠㅠ)  결국엔 laptop에서 듀얼 부트를 하는 편이 낫겠다 싶었다.
 
Windows Vista에서는 OS 운용 중에 HDD의 볼륨을 조정해 주는 기능이 있다. 제어판->관리도구-> 컴퓨터관리-> 디스크 관리를 선택해서,  볼륨 축소, 볼륨 확장을 할 수 있다. 축소/확장할 볼륨 크기는 시스템이 계산해서 변경 가능한 최대값을 알려주는데, 이 안의 값을 선택하면 된다. 그리고 나서 축소된 만큼의 볼륨을 새 파티션으로 만들고 Fedora를 설치하면 된다. OS 설치 시점에서부터 듀얼 부트를 고려하고 있다면, 초반부터 파티션을 나누어 놓고 설치를 해도 무방하겠다.
 반대로, Linux를 먼저 설치하고, Windows Vista를 설치하는 경우에는 Windows Vista가 부트로더 영역을 덮어 씌우는 관계로 듀얼 부트가 되지 못한다. 따라서, Windows Vista와 Fedora를 듀얼부트 하게 하려면, Windows Vista를 먼저 설치한 후, Fedora를 나중에 설치하여야 한다.
 

'코딩 관련 > ' 카테고리의 다른 글

Google Reader& DB 관련 블로그 주소들  (6) 2008.09.24
[Tip] ^M 없애기  (0) 2007.11.30
Posted by Bart
코딩 관련/팁2008. 9. 24. 22:25
아는 사람, 후배들의 블로그를 돌아다니면서 글들을 보려고 하면 일단 골치 아픈 것이 주소를 일일이 타자를 치던가 아니면 즐겨찾기로 추가를 해서 일일이 방문을 해야 하는 것이었다. 그리고 나서 어디서부터가 새글인지를 확인하고 역순으로 읽어내려간다.  그러다 보니, 이 작업이 내가 방문할 블로그가 많아지면 많아질 수록 아주 번거로운 일이 되어버린다는 것을 인식했다. 뭔가 좋은 방법이 없을까 싶던 중.. 저멀리 기억 저편으로 날려버렸던 RSS.. RSS 리더를 쓰면 이 문제가 간단히 해결되겠다 싶었다. 그리고, 이게 웹 상으로 서비스되는 것이 있다면, 여기저기 PC를 옮겨다니면서도 확인이 쉽게 되겠구나 싶었다.

아니나 다를까 Google에서 이미 Google Reader라는 이름으로 오랫동안 서비스를 해온 듯 싶다. (http://reader.google.com)
여기에 내가 방문할 블로그의 주소나 또는 블로그 제목을 검색해서 가입해 두면, 새 글이 올라왔는지 금방 확인이 가능하다.  내 블로그 주소로 검색을 해보니 이미 구독자가 3명이다. 나를 아는 사람 중에 3명이 이미 내 블로그를 Google reader로 읽고 있다는 얘기인데... 이거 독자층으로 봐야 하는건가... 누굴까..어여 자수하시라.

암튼 내친 김에 DB 관련한 블로그(또는 RSS 피드)들을 조사해 보았다.

  1. Stanford InfoLab Blog(http://infoblog.stanford.edu) : Hector Garcia-Molina, Jeffrey Ullman, Jeniffer Widom 교수가 Advisor로 있는 스탠포드 인포랩의 블로그. 이 랩에서 수행하는 과제들은 그 주제의 무게가 굵직하고 몇년 후에 유행할 주제들을 미리 예측하는 것으로 유명하다.
  2. ACM SIGMOD Online(http://www.sigmod.org/rss/index.rss ) ACM SIGMOD의 news RSS fee.
    DB 하는 사람들이라면 다들 자기 논문을 올려두고 싶어하는 #1 Conference& Journal
  3. Alon Halevy's Blog(http://alonhalevy.blogspot.com/) Alon Halevy 와싱톤대 교수의 블로그. 지금은 Google에서 Dataspace 관련 연구를 진행중...
  4. DBMS2(http://www.dbms2.com) DBMS2 is a blog for people who care about database and analytic technologies. Its author is Curt Monash, who has been following the industry since 1981. DBMS2 also features commentary from a broad range of industry participants, including senior executives of MySQL, EnterpriseDB, DATAllegro, Vertica, Infobright, StreamBase, and Progress Real-Time.
  5. IEEE TKDE (http://csdl.computer.org/rss/tkde.xml) IEEE Transactions on Knowledge and Data Engineering 저널의 RSS 피드 
  6. The database column (http://www.databasecolumn.com/) Michael Stonebreaker, David DeWitt 교수를 포함한 7명의 저자들이 공동으로 운영하는 DB 관련 블로그.

    Along Halevy' 교수의 아버지는 박사 학위를 2년도 안되서 땄다고... 어떻게 그리도 빨리 땄는지는 http://alonhalevy.blogspot.com/2007/12/my-dad-is-80.html에 나와 있다.

    역시 세상에 노력없이 그냥 똑똑한 사람은 없다.


'코딩 관련 > ' 카테고리의 다른 글

Windows Vista와 Fedora 10 리눅스의 듀얼 부트  (2) 2008.12.07
[Tip] ^M 없애기  (0) 2007.11.30
Posted by Bart
코딩 관련/Example codes2007. 12. 11. 21:29

보통 XML 문서에서는 DTD나 XSD를 참조할 수 있도록 해주는데,
그것이 없는 일반 문서를 DTD나 XSD 선언을 주고 이를 기준으로 유효성 검사를 수행하도록 하기 위한 방법을 찾아보려고 작성했던 sample code이다.
ps. 제발 매뉴얼 좀 잘 읽어보고 짜자고... ㅡㅡ^

/*
 * DTD 또는 스키마 참조가 없는 XML 문서를 DTD나 스키마를 지정하여
 * 유효성 검사를 수행할 수 있는 방법을 찾기 위해 작성된 예제 코드
 *        Dec 11, 2007
 *        from Bart
 */

#include <iostream>
#include <xercesc/sax2/XMLReaderFactory.hpp>
#include <xercesc/sax2/SAX2XMLReader.hpp>
#include <xercesc/sax2/DefaultHandler.hpp>
#include <xercesc/util/XMLUni.hpp>
#include <xercesc/util/XMLString.hpp>
#include <xercesc/validators/common/Grammar.hpp>
#include <iostream>

XERCES_CPP_NAMESPACE_USE

int main(int argc, char* args[]){
 std::cout << "Starting to validate an XML file with a DTD or an XMLSchema definition "<<std::endl;
 char* xmlFile = "personal.xml"; //XML 파일 이름
 char* dtdFile = "personal.dtd"; //DTD 파일 이름
 char* xsdFile = "personal.xsd"; //xsd 파일 이름

 XMLPlatformUtils::Initialize();

 SAX2XMLReader* parser = XMLReaderFactory::createXMLReader();
 parser->setFeature(XMLUni::fgSAX2CoreValidation, true); //validation 수행 여부 true이면 XML 문서는 반드시 문법을 설정해 주어야 한다.
 parser->setFeature(XMLUni::fgXercesDynamic, false); //문법이 기술된 경우에만 validation을 수행한다.
 parser->setFeature(XMLUni::fgSAX2CoreNameSpaces, true); //NS의 처리
 parser->setFeature(XMLUni::fgXercesSchema, true); //XMLSchema의 지원 여부 
 parser->setFeature(XMLUni::fgXercesSchemaFullChecking, true); //XMLSchema 선언 자체가 valid 한지 검사


 DefaultHandler* dh = new DefaultHandler();
 parser->setContentHandler(dh);
 parser->setErrorHandler(dh);


 /*
  * 아래의 property 설정은 XML 문서가 NS 선언과 함께 XSD 스키마 참조를 하는 경우를 함수로 설정하는 것이다.
  * e.g. XML 문서의 루트에서
  * <root xmlns:xsi="http:/www.w3.org/2001/XMLSchema-instance"
  *       xsi:schemaLocation="http://www.example.com
  *              personal.xsd">
  *  이 경우  xsi ns는 해당 XML 문서가 XS의 인스턴스임을 나타낼 뿐이며,
  *  실제 이 XML 문서의 NS는 schemaLocation 애트리뷰트 값 중 공백 문자로 구분되는 첫 URI http://www.example.com이 된다.
  *  그리고 해당 NS에 대한 스키마 정의는 공백 문자 다음의 personal.xsd이다.
  * 
  *  아래의 함수로 선언시에는  
  */
 //char* schLoc="www.example.com personal.xsd";
 //parser->setProperty(XMLUni::fgXercesSchemaExternalSchemaLocation, schLoc);

 /*
  * 아래의 함수는 schema를 참조하는 XML 문서가
  *  <root noNamespaceSchemalocation="personal.xsd">
  *  로 NS 없이 스키마 참조를 하는 것을 함수로 설정하는 것이다.
  *  더불어 DTD 에 대한 참조도 이와 동일하게 한다. (xsdFile 또는 dtdFile)
  */
 parser->setProperty(XMLUni::fgXercesSchemaExternalNoNameSpaceSchemaLocation, xsdFile);

 // 아래의 함수들은 스키마 참조 설정과 아무런 관계가 없다.
 // 하나의 스키마 정의를 가지고 이를 따르는 많은 XML 문서들을 반복적으로 validation 수행 시
 // 속도 향상을 위해 DTD/XSD를 캐싱하기 위한 함수일 뿐이다.

 /* parser->loadGrammar(dtdFile, Grammar::DTDGrammarType, true);
 parser->setFeature(XMLUni::fgXercesUseCachedGrammarInParse, true); 
 */
 try{
  parser->parse(xmlFile);
 }
 catch(const XMLException& toCatch){
  char* message= XMLString::transcode(toCatch.getMessage());
  std::cout << "Exception message is : "<<std::endl
     << message <<std::endl;
  XMLString::release(&message);
  return -1;
 }catch(const SAXParseException& toCatch){
  char* message =XMLString::transcode(toCatch.getMessage());
  std::cout << "Exception messsage is : " <<std::endl
   << message <<std::endl;
  XMLString::release(&message);
  return -1;
 }
 
 std::cout << "reading the XML file was finished"<<std::endl;
 delete parser;
 delete dh;
 return 0;

}

'코딩 관련 > Example codes' 카테고리의 다른 글

간단한 웹 페이지 복사 프로그램.  (0) 2007.11.14
Posted by Bart
* 이 글의 원 저자는 윤상배님(http://www.joinc.co.kr/modules/moniwiki/wiki.php/article/cvs%B8%A6_%C0%CC%BF%EB%C7%D1_%C7%C1%B7%CE%C1%A7%C6%AE%B0%FC%B8%AE#s-1) 이며, 정리를 위해 여기에 몇가지 내용을 추가하였음을 밝혀둡니다.


Contents

1 FAQ
1.1 snapshot
2 본문
3 관련 게시물

1 FAQ

cvs 사용중 궁금한 내용을 올려주세요. 어느정도 정리가 되면 본문에 정식 포함시킬 생각입니다.

1.1 snapshot

어느 한 시점에서 프로젝트 파일들에 동일한 꼬리표(TAG)를 붙여서, 프로젝트가 계속 진행되더라도 필요할 때 snapshot 시점의 프로젝트를 가져올 수 있다.

테스트 최종 완료시점, 1차 성능 완료 시점등의 프로젝트를 따로 유지해서 나중에 다시 가져오고 싶을 경우, CVS를 이용하지 않는다면 tar등으로 프로젝트 전체를 압축해서 보관해야 할 것이다. cvs는 명령어 한줄로 간단하게 이러한 문제를 해결할 수 있다.
# cvs tag TEST_REL_1 
 
이제 이 프로젝트는 TEST_REL_1 이라는 꼬리표(tag)가 붙게 된다. 프로젝트 진행중 TEST_REL_1 꼬리표가 붙은 프로젝트를 가져오길 원한다면 아래와 같이 하면 된다.
# cvs co -rTEST_REL_1 myprogdir 
 
이제 myprogdir에 TEST_REL_1의 snapshot 프로젝트가 생성이 되었다.


2 본문

cvs 를 이용한 프로젝트 관리

윤 상배

dreamyun@yahoo.co.kr

고친 과정
고침 1.0 2005년 6월 9일 13시
바이너리 파일 추가 방법, 디렉토리 upload 방법, 오타 수정
고침 0.9 2003년 10월 20일 21시
익명 CVS 설정 추가, diff관련 옵션 설명 추가
고침 0.8 2003년 8월 11일 23시
최초 문서 작성

1. CVS에 대한 소개

과거는 컴퓨터 시스템의 능력이 매우 제한적이였고 사용하는 유저 역시 제한적이거나 지극히 개인적인 용도로 사용하는 경우가 대부분 이였다. 때문에 소프트웨어역시 비교적 단순했으며 단지 한두명의 개발자 투입만으로도 꽤나 쓸만한 애플리케이션이 만들어지기도 했다.

도스 시절부터 컴퓨터를 다루어왔던 유저라면 "누구누구의 어떤 프로그램" 이라는 얘기를 많이 들어 보았을 것이다. 지금은 사정이 달라져서 아주 간단한 프로그램이 아니고서는 혼자 개발해서 그럭저럭 인지도 있는 프로그램을 만든다는것 자체가 매우 힘들어졌다.

요즘은 UI만 제작하는 것도 하나의 작업으로 분류된다. 인터넷이 일반적으로 보급되면서 대부분의 애플리케이션은 C/S환경하에서 작동하게 된다. 개발자는 데이터전송과 처리에 대한 부분까지 신경써야 하며 많은 경우 서버와 클라이언트가 서로 다른 운영체제하에 놓이며, 특히 서버 프로그램의 경우 여러가지의 전혀 달라보이는 운영체제를 지원해야하는 경우도 있다.

이런 이유로 왠만한 규모의 프로젝트라 할지라도 혼자서 개발을 진행한다는건 매우 힘들며, 대부분 팀단위로 프로젝트를 진행을 하게 된다. 이 팀이란건 또 어떤가 같은 사무실에서 같은 시간에 존재하면서 서로 의견교환을 통해서 프로젝트를 진행 시킬 수도 있지만 인터넷이라는 매체를 통해서 전혀 다른 공간에서 전혀 다른 시간대에 프로젝트를 진행 시켜야 하는 경우도 생긴다. 전 세계인이 참여하는 많은 오픈 프로젝트가 그러하다.

이런 경우 하나의 소스코드를 한명 이상이 접근해서 수정할 수 있는데, 프로젝트를 진행하다 보면 소스코드가 엉뚱하게 꼬일 수 있을 것이다. 같은 사무실에서 단지 몇명의 프로그래머가 작업을 한다면, 서로 의견조율을 하거나 처음부터 각각의 모듈만 담당하게 만들어서 어느정도 문제를 해결할 수 있을 거라고 생각할 수도 있을 것이다. 그러나 막상 프로젝트를 진행해보면 이게 결코 말처럼 쉬운일이 아니란걸 알게 될것이다. 프로젝트를 진행하다 보면 이쪽 시스템에서 테스트하고, 저쪽 시스템에서 테스트하고 문제가 생기면 즉각 수정을 하게 되는데, 이러다보면 소스코드가 여기저기 위치하게 되고 결국 어느 소스가 최근 소스코드인지 헷갈리는 사태가 발생하게 된다. 하물며 오픈소스와 같이 수많은 프로그래머가 느슨하게 묶여있는 경우는 더 말할 필요도 없다. 적당한 버젼관리 도구의 사용없이는 프로젝트의 진행자체가 불가능해질 것이다.

이러한 문제의 해결을 위해서 여러도구가 개발되었는데, 그중 하나가 CVS로 현재 가장 널리 사용되고 있는 버젼관리 도구이다.


2. CVS에 대한 기본지식

2.1. CVS 란?

CVS는 Concurrent Version System 의 줄임말로써 직역 하자면 공동 버젼 시스템, 의역하자면 "공동으로 진행하는 프로젝트의 버젼 관리 시스템" 정도가 될것이다.


2.2. 어떨때 CVS가 필요하죠?

CVS는 사용하기에 따라서 여러가지 용도로 사용할수 있다. 이번장에서는 CVS를 이용할수 있는 다양한 상황들에 대해서 알아보도록 하겠다.


2.2.1. 공동 프로젝트 관리

회사혹은 학교에서 프로젝트를 진행하다보면, 여러명이서 하나의 프로젝트를 진행하는 경우가 발생할것이다.

이럴경우 보통 모듈별로 개발을 하게 되겠지만, 또한 모듈은 전체 프로젝트에 영향을 미치게 되므로, 자신의 모듈버젼과 전체 프로젝트의 버젼을 컨트롤할수 있어야 한다. 그리고 코드가 충동하게 될경우(서로 같은 부분을 수정함으로써)의 문제를 해결할수 있어야 한다.

이러한 작업은 프로젝트 규모가 작고 개발 참여자 수가 적고, 개발 참여자가 가까운 지역(사무실 같은)에 모두 모여있다면, 한명의 버젼관리자(보통은 팀장)를 두고 그럭저럭 관리가 가능할것이다.

그러나 조금만 프로젝트가 커지고, 개발참여자 수가 많아지고 개발자가 지역적으로 떨어져 있는 상황에서는 거의 불가능 하다는 걸 알수 있게 될것이다. 특히 인터넷을 통해서 느슨하게 연결된 오픈 프로젝트의 경우 도구를 사용하지 않는 다면 거의 관리가 불가능 할 것이다.

CVS 를 사용하면 이러한 대규모의 프로젝트에서 각 모듈 개발자가 자신의 버젼을 유지하면서 전체 프로젝트에 참여할수 있도록 할수 있다.

실제로 KDE, GNOME, APACHE 서버 등 다양한 프로젝트가 CVS 를 이용해서 프로젝트를 관리하고 있다. 이러한 프로젝트는 규모가 작아도 수십명, 혹은 수백명이 프로젝트를 진행하게 되는데(게다가 지역적으로 멀리 떨어져 있다), CVS가 중간에서 프로젝트가 산으로 가지 않도록 중계해준다. 오픈 프로젝트를 하는데 있어서 CVS는 거의 표준적으로 사용되는 버젼관리 도구이다.


2.2.2. 프로젝트 백업

CVS 를 사용할경우 자동적으로 프로젝트 백업의 문제까지 해결이 가능하다. CVS 는 중간에 CVS 서버가 있어서, 프로젝트 데이터의 저장소 역할을하며 모든 개발자는 CVS 서버에서 최신의 프로젝트를 다운로드 받아서, 자신의 컴퓨터에서 테스트하고 코딩해서, 이걸 다시 CVS 서버에 업데이트 시키는 방식을 사용하게 된다.

그러므로 실수로 자신의 프로젝트 데이터가 날아간다고 해도 전혀 염려할 필요가 없다. 그냥 서버에서 다시 다운 받기만 하면 된다. 최악의 경우 CVS 서버가 날라갔다고 하더라도 가장 최근의 쏘쓰를 가진 개발자가 있을 것이므로 쉽게 복구 가능하다.

또한 CVS는 최신 버젼의 소스코드 뿐만 아니라 과거 버젼의 소스코드에 대한 정보를 가지고 있어서 최근의 몇개 버젼에 문제가 생겼다고 하더라도 쉽게 그이전의 소스코드를 얻어올 수 있다.


2.2.3. 데이터 동기화

요즘은 회사와 집과의 경계가 많이 허물어 졌다. (좋은 현상인지 나쁜 현상인지는 좀 생각해 봐야겠지만) 그러다 보니 회사에서 하는일을 가정에서 하기도 하고, 가정에서 했던 일을 회사로 가져가기도 한다. 그럴경우 회사의 컴터와 가정의 컴터에 있는 데이타의 동기화가 필수적이다. 이런 데이타 동기화를 위해서 "노트북", "PDA" 같은걸 사용할수 있겠지만, 이건 너무 비싸다. ftp 도 사용할수 있겠지만, 이거 잘못 사용하면 데이터가 꼬일수 있다. 또한 상당히 불편하다.

이럴때 CVS 를 사용하면 대단히 편하게 작업이 가능하다. 회사에서 작업을 마치고 CVS 서버에 등록하고, 가정으로 돌아가서 CVS 서버에 등록된 최신의 작업을 받아와서 작업을 하고 다시 CVS 서버에 등록만 하면 되기 때문이다. (물론 이왕이면 가정에까지 회사일을 가지고 가지 않으면 좋겠지만..)

또한 덤으로 자신의 중요한 자료까지 자동으로 백업된다.

필자 역시 이러한 방법으로 작업을 한다. 작업거리가 좀 남았는데, 회사에서는 일이 잘안되고(실은 일하기 싫어서겠지만 --;), 그냥 집에가서 느긋하게 TV도 보고, 웃통 벗어던지고 일하고 싶을때 매우 편하게 CVS를 이용할수 있다.


3. CVS 사용하기

3.1. CVS 서버 설치하기

지금 까지 CVS에 대한 개론적인 설명을 알아 보았다. 이제 본격적으로 CVS를 설치하고 운용하고 사용하는 방법을 알아보도록 하겠다. CVS 서버 설치는 redhat 8.x 리눅스를 기준으로 하겠다. 대부분의 redhat 리눅스 배포판은 cvs 를 기본적으로 포함하고 있다. rpm 패키지 관리자를 통해서 설치하자.

먼저 cvs로 접속할 포트가 사용가능한지를 확인한다.
# cat /etc/services |grep 2401
cvspserver 2401/tcp # CVS client/server operations
cvspserver 2401/udp # CVS client/server operations

이제 Internet services daemon 에 cvs 서버를 등록 시켜주기만 하면 된다. 레드헷 7.x 버젼부터는 inetd 대신에 xinetd 가 Internet services daemon 으로 사용되어 지고 있다. /etc/xinetd.d 디렉토리 밑에 cvspserver 이란 이름으로 서비스 설정파일을 만들도록 하자. 내용은 다음과 같다.

service cvspserver
{
    disable         = no
    flags           = REUSE
    socket_type     = stream
    wait            = no
    user            = root
    server          = /usr/bin/cvs
    server_args     = --allow-root=/home/cvs pserver
    log_on_failure  += USERID
}
			
만약 inetd를 사용하는 예전 버젼의 리눅스라면 아래와같이 설정하도록 한다.
# echo "2401 stream tcp nowait root /usr/bin/cvs cvs -f \
  --allow-root=/cvsroot pserver" >> /etc/inetd.conf
			

위의 설정값들중 --allow-root를 주목하기 바란다. 프로젝트가 서버에 저장된다면 당연히 이들 프로젝트가 서버의 어디 디렉토리로 저장되어야 하는지 지정할 수 있어야 할것이다. cvs에서는 이를 repository(이하 저장소)라고 한다. --allow-root는 cvs 저장소가 /home/cvs임을 명시해 주기 위해서 사용된다. 만약 새로운 프로젝트인 hello_world를 만들었다면 이 프로젝트는 /home/cvs/hello_world 디렉토리에 저장이 된다.

이제 xinetd 데몬(혹은 inetd데몬)을 다시 실행(/etc/rc.d/init.d/xinetd restart) 시키면 cvs 서버가 작동하게 될것이다. cvs 서비스는 2401 포트를 이용해서 서비스 된다(/etc/services 참조). 제대로 cvs 서비스가 되고 있는지 알아보기 위해서 포트 스캐닝 도구인 nmap 을 사용해서 확인해 보도록 하자.

[root@cvs xinetd.d]# nmap 192.168.0.4

Starting nmap V. 2.54BETA7 ( www.insecure.org/nmap/ )
Interesting ports on localhost.localdomain (127.0.0.1):
(The 1527 ports scanned but not shown below are in state: closed)
Port       State       Service
22/tcp     open        ssh                     
25/tcp     open        smtp                    
80/tcp     open        http                    
111/tcp    open        sunrpc                  
2401/tcp   open        cvspserver              
3306/tcp   open        mysql                   
		
2401 번 포트로 cvspserver 가 서비스 되고 있음을 알수 있다. 이로써 cvs 서버의 설치및 가동을 마쳤다.

다른 배포판을 사용하더라도 설치상에 있어서 문제점은 없을것이다. 데비안 이라면 전용 패키지관리자를 이용해서 설치하면 된다. 패키지 설치가 여의치 않다면 쏘쓰를 직접 컴파일 해서 설치 하면 된다.

cvs 는 cvspserver 를 이용한 서비스 외에도 rsh, ssh 를 이용한 서비스도 가능하다. 이에 대한 내용은 CVS 사용 문서를 참고 하기 바란다. 이문서에서는 가장 널리 사용되는 cvspserver 방식에 대해서만 설명하고 있다.


3.1.1. CVS저장소 만들기

위의 설정에서 우리는 /home/cvs를 프로젝트들을 위한 저장소로 사용한다고 했는데, 저장소로 사용하기 전에 저장소 터를 다지기 위한 사전작업이 필요하다.

이러한 작업은 cvs에서 제공하는 init옵션을 통해서 가능하다.

# cvs -d /home/cvs init
			
-d를 이용해서 저장소로 사용될 디렉토리를 지정하고 init를 명시하는 정도로 어렵잖게 저장소를 생성할 수 있다.


3.1.2. CVS 유저 환경 설정

CVS 에 서버를 만들어 놓았으면 이제 CVS 자원을 사용하도록 환경설정을 해주어야 한다. 가장 중요한 건 공동으로 작업할 프로젝트 파일들이 저장될 CVS 저장 디렉토리(저장소)를 설정하는 일이다.

일단 우리는 위의 /etc/xinetd.d/cvspserver 를 설정하면서 cvs 데몬이 뜨게될경우 --allow-root 옵션을 이용해서 /home/cvs 를 홈디렉토리(프로젝트가 저장될 디렉토리)를 지정했다. 이제 /home/cvs 를 실제 프로젝트 사용자들이 사용할수 있도록 권한 설정을 해주어야 한다.

cvs 의 권한 설정을 위해서 cvs 란 그룹을 만들도록 하고 /home/cvs 디렉토리에 cvs 그룹에 대해서 읽기/쓰기/실행 권한을 부여하도록 하자.

# groupadd cvs
# mkdir cvs
# chmod 770 cvs 
			
이제 cvs 그룹에 포함된 모든 사용자는 CVS 자원을 이용할수 있는 권한을 가지게 되며, 이후로는 프로젝트를 등록시켜서 공동작업에 CVS 를 이용하기만 하면 된다.


3.1.3. CVS 사용하기

이제 CVS 서버의 설정이 끝났음으로, 클라이언트의 입장에서 어떻게 프로젝트를 등록하고 공동으로 작업을 진행시킬수 있는지에 대해서 알아보도록 하겠다. 이러한 작업들은 클라이언트에게 제공되는 "cvs" 라는 프로그램을 통해서 이루어진다.

CVS 의 사용방법은 다음과 같은 환경하에서 테스트되었다.

              +------------+
              | @cvs       | project : hello_world
              | CVS SERVER |
              +------------+
                     |
                     |
        +------------+------------+ 
        |                         | 
   +---------+               +---------+
   | @myhome |               | @one    |
   +---------+               +---------+
@cvs 는 CVS 저장소를 가지고 있는 서버이며, @myhome 은 "팀원" @one 는 "팀장" 의 개발호스트(컴퓨터) 이다. 공동으로 진행될 프로젝트는 hello_world 이다.


3.1.3.1. 익명 CVS설정하기

익명 CVS란 말그대로 일반 사용자에게 CVS를 읽고/쓸수 있는 권한을 부여하는 것이며, 많은 오픈 프로젝트들이 익명 CVS를 허용해서 가능한한 많은 개발자가 참여할 수 있도록 길을 열어 놓고 있다.

이러한 익명 CVS사용자는 보통 프로젝트에 대한 읽기권한만을 부여한다. 익명 사용자에게 쓰기권한을 주면 프로젝트의 진행이 너무 산만해 질 수 있기 때문이다. 익명 사용자(혹은 개발자)는 프로젝트에 반영해야 될 내용이 있을 때 메일등을 통해서 프로젝트 메인 개발자에게 통보하는게 보통이다.

익명 CVS를 허용하기 위해서는 우선 시스템에 anonymous계정이 만들어져 있어야 한다. 쉘을 가지지 못하도록 설정한다.

# useradd anonymous -s /bin/false 
				
그리고 /cvsroot/CVSROOT/passwd 파일에 위의 계정을 등록하면 된다.
# echo anonymous: > /cvsroot/CVSROOT/passwd
				
정확하게는 [유저아이디]:[패스워드]의 형태가 되어야겠지만 익명 CVS의 경우 패스워드를 설정하지 않는게 일반적이므로 패스워드는 생략하도록 한다. 패스워드를 부여하고 싶다면 crypt된 문자열값을 사용하도록 한다.

익명 CVS사용자의 경우 아래와 같이 설정해서 읽기만 가능하도록 권한을 제한시킨다.

# echo anonymous > /cvsroot/CVSROOT/readers
				

3.1.3.2. 사용자별 암호 설정하기

이제 접속 포트는 열어두었으니, 개발자들에게 CVS 계정을 발급하는 일만 남았다.
암호 인증 방식을 이용하는 경우, 계정과 암호는 저장소의 CVSROOT 디렉토리 밑에 passwd란 이름의 파일에 저장된다.
여기에서는 /home/cvs/CVSROOT/passwd가 될 것이다.
하지만 이 파일은 처음에는 존재하지 않는다.
그러므로 직접 만들어주어야 한다
# cat /home/cvs/CVSROOT/passwd
chaeya:jAb80dIzmjBDU:cvs

각 줄은 한 사용자에 대한 정보를 담고 있다.
줄은 ':'을 경계로 다시 세 부분으로 나뉘는데
첫 부분이 사용자의 CVS 계정 이름(시스템 계정과는 무관하다), 그 다음은 암호, 그리고 마지막은 시스템 계정 이름이다.
즉, 이 파일에는 현재 chaeya 사용자가 등록되어 있고, 이들이 CVS 이용시에는 cvs란 씨스템 계정의 권한을 갖는 것이다. 암호부분은 유닉스 씨스템에서 전통적으로 사용되는 crypt 함수를 이용하여 변환된 값이 저장되어 있다.
새로운 사용자를 추가하기위한 스크립트를 하나 여기서 준비하자
#!/usr/bin/perl
#
# Simple script to take a username and password and
# return a line suitable for pasting into the CVS
# password file
#
($u, $p)=@ARGV;
@d=(A..Z,a..z);
$s=$d[rand(52)].$d[rand52];
print $u.":".crypt($p, $s).":cvs\n"

위의 내용으로 /usr/local/bin/cvspasswdgen 이라는 화일을 작성하고 실행권한을 준다
그리고 cvspasswdgen 명령을 이용해서 cvs유저계정을 생성한다
# chmod +x cvspasswdgen
# cd /home/cvs/CVSROOT
# cvspasswdgen id password >> passwd

여기사용한 perl 스크립트외에 다른방법으로 암호를 생성하는데는 htpasswd 를 이용할수있다.
아파치 패키지의 htpasswd 를 이용해서 암호를 생성한다면 아래처럼 할수있다
[chaeya@tsc14 ~/cvswork/demoproj]
$ /usr/local/apache/bin/htpasswd -n chaeya
New password:
Re-type new password:
chaeya:GsWCGVul8OlNM
-n 옵션은 화면에 출력만 하는것이다.
화면에 나온 암호를 CVSROOT안의 passwd 화일에 사용하면 된다


3.1.4. CVS 사용자 환경설정

cvs 서버에 프로젝트를 등록시키고, 프로젝트를 업데이트하고, 받아오기 위해서 우리는 "cvs" 라는 전용 클라이언트를 사용하게 된다. "cvs" 다음에 여러가지 명령행 옵션을 이용함으로써, 원하는 작업을 하게 된다.

"cvs" 프로그램을 사용하기 위해서 우리는 CVS 서버에 접근해서 지정된 디렉토리(프로젝트가 저장되는 디렉토리 다른말로 "저장소") 가 어디인지 cvs 프로그램에 알려주어야만 한다. 보통 환경변수인 CVSROOT 를 통해서 "cvs" 에게 CVS 서버의 정보를 알려준다. 그러므로 자신의 홈디렉토리의 .bash_profile 파일에 다음과 같은 내용을 추가시켜주어야 한다.

CVSROOT=:pserver:yundream@192.168.0.5:/home/cvs
				
위의 CVSROOT 에는 "pserver" 은 우리가 CVS 서버에 접근하기 위해서 pserver 방식을 사용할것이며, 서버의 IP는 192.168.0.5 접근 아이디는 yundream 그리고 프로젝트가 저장되어 있는 저장소 디렉토리는 /home/cvs 라는 정보를 가지고 있다.

환경변수 CVSROOT 를 사용하지 않고 -d 를 이용하는 방법도 있다.

cvs -d :pserver:yundream@192.168.0.5:/home/cvs [옵션]
				
그러나 이방법은 불편하므로 환경변수를 사용하도록 하자.


3.2. cvs 클라이언트 사용하기

3.2.1. 로그인 하기 : login

프로젝트를 아무나 접근해서 사용하게 해서는 당연히 안될 것이다. 그러므로 최초에 아이디와 패스워드를 이용한 인증절차를 거치게 된다.

CVS 서버에 로그인 하기 위해서는 "cvs" 다음에 옵션으로 login 을 주고 실행시키면 된다.

# cvs login
CVS password: 
				
그러면 패스워드를 묻는 프롬프트가 떨어질것이다. 자신의 패스워드를 입력하면 인증이 성공된다. 한번 인증이 성공되면 인증에 사용한 여러가지 정보가 .cvspass 에 저장되고 다음부터는 .cvspass 를 사용해서 자동적으로 인증을 하기 때문에 최초에 한번만 login 을 성공하면 된다.

물론 당연히 CVS 서버인 192.168.0.5 에는 yundream 이란 사용자가 등록되어 있어야 하며 이 사용자는 cvs 그룹에 포함되어 있어야 한다.


3.2.2. 프로젝트 만들고 등록하기 : import

가장 먼저 해야할일은 진행될 프로젝트를 만들고 등록하는 일이다. 우리가 진행하고자 하는 프로젝트는 hello_world 프로젝트이며, 여기에는 hello.c라는 하나의 파일이 포함되어 있다. 다음은 hello.c의 코드이다.

int main()
{
    printf("Hello World\n");
}
				
우리는 단지 hello.c코드가 있는 디렉토리로 이동해서 다음과 같이 import 시켜주면 된다.
# cd hello_world
# cvs -d:pserver:yundream@192.168.0.5:/home/cvs import hello_world project start
				
이 디렉토리는 프로젝트 임포트를 위해서 새로 준비된 디렉토리여야 한다. 그렇지 않고 잡다한 파일들이 있을경우 이들 파일들까지 몽땅 프로젝트에 등록되어 버린다. 만약 환경변수 CVSROOT가 설정되어 있다면 아래와 같이 간단하게 import를 실행할 수 있을 것이다.
# cvs import hello_world project start
N hello_world/hello.c

No conflicts created by this import
				
2번째 인자인 hello_world는 import할 프로젝트의 이름이며 project, start는 프로젝트의 부가 정보들이다 (별로 신경쓸 필요 없다).

이렇게 하면 cvs서버의 저장소에 hello_world란 디렉토리가 생기고 여기에 hello.c가 올라가게된다.

참고: cvs 저장소에 올라가는 파일 원래 파일이름뒤에 ',v'이 붙어서 저장된다. hello.c라면 hello.c,v라는 이름으로 저장되며 여기에는 hello.c의 원래 내용외에 버젼 관리를 위한 각종 정보가 들어가게 된다.


3.2.3. 프로젝트 가져오기 : checkout

프로젝트 관리자가 프로젝트를 만들었다면 이제 프로젝트 개발자들이 프로젝트를 받아와서 필요한 작업을 해야할 것이다. 프로젝트에 가져오는 방법은 간단하다. "cvs" 다음에 checkout(혹은 co) 옵션을 사용하면 된다. co 옵션 뒤에는 등록할 프로젝트 이름(hello_world)를 명시해 주면 된다.

 
# cvs co hello_world 
cvs server: 
Updating hello_world
U hello_world/hello.c
				
성공적으로 프로젝트가 등록되었다. 이제 ls 해보면 현재 디렉토리에 hello_world 라는 프로젝트이름을 가지는 디렉토리가 생겼음을 알 수 있다. 이와 더불어 hello.c역시 확인 가능할 것이다.


3.2.4. 프로젝트 수정후 적용(업데이트) : commit

그런데 원래의 hello.c를 보면 printf()함수가 선언되어 있는 stdio.h가 빠져있다. 대부분의 경우 문제가 없지만 컴파일러에 따라서 경고메시지를 보내거나 심한경우 컴파일 실패하는 경우도 있다. 그래서 헤더파일을 추가하기로 했다.

#include <stdio.h>

int main(int argc, char **argv)
{
    printf("Hello world");
}
				
헤더파일을 추가 시켰다. 이제 프로젝트 내용을 내가 수정한 내용으로 cvs를 업데이트 시켜야 할것이다. 이럴때는 "commit" 옵션을 이용하면 된다. -m 옵션을 이용하여 변경된 내용에 대한 간단한 로그도 남겨줄수 있다.
# cvs commit -m "stdio.h 헤더파일 인클루드" hello.c 
Checking in hello.c;
/home/cvs/hello_world/hello.c,v  <--  hello.c
new revision: 1.2; previous revision: 1.1
done
#
				
물론 cvs 업데이트를 시킬때는 반드시 컴파일이 되는지 정도는 확인하고 올려야 될것이다. 컴파일도 안되는 코드를 올렸다가는 팀원들에게 원망의 소리를 듣게 될것이다.


3.2.5. 파일 받아오기/로그 보기 : update, log

이제 여러분은 hello_world 프로젝트의 진행 일원이 되었다. 프로젝트 참여 개발자로서 여러분이 컴퓨터 앞에 앉았다면 가장 먼저 해야할일은 간밤에 누군가 프로젝트를 수정하지 않았는지 확인하고 테스트하는 일부터 해야 한다. 최근 프로젝트에 대한 변경사항은 "up"을 이용해서 확인할 수 있다.

[yundream@myhome hello_world]$ cvs up
cvs server: Updating .
P hello.c
[yundream@myhome hello_world]$ 
				
hello.c 라는 파일이 수정 돼었음을 알수 있다. hello.c 는 새로이 수정 되었음으로 누가 어떤 이유로 코드를 수정했는지 확인해볼 필요가 있을 것이다. 이럴때는 "log" 명령을 사용한다. "log"명령을 사용하면 해당 파일에 대한 간략한 로그정보를 얻어 올수 있다.
[yundream@myhome hello_world]$ cvs log hello.c
cvs server: Logging .

RCS file: /home/cvs/hello_world/hello.c,v
Working file: hello.c
head: 1.1
branch:
locks: strict
access list:
symbolic names:
keyword substitution: kv
total revisions: 1;     selected revisions: 1
description:
----------------------------
revision 1.1
date: 2002/06/07 01:36:27;  author: yundream;  state: Exp;
stdio.h 헤더파일 인클루드
=============================================================================
				
그러면 버젼정보에서 부터, 누가 수정을 했는지, 언제 수정을 했는지와 commit하면서 남긴 로그메시지 등이 표시 됨으로 hello.c 가 어떤식으로 변경되었는지 대략의 정보를 얻어올 수 있다.


3.2.6. 버젼별 차이 확인 : diff

이렇게 해서 새로운 소스 파일을 가져왔는데, 어느 코드의 어떤 라인이 수정되었는지 확인하고 싶을 때가 생길 것이다. 이러한 버젼별 코드 변경사항의 히스토리가 남게 된다면 나중에 문제가 생겼을 때 이전버젼의 코드를 쉽게 얻어낼수도 있을 것이다.

cvs는 버젼별 코드변경사항을 모두 저장하며 개발자는 이 내용을 이용해서 코드변경사항을 확인할 수 있을 뿐 아니라 문제가 생겼을시 이전 버젼의 코드를 얻어올 수도 있다.

참고: 물론 이러한 작업을 위해서는 약간의 수작업이 필요한데, 공개된 CVS프론트엔드들은 자동으로 이러한 귀찮은 일을 대신 해준다. 아마도 웹기반의 CVS프론트엔드가 가장 좋은 선택이 될 것이다.

버젼별 변경내용은 diff를 통해서 확인할 수 있다.

//hello.cc에 대해서 1.1버젼과 1.2버젼과의 변경내용을 출력하시오.
# cvs diff -r 1.2 -r 1.1 hello.cc
bash-2.04# cvs diff -r 1.2 -r 1.1 hello.c 
Index: hello.c
===================================================================
RCS file: /home/cvs/hello_world/hello.c,v
retrieving revision 1.2
retrieving revision 1.1
diff -r1.2 -r1.1
1,2d0
< #include <stdio.h>
< 
				

단지 diff만 사용하면 가장최근의 변경사항을 출력한다.

#cvs diff test.cc
				


3.2.7. 파일 추가하기 : add

이렇게 해서 프로젝트를 진행하다 보니 README파일을 추가시켜야 할 필요성을 느끼게 되었다. 이처럼 프로젝트를 진행하다 보면 중간중간 여러개의 파일을 추가해야 될건데 이럴경우 add를 이용해서 파일을 추가하면 된다.

# cvs add README
cvs server: scheduling file `README' for addition
cvs server: use 'cvs commit' to add this file permanently
				
이렇게 해서 파일을 추가하긴 했는데 이렇게 했다고해서 바로 파일이 cvs에 등록되지는 않는다. 반드시 commit 를 해서 파일이 프로젝트에 사용될것이라는것을 승인 시켜줘야 한다.

이제 다른 개발자가 cvs up을 하면 README파일이 추가된것을 확인할 수 있게 된다.

# cvs up
cvs server: Updating .
U README
				


3.2.8. 충돌의 해결

위의 경우는 "팀장"이 파일을 올리고, "팀원"이 파일을 받아서 수정하고 다시 파일을 올리는 과정을 밟고 있다. 그러나 하나의 프로젝트파일에 대해서 2명이 동시에 작업을 하다보면 충돌되는 경우가 생길수 있을것이다.

예를 들어서 "팀장"과 "팀원"이 동시에 같은 라인을 수정하고 있는데, 팀장이 먼저 commit 를 했다고 하자. 잠시 후에 팀원이 commit 하면, 같은 코드라인에 대해서 충돌이 일어나게 될것이다.

그렇지만 걱정할것 없다. cvs 가 알아서 자동적으로 관리해주기 때문이다. 팀원이 자신의 쏘쓰를 commit 하려고 하면 cvs 는 자동적으로 "코드라인에 충돌이 일어났음으로 먼저 update 를 하시오" 라는 메시지를 출력한다. 그래서 update 를 시키면 어느 부분이 충돌을 일으키는지 표시해준다.

다시 프로젝트 쏘쓰인 hello.c 로 돌아가서, 팀장이 다음과 같이 코드를 약간 수정했다고 가정하자.

#include <stdio.h>

int main(int argc, char **argv)
{
    printf("Hello World!!!!!\n");
}
				
그런데 그때 "팀원" 도 동일한 코드를 수정했다.
#include <stdio.h>

int main(int argc, char **argv)
{
    printf("Hello World??\n");
}
				
이상태에서 팀장이 commit 시키고, 잠시후에 팀원이 commit 시키려고 하면 다음과 같은 에러가 발생한다. (팀장은 아무 이상없이 commit 시킬수 있다)
[yundream@myhome hello_world]$ cvs commit -m hello.c
cvs commit: Examining .
cvs server: Up-to-date check failed for `hello.c'
cvs [server aborted]: correct above errors first!
				
이문제를 해결하기 위해서 먼저 update 명령을 이용해서 현재 CVS서버에 저장된 프로젝트 내용을 가져오도록 한다. 그러면 다음과 같은 메시지를 받아볼수 있을것이다.
# cvs up
cvs server: Updating .
RCS file: /home/cvs/hello_world/hello.c,v
retrieving revision 1.3
retrieving revision 1.4
Merging differences between 1.3 and 1.4 into hello.c
rcsmerge: warning: conflicts during merge
cvs server: conflicts found in hello.c
C hello.c
				
기존의 1.3 버젼과 지금 버젼의 프로젝트와 충돌이 생겼음을 알수 있다. 'C' 는 충돌(confilict)이 있음을 나타내는 단어이다. 이제 hello.c 쏘쓰를 보면 다음과 같이 충돌되는 부분을 표시해줄것이다.
#include <stdio.h>

int main(int main, char **argv)
{
<<<<<<< hello.c
    printf("Hello World!!!!!\n");
=======
    printf("Hello World??\n");
>>>>>>> 1.3
}
				
이럴경우는 팀장과 연락을 취해서(메일, 메신저, 전화로) 충돌되는 코드에 대해서 서로 조율해야 할것이다. 굳이 조율까지 할 필요 없이 그냥 팀장의 코드를 사용하기로 했다면 충돌된 부분을 팀장의 코드로 조정한다음에 commit시키면 된다.


3.2.9. 필요없는 파일지우기 : delete

쏘쓰파일중에 더이상 필요가 없어진 파일에 대해서는 delete 명령을 이용할수 있다. README 파일이 더이상 필요 없다면 우선 README 파일을 rm을 이용해서 지워주고.. delete한 후 commit시키면 된다.

# rm README
# cvs delete README
cvs server: scheduling `README' for removal
cvs server: use 'cvs commit' to remove this file permanently

# cvs commit -m "리드미 더이상 필요 없어서 삭제했음" README
Removing README;
/home/cvs/hello_world/README,v  <--  README
new revision: delete; previous revision: 1.1
done
				
이제 다른 프로젝트 개발자가 cvs up을 하면 다음과 같이 README가 삭제되었음을 확인할 수 있게 된다.
# cvs up
cvs server: Updating .
cvs server: README is no longer in the repository
				
이후 ls를 하면 실제 README파일이 지워져있음을 확인 할 수 있을 것이다. 그렇다면 어떤 악의적인 개발자가 중요한 파일을 지워버리면 잘못해서 해당 파일을 영원히 잃어 버리게 되는 사태가 발생하지 않을지 걱정이 될 수도 있을 것이다. 그러나 이 문제는 그리 크게 걱정할 필요가 없다. 해당 파일은 이미 다른 여러 개발자들이 가지고 있을 것이며, 설혹 그렇지 않다 하더라도 cvs 서버에는 파일이 보존이 되어 있기 때문에 언제든지 복구 가능하기 때문이다. 또한 누가 파일을 삭제 했는지 log를 통해서 쉽게 알아 낼 수 도있다.


3.2.10. 프로젝트를 완료했을때 : release

프로젝트를 완료했다면, release 옵션을 사용해서 프로젝트를 닫을수 있다. 프로젝트를 닫는다고 해서 저장소의 파일을 완전히 지우는 것은 아니다. 단지 저장소의 프로젝트에 어떠한 수정도 할수없는 상태다.


3.3. 효율적인 프로젝트 관리를 위한 CVS 사용법

CVS는 여러명의 개발자가 참여한다는 가정하게 사용되어 지므로 이를 효율적으로 프로젝트가 진행되도록 하기 위한 몇가지 지켜야할 사항이 있는데 이들에 대해서 알아보도록 하겠다.

아침에 와서 컴퓨터 앞에 앉았다면, 가장먼저 cvs up 을 이용해서 밤사이에 업데이트 된 내용이 있는지 확인을 한다. 그다음 작업에 들어간다. 공동작업을 할때 가장 중요한것은 상대편 작업자가 무슨일을 하고 있는지에 대해 알아야 하는 것이다. 습관적으로 cvs up을 해줘야 한다.

그리고 꽤 중요한 수정이 있었다고 하면 중간중간에 commit하도록 한다. commit할때도 우선 cvs up을 이용해서 수정사항이 있었는지 확인하도록 하고 당연하지만 반드시 컴파일이 되는지 확인한 후 commit 시켜야 한다.

CVS 를 제대로 사용하기 위해서는 CVS 저장소의 구성을 잘해놓아야 한다. 예를 들어 Project_A 란 프로젝트를 시작한다면 Project_A 란 디렉토리를 만들고 그 아래에 프로젝트에 필요한 각종 디렉토리, 즉 작업문서의 저장을 위한 doc 디렉토리, 실제 쏘쓰가 포함될 src 디렉토리, 인크루드 파일이 존재하게 될 src/include 디렉토리 등, 체계적으로 프로젝트를 관리할수 있도록 미리 저장소를 세팅해 놓아야한다. 물론 나중에라도 디렉토리를 추가할수도 있지만, 이왕이면 미리 프로젝트에 대한 계획을 잘 세워두는게 좋을것이다.

Project_A --+-- doc -+-- readme.txt
            |        | 
            |        +-- todo_list.txt
            |        |
            |        +-- schedule.xls 
            |
            +-- src --+-- Makefile
                      |  
                      +-- main.cc  
                      |  
                      +-- io.cc  
                      |  
                      +-- include --+-- io.h 
                      |             | 
                      |             +-- common.h 
                      +-- lib     --+-- crypt.a 
			
그리고 가능하면 모듈별로 쏘쓰코드를 나누고, 각 모듈별로 분담해서 개발을 하도록 하는게 프로젝트를 쉽게 관리 하는 방법이다. 그러면 개발자는 자신의 모듈만 신경쓰면서도, 전체의 프로젝트의 흐름에 유연하게 대처할수 있게 된다.


3.4. 기타 옵션

3.4.1. 디렉토리 upload하기

작업을 하다보면 디렉토리를 만들어서 cvs서버에 등록시켜야 하는 경우가 있다. 등록은 add와 commit를 이용해서 가능한데, 이를 upload하기 위해서는 -d 옵션을 이용해야 한다. -d 옵션을 사용하지 않을경우 일반 파일들만 upload하고 디렉토리는 upload 하지 않게 된다.

# cvs up -d
				


3.4.2. 바이너리 파일 추가하기

cvs는 일반텍스트로 이루어진 파일을 잘 관리하게끔 최적화 되어있다. 그러다 보니 바이너리 파일을 추가할경우 이를 텍스트로 인식해서 힘들게 추가했더니 파일이 깨지는 경우가 발생한다. 바이너리 파일은 다음과 같은 방법으로 추가하도록 하자.

# cvs add -kb filename.bin 
				


4. CVS 의 다른 활용들

필자는 개인 정보의 관리를 위해서 CVS 를 사용한다. 회사에서 일하다가 좋은 사이트를 찾았다거나, 중요한 메모 상황이 발생했다거나, 좋은 자료를 찾았을경우, 일정관리 까지 모두 CVS 로 저장해두고 집에 있건 회사에 있건 개인 자료를 공유할수 있도록 만들어 놓았다.

모질라의 북마킹 데이타를 CVS 로 연결시켜 놓았기 때문에, 집에 있건 회사에 있건 동일한 북마킹 데이타를 유지할수 있으며, 일정관리(Koraganizer 을 이용한다), 메모데이타(knotes) 역시 공유 가능하도록 해두어서 편하게 사용하고 있다.

이렇게 CVS를 사용함으로써 PDA, email, 노트북(돈이 좀 있어야 한다) 매체를 이용하지 않고도, 간단하게 데이타 동기화를 이루어 낼수 있다.


5. 결론

이상 CVS의 설치와 간단한 사용방법과 활용방법에 대해서 얘기 해보았다. 여기에 있는 내용은 CVS 의 많은 기능중 꼭 필요한 10% 정도의 내용만을 다루고 있다. CVS 사용에 대한 깊은 내용을 알기 원한다면 cvs 와 함께 배포되는 man 페이지를 활용하거나 CVS 이야기, CVS 사용 등의 문서를 참고하기 바란다.


Posted by Bart
코딩 관련/팁2007. 11. 30. 12:13
문서출처 : http://www.lug.or.kr/2005/bbs.php?table=community_tip&query=view&uid=109&p=1


생기는 이유는 도스용 문서를 유닉스에서 보면 ^M이 행 끝에 나와 있는 경우를 볼 수 있다.
또한 editplus, ultraedit 등 문서편집기에서 바로 ftp 에 업로드 하는 경우도 발생 할 수 있다.


Vi(m) 사용시 ^M 없애기

vi(m)을 사용하다보면 윈도우즈에서 작업한 텍스트 파일에 ^M 문자가 붙는 경우를 종종 봅니다.
현재 vim에서는 도스 문서로 문서를 읽게 되면 이 문자가 붙지 않기도 하지만
종종 이 문자 때문에 문서 보기가 상당히 귀찮아 보일 때가 있습니다.
특히, SQL 백업 파일을 볼 때 이 문자가 붙기도 하는데 특정한 내용을 볼 때
이런 문자가 붙은 것을 보기는 상당히 힘들죠.

1. vi(m)이 열린 상태
1.1 문자 치환 명령 이용
다음과 같이 문자 치환 명령으로 이 문자를 없애도록 합니다.
:1,$s/^M//g

1,$ -- 범위지정
s -- substitute(치환)하라
^M -- Control Key 를 누른 상태에서 v key 를 누르고 동시에 m key 를 누릅니다.
// -- 공백문자로 치환
g -- 모든 매치된 case를 치환하라.


물론 이것의 입력은 다음과 같이 하도록 합니다.
:1,$s/[Control]+v+m을 누른다.//g

1.2 파일포맷 변경
vim 5.0 이상에서는 다음과 같이 명령을 내리도록 합니다

:set fileformat=unix
혹은
:set ff=unix
그리고, 저장하고 종료합니다.
:wq

이와 반대로 끝에 ^M붙이려면 다음과 같이 합니다.
:set fileformat=dos

이전버전에서는 다음과 같이 합니다.
:set textmode


2. 프롬프트 상에서
2.1 dos2unix 이용
dos2unix와 unix2dos를 이용하여 변환할 수 있습니다.

다음과 같이 하면 ^M 문자가 없어집니다.
$ dos2unix [문서명](여러 파일의 경우 *.확장자)

그리고 다음과 같이 하면 ^M 문자가 생기죠.
$ unix2dos [문서명]

2.2 cat과 tr을 이용한 방법
다음과 같이 하면 해당 파일의 ^M 문자가 없어집니다.
$ cat dos-file | tr -d ^M > unix-file

2.3 perl 스크립트
다음과 같이 하면 여러파일을 한번에 바꾸게 됩니다.

# perl -i -pe 's/015//g' *.html

3. 다른 에디터의 사용
pico 라는 에디터 아실겁니다. 기능은 별로 없지만 빠르고 간편하죠.
vi나 emacs에서 ^M으로 잡히는 것이 pico에서는 안 잡히는데,
파일을 pico로 열어서 다시 이 문서를 저장합니다.
그리고 vi나 emacs로 읽으면 ^M이 모두 없어진 것을 알 수 있습니다.

4. man 페이지의 ^M, ^H 문자
man 페이지를 일반 화일로 바꾸면 ^M, ^H 등이 생기는데(예, 한컴리눅스 ^H)
다음과 같이 하면 이 문자를 없애고 볼 수 있습니다.

다음은 ls의 예입니다.
4.1 ps 또는 pdf로 변환하기
man -t 변경 시킬 화일 > 변경후 화일명.ps
man -t 변경 시킬 화일 > 변경후 화일명.pdf
예)
man -t ls > ls.ps 또는 ls.pdf로 보시면 됩니다.

4.2 텍스로 변환하기
man 변환시킬화일 | col -b> 변환후 화일명.txt
예)
man ls | col -b > ls.txt
Posted by Bart
코딩 관련/Example codes2007. 11. 14. 17:43

동생 선배의 부탁으로 방금 짜보았다.
오랫만에 Java 코딩하려니 30분이나 걸렸다는...... 그중 25분은  URL을 파일명으로 기록하는데 있어 // 이 파일명에 들어가서 발생하는 예외사항을 깜빡해서 ㅎㅎㅎ

import java.net.*;
import java.io.*;

public class pullPages {

 public static void main(String[] args)
 {
  subClass instance = new subClass();

  if (args.length==0) { // 실행 시 인수 있는지 확인하여 없으면 오류 처리
   System.out.println("Usage: java pullPages <your URL>");
   System.out.println("For example, java pullPages http://bart7449.tistory.com");
   System.exit(0);
  }
  else if(args.length==1){  // 접근하고자 하는 URL이 정확히 하나인 경우
   instance.accessByURL(args[0]);

  }else{ //여러 개일 경우 반복하여 처리
   for(int i=0; i<args.length; i++){
    instance.accessByURL(args[i]);  
   }
  }
 
 }
}

class subClass{ //실제 URL에 접근해서 데이터를 가져오는  클래스

 void accessByURL(String s){
  try{
   URL newAccess= new URL(s); 
   URLConnection newCon = newAccess.openConnection();
   BufferedReader in = new BufferedReader(new InputStreamReader(newCon.getInputStream()));
   int charRead;
   File newFile= new File(s.trim().substring(7)); //파일 객체의 생성
   newFile.createNewFile();
   BufferedWriter out = new BufferedWriter(new FileWriter(newFile));

   while((charRead = in.read()) != -1){ //페이지가 종료될때까지 코드표로 한자 씩 읽어서
    out.write(charRead); //파일에 한자씩 기록
   }
   in.close();
   out.close(); // 다 기록 후 입출력 스트림을 닫음
  }catch(Exception e){
   System.out.println("Exceptional case has arisen during this execution");
   //e.printStackTrace();
  }

 }
}


Posted by Bart