次のような多くのXML文書を含むファイルがあるとします。
<a>
<b>
...
</a>
in between xml documents there may be plain text log messages
<x>
...
</x>
...
与えられた正規表現がこのxml文書の行に一致するxml文書のみを表示するようにこのファイルをフィルタリングするにはどうすればよいですか?ここでは簡単なテキストマッチングについて話しているので、正規表現マッチング部分はデフォルトのフォーマットであるxmlを完全に認識しないかもしれません。
ルート要素の開いたタグと閉じるタグは常に独自の行にあり(空白が埋められる可能性があります)、ルート要素としてのみ使用されると想定できます。つまり、同じ名前のタグはルート要素の下には表示されません。 。これにより、XML認識ツールに頼らずに作業が完了します。
答え1
一般化する
私はPythonソリューション、Bashソリューション、Awkソリューションを作成しました。アイデアはすべてのスクリプトで同じです。 1行ずつ移動し、フラグ変数を使用して状態(つまり、現在XMLサブ文書にあるかどうかに一致する行が見つかったかどうか)を追跡します。
Pythonスクリプトでは、すべての行をリストとして読み込み、現在のXMLサブドキュメントが開始されているリストインデックスを追跡して、閉じるタグに達すると現在のサブドキュメントを印刷できます。各行の正規表現パターンを確認し、フラグを使用して処理が完了したときに現在のサブ文書が出力されているかどうかを追跡します。
Bashスクリプトは一時ファイルをバッファとして使用し、現在のXMLサブドキュメントを保存し、書き込みが完了するのを待ってから、指定された正規表現に一致する行がgrep
含まれていることを確認します。
AwkスクリプトはBaseスクリプトと似ていますが、ファイルの代わりにAwk配列をバッファとして使用します。
テストデータファイル
data.xml
あなたの質問に提供されているサンプルデータに基づいて、次のデータファイル()の両方のスクリプトを確認しました。
<a>
<b>
string to search for: stuff
</b>
</a>
in between xml documents there may be plain text log messages
<x>
unicode string: øæå
</x>
Pythonソリューション
以下は、目的のタスクを実行する簡単なPythonスクリプトです。
#!/usr/bin/env python2
# -*- encoding: ascii -*-
"""xmlgrep.py"""
import sys
import re
invert_match = False
if sys.argv[1] == '-v' or sys.argv[1] == '--invert-match':
invert_match = True
sys.argv.pop(0)
regex = sys.argv[1]
# Open the XML-ish file
with open(sys.argv[2], 'r') if len(sys.argv) > 2 else sys.stdin as xmlfile:
# Read all of the data into a list
lines = xmlfile.readlines()
# Use flags to keep track of which XML subdocument we're in
# and whether or not we've found a match in that document
start_index = closing_tag = regex_match = False
# Iterate through all the lines
for index, line in enumerate(lines):
# Remove trailing and leading white-space
line = line.strip()
# If we have a start_index then we're inside an XML document
if start_index is not False:
# If this line is a closing tag then reset the flags
# and print the document if we found a match
if line == closing_tag:
if regex_match != invert_match:
print(''.join(lines[start_index:index+1]))
start_index = closing_tag = regex_match = False
# If this line is NOT a closing tag then we
# search the current line for a match
elif re.search(regex, line):
regex_match = True
# If we do NOT have a start_index then we're either at the
# beginning of a new XML subdocument or we're inbetween
# XML subdocuments
else:
# Check for an opening tag for a new XML subdocument
match = re.match(r'^<(\w+)>$', line)
if match:
# Store the current line number
start_index = index
# Construct the matching closing tag
closing_tag = '</' + match.groups()[0] + '>'
文字列「stuff」を検索するスクリプトを実行する方法は次のとおりです。
python xmlgrep.py stuff data.xml
出力は次のとおりです。
<a>
<b>
string to search for: stuff
</b>
</a>
「øæå」文字列を取得するスクリプトを実行する方法は次のとおりです。
python xmlgrep.py øæå data.xml
出力は次のとおりです。
<x>
unicode string: øæå
</x>
一致しない文書を指定-v
または検索し--invert-match
、標準入力で作業することもできます。
cat data.xml | python xmlgrep.py -v stuff
カンクンソリューション
これは同じ基本アルゴリズムをbashとして実装したものです。フラグを使用して、現在の行がXML文書に属しているかどうかを追跡し、一時ファイルをバッファーとして使用して、処理中の各XML文書を保管します。
#!/usr/bin/env bash
# xmlgrep.sh
# Get the filename and search pattern from the command-line
FILENAME="$1"
REGEX="$2"
# Use flags to keep track of which XML subdocument we're in
XML_DOC=false
CLOSING_TAG=""
# Use a temporary file to store the current XML subdocument
TEMPFILE="$(mktemp)"
# Reset the internal field separator to preserver white-space
export IFS=''
# Iterate through all the lines of the file
while read LINE; do
# If we're already in an XML subdocument then update
# the temporary file and check to see if we've reached
# the end of the document
if "${XML_DOC}"; then
# Append the line to the temp-file
echo "${LINE}" >> "${TEMPFILE}"
# If this line is a closing tag then reset the flags
if echo "${LINE}" | grep -Pq '^\s*'"${CLOSING_TAG}"'\s*$'; then
XML_DOC=false
CLOSING_TAG=""
# Print the document if it contains the match pattern
if grep -Pq "${REGEX}" "${TEMPFILE}"; then
cat "${TEMPFILE}"
fi
fi
# Otherwise we check to see if we've reached
# the beginning of a new XML subdocument
elif echo "${LINE}" | grep -Pq '^\s*<\w+>\s*$'; then
# Extract the tag-name
TAG_NAME="$(echo "${LINE}" | sed 's/^\s*<\(\w\+\)>\s*$/\1/;tx;d;:x')"
# Construct the corresponding closing tag
CLOSING_TAG="</${TAG_NAME}>"
# Set the XML_DOC flag so we know we're inside an XML subdocument
XML_DOC=true
# Start storing the subdocument in the temporary file
echo "${LINE}" > "${TEMPFILE}"
fi
done < "${FILENAME}"
文字列「stuff」を検索するスクリプトを実行する方法は次のとおりです。
bash xmlgrep.sh data.xml 'stuff'
対応する出力は次のとおりです。
<a>
<b>
string to search for: stuff
</b>
</a>
「øæå」文字列を取得するスクリプトを実行する方法は次のとおりです。
bash xmlgrep.sh data.xml 'øæå'
対応する出力は次のとおりです。
<x>
unicode string: øæå
</x>
奇妙なソリューション
awk
解決策は次のとおりです。awk
私の状態はあまり良くないので荒いです。 BashスクリプトとPythonスクリプトと同じ基本的なアイデアを使用します。各XML文書をバッファ(awk
配列)に保存し、フラグを使用してステータスを追跡します。文書の処理が終了すると、指定された正規表現に一致する行が含まれている場合は文書を印刷します。スクリプトは次のとおりです。
#!/usr/bin/env gawk
# xmlgrep.awk
# Variables:
#
# XML_DOC
# XML_DOC=1 if the current line is inside an XML document.
#
# CLOSING_TAG
# Stores the closing tag for the current XML document.
#
# BUFFER_LENGTH
# Stores the number of lines in the current XML document.
#
# MATCH
# MATCH=1 if we found a matching line in the current XML document.
#
# PATTERN
# The regular expression pattern to match against (given as a command-line argument).
#
# Initialize Variables
BEGIN{
XML_DOC=0;
CLOSING_TAG="";
BUFFER_LENGTH=0;
MATCH=0;
}
{
if (XML_DOC==1) {
# If we're inside an XML block, add the current line to the buffer
BUFFER[BUFFER_LENGTH]=$0;
BUFFER_LENGTH++;
# If we've reached a closing tag, reset the XML_DOC and CLOSING_TAG flags
if ($0 ~ CLOSING_TAG) {
XML_DOC=0;
CLOSING_TAG="";
# If there was a match then output the XML document
if (MATCH==1) {
for (i in BUFFER) {
print BUFFER[i];
}
}
}
# If we found a matching line then update the MATCH flag
else {
if ($0 ~ PATTERN) {
MATCH=1;
}
}
}
else {
# If we reach a new opening tag then start storing the data in the buffer
if ($0 ~ /<[a-z]+>/) {
# Set the XML_DOC flag
XML_DOC=1;
# Reset the buffer
delete BUFFER;
BUFFER[0]=$0;
BUFFER_LENGTH=1;
# Reset the match flag
MATCH=0;
# Compute the corresponding closing tag
match($0, /<([a-z]+)>/, match_groups);
CLOSING_TAG="</" match_groups[1] ">";
}
}
}
次のように呼び出すことができます。
gawk -v PATTERN="øæå" -f xmlgrep.awk data.xml
対応する出力は次のとおりです。
<x>
unicode string: øæå
</x>