Eu precisaria de uma maneira de extrair determinados dados e armazená-los em arquivo para auditoria e migração. Aqui estão meus dados.
network vlan internal-vlan {
description "internal-use"
interfaces {
1.1 { }
1.2 { }
}
MTU {
1500
}
}
network vlan external-vlan {
description "external-use"
interfaces {
2.1 { }
2.2 { }
}
MTU {
1500
}
}
Preciso extrair os nomes das vlans e suas descrições e detalhes da interface em formato de tabela, algo como abaixo. Se eu pudesse executar um script bash ou talvez perl nesse arquivo de dados, gerarei a saída em um arquivo csv e poderia abri-lo para auditoria.
Atualizando os dados de entrada e requisitos (desculpe),se você notar que há uma vlan fictícia que não possui MTU, quero que o valor das colunas MTU seja nenhum.
network vlan internal-vlan {
description "internal-use"
interfaces {
1.1 { }
1.2 { }
}
MTU {
1500
}
}
network vlan external-vlan {
description "external-use"
interfaces {
2.1 { }
2.2 { }
}
MTU {
1500
}
}
network vlan dummy-vlan {
description "dummy-use"
interfaces {
1.1 { }
}
}
network interface 1.1 {
Status {
UP
}
}
network interface 1.2 {
Status {
UP
}
}
network interface 2.1 {
Status {
DOWN
}
}
network interface 2.2 {
Status {
UP
}
}
Responder1
Na verdade, este é um uso perfeito paraTcl- esses dados são uma sintaxe Tcl válida, então só precisamos definir uma network
função e temos uma DSL válida:
#!/usr/bin/env tclsh
proc network {_ name data} {
set values [lmap val [concat $name [dict values $data]] {
format {"%s"} [regsub -all {\s+} [string trim $val] " "]
}]
puts [join $values ,]
}
puts {"Vlan","Description","Interfaces","MTU"}
source [lindex $argv 0]
então
$ tclsh parse.tcl datafile
"Vlan","Description","Interfaces","MTU"
"internal-vlan","internal-use","1.1 { } 1.2 { }","1500"
"external-vlan","external-use","2.1 { } 2.2 { }","1500"
Ou instaletcllibpara permitir que ele lide com a saída CSV:
#!/usr/bin/env tclsh
package require csv
proc network {_ name data} {
puts [csv::join [lmap val [concat $name [dict values $data]] {
regsub -all {\s+} [string trim $val] " "
}]]
}
puts [csv::join {Vlan Description Interfaces MTU}]
source [lindex $argv 0]
$ tclsh parse.tcl datafile
Vlan,Description,Interfaces,MTU
internal-vlan,internal-use,1.1 { } 1.2 { },1500
external-vlan,external-use,2.1 { } 2.2 { },1500
Para a versão mais antiga do Tcl 8.5, adicione este proc ao topo do programa:
proc lmap {varname list body} {
upvar 1 $varname element
set result {}
foreach element $list {
lappend result [uplevel 1 $body]
}
return $result
}
Para lidar com diferentes tipos de rede, bem como com o valor "nenhum", e também para garantir que os campos estejam na ordem correta:
proc network {type name data} {
if {$type ne "vlan"} {
return
}
set values [list $name]
foreach key {description interfaces MTU} {
set val [expr {[dict exists $data $key] ? [dict get $data $key] : "none"}]
lappend values [regsub -all {\s+} [string trim $val] " "]
}
puts [join $values ,]
}
Responder2
Se seus dados de entrada forem uniformes, você pode tentar algo como o seguinte:
#! /usr/bin/perl
use strict;
use warnings;
use feature qw{ say };
say 'Vlan,Description,Interfaces,MTU';
my @interfaces;
while (<>) {
if (/(?|network vlan (.*) \{|description (".*"))/) {
print "$1,";
} elsif (/interfaces/) {
@interfaces = ();
} elsif (/ *(.* \{ \})/) {
push @interfaces, "$1";
} elsif (/MTU \{/) {
my $next = <>;
say "@interfaces,$1" if $next =~ /^ *(.*)/;
}
}
Se for permitida mais variação nos dados (ou seja, as novas linhas são opcionais), você terá que escrever um analisador real.
Responder3
Desde que o formato do arquivo de configuração seja semelhante para todas as configurações:
Usandoestranho:
$ awk '
BEGIN { printf "Vlan,descripton,interfaces,MTU\n" }
{
if ($1 ~ /^network/ ) { printf $3","; }
if ( $1 ~ /description/ ) { printf $2","; }
if ( $1 ~ /^[0-9]\.[0-9]/ ) { printf $1$2$3" "; }
if ( $1 ~ /^[0-9]{2,}/ ) { printf ","$1"\n"; }
} ' input.conf
Vlan,descripton,interfaces,MTU
internal-vlan,"internal-use",1.1{} 1.2{} ,1500
external-vlan,"external-use",2.1{} 2.2{} ,1500