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생각하기 때문입니다 . 하지만 사용하기 쉬운 경우 예외를 만들 수 있습니다 !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

내가 올바르게 이해했다면 다음과 같은 것을 찾고 있습니다.

$ 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;}: 이 줄이 server0개 이상의 공백과 뒤에 a가 일치하면 {변수는 c1로 설정됩니다. 따라서 c새로운 블록을 찾을 때마다 server{1이 됩니다. 그런 다음 print해당 라인을 클릭하고 해당 next라인으로 점프합니다.
  • c&&/{/{c++}: c정의되어 있고 0이 아닌 경우 행이 일치하면 1( ) {씩 증가합니다 . 이는 다음과 같이 쓸 수 있습니다 . 따라서 발견된 각 중첩 블록에 대해 값이 증가합니다 .cc++if(c && /{/){ c=c+1}c{

  • c&&/}/{c--}: c정의되어 있고 0이 아닌 경우 이 줄이 일치하면 1 }씩 감소합니다 . c이는 다음과 같이 쓸 수 있습니다 if(c && /}/){c=c-1}. 이렇게 하면 중첩된 각 블록이 닫힐 때 값이 c삭제됩니다 .{}

  • c: 이건 awk트릭이에요. 어떤 것이 true로 평가되면 기본 작업은 awk현재 줄을 인쇄하는 것입니다. 여기서는 c0(true) 대신 정의된 경우 인쇄됩니다 . 위의 명령은 블록 c에 있는지 server여부에 따라 증가 및 감소하므로 다음 줄만 인쇄됩니다.

관련 정보