从 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 上找到的一个例子。

目标是提取所有服务器的块以进行各种处理。在这种特殊情况下,有两个我感兴趣的服务器块。

因为我的工作环境非常有限,所以我只能使用 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

如果我理解正确的话,你正在寻找类似的东西:

$ 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, 0 个或多个空格,然后是 a {,则将变量设置c为 1。因此,c每次我们找到新server{块时,都会为 1。然后print该行并跳至该next行。
  • c&&/{/{c++}:如果c定义了 且不为 0,并且该行与 匹配{,则增加c1 ( c++)。这可以写成if(c && /{/){ c=c+1}。因此,对于找到的每个嵌套块, 的值c都会增加{

  • c&&/}/{c--}:如果c已定义且不为 0,并且此行匹配},则减c1。这可以写为if(c && /}/){c=c-1}。这确保了当每个嵌套块关闭时的值c会下降。{}

  • c: 这是一个awk技巧。当某些内容评估为 true 时,默认操作awk是打印当前行。在这里,它将打印是否c已定义而不是 0(真)。由于上面的命令c根据我们是否在一个server块中而递增和递减,这将导致仅打印您后面的行。

相关内容