patch파일 수동으로 다루기

wkpark의 이미지

개발을 하다보면 수많은 패치가 섞여있어서 커밋을 그냥 할 수 없는 경우가 많습니다.

이런 경우 저는 보통 패치 파일을 열어서 커밋하면 안되는 hunk들을 지우고, 커밋할 hunk만 따로 추려서 패치를 만들고 난 후에
그 패치를 1) 깨끗한 트리로 가서 먹인 후에 커밋을 하거나 2) 어떤 경우는 all.diff를 보관할패치 a.diff / 보낼패치 b.diff로 분리한 후에

patch -p0 -R <all.diff (지운 후에 그냥 svn up해도 되나 확인 차원에서)
patdh -p0 < b.diff (보낼패치 적용)
svn commit ... (커밋)
patch -p0 < a.diff (보존패치 되먹임)

하거나, 3) 원본을 보존하고 svn up 한후에 보낼패치를 먹이고 커밋하거나 했습니다.

이렇게 하는 경우는 대부분 패치가 알맞게 분리되어 있는 경우이지만, 간혹 패치의 hunk를 쉽게 분리할 수 없게 섞여있으면 수동으로 패치를 고쳐서 커밋하곤 했습니다.
간단한 경우는

--- Theme.class (revision 370)
+++ Theme.class (working copy)
@@ -293,9 +297,10 @@
                        if (empty($message_box_is_alert)) {
                                $js_message_box = "message_box_after_loading('$feedback');";
                        } else {
+                               $feedback = preg_replace("/<br[\s\/]*>/i", "\\n", $feedback);
                                $js_message_box = "alert('$feedback')";
                        }
-                       $this->smarty->assign( 'feedback', $js_message_box );
+                       $this->assign( 'feedback', $js_message_box );
                }
 
                /* get the per page contents */

여기서 첫번째 패치라인은 커밋하고 싶고, 두번째 부분은 집어넣으면 안되는 경우는 간단히 다음과 같이 하면 됩니다.
--- Theme.class (revision 370)
+++ Theme.class (working copy)
@@ -293,9 +297,10 @@
                        if (empty($message_box_is_alert)) {
                                $js_message_box = "message_box_after_loading('$feedback');";
                        } else {
+                               $feedback = preg_replace("/<br[\s\/]*>/i", "\\n", $feedback);
                                $js_message_box = "alert('$feedback')";
                        }
-                       $this->smarty->assign( 'feedback', $js_message_box );
+                       $this->smarty->assign( 'feedback', $js_message_box ); (고치기 싫은 라인)
                }
 
                /* get the per page contents */

그런데 만약 이것을 그냥 수동으로 라인을 지우면 어떻게 될까요?

@@ -293,9 +297,10 @@
                        if (empty($message_box_is_alert)) {
                                $js_message_box = "message_box_after_loading('$feedback');";
                        } else {
+                               $feedback = preg_replace("/<br[\s\/]*>/i", "\\n", $feedback);
                                $js_message_box = "alert('$feedback')";
                        }
                        $this->smarty->assign( 'feedback', $js_message_box );

이 패치는 그냥 쓰지 못하고 패치된 라인번호 (293, 297) 및 라인 카운팅 번호 9,10을 보정해줘야 합니다.
(위의 경우는 간단해서 눈짐작으로 대충 고쳐줘도 되지만)

라인과 카운팅 번호를 보정해주는 유틸리티가 바로 recountdiff입니다. (혹은 rediff)
recountdiff a.diff 라고 하면 hunk의 줄번호와 카운팅 번호를 고쳐줍니다.

$ recountdiff a.diff
Index: Theme.class
===================================================================
--- Theme.class (revision 370)
+++ Theme.class (working copy)
@@ -293,5 +293,6 @@
                        if (empty($message_box_is_alert)) {
                                $js_message_box = "message_box_after_loading('$feedback');";
                        } else {
+                               $feedback = preg_replace("/<br[\s\/]*>/i", "\\n", $feedback);
                                $js_message_box = "alert('$feedback')";
                        }
                        $this->assign( 'feedback', $js_message_box );

이걸 패치해보면 다음과 같이 나오면서 성공적으로 패치가 됩니다. 이렇게 하면 분리된 패치를 커밋할 수 있게 되겠지요.

$ recountdiff a.diff |patch -p0
patching file Theme.class
Hunk #1 succeeded at 297 with fuzz 1 (offset 4 lines).

P.S.: patchutils에 관련된 문서를 찾아보니 KLDP에는 단 한건도 없고, 구글링해도 한글로 된 설명이 별로 없어서 올려봅니다 :)

댓글

이응준의 이미지

이런 좋은 툴이 있었군요! 그것도 모르고 수동으로 고치다가 잘못해서 패치가 엉망되고...

좋은 정보 감사합니다~ 적어두어야겠네요.

권순선의 이미지

아 좋은 내용이에요... 리눅스 커널 쪽에서 나온 quilt 라는 툴도 비슷하다고 알고 있는데 언제 한번 써보시고 사용법 알려 주심 감사... ^^

wkpark의 이미지

guilt는 처음 보는거네요.
살펴보니 이건 패치 덩어리 그 자체를 좀 더 쉽게 관리하는 git 위에서 돌아가는 툴인것 같구요. bazaar같은데서 지원하는 패치큐 기능인듯.

