1
0
mirror of https://github.com/cmur2/openvpn-status-web.git synced 2025-09-28 21:52:05 +02:00

37 Commits

Author SHA1 Message Date
cn
ea7b3a947c gem: release 3.2.1 2022-01-18 23:18:00 +01:00
cn
eb2bb4d632 gem: initialize new status attributes with default values
- prevents 'undefined method each_with_index for nil:NilClass' errors
2022-01-18 23:17:42 +01:00
cn
2a9bacf196 gem: release 3.2.0 2022-01-18 17:09:03 +01:00
cn
39adc74fbc gem: Date::Error is not in Ruby 2.6 and older, use ArgumentError 2022-01-18 17:07:33 +01:00
cn
9ad59194bd gem: support date format used by OpenVPN 2.5
- smooth fallback
2022-01-18 17:07:33 +01:00
cn
528709c895 gem: parse any number of list headers in status-version 2/3 2022-01-18 17:07:33 +01:00
cn
39d2b3ce94 gem: refactor to prepare supporting flexible list headers 2022-01-18 17:07:33 +01:00
406ed867f7 gem: add Ruby 3.1 support 2022-01-07 09:31:53 +01:00
depfu[bot]
22d9797f55 gems: update rubocop-rspec to version 2.7.0 2021-12-28 00:29:52 +01:00
depfu[bot]
55aecedef7 gems: update rubocop to version 1.24.0 2021-12-24 16:21:27 +01:00
depfu[bot]
d401074524 gems: update rubocop to version 1.23.0 2021-11-17 13:53:41 +01:00
depfu[bot]
ff6572bdac gems: update rubocop-rspec to version 2.6.0 2021-11-09 09:24:58 +01:00
depfu[bot]
b4bf74240b gems: update rubocop to version 1.22.0 2021-09-30 12:09:33 +02:00
depfu[bot]
fb4b3ca461 gems: update solargraph to version 0.44.0 2021-09-28 11:42:56 +02:00
depfu[bot]
8a6722f981 gems: update rubocop-rspec to version 2.5.0 2021-09-23 08:35:49 +02:00
depfu[bot]
8249b01c95 gems: update rubocop to version 1.21.0 2021-09-14 22:17:23 +02:00
depfu[bot]
5d04255a95 gems: update bundler-audit to version 0.9.0.1 2021-09-02 00:53:22 +02:00
depfu[bot]
846d41d35d gems: update rubocop to version 1.20.0 2021-08-27 12:50:41 +02:00
depfu[bot]
ce0e80f35f gems: update rubocop to version 1.19.0 2021-08-13 15:43:58 +02:00
depfu[bot]
0ec3faf5b4 gems: update solargraph to version 0.43.0 2021-07-27 00:54:44 +02:00
depfu[bot]
10a4e757f3 gems: update rubocop to version 1.18.1 2021-07-01 09:58:43 +02:00
depfu[bot]
f95134c315 gems: update rubocop-rake to version 0.6.0 2021-06-29 22:24:25 +02:00
depfu[bot]
7da4cb910f gems: update rubocop to version 1.17.0 2021-06-16 16:13:00 +02:00
depfu[bot]
382daa34c6 gems: update solargraph to version 0.42.1 2021-06-12 15:25:00 +02:00
depfu[bot]
5338b30e32 gems: update rubocop-rspec to version 2.4.0 2021-06-10 23:59:30 +02:00
depfu[bot]
47ee407af2 gems: update solargraph to version 0.41.1 2021-06-02 19:59:48 +02:00
depfu[bot]
281750d6ea gems: update rubocop to version 1.16.0 2021-06-02 19:55:34 +02:00
depfu[bot]
42743a1786 gems: update rubocop to version 1.15.0 2021-05-18 12:05:21 +02:00
depfu[bot]
7c94192765 gems: update rubocop to version 1.14.0 2021-05-06 12:44:25 +02:00
depfu[bot]
a3a65f05bc gems: update rubocop-rspec to version 2.3.0 2021-04-29 15:09:21 +02:00
depfu[bot]
e87a4d86f6 gems: update rubocop to version 1.13.0 2021-04-21 12:29:53 +02:00
depfu[bot]
2145e183e3 gems: update rubocop to version 1.12.0 2021-03-25 17:22:09 +01:00
depfu[bot]
9e5aa1ff71 gems: update bundler-audit to version 0.8.0 2021-03-11 17:18:44 +01:00
depfu[bot]
a99ba901c9 gems: update rubocop to version 1.11.0 2021-03-02 12:31:48 +01:00
depfu[bot]
cc21291a5c gems: update rubocop to version 1.10.0 2021-02-16 16:59:41 +01:00
depfu[bot]
ba00407d55 gems: update rubocop-rspec to version 2.2.0 2021-02-04 10:11:43 +01:00
depfu[bot]
80bd4f8584 gems: update rubocop to version 1.9.0 2021-01-29 14:29:43 +01:00
15 changed files with 137 additions and 44 deletions

