Compare commits

...

170 Commits

Author SHA1 Message Date
Christian Nicolai cc0de6f563
docs: remove Depfu badge 2024-05-16 21:42:48 +02:00
Christian Nicolai 9933baed66
gems: update Renovate configuration 2024-05-16 21:39:08 +02:00
dependabot[bot] 6f106dd8ea gems: update rubocop-rspec requirement from ~> 2.26.1 to ~> 2.29.2
Updates the requirements on [rubocop-rspec](https://github.com/rubocop/rubocop-rspec) to permit the latest version.
- [Release notes](https://github.com/rubocop/rubocop-rspec/releases)
- [Changelog](https://github.com/rubocop/rubocop-rspec/blob/master/CHANGELOG.md)
- [Commits](https://github.com/rubocop/rubocop-rspec/compare/v2.26.1...v2.29.2)

---
updated-dependencies:
- dependency-name: rubocop-rspec
  dependency-type: direct:development
...

Signed-off-by: dependabot[bot] <support@github.com>
2024-05-16 21:37:40 +02:00
dependabot[bot] e17c5bef2d gems: update rubocop requirement from ~> 1.61.0 to ~> 1.63.5
Updates the requirements on [rubocop](https://github.com/rubocop/rubocop) to permit the latest version.
- [Release notes](https://github.com/rubocop/rubocop/releases)
- [Changelog](https://github.com/rubocop/rubocop/blob/master/CHANGELOG.md)
- [Commits](https://github.com/rubocop/rubocop/compare/v1.61.0...v1.63.5)

---
updated-dependencies:
- dependency-name: rubocop
  dependency-type: direct:development
...

Signed-off-by: dependabot[bot] <support@github.com>
2024-05-16 21:37:20 +02:00
Christian Nicolai a65bf15ab3
gems: disable Depfu for Dependabot since PR creation fails 2024-05-16 21:34:21 +02:00
renovate[bot] 85791c1b74 docker: update alpine Docker tag to v3.19 2024-03-03 11:31:17 +01:00
renovate[bot] 5162bdaa8a ci: update actions/checkout action to v4 2024-03-03 11:31:01 +01:00
depfu[bot] 4ead5d4972 gems: update rubocop to version 1.61.0 2024-03-03 11:30:47 +01:00
cn d75cf2aade gem: add Ruby 3.3 support, drop Ruby 2.7 due to EOL 2024-03-03 11:28:40 +01:00
Christian Nicolai 88b41c8998
project: add renovate config 2024-03-03 11:13:31 +01:00
depfu[bot] 6d3d6e8242 gems: update rubocop to version 1.60.0 2024-01-16 11:27:22 +01:00
depfu[bot] 45e9fb95cb gems: update rubocop-rspec to version 2.26.1 2024-01-06 23:17:28 +01:00
depfu[bot] 8d9aac329f gems: update rubocop to version 1.59.0 2023-12-12 15:16:50 +01:00
depfu[bot] d2f0de2d8a gems: update solargraph to version 0.50.0 2023-12-09 10:17:28 +01:00
depfu[bot] a330744dde gems: update rubocop to version 1.58.0 2023-12-02 21:48:04 +01:00
depfu[bot] 932c733203 gems: update rubocop-rspec to version 2.25.0 2023-10-28 13:35:16 +02:00
depfu[bot] c640431eda gems: update rubocop to version 1.57.0 2023-10-12 15:42:19 +02:00
cn b3c776a09d gem: fix rubocop warning by disabling redundant cop 2023-09-28 12:14:32 +02:00
depfu[bot] 38375fbdf7 gems: update rubocop-rspec to version 2.24.0 2023-09-09 17:22:16 +02:00
depfu[bot] fc25d49719 gems: update rubocop to version 1.56.0 2023-08-10 11:08:13 +02:00
depfu[bot] 0322f9c97d gems: update rubocop-rspec to version 2.23.0 2023-07-31 19:28:02 +02:00
depfu[bot] 35d78740a7 gems: update rubocop to version 1.55.0 2023-07-26 20:14:14 +02:00
depfu[bot] 097f3ea988 gems: update rubocop to version 1.54.0 2023-07-02 13:22:29 +02:00
depfu[bot] cb8698b80f gems: update rubocop to version 1.53.0 2023-06-24 16:23:20 +02:00
depfu[bot] 90ec76a503 gems: update rubocop to version 1.52.0 2023-06-04 12:02:04 +02:00
depfu[bot] 2f02388799 gems: update rubocop to version 1.51.0 2023-05-14 12:48:10 +02:00
depfu[bot] 9d80258a8f gems: update rubocop-rspec to version 2.22.0 2023-05-07 21:46:11 +02:00
depfu[bot] a189922123 gems: update rubocop-rspec to version 2.21.0 2023-05-06 14:59:28 +02:00
depfu[bot] 9a795fe2f9 gems: update rubocop-rspec to version 2.20.0 2023-04-19 13:17:43 +02:00
depfu[bot] 4df29aded8 gems: update rubocop to version 1.50.0 2023-04-12 11:55:17 +02:00
depfu[bot] 0182d784ce gems: update solargraph to version 0.49.0 2023-04-11 09:38:53 +02:00
depfu[bot] b3a5bbb244 gems: update rubocop to version 1.49.0 2023-04-04 11:59:50 +02:00
cn 184e0c2a9b gem: fix rubocop warning 2023-03-07 22:56:27 +01:00
depfu[bot] 91a414c9c4 gems: update rubocop-rspec to version 2.19.0 2023-03-07 22:56:27 +01:00
depfu[bot] b8f7390ddc gems: update rubocop to version 1.48.0 2023-03-07 13:59:17 +01:00
depfu[bot] 3d6bb9c662 gems: update rubocop to version 1.47.0 2023-03-02 15:12:23 +01:00
depfu[bot] 89ac867f6e gems: update rubocop to version 1.46.0 2023-02-23 22:31:45 +01:00
Christian Nicolai 8e2ac7e677
docs: add example docker-compose usage 2023-02-16 16:00:18 +01:00
depfu[bot] 4a5e258a86 gems: update rubocop to version 1.45.1 2023-02-09 20:58:30 +01:00
depfu[bot] 1237f144d5 gems: update rubocop to version 1.44.0 2023-01-24 15:20:52 +01:00
cn 65a65081ae gem: release 3.4.0 2023-01-20 14:57:35 +01:00
cn 194137adc8 gem: add Ruby 3.2 support 2023-01-20 14:55:36 +01:00
cn ee06f65e81 gems: add constraint for rackup version 2023-01-19 16:36:35 +01:00
depfu[bot] 075922e484 gems: update rubocop-rspec to version 2.18.0 2023-01-17 14:01:47 +01:00
depfu[bot] 9ece1e125b gems: update rubocop-rspec to version 2.17.0 2023-01-14 20:15:19 +01:00
depfu[bot] ba50185f44 gems: update rubocop to version 1.43.0 2023-01-11 13:54:46 +01:00
depfu[bot] d57e8b7d2e gems: update rubocop to version 1.42.0 2023-01-02 19:30:53 +01:00
depfu[bot] 2d77c0efee gems: update rubocop to version 1.41.0 2022-12-21 14:30:56 +01:00
depfu[bot] 4f6725825a gems: update solargraph to version 0.48.0 2022-12-20 12:28:50 +01:00
depfu[bot] 8af23da209 gems: update rubocop-rspec to version 2.16.0 2022-12-14 14:35:12 +01:00
depfu[bot] 78d4c3c8cb gems: update rubocop to version 1.40.0 2022-12-09 22:25:28 +01:00
depfu[bot] 39f570ec48 gems: update rubocop to version 1.39.0 2022-11-15 10:20:02 +01:00
depfu[bot] 1eb6202a8f gems: update rubocop-rspec to version 2.15.0 2022-11-04 23:39:42 +01:00
depfu[bot] a329966bc5 gems: update rubocop to version 1.38.0 2022-11-02 10:59:06 +01:00
depfu[bot] 575516bd27 gems: update rubocop-rspec to version 2.14.1 2022-10-25 11:53:40 +02:00
depfu[bot] af253c61f0 gems: update rubocop to version 1.37.0 2022-10-21 12:49:20 +02:00
depfu[bot] a804f842df gems: update solargraph to version 0.47.0 2022-09-26 21:23:32 +02:00
Christian Nicolai 648b4d94b6 gem: release 3.3.0 2022-09-16 06:48:21 +02:00
depfu[bot] 117f70e2e3 gems: update rack to version 3.0.0 2022-09-15 21:41:54 +02:00
depfu[bot] 1e0444a71d gems: update rubocop-rspec to version 2.13.1 2022-09-13 21:03:28 +02:00
depfu[bot] 20c6c9176a gems: update rubocop to version 1.36.0 2022-09-02 12:36:16 +02:00
depfu[bot] a0ad379b00 gems: update solargraph to version 0.46.0 2022-08-23 15:10:14 +02:00
depfu[bot] d8f4931657 gems: update rubocop to version 1.35.0 2022-08-13 19:24:55 +02:00
depfu[bot] 6676f47c97 gems: update rubocop to version 1.34.1 2022-08-10 16:35:06 +02:00
depfu[bot] 811b928403 gems: update rubocop to version 1.33.0 2022-08-05 14:09:34 +02:00
depfu[bot] f4a48b8ce0 gems: update rubocop to version 1.32.0 2022-07-22 15:49:47 +02:00
depfu[bot] f985ec1c12 gems: update rubocop-rspec to version 2.12.0 2022-07-03 14:28:49 +02:00
depfu[bot] 3d10b7d128 gems: update rubocop to version 1.31.0 2022-06-28 11:02:32 +02:00
depfu[bot] 27876c7f12 gems: update solargraph to version 0.45.0 2022-05-28 17:03:37 +02:00
depfu[bot] 9d584ceb71 gems: update rubocop to version 1.30.0 2022-05-28 17:01:23 +02:00
depfu[bot] 1996f1d9b9 gems: update rubocop-rspec to version 2.11.1 2022-05-22 19:08:44 +02:00
depfu[bot] f342b07495 gems: update rubocop to version 1.29.0 2022-05-08 16:59:25 +02:00
cn 557a328336 gem: drop support for Rubies < 2.7 2022-05-08 16:56:46 +02:00
cn 057d243db8 gem: make intention of ENV.fetch clear with explicit default value 2022-04-22 21:35:26 +02:00
depfu[bot] e3a06fdf6c gems: update rubocop to version 1.28.1 2022-04-22 21:35:26 +02:00
depfu[bot] bfe39adaf8 gems: update rubocop-rspec to version 2.10.0 2022-04-20 10:50:18 +02:00
depfu[bot] 9159a53cce gems: update rubocop to version 1.27.0 2022-04-09 11:59:16 +02:00
depfu[bot] 51825eb2da gems: update rubocop to version 1.26.0 2022-03-10 20:44:34 +01:00
depfu[bot] 280e2b9ae1 gems: update rubocop-rspec to version 2.9.0 2022-03-01 14:07:18 +01:00
depfu[bot] 6d56db855f gems: update rubocop-rspec to version 2.8.0 2022-01-25 13:36:48 +01:00
depfu[bot] 0b782b7dd3 gems: update rubocop to version 1.25.0 2022-01-19 11:38:45 +01:00
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
Christian Nicolai 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
cn e4034073a6 release: 3.1.0
- Ruby 3.0
- move from Travis CI to Github Actions
- dependency updates
2021-01-18 20:56:53 +01:00
depfu[bot] f4d17809a2 gems: update rubocop to version 1.8.1 2021-01-18 20:51:39 +01:00
cn 08b50d5974 docs: update Dockerfile to Alpine 3.13 2021-01-18 15:00:41 +01:00
depfu[bot] f287906240 gems: update rubocop to version 1.7.0 2020-12-28 21:42:19 +01:00
cn 27306b831f ci: use bundler cache to save resources
- see https://github.com/ruby/setup-ruby#caching-bundle-install-automatically
2020-12-26 15:17:06 +01:00
cn 1e12701fc3 gems: include rubocop-rspec and fix linting 2020-12-25 18:17:01 +01:00
cn 597b619b0b gems: include rubocop-rake and fix linting 2020-12-25 18:16:27 +01:00
cn 61f7cbd76c gem: add Ruby 3.0 support 2020-12-25 17:52:10 +01:00
cn 2628c08975 gems: revert back to upstream solargraph now with rubocop 1.0 compat
See https://github.com/castwide/solargraph/issues/380 and previous commit 0ffe369ef1
2020-12-25 17:51:25 +01:00
depfu[bot] 7343081406 gems: update rubocop to version 1.6.1 2020-12-11 12:15:59 +01:00
cn ec4d028c03 ci: drop Travis CI
- end of an era
2020-12-10 11:03:44 +01:00
cn 12b5153604 gem: refactor Rakefile solargraph tasks, include in default task 2020-12-04 09:17:11 +01:00
cn 7c0722e875 ci: use Github Actions in parallel to Travis CI
- this adds a new workflow for Github Actions that mirrors what the existing Travis CI workflow tests
- Travis CI might become unfriendly to opensource soonish so migration might be necessary
2020-12-04 09:14:32 +01:00
depfu[bot] afd7f4fa98 gems: update rubocop to version 1.5.1 2020-12-03 21:23:22 +01:00
depfu[bot] 63272da3e3 gems: update rubocop to version 1.4.1 2020-11-24 20:56:04 +01:00
depfu[bot] ae7fc1137b gems: update rubocop to version 1.3.0 2020-11-13 13:09:57 +01:00
depfu[bot] 1f69dabf5f gems: update rubocop to version 1.2.0 2020-11-06 12:28:33 +01:00
cn 0e9c9fea68 docs: use travis-ci.com links 2020-10-31 00:05:28 +01:00
cn 0ffe369ef1 gems: use solargraph fork until gem metadata allows rubocop 1.0 compat
See https://github.com/castwide/solargraph/issues/380
2020-10-30 23:59:05 +01:00
depfu[bot] c2cac319ae gems: update rubocop to version 1.1.0 2020-10-30 22:56:37 +01:00
depfu[bot] 5f343d7873 gems: update rubocop to version 1.0.0 2020-10-22 15:56:12 +02:00
depfu[bot] 0a45726857 gems: update rubocop to version 0.93.0 2020-10-10 16:46:18 +02:00
cn 109b27fecc release: 3.0.1
- use webrick gem which contains fixes against [CVE-2020-25613](https://www.ruby-lang.org/en/news/2020/09/29/http-request-smuggling-cve-2020-25613/)
2020-10-03 11:04:48 +02:00
cn b44517ec21 gems: update webrick to version 1.6.1
- explicitly use webrick gem version with patch against CVE-2020-25613
- https://www.ruby-lang.org/en/news/2020/09/29/http-request-smuggling-cve-2020-25613/
2020-10-03 11:03:01 +02:00
depfu[bot] 520da15739 gems: update rubocop to version 0.92.0 2020-09-26 12:40:07 +02:00
depfu[bot] 9b3a13abef gems: update rubocop to version 0.91.0 2020-09-16 09:39:25 +02:00
depfu[bot] b81d61b8d7
gems: update rubocop to version 0.90.0 (#16)
Update rubocop to version 0.90.0 (#16)

Co-authored-by: depfu[bot] <23717796+depfu[bot]@users.noreply.github.com>
2020-09-02 10:35:06 +02:00
cn 147a905bfc gem: fix solargraph typecheck problem in CI + remove downpin 2020-08-14 17:39:02 +02:00
cn eeed0fa089 gems: downpin solargraph
- releases newer than 0.39.12 fail on Travis CI although they work locally with RVM Ruby 2.7.1
2020-08-14 14:57:08 +02:00
cn 586ae8ae0f gem: add editorconfig 2020-08-14 14:07:47 +02:00
depfu[bot] 075ec1e548
gems: update rubocop to version 0.89.0
Update rubocop to version 0.89.0 (#15)
Co-authored-by: depfu[bot] <23717796+depfu[bot]@users.noreply.github.com>
2020-08-07 08:32:29 +02:00
cn 64ec5e1e56 release: 3.0.0
- drop Ruby 2.3 and 2.4 compatibility
- add Dockerfile
2020-07-14 20:08:06 +02:00
depfu[bot] 7597c17e93
gem: upgrade rubocop to version 0.88.0
Update rubocop to version 0.88.0 (#13)

Co-authored-by: depfu[bot] <23717796+depfu[bot]@users.noreply.github.com>
Co-authored-by: Christian Nicolai <cn@mycrobase.de>
2020-07-14 20:06:06 +02:00
Christian Nicolai ac647a4f21
gem: bump minimum required Ruby version to Ruby 2.5 (#14) 2020-07-14 19:57:25 +02:00
cn 78f45f5080 docs: add example Dockerfile 2020-07-13 10:39:30 +02:00
Christian Nicolai 9aeb1a9ad5
docs: fix typos 2020-07-13 10:33:54 +02:00
Christian Nicolai 88c22a6504
docs: fix Travis CI build badge 2020-07-10 23:09:34 +02:00
depfu[bot] 289535d753
gems: update bundler-audit to version 0.7.0.1
(#9)
Co-authored-by: depfu[bot] <23717796+depfu[bot]@users.noreply.github.com>
2020-06-14 12:42:05 +02:00
depfu[bot] 8d42bdafc1
gems: update rubocop to version 0.81.0
(#3)
Co-authored-by: depfu[bot] <23717796+depfu[bot]@users.noreply.github.com>
2020-04-02 13:51:52 +02:00
cn f8666fdfc2 release: 2.1.0 2020-03-07 01:32:22 +01:00
cn 41f5d4ed62 gem: adopt frozen string literals 2020-03-07 01:30:57 +01:00
cn 92ee01e5de gem: port fixes from Sorbet from dyndnsd 2020-03-07 01:29:12 +01:00
cn bf3ba8f7cd gem: refactor gemspec, exclude tests from gem, move binaries to ./exe
- based on recommendations of https://piotrmurach.com/articles/writing-a-ruby-gem-specification/
2020-03-07 01:28:17 +01:00
cn 140c60c753 gem: add rubocop and fix style 2020-03-02 01:57:58 +01:00
cn d959e7fe62 gem: fix gemspec using rubocop hints
- especially the wrong homepage URL
2020-03-01 22:13:23 +01:00
cn 4cde78fe96 gem: add solargraph support 2020-03-01 22:12:38 +01:00
Christian Nicolai 23dd4aa63f
docs: add depfu to README 2020-02-28 21:40:09 +01:00
Christian Nicolai 1547db7bc6
travis: fix badge image URL 2020-02-28 13:31:36 +01:00
Christian Nicolai b1aa38059f travis: add Ruby 2.7 2020-02-28 13:30:49 +01:00
Christian Nicolai 17054794da
travis: fix build config validation problems
- https://docs.travis-ci.com/user/reference/overview/#deprecated-virtualization-environments
2020-02-28 13:25:02 +01:00
cn ab81c8975e gem: use bundler-audit 2019-12-18 20:20:14 +01:00
cn 1804693c15 travis: add Ruby 2.6 2019-01-04 18:32:32 +01:00
31 changed files with 607 additions and 174 deletions

9
.editorconfig Normal file
View File

@ -0,0 +1,9 @@
root = true
[*]
indent_style = space
indent_size = 2
end_of_line = lf
charset = utf-8
trim_trailing_whitespace = true
insert_final_newline = true

13
.github/dependabot.yml vendored Normal file
View File

@ -0,0 +1,13 @@
---
version: 2
updates:
- package-ecosystem: "bundler"
directory: "/"
schedule:
interval: "weekly"
commit-message:
prefix: "gems"
labels: ["dependabot"]
open-pull-requests-limit: 10
pull-request-branch-name:
separator: "-"

42
.github/renovate.json5 vendored Normal file
View File

@ -0,0 +1,42 @@
{
extends: [
"config:recommended",
":dependencyDashboard",
":prHourlyLimitNone",
":prConcurrentLimitNone",
":label(dependency-upgrade)",
],
schedule: ["before 8am on thursday"],
branchPrefix: "renovate-",
dependencyDashboardHeader: "View repository job log [here](https://app.renovatebot.com/dashboard#github/cmur2/dyndnsd).",
separateMinorPatch: true,
commitMessagePrefix: "project: ",
commitMessageAction: "update",
commitMessageTopic: "{{depName}}",
commitMessageExtra: "to {{#if isSingleVersion}}v{{{newVersion}}}{{else}}{{{newValue}}}{{/if}}",
packageRules: [
// Ruby dependencies are managed by dependabot (previously depfu, until PR creation failed)
{
matchManagers: ["bundler"],
enabled: false,
},
// Commit message formats
{
matchDatasources: ["docker"],
commitMessagePrefix: "docker: ",
},
{
matchManagers: ["github-actions"],
commitMessagePrefix: "ci: ",
},
],
customManagers: [
{
customType: "regex",
fileMatch: ["\.rb$", "^Rakefile$"],
matchStrings: [
"renovate: datasource=(?<datasource>.*?) depName=(?<depName>.*?)\\s.*_version = '(?<currentValue>.*)'\\s"
]
},
],
}

32
.github/workflows/ci.yml vendored Normal file
View File

@ -0,0 +1,32 @@
---
name: ci
on:
push:
branches: [master]
pull_request:
branches: [master]
workflow_dispatch:
schedule:
- cron: '35 4 * * 4' # weekly on thursday morning
jobs:
build:
runs-on: ubuntu-latest
strategy:
matrix:
ruby-version:
- '3.0'
- '3.1'
- '3.2'
- '3.3'
steps:
- uses: actions/checkout@v4
- name: Set up Ruby ${{ matrix.ruby-version }}
uses: ruby/setup-ruby@v1
with:
ruby-version: ${{ matrix.ruby-version }}
bundler-cache: true # runs 'bundle install' and caches installed gems automatically
- name: Lint and Test
run: |
bundle exec rake ci

1
.gitignore vendored
View File

@ -1,3 +1,4 @@
.DS_Store
*.lock
pkg/*
.yardoc

111
.rubocop.yml Normal file
View File

@ -0,0 +1,111 @@
require:
- rubocop-rake
- rubocop-rspec
AllCops:
TargetRubyVersion: 3.0'
NewCops: enable
Gemspec/DevelopmentDependencies:
EnforcedStyle: gemspec
Gemspec/RequireMFA:
Enabled: false
Layout/EmptyLineAfterGuardClause:
Enabled: false
# allows nicer usage of private_class_method
Layout/EmptyLinesAroundArguments:
Enabled: false
Layout/HashAlignment:
Enabled: false
Layout/LeadingEmptyLines:
Enabled: false
Layout/LineLength:
Max: 200
Layout/SpaceInsideHashLiteralBraces:
Enabled: false
Metrics/AbcSize:
Enabled: false
Metrics/BlockLength:
Enabled: false
Metrics/ClassLength:
Enabled: false
Metrics/CyclomaticComplexity:
Enabled: false
Metrics/MethodLength:
Enabled: false
Metrics/PerceivedComplexity:
Enabled: false
Naming/MethodParameterName:
Enabled: false
Naming/MemoizedInstanceVariableName:
Enabled: false
Style/AccessorGrouping:
Enabled: false
Style/ConditionalAssignment:
Enabled: false
Style/Documentation:
Enabled: false
Style/FormatStringToken:
Enabled: false
Style/GuardClause:
Enabled: false
Style/HashEachMethods:
Enabled: true
Style/HashTransformKeys:
Enabled: true
Style/HashTransformValues:
Enabled: true
Style/IdenticalConditionalBranches:
Enabled: false
Style/InverseMethods:
Enabled: false
Style/NegatedIf:
Enabled: false
Style/RescueModifier:
Enabled: false
Style/Semicolon:
AllowAsExpressionSeparator: true
Style/SymbolArray:
Enabled: false
RSpec/ExampleLength:
Max: 10
RSpec/FilePath:
Enabled: false
RSpec/SpecFilePathFormat:
CustomTransform:
OpenVPNStatusWeb: openvpn-status-web
RSpec/MultipleExpectations:
Max: 5

16
.solargraph.yml Normal file
View File

@ -0,0 +1,16 @@
---
include:
- "**/*.rb"
- "bin/openvpn-status-web"
exclude:
- spec/**/*
- test/**/*
- vendor/**/*
- ".bundle/**/*"
require: []
domains: []
reporters:
- rubocop
- require_not_found
require_paths: []
max_files: 5000

View File

@ -1,7 +0,0 @@
---
sudo: false
language: ruby
rvm:
- 2.5
- 2.4
- 2.3

View File

@ -1,3 +1,5 @@
# frozen_string_literal: true
source 'https://rubygems.org'
gemspec

View File

@ -1,6 +1,6 @@
# openvpn-status-web
[![Build Status](https://travis-ci.org/cmur2/openvpn-status-web.png)](https://travis-ci.org/cmur2/openvpn-status-web)
![ci](https://github.com/cmur2/openvpn-status-web/workflows/ci/badge.svg)
## Description
@ -57,10 +57,36 @@ For more information about OpenVPN status file and version, see their [man page]
If the information exposed is important to you serve it via the VPN or use a webserver as a proxy to handle SSL and/or HTTP authentication.
### Init scripts
### Startup
The [Debian 6 init.d script](init.d/debian-6-openvpn-status-web) assumes that openvpn-status-web is installed into the system ruby (no RVM support) and the config.yaml is at /opt/openvpn-status-web/config.yaml. Modify to your needs.
There is a [Dockerfile](docs/Dockerfile) that can be used to build a Docker image for running openvpn-status-web.
This can for example be used with `docker-compose` via:
```yaml
version: "2.4"
services:
openvpn-status-web:
image: your-selfbuilt-docker-image
user: root # needed since the default status files are chmod 600
volumes:
- /path/to/host/config.yml:/etc/openvpn-status-web/config.yml:ro
- /run/openvpn-server:/run/openvpn-server
ports:
- "8080:8080"
```
The `/path/to/host/config.yml` could be:
```yaml
host: "0.0.0.0"
port: "8080"
vpns:
my-cool-vpn: # the following depends on your setup
version: 2
status_file: "/run/openvpn-server/status-my-cool-vpn.log"
```
## License
openvpn-statsu-web is licensed under the Apache License, Version 2.0. See LICENSE for more information.
openvpn-status-web is licensed under the Apache License, Version 2.0. See LICENSE for more information.

View File

@ -1,6 +1,27 @@
# frozen_string_literal: true
require 'bundler/gem_tasks'
require 'rspec/core/rake_task'
require 'rubocop/rake_task'
require 'bundler/audit/task'
RSpec::Core::RakeTask.new(:spec)
RuboCop::RakeTask.new
Bundler::Audit::Task.new
task :default => :spec
desc 'Run experimental solargraph type checker'
task :solargraph do
sh 'solargraph typecheck'
end
namespace :solargraph do
desc 'Should be run by developer once to prepare initial solargraph usage (fill caches etc.)'
task :init do
sh 'solargraph download-core'
end
end
task default: [:rubocop, :spec, 'bundle:audit', :solargraph]
desc 'Run all tasks desired for CI'
task ci: ['solargraph:init', :default]

15
docs/Dockerfile Normal file
View File

@ -0,0 +1,15 @@
FROM alpine:3.19
EXPOSE 8080
ENV VERSION=3.4.0
RUN apk --no-cache add openssl ca-certificates && \
apk --no-cache add ruby ruby-etc ruby-webrick && \
apk --no-cache add --virtual .build-deps ruby-dev build-base tzdata && \
gem install --no-document openvpn-status-web -v ${VERSION} && \
# set timezone to Berlin
cp /usr/share/zoneinfo/Europe/Berlin /etc/localtime && \
apk del .build-deps
ENTRYPOINT ["openvpn-status-web", "/etc/openvpn-status-web/config.yml"]

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

@ -1,3 +1,5 @@
#!/usr/bin/env ruby
# frozen_string_literal: true
require 'openvpn-status-web'

78
lib/openvpn-status-web.rb Normal file → Executable file
View File

@ -1,4 +1,5 @@
#!/usr/bin/env ruby
# frozen_string_literal: true
require 'date'
require 'etc'
@ -6,9 +7,10 @@ require 'logger'
require 'ipaddr'
require 'yaml'
require 'rack'
require 'rackup'
require 'erb'
require 'metriks'
require 'better_errors' if ENV['RACK_ENV'] == "development"
require 'better_errors' if ENV.fetch('RACK_ENV', nil) == 'development'
require 'openvpn-status-web/status'
require 'openvpn-status-web/parser/v1'
@ -18,17 +20,25 @@ require 'openvpn-status-web/int_patch'
require 'openvpn-status-web/version'
module OpenVPNStatusWeb
# @return [Logger]
def self.logger
@logger
end
# @param logger [Logger]
# @return [Logger]
def self.logger=(logger)
@logger = logger
end
class LogFormatter
def call(lvl, time, progname, msg)
"[%s] %-5s %s\n" % [Time.now.strftime('%Y-%m-%d %H:%M:%S'), lvl, msg.to_s]
# @param lvl [Object]
# @param _time [DateTime]
# @param _progname [String]
# @param msg [Object]
# @return [String]
def call(lvl, _time, _progname, msg)
format("[%s] %-5s %s\n", Time.now.strftime('%Y-%m-%d %H:%M:%S'), lvl, msg.to_s)
end
end
@ -40,29 +50,29 @@ module OpenVPNStatusWeb
end
def call(env)
return [405, {"Content-Type" => "text/plain"}, ["Method Not Allowed"]] if env["REQUEST_METHOD"] != "GET"
return [404, {"Content-Type" => "text/plain"}, ["Not Found"]] if env["PATH_INFO"] != "/"
return [405, {'Content-Type' => 'text/plain'}, ['Method Not Allowed']] if env['REQUEST_METHOD'] != 'GET'
return [404, {'Content-Type' => 'text/plain'}, ['Not Found']] if env['PATH_INFO'] != '/'
# variables for template
vpns = @vpns
stati = {}
@vpns.each do |name,config|
@vpns.each do |name, config|
stati[name] = parse_status_log(config)
end
# eval
html = @main_tmpl.result(binding)
[200, {"Content-Type" => "text/html"}, [html]]
[200, {'Content-Type' => 'text/html'}, [html]]
end
def read_template(file)
text = File.open(file, 'rb') do |f| f.read end
text = File.read(file, mode: 'rb')
ERB.new(text)
end
def parse_status_log(vpn)
text = File.open(vpn['status_file'], 'rb') do |f| f.read end
text = File.read(vpn['status_file'], mode: 'rb')
case vpn['version']
when 1
@ -76,56 +86,64 @@ module OpenVPNStatusWeb
end
end
# @return [void]
def self.run!
if ARGV.length != 1
puts "Usage: openvpn-status-web config_file"
puts 'Usage: openvpn-status-web config_file'
exit 1
end
config_file = ARGV[0]
if not File.file?(config_file)
puts "Config file not found!"
if !File.file?(config_file)
puts 'Config file not found!'
exit 1
end
puts "openvpn-status-web version #{OpenVPNStatusWeb::VERSION}"
puts "Using config file #{config_file}"
config = YAML::load(File.open(config_file, 'r') { |f| f.read })
config = YAML.safe_load(File.read(config_file, mode: 'r'))
if config['logfile']
OpenVPNStatusWeb.logger = Logger.new(config['logfile'])
else
OpenVPNStatusWeb.logger = Logger.new(STDOUT)
OpenVPNStatusWeb.logger = Logger.new($stdout)
end
OpenVPNStatusWeb.logger.progname = "openvpn-status-web"
OpenVPNStatusWeb.logger.progname = 'openvpn-status-web'
OpenVPNStatusWeb.logger.formatter = LogFormatter.new
OpenVPNStatusWeb.logger.info "Starting..."
OpenVPNStatusWeb.logger.info 'Starting...'
# drop privs (first change group than user)
Process::Sys.setgid(Etc.getgrnam(config['group']).gid) if config['group']
Process::Sys.setuid(Etc.getpwnam(config['user']).uid) if config['user']
# drop priviliges as soon as possible
# NOTE: first change group than user
if config['group']
group = Etc.getgrnam(config['group'])
Process::Sys.setgid(group.gid) if group
end
if config['user']
user = Etc.getpwnam(config['user'])
Process::Sys.setuid(user.uid) if user
end
# configure rack
app = Daemon.new(config['vpns'])
if ENV['RACK_ENV'] == "development"
if ENV.fetch('RACK_ENV', nil) == 'development'
app = BetterErrors::Middleware.new(app)
BetterErrors.application_root = File.expand_path("..", __FILE__)
BetterErrors.application_root = File.expand_path(__dir__)
end
Signal.trap('INT') do
OpenVPNStatusWeb.logger.info "Quitting..."
Rack::Handler::WEBrick.shutdown
OpenVPNStatusWeb.logger.info 'Quitting...'
Rackup::Handler::WEBrick.shutdown
end
Signal.trap('TERM') do
OpenVPNStatusWeb.logger.info "Quitting..."
Rack::Handler::WEBrick.shutdown
OpenVPNStatusWeb.logger.info 'Quitting...'
Rackup::Handler::WEBrick.shutdown
end
Rack::Handler::WEBrick.run app, :Host => config['host'], :Port => config['port']
Rackup::Handler::WEBrick.run app, Host: config['host'], Port: config['port']
end
end
end

View File

@ -1,16 +1,17 @@
# frozen_string_literal: true
class Integer
def as_bytes
return "1 Byte" if self == 1
label = ["Bytes", "KiB", "MiB", "GiB", "TiB"]
return '1 Byte' if self == 1
label = %w[Bytes KiB MiB GiB TiB]
i = 0
num = self.to_f
while num >= 1024 do
num = num / 1024
num = to_f
while num >= 1024
num /= 1024
i += 1
end
"#{format('%.2f', num)} #{label[i]}"
end
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

@ -1,41 +1,55 @@
# frozen_string_literal: true
module OpenVPNStatusWeb
module Parser
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.global_stats << parse_global(parts[1..2]) if parts[0] == "GLOBAL_STATS"
status.client_list_headers = parts[2..] if parts[0] == 'HEADER' && parts[1] == 'CLIENT_LIST'
status.client_list << parse_client(parts[1..], status.client_list_headers) if parts[0] == 'CLIENT_LIST'
status.routing_table_headers = parts[2..] if parts[0] == 'HEADER' && parts[1] == 'ROUTING_TABLE'
status.routing_table << parse_route(parts[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
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
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')
client
end
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
def self.parse_global(global)
private_class_method def self.parse_global(global)
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

@ -1,3 +1,4 @@
# frozen_string_literal: true
module OpenVPNStatusWeb
module Parser
@ -13,7 +14,7 @@ module OpenVPNStatusWeb
(current_section = :rt; next) if line == "ROUTING TABLE\n"
(current_section = :gs; next) if line == "GLOBAL STATS\n"
(current_section = :end; next) if line == "END\n"
case current_section
when :cl
client_list << line.strip.split(',')
@ -25,8 +26,10 @@ module OpenVPNStatusWeb
end
status = Status.new
status.client_list = client_list[2..-1].map { |client| parse_client(client) }
status.routing_table = routing_table[1..-1].map { |route| parse_route(route) }
status.client_list_headers = ['Common Name', 'Real Address', 'Data Received', 'Data Sent', 'Connected Since']
status.client_list = client_list[2..].map { |client| parse_client(client) }
status.routing_table_headers = ['Virtual Address', 'Common Name', 'Real Address', 'Last Ref']
status.routing_table = routing_table[1..].map { |route| parse_route(route) }
status.global_stats = global_stats.map { |global| parse_global(global) }
status
end

View File

@ -1,5 +1,6 @@
# frozen_string_literal: true
require 'openvpn-status-web/parser/modern_stateless'
require_relative 'modern_stateless'
module OpenVPNStatusWeb
module Parser

View File

@ -1,5 +1,6 @@
# frozen_string_literal: true
require 'openvpn-status-web/parser/modern_stateless'
require_relative 'modern_stateless'
module OpenVPNStatusWeb
module Parser

View File

@ -1,7 +1,10 @@
# frozen_string_literal: true
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,4 +1,5 @@
# frozen_string_literal: true
module OpenVPNStatusWeb
VERSION = "2.0.0"
VERSION = '3.4.0'
end

View File

@ -1,32 +1,45 @@
# frozen_string_literal: true
$LOAD_PATH.push File.expand_path('lib', __dir__)
require 'openvpn-status-web/version'
require_relative 'lib/openvpn-status-web/version'
Gem::Specification.new do |s|
s.name = 'openvpn-status-web'
s.name = 'openvpn-status-web'
s.version = OpenVPNStatusWeb::VERSION
s.summary = 'openvpn-status-web'
s.description = 'Small Rack (Ruby) application serving OpenVPN status file.'
s.author = 'Christian Nicolai'
s.email = 'chrnicolai@gmail.com'
s.homepage = 'https://github.com/cmur2/dyndnsd'
s.homepage = 'https://github.com/cmur2/openvpn-status-web'
s.license = 'Apache-2.0'
s.metadata = {
'bug_tracker_uri' => "#{s.homepage}/issues",
'source_code_uri' => s.homepage
}
s.files = `git ls-files`.split($INPUT_RECORD_SEPARATOR)
s.test_files = s.files.grep(%r{^(test|spec|features)/})
s.files = `git ls-files -z`.split("\x0").select do |f|
f.match(%r{^(init.d|lib)/})
end
s.require_paths = ['lib']
s.bindir = 'exe'
s.executables = ['openvpn-status-web']
s.extra_rdoc_files = Dir['README.md', 'LICENSE']
s.required_ruby_version = '>= 2.3'
s.required_ruby_version = '>= 3.0'
s.add_runtime_dependency 'rack', '~> 2.0'
s.add_runtime_dependency 'metriks'
s.add_runtime_dependency 'rack', '~> 3.0'
s.add_runtime_dependency 'rackup', '~> 2'
s.add_runtime_dependency 'webrick', '>= 1.6.1'
s.add_development_dependency 'bundler'
s.add_development_dependency 'rake'
s.add_development_dependency 'rspec'
s.add_development_dependency 'rack-test'
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.9.0'
s.add_development_dependency 'rack-test'
s.add_development_dependency 'rake'
s.add_development_dependency 'rspec'
s.add_development_dependency 'rubocop', '~> 1.63.5'
s.add_development_dependency 'rubocop-rake', '~> 0.6.0'
s.add_development_dependency 'rubocop-rspec', '~> 2.29.2'
s.add_development_dependency 'solargraph', '~> 0.50.0'
end

View File

@ -0,0 +1,83 @@
# frozen_string_literal: true
require_relative '../../spec_helper'
describe OpenVPNStatusWeb::Parser::ModernStateless do
{
2 => status_v2,
3 => status_v3
}.each do |version, status|
context "when status-version #{version}" do
context 'with client list' do
it 'parses common names' do
expect(status.client_list.map { |client| client[0] }).to eq(%w[foo bar])
end
it 'parses real addresses' 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[3] }).to eq([11_811_160_064, 512])
end
it 'parses sent bytes' do
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[5] }).to eq(
[
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
it 'parses virtual addresses' do
expect(status.routing_table.map { |route| route[0] }).to eq(['192.168.0.0/24', '192.168.66.2', '192.168.66.3', '2001:db8:0:0::1000'])
end
it 'parses common names' do
expect(status.routing_table.map { |route| route[1] }).to eq(%w[foo bar foo bar])
end
it 'parses real addresses' do
expect(status.routing_table.map { |route| route[2] }).to eq(['1.2.3.4:1234', '1.2.3.5:1235', '1.2.3.4:1234', '1.2.3.5:1235'])
end
it 'parses last ref date' do
expect(status.routing_table.map { |route| route[3] }).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.routing_table[0].size).to eq(status.routing_table_headers.size)
end
end
it 'parses global stats' do
expect(status.global_stats.size).to eq(1)
expect(status.global_stats.first).to eq(['Max bcast/mcast queue length', 42])
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

@ -1,50 +1,71 @@
require 'spec_helper'
# frozen_string_literal: true
require_relative '../../spec_helper'
describe OpenVPNStatusWeb::Parser::V1 do
def status; status_v1; end
def status
status_v1
end
context 'for client list' do
context 'with client list' do
it 'parses common names' do
expect(status.client_list.map { |client| client[0] }).to eq(["foo", "bar"])
expect(status.client_list.map { |client| client[0] }).to eq(%w[foo bar])
end
it 'parses real addresses' do
expect(status.client_list.map { |client| client[1] }).to eq(["1.2.3.4:1234", "1.2.3.5:1235"])
expect(status.client_list.map { |client| client[1] }).to eq(['1.2.3.4:1234', '1.2.3.5:1235'])
end
it 'parses received bytes' do
expect(status.client_list.map { |client| client[2] }).to eq([11811160064, 512])
expect(status.client_list.map { |client| client[2] }).to eq([11_811_160_064, 512])
end
it 'parses sent bytes' do
expect(status.client_list.map { |client| client[3] }).to eq([4194304, 2048])
expect(status.client_list.map { |client| client[3] }).to eq([4_194_304, 2048])
end
it 'parses connected since date' do
expect(status.client_list.map { |client| client[4] }).to eq([DateTime.new(2012,1,1,23,42,0), DateTime.new(2012,1,1,23,42,0)])
expect(status.client_list.map { |client| client[4] }).to eq(
[
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].length).to eq(status.client_list_headers.length)
end
end
context 'for routing table' do
context 'with routing table' do
it 'parses virtual addresses' do
expect(status.routing_table.map { |route| route[0] }).to eq(["192.168.0.0/24", "192.168.66.2", "192.168.66.3", "2001:db8:0:0::1000"])
expect(status.routing_table.map { |route| route[0] }).to eq(['192.168.0.0/24', '192.168.66.2', '192.168.66.3', '2001:db8:0:0::1000'])
end
it 'parses common names' do
expect(status.routing_table.map { |route| route[1] }).to eq(["foo", "bar", "foo", "bar"])
expect(status.routing_table.map { |route| route[1] }).to eq(%w[foo bar foo bar])
end
it 'parses real addresses' do
expect(status.routing_table.map { |route| route[2] }).to eq(["1.2.3.4:1234", "1.2.3.5:1235", "1.2.3.4:1234", "1.2.3.5:1235"])
expect(status.routing_table.map { |route| route[2] }).to eq(['1.2.3.4:1234', '1.2.3.5:1235', '1.2.3.4:1234', '1.2.3.5:1235'])
end
it 'parses last ref date' do
expect(status.routing_table.map { |route| route[3] }).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)])
expect(status.routing_table.map { |route| route[3] }).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.routing_table[0].length).to eq(status.routing_table_headers.length)
end
end
it 'parses global stats' do
expect(status.global_stats.size).to eq(1)
expect(status.global_stats.first).to eq(["Max bcast/mcast queue length", 42])
expect(status.global_stats.first).to eq(['Max bcast/mcast queue length', 42])
end
end

View File

@ -1,55 +0,0 @@
require 'spec_helper'
describe OpenVPNStatusWeb::Parser::ModernStateless do
{
2 => status_v2,
3 => status_v3
}.each do |version, status|
context "for status-version #{version}" do
context 'for client list' do
it 'parses common names' do
expect(status.client_list.map { |client| client[0] }).to eq(["foo", "bar"])
end
it 'parses real addresses' do
expect(status.client_list.map { |client| client[1] }).to eq(["1.2.3.4:1234", "1.2.3.5:1235"])
end
it 'parses received bytes' do
expect(status.client_list.map { |client| client[2] }).to eq([11811160064, 512])
end
it 'parses sent bytes' do
expect(status.client_list.map { |client| client[3] }).to eq([4194304, 2048])
end
it 'parses connected since date' do
expect(status.client_list.map { |client| client[4] }).to eq([DateTime.new(2012,1,1,23,42,0), DateTime.new(2012,1,1,23,42,0)])
end
end
context 'for routing table' do
it 'parses virtual addresses' do
expect(status.routing_table.map { |route| route[0] }).to eq(["192.168.0.0/24", "192.168.66.2", "192.168.66.3", "2001:db8:0:0::1000"])
end
it 'parses common names' do
expect(status.routing_table.map { |route| route[1] }).to eq(["foo", "bar", "foo", "bar"])
end
it 'parses real addresses' do
expect(status.routing_table.map { |route| route[2] }).to eq(["1.2.3.4:1234", "1.2.3.5:1235", "1.2.3.4:1234", "1.2.3.5:1235"])
end
it 'parses last ref date' do
expect(status.routing_table.map { |route| route[3] }).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
end
it 'parses global stats' do
expect(status.global_stats.size).to eq(1)
expect(status.global_stats.first).to eq(["Max bcast/mcast queue length", 42])
end
end
end
end

View File

@ -1,3 +1,4 @@
# frozen_string_literal: true
require 'rubygems'
require 'bundler/setup'
@ -6,16 +7,21 @@ require 'rack/test'
require 'openvpn-status-web'
def status_v1
text = File.open('examples/status.v1', 'rb') do |f| f.read end
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') do |f| f.read end
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') do |f| f.read end
text = File.binread('examples/status.v3')
OpenVPNStatusWeb::Parser::V3.new.parse_status_log text
end