patchutils는 패치 그 자체를 쪼개거나 살펴보고, 패치 자체를 편집하는 것을 도와주는 기능도 있다는 것입니다.

온갖 참된 삶은 만남이다 --Martin Buber

feanor의 이미지

quilt에 대해 예전에 적었습니다.
http://kldp.org/node/81228

댓글 달기

Filtered HTML

  • 텍스트에 BBCode 태그를 사용할 수 있습니다. URL은 자동으로 링크 됩니다.
  • 사용할 수 있는 HTML 태그: <p><div><span><br><a><em><strong><del><ins><b><i><u><s><pre><code><cite><blockquote><ul><ol><li><dl><dt><dd><table><tr><td><th><thead><tbody><h1><h2><h3><h4><h5><h6><img><embed><object><param><hr>
  • 다음 태그를 이용하여 소스 코드 구문 강조를 할 수 있습니다: <code>, <blockcode>, <apache>, <applescript>, <autoconf>, <awk>, <bash>, <c>, <cpp>, <css>, <diff>, <drupal5>, <drupal6>, <gdb>, <html>, <html5>, <java>, <javascript>, <ldif>, <lua>, <make>, <mysql>, <perl>, <perl6>, <php>, <pgsql>, <proftpd>, <python>, <reg>, <spec>, <ruby>. 지원하는 태그 형식: <foo>, [foo].
  • web 주소와/이메일 주소를 클릭할 수 있는 링크로 자동으로 바꿉니다.

BBCode

  • 텍스트에 BBCode 태그를 사용할 수 있습니다. URL은 자동으로 링크 됩니다.
  • 다음 태그를 이용하여 소스 코드 구문 강조를 할 수 있습니다: <code>, <blockcode>, <apache>, <applescript>, <autoconf>, <awk>, <bash>, <c>, <cpp>, <css>, <diff>, <drupal5>, <drupal6>, <gdb>, <html>, <html5>, <java>, <javascript>, <ldif>, <lua>, <make>, <mysql>, <perl>, <perl6>, <php>, <pgsql>, <proftpd>, <python>, <reg>, <spec>, <ruby>. 지원하는 태그 형식: <foo>, [foo].
  • 사용할 수 있는 HTML 태그: <p><div><span><br><a><em><strong><del><ins><b><i><u><s><pre><code><cite><blockquote><ul><ol><li><dl><dt><dd><table><tr><td><th><thead><tbody><h1><h2><h3><h4><h5><h6><img><embed><object><param>
  • web 주소와/이메일 주소를 클릭할 수 있는 링크로 자동으로 바꿉니다.

Textile

  • 다음 태그를 이용하여 소스 코드 구문 강조를 할 수 있습니다: <code>, <blockcode>, <apache>, <applescript>, <autoconf>, <awk>, <bash>, <c>, <cpp>, <css>, <diff>, <drupal5>, <drupal6>, <gdb>, <html>, <html5>, <java>, <javascript>, <ldif>, <lua>, <make>, <mysql>, <perl>, <perl6>, <php>, <pgsql>, <proftpd>, <python>, <reg>, <spec>, <ruby>. 지원하는 태그 형식: <foo>, [foo].
  • You can use Textile markup to format text.
  • 사용할 수 있는 HTML 태그: <p><div><span><br><a><em><strong><del><ins><b><i><u><s><pre><code><cite><blockquote><ul><ol><li><dl><dt><dd><table><tr><td><th><thead><tbody><h1><h2><h3><h4><h5><h6><img><embed><object><param><hr>

Markdown

  • 다음 태그를 이용하여 소스 코드 구문 강조를 할 수 있습니다: <code>, <blockcode>, <apache>, <applescript>, <autoconf>, <awk>, <bash>, <c>, <cpp>, <css>, <diff>, <drupal5>, <drupal6>, <gdb>, <html>, <html5>, <java>, <javascript>, <ldif>, <lua>, <make>, <mysql>, <perl>, <perl6>, <php>, <pgsql>, <proftpd>, <python>, <reg>, <spec>, <ruby>. 지원하는 태그 형식: <foo>, [foo].
  • Quick Tips:
    • Two or more spaces at a line's end = Line break
    • Double returns = Paragraph
    • *Single asterisks* or _single underscores_ = Emphasis
    • **Double** or __double__ = Strong
    • This is [a link](http://the.link.example.com "The optional title text")
    For complete details on the Markdown syntax, see the Markdown documentation and Markdown Extra documentation for tables, footnotes, and more.
  • web 주소와/이메일 주소를 클릭할 수 있는 링크로 자동으로 바꿉니다.
  • 사용할 수 있는 HTML 태그: <p><div><span><br><a><em><strong><del><ins><b><i><u><s><pre><code><cite><blockquote><ul><ol><li><dl><dt><dd><table><tr><td><th><thead><tbody><h1><h2><h3><h4><h5><h6><img><embed><object><param><hr>

Plain text

  • HTML 태그를 사용할 수 없습니다.
  • web 주소와/이메일 주소를 클릭할 수 있는 링크로 자동으로 바꿉니다.
  • 줄과 단락은 자동으로 분리됩니다.
댓글 첨부 파일
이 댓글에 이미지나 파일을 업로드 합니다.
파일 크기는 8 MB보다 작아야 합니다.
허용할 파일 형식: txt pdf doc xls gif jpg jpeg mp3 png rar zip.
CAPTCHA
이것은 자동으로 스팸을 올리는 것을 막기 위해서 제공됩니다.