View File

@@ -20,6 +20,7 @@ jobs:
- '2.6'
- '2.7'
- '3.0'
- '3.1'
steps:
- uses: actions/checkout@v2
- name: Set up Ruby ${{ matrix.ruby-version }}

View File

@@ -6,6 +6,9 @@ AllCops:
TargetRubyVersion: '2.5'
NewCops: enable
Gemspec/RequireMFA:
Enabled: false
Layout/EmptyLineAfterGuardClause:
Enabled: false

View File

@@ -2,7 +2,7 @@ FROM alpine:3.13
EXPOSE 8080
ENV VERSION=3.1.0
ENV VERSION=3.2.1
RUN apk --no-cache add openssl ca-certificates && \
apk --no-cache add ruby ruby-etc ruby-webrick && \

View File

@@ -1,8 +1,8 @@
TITLE,OpenVPN 2.1_rc15 mipsel-unknown-linux-gnu [SSL] [LZO1] [EPOLL] built on Mar 27 2009
TIME,Sun Jan 1 23:42:00 2012,1238702330
HEADER,CLIENT_LIST,Common Name,Real Address,Virtual Address,Bytes Received,Bytes Sent,Connected Since,Connected Since (time_t)
CLIENT_LIST,foo,1.2.3.4:1234,11811160064,4194304,Sun Jan 1 23:42:00 2012,1238702330
CLIENT_LIST,bar,1.2.3.5:1235,512,2048,Sun Jan 1 23:42:00 2012,1238702330
CLIENT_LIST,foo,1.2.3.4:1234,192.168.66.2,11811160064,4194304,Sun Jan 1 23:42:00 2012,1238702330
CLIENT_LIST,bar,1.2.3.5:1235,2001:db8:0:0::1000,512,2048,Sun Jan 1 23:42:00 2012,1238702330
HEADER,ROUTING_TABLE,Virtual Address,Common Name,Real Address,Last Ref,Last Ref (time_t)
ROUTING_TABLE,192.168.0.0/24,foo,1.2.3.4:1234,Sun Jan 1 23:42:00 2012,1238702330
ROUTING_TABLE,192.168.66.2,bar,1.2.3.5:1235,Sun Jan 1 23:42:00 2012,1238702330

View File

@@ -1,8 +1,8 @@
TITLE OpenVPN 2.1_rc15 mipsel-unknown-linux-gnu [SSL] [LZO1] [EPOLL] built on Mar 27 2009
TIME Sun Jan 1 23:42:00 2012 1238702330
HEADER CLIENT_LIST Common Name Real Address Virtual Address Bytes Received Bytes Sent Connected Since Connected Since (time_t)
CLIENT_LIST foo 1.2.3.4:1234 11811160064 4194304 Sun Jan 1 23:42:00 2012 1238702330
CLIENT_LIST bar 1.2.3.5:1235 512 2048 Sun Jan 1 23:42:00 2012 1238702330
CLIENT_LIST foo 1.2.3.4:1234 192.168.66.2 11811160064 4194304 Sun Jan 1 23:42:00 2012 1238702330
CLIENT_LIST bar 1.2.3.5:1235 2001:db8:0:0::1000 512 2048 Sun Jan 1 23:42:00 2012 1238702330
HEADER ROUTING_TABLE Virtual Address Common Name Real Address Last Ref Last Ref (time_t)
ROUTING_TABLE 192.168.0.0/24 foo 1.2.3.4:1234 Sun Jan 1 23:42:00 2012 1238702330
ROUTING_TABLE 192.168.66.2 bar 1.2.3.5:1235 Sun Jan 1 23:42:00 2012 1238702330

