Bash脚本更新pom文件中的标签之间的文本?

时间:2013-09-20 14:14:34

标签: regex sed pom.xml

我用谷歌搜索了这个,但找不到具体的答案。

我有一个带有以下标记的pom.xml文件

<properties>
    <rpm.release>10</rpm.release>
</properties>

然后,我必须遵循脚本,寻找标签之间的标签,该标签有效。

update_rpm_release_in_pom()
{
  BUILD_NUM=$1
  POM_FILE="./template/iptools/pom.xml"
  BEGIN_PROPERTIES="n"
  # Look for the <rpm.release> tag in the <properties>
  # and update with with passed-in build_num
  cat $POM_FILE | while read LINE
  do
    case $BEGIN_PROPERTIES in
      n) # <property> tag not found yet
        if [[ "$LINE" == *\<properties\>* ]]; then
          BEGIN_PROPERTIES="y"
        fi
        continue
      ;;
      y) # <property> tag found
        if [[ "$LINE" == *\</properties\>* ]]; then
          # </property> tag found, stop searching
          break
        fi
        if [[ "$LINE" == *rpm.release* ]]; then
          # Update value
          ####################################
          # How do I update value??????
          ####################################
          break
        fi
      ;;
    esac
  done
}

# Call above function    
update_rpm_release_in_pom 11

我应该在“#我如何更新价值???”中加入什么所以当我运行脚本时,pom.xml现在有了

<properties>
    <rpm.release>11</rpm.release>
</properties>

感谢。

2 个答案:

答案 0 :(得分:2)

由于问题被标记为,我认为awk将是有效的替代方案。

在这种情况下,您可以在awk中处理pom文件,将结果保存在tmpfile中并使用修改后的版本覆盖原始文件:

#!/bin/sh

# error function
error(){
 echo -e >&2 "error: $@"
 exit 1
}

update_rpm_release_in_pom()
{
  #################
  # set constants #
  #################

  POM_FILE="./template/iptools/pom.xml"
  USAGE="usage: update_rpm_release_in_pom <build num>"

  #################
  # sanity checks #
  #################

  # argument given?
  [[ $# -eq 1 ]] || \
    error "wrong number of parameters\n$USAGE"

  # pom file exists?
  [[ -e "$POM_FILE" ]] || \
    error "pom file $POM_FILE does not exist"

  # pom file readable?
  [[ -r "$POM_FILE" ]] || \
    error "pom file $POM_FILE is not readable"

  # pom file writeable?
  [[ -w "$POM_FILE" ]] || \
    error "pom file $POM_FILE is not writeable"

  ##################
  # set parameters #
  ##################

  BUILD_NUM=$1

  ###############################
  # create temporal output file #
  # and ensure its removal      #
  ###############################

  # cleanup function
  cleanup(){
    [[ -e "$TMP_FILE" ]] && rm -- "$TMP_FILE"
  }

  # ensure cleanup on exit
  trap 'cleanup' EXIT

  # create temporal output file or exit with error
  TMP_FILE=`mktemp` || \
    error "could not create temporal output file"

  #############################
  # process pom file with awk #
  # and save modified version #
  # in temporal output file   #
  #############################

  awk \
    -F "rpm.release>" \
    -v OFS="rpm.release>" \
    -v BUILD_NUM="$BUILD_NUM" \
    '
      /<properties>/{
        properties=1
      }
      /<\/properties>/{
        properties=0
      }
      properties&&$1~/<$/&&$2~/<\/$/{
        $2=BUILD_NUM"</"
      }
      1
    ' "$POM_FILE" > "$TMP_FILE" || \
    error "processing pom file $POM_FILE with awk failed"

  #########################
  # move modified version #
  # back to original path #
  #########################

  mv -- "$TMP_FILE" "$POM_FILE" || \
    error "could not mv temporal output file $TMP_FILE to pom file $POM_FILE"
}

# Call above function
update_rpm_release_in_pom 11

请注意,这应该适用于awk的所有版本,因为那里没有GNUisms。

可能有一种方法可以在sed中执行此操作并使用sed的inplace(-i)选项来解决tmpfile问题,但是a)我不知道如何要在sed中以安全的方式执行此操作,并且b)我甚至不知道您是否运行了支持sed选项的-i版本。

