Nginx confからサーバーチャンクを抽出する

Nginx confからサーバーチャンクを抽出する

次の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と思うからです。しかし、使いやすい場合は例外を作成できます!awkgrepawk

ありがとうございます:)

編集する:

出力は次のようになります。

"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が一致すると、{変数はc1に設定されます。したがって、c新しいブロックが見つかるたびにserver{1になります。次に、printその行をクリックしてそのnext行にジャンプします。
  • c&&/{/{c++}c定義されていてゼロでない場合、行が一致すると1(){ずつ増加します。これは次のように書くことができます。したがって、見つかった各ネストされたブロックについて値が増加します。cc++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かどうかに応じて増減するため、次の行だけが印刷されます。

関連情報