8
examples/status2_5.v2 Normal file
View File

@@ -0,0 +1,8 @@
TITLE,OpenVPN 2.5.1 x86_64-pc-linux-gnu [SSL (OpenSSL)] [LZO] [LZ4] [EPOLL] [PKCS11] [MH/PKTINFO] [AEAD] built on May 14 2021
TIME,2012-01-01 23:42:00,1238702330
HEADER,CLIENT_LIST,Common Name,Real Address,Virtual Address,Virtual IPv6 Address,Bytes Received,Bytes Sent,Connected Since,Connected Since (time_t),Username,Client ID,Peer ID,Data Channel Cipher
CLIENT_LIST,foo,1.2.3.4:1234,192.168.66.2,,11811160064,4194304,2012-01-01 23:42:00,1238702330,UNDEF,1,0,AES-256-GCM
HEADER,ROUTING_TABLE,Virtual Address,Common Name,Real Address,Last Ref,Last Ref (time_t)
ROUTING_TABLE,192.168.66.2,foo,1.2.3.4:1234,2012-01-01 23:42:00,1238702330
GLOBAL_STATS,Max bcast/mcast queue length,42
END

View File

@@ -50,20 +50,36 @@ thead {
<div>
<table>
<thead>
<td class="first">Common Name</td>
<td class="middle">Real Address</td>
<td class="middle">Data Received</td>
<td class="middle">Data Sent</td>
<td class="last">Connected Since</td>
<% status.client_list_headers.each_with_index do |header,i| %>
<% if i == 0 %>
<td class="first">
<% elsif i == status.client_list_headers.size-1 %>
<td class="last">
<% else %>
<td class="middle">
<% end %>
<%= header %></td>
<% end %>
</thead>
<tbody>
<% status.client_list.each do |client| %>
<tr>
<td class="first"><%= client[0] %></td>
<td class="middle"><%= client[1] %></td>
<td class="middle"><%= client[2].to_i.as_bytes %></td>
<td class="middle"><%= client[3].to_i.as_bytes %></td>
<td class="last"><%= client[4].strftime('%-d.%-m.%Y %H:%M:%S') %></td>
<% status.client_list_headers.each_with_index do |header,i| %>
<% if i == 0 %>
<td class="first">
<% elsif i == status.client_list_headers.size-1 %>
<td class="last">
<% else %>
<td class="middle">
<% end %>
<% if header =~ /(Received|Sent)/ %>
<%= client[i].as_bytes %></td>
<% elsif client[i].is_a? DateTime %>
<%= client[i].strftime('%-d.%-m.%Y %H:%M:%S') %></td>
<% else %>
<%= client[i] %></td>
<% end %>
<% end %>
</tr>
<% end %>
</tbody>
@@ -74,18 +90,34 @@ thead {
<div>
<table>
<thead>
<td class="first">Virtual Address</td>
<td class="middle">Common Name</td>
<td class="middle">Real Address</td>
<td class="last">Last Ref</td>
<% status.routing_table_headers.each_with_index do |header,i| %>
<% if i == 0 %>
<td class="first">
<% elsif i == status.routing_table_headers.size-1 %>
<td class="last">
<% else %>
<td class="middle">
<% end %>
<%= header %></td>
<% end %>
</thead>
<tbody>
<% status.routing_table.each do |route| %>
<tr>
<td class="first"><%= route[0] %></td>
<td class="middle"><%= route[1] %></td>
<td class="middle"><%= route[2] %></td>
<td class="last"><%= route[3].strftime('%-d.%-m.%Y %H:%M:%S') %></td>
<% status.routing_table_headers.each_with_index do |header,i| %>
<% if i == 0 %>
<td class="first">
<% elsif i == status.routing_table_headers.size-1 %>
<td class="last">
<% else %>
<td class="middle">
<% end %>
<% if route[i].is_a? DateTime %>
<%= route[i].strftime('%-d.%-m.%Y %H:%M:%S') %></td>
<% else %>
<%= route[i] %></td>
<% end %>
<% end %>
</tr>
<% end %>
</tbody>

View File

@@ -5,29 +5,38 @@ module OpenVPNStatusWeb
class ModernStateless
def self.parse_status_log(text, sep)
status = Status.new
status.client_list_headers = []
status.client_list = []
status.routing_table_headers = []
status.routing_table = []
status.global_stats = []
text.lines.each do |line|
parts = line.strip.split(sep)
status.client_list << parse_client(parts[1..5]) if parts[0] == 'CLIENT_LIST'
status.routing_table << parse_route(parts[1..4]) if parts[0] == 'ROUTING_TABLE'
status.client_list_headers = parts[2..-1] if parts[0] == 'HEADER' && parts[1] == 'CLIENT_LIST'
status.client_list << parse_client(parts[1..-1], status.client_list_headers) if parts[0] == 'CLIENT_LIST'
status.routing_table_headers = parts[2..-1] if parts[0] == 'HEADER' && parts[1] == 'ROUTING_TABLE'
status.routing_table << parse_route(parts[1..-1], status.routing_table_headers) if parts[0] == 'ROUTING_TABLE'
status.global_stats << parse_global(parts[1..2]) if parts[0] == 'GLOBAL_STATS'
end
status
end
private_class_method def self.parse_client(client)
client[2] = client[2].to_i
client[3] = client[3].to_i
client[4] = DateTime.strptime(client[4], '%a %b %d %k:%M:%S %Y')
private_class_method def self.parse_client(client, headers)
headers.each_with_index do |header, i|
client[i] = parse_date(client[i]) if header.end_with?('Since')
client[i] = client[i].to_i if header.start_with?('Bytes')
end
client
end
private_class_method def self.parse_route(route)
route[3] = DateTime.strptime(route[3], '%a %b %d %k:%M:%S %Y')
private_class_method def self.parse_route(route, headers)
headers.each_with_index do |header, i|
route[i] = parse_date(route[i]) if header.end_with?('Last Ref')
end
route
end
@@ -35,6 +44,12 @@ module OpenVPNStatusWeb
global[1] = global[1].to_i
global
end
private_class_method def self.parse_date(date_string)
DateTime.strptime(date_string, '%a %b %d %k:%M:%S %Y')
rescue ArgumentError
DateTime.strptime(date_string, '%Y-%m-%d %k:%M:%S')
end
end
end
end

View File

@@ -26,7 +26,9 @@ module OpenVPNStatusWeb
end
status = Status.new
status.client_list_headers = ['Common Name', 'Real Address', 'Data Received', 'Data Sent', 'Connected Since']
status.client_list = client_list[2..-1].map { |client| parse_client(client) }
status.routing_table_headers = ['Virtual Address', 'Common Name', 'Real Address', 'Last Ref']
status.routing_table = routing_table[1..-1].map { |route| parse_route(route) }
status.global_stats = global_stats.map { |global| parse_global(global) }
status

View File

@@ -2,7 +2,9 @@
module OpenVPNStatusWeb
class Status
attr_accessor :client_list_headers
attr_accessor :client_list
attr_accessor :routing_table_headers
attr_accessor :routing_table
attr_accessor :global_stats
end

View File

@@ -1,5 +1,5 @@
# frozen_string_literal: true
module OpenVPNStatusWeb
VERSION = '3.1.0'
VERSION = '3.2.1'
end

View File

@@ -33,12 +33,12 @@ Gem::Specification.new do |s|
s.add_development_dependency 'better_errors'
s.add_development_dependency 'binding_of_caller'
s.add_development_dependency 'bundler'
s.add_development_dependency 'bundler-audit', '~> 0.7.0'
s.add_development_dependency 'bundler-audit', '~> 0.9.0'
s.add_development_dependency 'rack-test'
s.add_development_dependency 'rake'
s.add_development_dependency 'rspec'
s.add_development_dependency 'rubocop', '~> 1.8.1'
s.add_development_dependency 'rubocop-rake', '~> 0.5.1'
s.add_development_dependency 'rubocop-rspec', '~> 2.1.0'
s.add_development_dependency 'solargraph', ' ~> 0.40.0'
s.add_development_dependency 'rubocop', '~> 1.24.0'
s.add_development_dependency 'rubocop-rake', '~> 0.6.0'
s.add_development_dependency 'rubocop-rspec', '~> 2.7.0'
s.add_development_dependency 'solargraph', '~> 0.44.0'
end

View File

@@ -17,21 +17,30 @@ describe OpenVPNStatusWeb::Parser::ModernStateless do
expect(status.client_list.map { |client| client[1] }).to eq(['1.2.3.4:1234', '1.2.3.5:1235'])
end
it 'parses virtual addresses' do
expect(status.client_list.map { |client| client[2] }).to eq(['192.168.66.2', '2001:db8:0:0::1000'])
end
it 'parses received bytes' do
expect(status.client_list.map { |client| client[2] }).to eq([11_811_160_064, 512])
expect(status.client_list.map { |client| client[3] }).to eq([11_811_160_064, 512])
end
it 'parses sent bytes' do
expect(status.client_list.map { |client| client[3] }).to eq([4_194_304, 2048])
expect(status.client_list.map { |client| client[4] }).to eq([4_194_304, 2048])
end
it 'parses connected since date' do
expect(status.client_list.map { |client| client[4] }).to eq(
expect(status.client_list.map { |client| client[5] }).to eq(
[
DateTime.new(2012, 1, 1, 23, 42, 0), DateTime.new(2012, 1, 1, 23, 42, 0)
DateTime.new(2012, 1, 1, 23, 42, 0),
DateTime.new(2012, 1, 1, 23, 42, 0)
]
)
end
it 'has the same number of headers' do
expect(status.client_list[0].size).to eq(status.client_list_headers.size)
end
end
context 'with routing table' do
@@ -55,6 +64,10 @@ describe OpenVPNStatusWeb::Parser::ModernStateless do
]
)
end
it 'has the same number of headers' do
expect(status.routing_table[0].size).to eq(status.routing_table_headers.size)
end
end
it 'parses global stats' do
@@ -63,4 +76,8 @@ describe OpenVPNStatusWeb::Parser::ModernStateless do
end
end
end
it 'parses status-version 2 of OpenVPN 2.5' do
expect(status_2_5_v2).not_to be_nil
end
end

View File

@@ -31,6 +31,10 @@ describe OpenVPNStatusWeb::Parser::V1 do
]
)
end
it 'has the same number of headers' do
expect(status.client_list[0].length).to eq(status.client_list_headers.length)
end
end
context 'with routing table' do
@@ -54,6 +58,10 @@ describe OpenVPNStatusWeb::Parser::V1 do
]
)
end
it 'has the same number of headers' do
expect(status.routing_table[0].length).to eq(status.routing_table_headers.length)
end
end
it 'parses global stats' do

View File

@@ -7,16 +7,21 @@ require 'rack/test'
require 'openvpn-status-web'
def status_v1
text = File.open('examples/status.v1', 'rb', &:read)
text = File.binread('examples/status.v1')
OpenVPNStatusWeb::Parser::V1.new.parse_status_log text
end
def status_v2
text = File.open('examples/status.v2', 'rb', &:read)
text = File.binread('examples/status.v2')
OpenVPNStatusWeb::Parser::V2.new.parse_status_log text
end
def status_2_5_v2
text = File.binread('examples/status2_5.v2')
OpenVPNStatusWeb::Parser::V2.new.parse_status_log text
end
def status_v3
text = File.open('examples/status.v3', 'rb', &:read)
text = File.binread('examples/status.v3')
OpenVPNStatusWeb::Parser::V3.new.parse_status_log text
end