diff --git a/README.md b/README.md index 0446fad..bbe701c 100644 --- a/README.md +++ b/README.md @@ -88,6 +88,7 @@ Usage: all configuration is done via commandline parameters or environment varia | --title-brand | BIRDLG_TITLE_BRAND | prefix of page titles in browser tabs (default "Bird-lg Go") | | --navbar-brand | BIRDLG_NAVBAR_BRAND | brand to show in the navigation bar (default "Bird-lg Go") | | --net-specific-mode | BIRDLG_NET_SPECIFIC_MODE | apply network-specific changes for some networks, use "dn42" for BIRD in dn42 network | +| --protocol-filter | BIRDLG_PROTOCOL_FILTER | protocol types to show in summary tables (comma separated list); defaults to all if not set | Example: the following command starts the frontend with 2 BIRD nodes, with domain name "gigsgigscloud.dn42.lantian.pub" and "hostdare.dn42.lantian.pub", and proxies are running on port 8000 on both nodes. diff --git a/frontend/main.go b/frontend/main.go index ab1b889..e6bd39c 100644 --- a/frontend/main.go +++ b/frontend/main.go @@ -19,6 +19,7 @@ type settingType struct { titleBrand string navBarBrand string telegramBotName string + protocolFilter []string } var setting settingType @@ -33,6 +34,7 @@ func main() { titleBrand: "Bird-lg Go", navBarBrand: "Bird-lg Go", telegramBotName: "", + protocolFilter: []string{}, } if env := os.Getenv("BIRDLG_SERVERS"); env != "" { @@ -69,6 +71,9 @@ func main() { if env := os.Getenv("BIRDLG_TELEGRAM_BOT_NAME"); env != "" { settingDefault.telegramBotName = env } + if env := os.Getenv("BIRDLG_PROTOCOL_FILTER"); env != "" { + settingDefault.protocolFilter = strings.Split(env, ",") + } serversPtr := flag.String("servers", strings.Join(settingDefault.servers, ","), "server name prefixes, separated by comma") domainPtr := flag.String("domain", settingDefault.domain, "server name domain suffixes") @@ -80,6 +85,8 @@ func main() { titleBrandPtr := flag.String("title-brand", settingDefault.titleBrand, "prefix of page titles in browser tabs") navBarBrandPtr := flag.String("navbar-brand", settingDefault.navBarBrand, "brand to show in the navigation bar") telegramBotNamePtr := flag.String("telegram-bot-name", settingDefault.telegramBotName, "telegram bot name (used to filter @bot commands)") + protocolFilterPtr := flag.String("protocol-filter", strings.Join(settingDefault.protocolFilter, ","), + "protocol types to show in summary tables (comma separated list); defaults to all if not set") flag.Parse() if *serversPtr == "" { @@ -89,6 +96,12 @@ func main() { servers := strings.Split(*serversPtr, ",") serversDisplay := strings.Split(*serversPtr, ",") + protocolFilter := []string{} + // strings.Split returns [""] for empty inputs; we want the list to remain empty in these cases + if len(*protocolFilterPtr) > 0 { + protocolFilter = strings.Split(*protocolFilterPtr, ",") + } + // Split server names of the form "DisplayName" for i, server := range servers { pos := strings.Index(server, "<") @@ -110,6 +123,7 @@ func main() { *titleBrandPtr, *navBarBrandPtr, *telegramBotNamePtr, + protocolFilter, } ImportTemplates() diff --git a/frontend/render.go b/frontend/render.go index dd581d3..ebb33b5 100644 --- a/frontend/render.go +++ b/frontend/render.go @@ -156,6 +156,18 @@ func summaryParse(data string, serverName string) (TemplateSummary, error) { } if len(lineSplitted) >= 4 { row.Proto = strings.TrimSpace(lineSplitted[3]) + // Filter away unwanted protocol types, if setting.protocolFilter is non-empty + found := false + for _, protocol := range setting.protocolFilter { + if strings.EqualFold(row.Proto, protocol) { + found = true + break + } + } + + if len(setting.protocolFilter) > 0 && !found { + continue + } } if len(lineSplitted) >= 6 { row.Table = strings.TrimSpace(lineSplitted[5]) diff --git a/frontend/render_test.go b/frontend/render_test.go index 95bc86b..b93edca 100644 --- a/frontend/render_test.go +++ b/frontend/render_test.go @@ -77,3 +77,36 @@ func TestSummaryTableXSS(t *testing.T) { t.Errorf("XSS injection succeeded: %s", result) } } + +func TestSummaryTableProtocolFilter(t *testing.T) { + initSettings() + setting.protocolFilter = []string{"Static", "Direct", "Babel"} + data := `BIRD 2.0.8 ready. +Name Proto Table State Since Info +static1 Static master4 up 2021-08-27 +static2 Static master6 up 2021-08-27 +device1 Device --- up 2021-08-27 +kernel1 Kernel master6 up 2021-08-27 +kernel2 Kernel master4 up 2021-08-27 +direct1 Direct --- up 2021-08-27 +int_babel Babel --- up 2021-08-27 ` + + result := summaryTable(data, "testserver") + expectedInclude := []string{"static1", "static2", "int_babel", "direct1"} + expectedExclude := []string{"device1", "kernel1", "kernel2"} + + for _, item := range expectedInclude { + if !strings.Contains(result, item) { + t.Errorf("Did not find expected %s in summary table output", result) + } + } + for _, item := range expectedExclude { + if strings.Contains(result, item) { + t.Errorf("Found unexpected %s in summary table output", result) + } + } + + t.Cleanup(func() { + setting.protocolFilter = []string{} + }) +}