次のnginx confファイルがあります。
user www www; ## Default: nobody
...
events {
worker_connections 4096; ## Default: 1024
}
http {
include conf/mime.types;
...
server { # php/fastcgi
listen 80;
server_name domain1.com www.domain1.com;
access_log logs/domain1.access.log main;
root html;
location ~ \.php$ {
fastcgi_pass 127.0.0.1:1025;
}
}
server { # simple reverse-proxy
listen 80;
server_name domain2.com www.domain2.com;
access_log logs/domain2.access.log main;
# serve static files
location ~ ^/(images|javascript|js|css|flash|media|static)/ {
root /var/www/virtual/big.server.com/htdocs;
expires 30d;
}
# pass requests for dynamic content to rails/turbogears/zope, et al
location / {
proxy_pass http://127.0.0.1:8080;
}
}
}
foo
以下はwww.nginx.comで見つけた例です。
さまざまな処理のために、すべてのサーバーからブロックを抽出することが目標です。この特別なケースでは、私が興味を持っている2つのサーバーブロックがあります。
私の作業環境は非常に限られているので、sed / grep / awk / unixシステムコマンドのみを使用できます。ボアヘビもなく真珠もなく...
この構成ファイルの問題は次のとおりです。たぶん可能かもしれません。ブロックserver
自体には、いくつかのサブブロック(例えば、server { ... directive { ... }}
これを念頭に置いて、単にgrep -oP "server {.*?}"
。
あなたはgrep
これを行うことができますか?私はさまざまな正規表現を試しましたが、良い正規表現を見つけることができませんでした。現在私はバックスラッシュcat $FILE | tr -d "\n"
なしで上記のファイルを使用しています。私は似たようなことをしましたが、grep -oP "server\s{1,}{.*?({.*?}){0,}}"
私の味には適していません。
私が使用したいのは、後で読みやすくなく、コードを維持する方が簡単だgrep
と思うからです。しかし、使いやすい場合は例外を作成できます!awk
grep
awk
ありがとうございます:)
編集する:
出力は次のようになります。
"server { # php/fastcgi
listen 80;
server_name domain1.com www.domain1.com;
access_log logs/domain1.access.log main;
root html;
location ~ \.php$ {
fastcgi_pass 127.0.0.1:1025;
}
}",
"server { # simple reverse-proxy
listen 80;
server_name domain2.com www.domain2.com;
access_log logs/domain2.access.log main;
# serve static files
location ~ ^/(images|javascript|js|css|flash|media|static)/ {
root /var/www/virtual/big.server.com/htdocs;
expires 30d;
}
# pass requests for dynamic content to rails/turbogears/zope, et al
location / {
proxy_pass http://127.0.0.1:8080;
}
}"
これらのコマンドを使用してジョブを処理できますecho $OUTPUT | tr -d ... | grep -v ...
。可能であれば、各サーバーブロックを1行に抽出して結果を繰り返したい!
答え1
私が正しく理解した場合は、次のようなものを探しています。
$ awk '/server *{/{c=1; print;next} c&&/{/{c++} c&&/}/{c--} c' file
server { # php/fastcgi
listen 80;
server_name domain1.com www.domain1.com;
access_log logs/domain1.access.log main;
root html;
location ~ \.php$ {
fastcgi_pass 127.0.0.1:1025;
}
}
server { # simple reverse-proxy
listen 80;
server_name domain2.com www.domain2.com;
access_log logs/domain2.access.log main;
# serve static files
location ~ ^/(images|javascript|js|css|flash|media|static)/ {
root /var/www/virtual/big.server.com/htdocs;
expires 30d;
}
# pass requests for dynamic content to rails/turbogears/zope, et al
location / {
proxy_pass http://127.0.0.1:8080;
}
}
説明する
/server *{/{c=1; print; next;}
:この行がserver
ゼロ個以上の空白の後にaが一致すると、{
変数はc
1に設定されます。したがって、c
新しいブロックが見つかるたびにserver{
1になります。次に、print
その行をクリックしてそのnext
行にジャンプします。c&&/{/{c++}
:c
定義されていてゼロでない場合、行が一致すると1(){
ずつ増加します。これは次のように書くことができます。したがって、見つかった各ネストされたブロックについて値が増加します。c
c++
if(c && /{/){ c=c+1}
c
{
c&&/}/{c--}
:c
定義されていてゼロ以外の場合、この行が一致すると1}
ずつ減少します。c
これは次のように書くことができますif(c && /}/){c=c-1}
。これにより、入れ子になった各ブロックが閉じたときに値がc
削除されます。{}
c
:これはawk
トリックです。何かが true と評価された場合、基本的な操作はawk
現在の行を印刷することです。ここではc
、0(true)の代わりに定義されている場合に印刷されます。上記のコマンドはブロックc
内にあるserver
かどうかに応じて増減するため、次の行だけが印刷されます。