如果您需要进一步的帮助,调整和/或解释,请随时添加评论。

<强>附录:

如果确实需要在没有任何脚本语言的情况下执行此操作,我修改了您的代码并用{:}代替了我的awk代码:

#!/bin/bash

# error function
error(){
 echo -e >&2 "error: $@"
 exit 1
}

update_rpm_release_in_pom()
{
  #################
  # set constants #
  #################

  POM_FILE="./template/iptools/pom.xml"
  USAGE="usage: update_rpm_release_in_pom <build num>"

  #################
  # sanity checks #
  #################

  # argument given?
  [[ $# -eq 1 ]] || \
    error "wrong number of parameters\n$USAGE"

  # pom file exists?
  [[ -e "$POM_FILE" ]] || \
    error "pom file $POM_FILE does not exist"

  # pom file readable?
  [[ -r "$POM_FILE" ]] || \
    error "pom file $POM_FILE is not readable"

  # pom file writeable?
  [[ -w "$POM_FILE" ]] || \
    error "pom file $POM_FILE is not writeable"

  ##################
  # set parameters #
  ##################

  BUILD_NUM=$1

  ###############################
  # create temporal output file #
  # and ensure its removal      #
  ###############################

  # cleanup function
  cleanup(){
    [[ -e "$TMP_FILE" ]] && rm -- "$TMP_FILE"
  }

  # ensure cleanup on exit
  trap 'cleanup' EXIT

  # create temporal output file or exit with error
  TMP_FILE=`mktemp` || \
    error "could not create temporal output file"

  ##############################
  # process pom file with bash #
  # and save modified version  #
  # in temporal output file    #
  ##############################

  BEGIN_PROPERTIES="n"
  while read LINE
  do
    case $BEGIN_PROPERTIES in
      n) # <property> tag not found yet
        [[ "$LINE" == *\<properties\>* ]] && BEGIN_PROPERTIES="y"
        echo "$LINE"
        continue
      ;;
      y) # <property> tag found
        [[ "$LINE" == *\</properties\>* ]] && BEGIN_PROPERTIES="n"
        if [[ "$LINE" == *rpm.release* ]]; then
          # Update value
          echo "${LINE/<rpm\.release>*<\/rpm\.release>/<rpm.release>$BUILD_NUM<\\rpm.release>}"
          continue
        fi
        echo "$LINE"
      ;;
    esac
  done < "$POM_FILE" > "$TMP_FILE" || \
    error "processing pom file $POM_FILE with bash failed"

  #########################
  # move modified version #
  # back to original path #
  #########################

  mv -- "$TMP_FILE" "$POM_FILE" || \
    error "could not mv temporal output file $TMP_FILE to pom file $POM_FILE"
}

# Call above function
update_rpm_release_in_pom 11

然而,这不如上述解决方案稳定。 例如,在我的机器上,这将从pom文件中删除前导空格。 我不知道这是由于某些引用问题还是您需要调整readecho的参数。 因为我永远不会使用这样的灵魂,所以我不会对此进行进一步的调查,但如果您坚持bash唯一的解决方案,这可能会对您有所帮助。

答案 1 :(得分:-1)

您可以使用

[[ $LINE =~ ^([^>]+>)([0-9]+)(.*) ]] && printf -v LINE "%s%d%s" \
    "${BASH_REMATCH[1]}" \
    $(( ${BASH_REMATCH[2]} + 1 )) \
    "${BASH_REMATCH[3]}"

BASH_REMATCH是由=~运算符设置的数组,用于将值存储在捕获括号中。


它只是更新了这一行。我会使用我最喜欢的命令行xml工具重写它:

update_rpm_release_in_pom() {
    local f=pom.xml
    # get the current value
    local current=$(xmlstarlet sel -t -v '/properties/rpm.release' $f)
    local new=$(( current + 1 ))
    # update the xml with the new value and save it
    xmlstarlet ed -u '/properties/rpm.release' -v $new $f > $f.$new &&
      mv $f $f.$current &&
      mv $f.$new $f
}