tac_scribe 0.2.0-java
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +7 -0
- data/.gitignore +11 -0
- data/.gitlab-ci.yml +22 -0
- data/.rspec +1 -0
- data/.rubocop.yml +31 -0
- data/.ruby-gemset +1 -0
- data/.ruby-version +1 -0
- data/.travis.yml +5 -0
- data/CHANGELOG.md +20 -0
- data/Gemfile +7 -0
- data/LICENSE.txt +15 -0
- data/README.md +53 -0
- data/Rakefile +11 -0
- data/bin/console +20 -0
- data/bin/setup +8 -0
- data/data/airfields.json +121 -0
- data/db/001_create_unit_table.rb +23 -0
- data/db/README.md +10 -0
- data/exe/start_scribe +103 -0
- data/lib/tac_scribe/daemon.rb +98 -0
- data/lib/tac_scribe/datastore.rb +140 -0
- data/lib/tac_scribe/event_processor.rb +86 -0
- data/lib/tac_scribe/event_queue.rb +81 -0
- data/lib/tac_scribe/version.rb +5 -0
- data/lib/tac_scribe.rb +8 -0
- data/scratchpad +11 -0
- data/tac_scribe.gemspec +49 -0
- metadata +226 -0
checksums.yaml
ADDED
@@ -0,0 +1,7 @@
|
|
1
|
+
---
|
2
|
+
SHA256:
|
3
|
+
metadata.gz: e9e9df45f82c6c95147899eb06561c0d4a6f250585500dd7ac772e40c29f51b1
|
4
|
+
data.tar.gz: 9f40713098afbe71d7ccd486e8a0913b5f7245744c5b3a87753401bbd8c5596c
|
5
|
+
SHA512:
|
6
|
+
metadata.gz: 148a8beed1d393b3ab8fb5a7e310e7aa913e0b3de237093681beb66f6a74f85c60292cdfed2a4ab439a6df4b158eb0cfb84611ed7582b68261481185e52504a1
|
7
|
+
data.tar.gz: be3183bf59c00cfa169431c383f282838acdf572ae9b7ac3fd425088b9a9c1d7c93610252c367041d3e3adb9768746c04a530a6a55040ccd9f200fcbfbdb72e5
|
data/.gitignore
ADDED
data/.gitlab-ci.yml
ADDED
@@ -0,0 +1,22 @@
|
|
1
|
+
.tests:
|
2
|
+
image: "ruby:2.5"
|
3
|
+
before_script:
|
4
|
+
- apt-get update && apt-get install -y git
|
5
|
+
- ruby -v
|
6
|
+
- which ruby
|
7
|
+
- gem install bundler --no-document
|
8
|
+
- bundle install --jobs $(nproc) "${FLAGS[@]}"
|
9
|
+
|
10
|
+
.rspec:
|
11
|
+
extends: .tests
|
12
|
+
script:
|
13
|
+
- bundle exec rake spec
|
14
|
+
|
15
|
+
rubocop:
|
16
|
+
extends: .tests
|
17
|
+
script:
|
18
|
+
- bundle exec rake rubocop
|
19
|
+
|
20
|
+
rspec-ruby:
|
21
|
+
extends: .rspec
|
22
|
+
image: ruby:latest
|
data/.rspec
ADDED
@@ -0,0 +1 @@
|
|
1
|
+
--require spec_helper
|
data/.rubocop.yml
ADDED
@@ -0,0 +1,31 @@
|
|
1
|
+
Style/BlockComments:
|
2
|
+
Exclude:
|
3
|
+
# This is the default rspec generated file so leave it be for consistency
|
4
|
+
- spec/spec_helper.rb
|
5
|
+
Layout/CommentIndentation:
|
6
|
+
Exclude:
|
7
|
+
# This is the default rspec generated file so leave it be for consistency
|
8
|
+
- spec/spec_helper.rb
|
9
|
+
|
10
|
+
Metrics/BlockLength:
|
11
|
+
Exclude:
|
12
|
+
# Rubocop does not like the rspec describe block style which is always long
|
13
|
+
- spec/**/*_spec.rb
|
14
|
+
# There isn't much sense breaking up the OptionParser block since splitting
|
15
|
+
# into db and tacview option methods will just break the method length cop
|
16
|
+
# and splitting further doesn't aid readability
|
17
|
+
- exe/start_scribe
|
18
|
+
Metrics/MethodLength:
|
19
|
+
Exclude:
|
20
|
+
# Breaking up the initializer doesn't really do much for readability
|
21
|
+
- lib/tac_scribe/daemon.rb
|
22
|
+
|
23
|
+
Metrics/ParameterLists:
|
24
|
+
Exclude:
|
25
|
+
# The nature of the beast. Being the entry point into the program
|
26
|
+
# necessitates this
|
27
|
+
- lib/tac_scribe/daemon.rb
|
28
|
+
Style/GlobalVars:
|
29
|
+
Exclude:
|
30
|
+
# Using a global variable makes it available to the Pry console
|
31
|
+
- bin/console
|
data/.ruby-gemset
ADDED
@@ -0,0 +1 @@
|
|
1
|
+
tac_scribe
|
data/.ruby-version
ADDED
@@ -0,0 +1 @@
|
|
1
|
+
jruby-9.2.6.0
|
data/.travis.yml
ADDED
data/CHANGELOG.md
ADDED
@@ -0,0 +1,20 @@
|
|
1
|
+
# Changelog
|
2
|
+
All notable changes to this project will be documented in this file.
|
3
|
+
|
4
|
+
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
|
5
|
+
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
|
6
|
+
|
7
|
+
## [Unreleased]
|
8
|
+
|
9
|
+
## [0.2.0] - 2019-10-13
|
10
|
+
### Added
|
11
|
+
- JRuby support
|
12
|
+
|
13
|
+
### Changed
|
14
|
+
- Improve logging of events with number of events processed.
|
15
|
+
|
16
|
+
|
17
|
+
## [0.1.1] - 2019-09-05
|
18
|
+
### Added
|
19
|
+
- Added Net::HTTP error handling
|
20
|
+
|
data/Gemfile
ADDED
data/LICENSE.txt
ADDED
@@ -0,0 +1,15 @@
|
|
1
|
+
Tac Scribe - Record Tacview events to PostgreSQL
|
2
|
+
Copyright (C) 2019 Jeffrey Jones
|
3
|
+
|
4
|
+
This program is free software: you can redistribute it and/or modify
|
5
|
+
it under the terms of the GNU Affero General Public License as
|
6
|
+
published by the Free Software Foundation, either version 3 of the
|
7
|
+
License, or (at your option) any later version.
|
8
|
+
|
9
|
+
This program is distributed in the hope that it will be useful,
|
10
|
+
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
11
|
+
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
12
|
+
GNU Affero General Public License for more details.
|
13
|
+
|
14
|
+
You should have received a copy of the GNU Affero General Public License
|
15
|
+
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
data/README.md
ADDED
@@ -0,0 +1,53 @@
|
|
1
|
+
# TacScribe
|
2
|
+
|
3
|
+
Writes object state from a tacview server to a PostGIS extended Postgresql
|
4
|
+
database for further processing by other systems.
|
5
|
+
|
6
|
+
## Installation
|
7
|
+
|
8
|
+
Add this line to your application's Gemfile:
|
9
|
+
|
10
|
+
```ruby
|
11
|
+
gem 'tac_scribe'
|
12
|
+
```
|
13
|
+
|
14
|
+
And then execute:
|
15
|
+
|
16
|
+
$ bundle
|
17
|
+
|
18
|
+
Or install it yourself as:
|
19
|
+
|
20
|
+
$ gem install tac_scribe
|
21
|
+
|
22
|
+
## Usage
|
23
|
+
|
24
|
+
You can run the tool using the `start_scribe` command. Use the `--help`
|
25
|
+
option for information on required arguments
|
26
|
+
|
27
|
+
## Development
|
28
|
+
|
29
|
+
After checking out the repo, run `bin/setup` to install dependencies. Then,
|
30
|
+
run `rake test` to run the tests. You can also run `bin/console` for an
|
31
|
+
interactive prompt that will allow you to experiment.
|
32
|
+
|
33
|
+
### Postgresql and PostGIS
|
34
|
+
|
35
|
+
This gem requires postgresql and PostGIS available and listening on an
|
36
|
+
accessible network port.
|
37
|
+
|
38
|
+
See the `db` folder for information on running the database migrations.
|
39
|
+
|
40
|
+
## Contributing
|
41
|
+
|
42
|
+
Bug reports and pull requests are welcome on GitLab at https://gitlab.com/overlord-bot/tac-scribe.
|
43
|
+
|
44
|
+
### Adding Data
|
45
|
+
|
46
|
+
There are incomplete data-files for this project whose completion would
|
47
|
+
be very helpful. See the `data` folder. Current tasks needed are:
|
48
|
+
|
49
|
+
* Complete the list of all airfields in DCS with their position information
|
50
|
+
|
51
|
+
## License
|
52
|
+
|
53
|
+
The gem is available as open source under the terms of the [MIT License](https://opensource.org/licenses/MIT).
|
data/Rakefile
ADDED
data/bin/console
ADDED
@@ -0,0 +1,20 @@
|
|
1
|
+
#!/usr/bin/env ruby
|
2
|
+
# frozen_string_literal: true
|
3
|
+
|
4
|
+
require 'bundler/setup'
|
5
|
+
require 'tac_scribe'
|
6
|
+
require 'tac_scribe/datastore'
|
7
|
+
|
8
|
+
TacScribe::Datastore.instance.connect
|
9
|
+
|
10
|
+
$DB = TacScribe::Datastore.instance.db
|
11
|
+
|
12
|
+
# You can add fixtures and/or initialization code here to make experimenting
|
13
|
+
# with your gem easier. You can also use a different console, if you like.
|
14
|
+
|
15
|
+
# (If you use this, don't forget to add pry to your Gemfile!)
|
16
|
+
# require "pry"
|
17
|
+
# Pry.start
|
18
|
+
|
19
|
+
require 'irb'
|
20
|
+
IRB.start(__FILE__)
|
data/bin/setup
ADDED
data/data/airfields.json
ADDED
@@ -0,0 +1,121 @@
|
|
1
|
+
[{
|
2
|
+
"lon": 45.01909093846007,
|
3
|
+
"lat": 41.637735936261556,
|
4
|
+
"alt": 464.5004577636719,
|
5
|
+
"name": "Vaziani"
|
6
|
+
},
|
7
|
+
{
|
8
|
+
"lon": 44.94687659192431,
|
9
|
+
"lat": 41.67471935873423,
|
10
|
+
"alt": 479.69482421875,
|
11
|
+
"name": "Tbilisi-Lochini"
|
12
|
+
},
|
13
|
+
{
|
14
|
+
"lon": 43.62488628801948,
|
15
|
+
"lat": 43.50998473505967,
|
16
|
+
"alt": 430.01043701171875,
|
17
|
+
"name": "Nalchik"
|
18
|
+
},
|
19
|
+
{
|
20
|
+
"lon": 38.92520230077506,
|
21
|
+
"lat": 45.087429883845076,
|
22
|
+
"alt": 30.010032653808594,
|
23
|
+
"name": "Krasnodar-Center"
|
24
|
+
},
|
25
|
+
{
|
26
|
+
"lon": 37.35978347755592,
|
27
|
+
"lat": 45.01317473377168,
|
28
|
+
"alt": 43.00004196166992,
|
29
|
+
"name": "Anapa-Vityazevo"
|
30
|
+
},
|
31
|
+
{
|
32
|
+
"lon": 41.876483823101026,
|
33
|
+
"lat": 41.93210535345338,
|
34
|
+
"alt": 18.01001739501953,
|
35
|
+
"name": "Kobuleti"
|
36
|
+
},
|
37
|
+
{
|
38
|
+
"lon": 43.100679733081456,
|
39
|
+
"lat": 44.21864682380681,
|
40
|
+
"alt": 320.01031494140625,
|
41
|
+
"name": "Mineralnye Vody"
|
42
|
+
},
|
43
|
+
{
|
44
|
+
"lon": 44.62032726210201,
|
45
|
+
"lat": 43.79130325093825,
|
46
|
+
"alt": 154.61184692382812,
|
47
|
+
"name": "Mozdok"
|
48
|
+
},
|
49
|
+
{
|
50
|
+
"lon": 44.94718306531669,
|
51
|
+
"lat": 41.64116326678661,
|
52
|
+
"alt": 449.4102478027344,
|
53
|
+
"name": "Soganlug"
|
54
|
+
},
|
55
|
+
{
|
56
|
+
"lon": 44.588922553542936,
|
57
|
+
"lat": 43.20850098738094,
|
58
|
+
"alt": 524.0057983398438,
|
59
|
+
"name": "Beslan"
|
60
|
+
},
|
61
|
+
{
|
62
|
+
"lon": 42.49568635853585,
|
63
|
+
"lat": 42.179154028210135,
|
64
|
+
"alt": 45.010047912597656,
|
65
|
+
"name": "Kutaisi"
|
66
|
+
},
|
67
|
+
{
|
68
|
+
"lon": 37.786226060479564,
|
69
|
+
"lat": 44.6733296041269,
|
70
|
+
"alt": 40.010040283203125,
|
71
|
+
"name": "Novorossiysk"
|
72
|
+
},
|
73
|
+
{
|
74
|
+
"lon": 40.021427482235985,
|
75
|
+
"lat": 44.67144025735508,
|
76
|
+
"alt": 180.01019287109375,
|
77
|
+
"name": "Maykop-Khanskaya"
|
78
|
+
},
|
79
|
+
{
|
80
|
+
"lon": 39.924231880466095,
|
81
|
+
"lat": 43.43937843405085,
|
82
|
+
"alt": 30.010034561157227,
|
83
|
+
"name": "Sochi-Adler"
|
84
|
+
},
|
85
|
+
{
|
86
|
+
"lon": 41.142447588488196,
|
87
|
+
"lat": 42.852741071634995,
|
88
|
+
"alt": 13.339526176452637,
|
89
|
+
"name": "Sukhumi-Babushara"
|
90
|
+
},
|
91
|
+
{
|
92
|
+
"lon": 42.061021312855914,
|
93
|
+
"lat": 42.23872808157328,
|
94
|
+
"alt": 13.23994255065918,
|
95
|
+
"name": "Senaki-Kolkhi"
|
96
|
+
},
|
97
|
+
{
|
98
|
+
"lon": 40.56417576840064,
|
99
|
+
"lat": 43.124233340197144,
|
100
|
+
"alt": 21.01003074645996,
|
101
|
+
"name": "Gudauta"
|
102
|
+
},
|
103
|
+
{
|
104
|
+
"lon": 37.985886938697085,
|
105
|
+
"lat": 44.961383022734175,
|
106
|
+
"alt": 20.010303497314453,
|
107
|
+
"name": "Krymsk"
|
108
|
+
},
|
109
|
+
{
|
110
|
+
"lon": 38.0041463505281,
|
111
|
+
"lat": 44.56767458600406,
|
112
|
+
"alt": 22.00992202758789,
|
113
|
+
"name": "Gelendzhik"
|
114
|
+
},
|
115
|
+
{
|
116
|
+
"lon": 39.20306690632454,
|
117
|
+
"lat": 45.0460996415433,
|
118
|
+
"alt": 34.01003646850586,
|
119
|
+
"name": "Krasnodar-Pashkovsky"
|
120
|
+
}
|
121
|
+
]
|
@@ -0,0 +1,23 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
Sequel.migration do
|
4
|
+
up do
|
5
|
+
create_table(:units) do
|
6
|
+
String :id, primary_key: true
|
7
|
+
column :position, :geography, null: false
|
8
|
+
Float :altitude, default: 0
|
9
|
+
String :type
|
10
|
+
String :name, null: true
|
11
|
+
String :pilot, null: true
|
12
|
+
String :group, null: true
|
13
|
+
Integer :coalition
|
14
|
+
Integer :heading
|
15
|
+
Time :updated_at
|
16
|
+
# TODO: GIST Index on the position
|
17
|
+
end
|
18
|
+
end
|
19
|
+
|
20
|
+
down do
|
21
|
+
drop_table(:units)
|
22
|
+
end
|
23
|
+
end
|
data/db/README.md
ADDED
@@ -0,0 +1,10 @@
|
|
1
|
+
# Database Setup
|
2
|
+
|
3
|
+
See example instructions for ubuntu here [here](https://kitcharoenp.github.io/postgresql/postgis/2018/05/28/set_up_postgreSQL_postgis.html)
|
4
|
+
|
5
|
+
# Database Migrations
|
6
|
+
|
7
|
+
Run migrations using
|
8
|
+
|
9
|
+
`sequel -m db/ postgres://username:password@database_ip:database_port/tac_scribe`
|
10
|
+
|
data/exe/start_scribe
ADDED
@@ -0,0 +1,103 @@
|
|
1
|
+
#!/usr/bin/env ruby
|
2
|
+
# frozen_string_literal: true
|
3
|
+
|
4
|
+
require 'optparse'
|
5
|
+
require 'bundler/setup'
|
6
|
+
require 'tac_scribe'
|
7
|
+
|
8
|
+
options = {
|
9
|
+
tacview_host: 'localhost',
|
10
|
+
tacview_port: 42_674,
|
11
|
+
tacview_password: nil,
|
12
|
+
tacview_client_name: 'TacScribe',
|
13
|
+
db_host: 'localhost',
|
14
|
+
db_port: 5432,
|
15
|
+
db_name: 'tac_scribe',
|
16
|
+
verbose_logging: false,
|
17
|
+
threads: 1,
|
18
|
+
populate_airfields: false
|
19
|
+
}
|
20
|
+
|
21
|
+
OptionParser.new do |opts|
|
22
|
+
opts.banner = 'Usage: ruby start_scribe [options]'
|
23
|
+
|
24
|
+
opts.separator "\nTacview Options"
|
25
|
+
opts.on('-h', '--tacview-host=host',
|
26
|
+
'Tacview server hostname / IP (Default: localhost)') do |v|
|
27
|
+
options[:tacview_host] = v
|
28
|
+
end
|
29
|
+
|
30
|
+
opts.on('-p', '--tacview-port=port',
|
31
|
+
'Tacview server port (Default: 42674)') do |v|
|
32
|
+
options[:tacview_port] = v
|
33
|
+
end
|
34
|
+
|
35
|
+
opts.on('-a', '--tacview-password=password',
|
36
|
+
'Tacview server password (Optional)') do |v|
|
37
|
+
options[:tacview_password] = v
|
38
|
+
end
|
39
|
+
|
40
|
+
opts.on('-c', '--tacview-client-name=client',
|
41
|
+
'Client name (Default: TacScribe)') do |v|
|
42
|
+
options[:tacview_client_name] = v
|
43
|
+
end
|
44
|
+
|
45
|
+
opts.separator "\nDatabase Options"
|
46
|
+
opts.on('-o', '--db-host=host',
|
47
|
+
'Postgresql server hostname / IP (Default: localhost)') do |v|
|
48
|
+
options[:db_host] = v
|
49
|
+
end
|
50
|
+
opts.on('-r', '--db-port=port',
|
51
|
+
'Postgresql server port (Default: 5432)') do |v|
|
52
|
+
options[:db_port] = v
|
53
|
+
end
|
54
|
+
opts.on('-u', '--db-username=username',
|
55
|
+
'Postgresql username (Required)') do |v|
|
56
|
+
options[:db_user] = v
|
57
|
+
end
|
58
|
+
opts.on('-s', '--db-password=password',
|
59
|
+
'Postgresql password (Required)') do |v|
|
60
|
+
options[:db_password] = v
|
61
|
+
end
|
62
|
+
opts.on('-n', '--db-name=name',
|
63
|
+
'Postgresql database name (Default: tac_scribe)') do |v|
|
64
|
+
options[:db_name] = v
|
65
|
+
end
|
66
|
+
opts.separator "\nMisc options"
|
67
|
+
opts.on('-f', '--populate-airfields',
|
68
|
+
'Populate DCS airfield locations') do |_v|
|
69
|
+
options[:populate_airfields] = true
|
70
|
+
end
|
71
|
+
opts.on('-t', '--threads=threads',
|
72
|
+
'Thread Count (Default: 1)') do |v|
|
73
|
+
options[:threads] = v
|
74
|
+
end
|
75
|
+
opts.on('-v', '--verbose',
|
76
|
+
'Verbose logging') do |_v|
|
77
|
+
options[:verbose] = true
|
78
|
+
end
|
79
|
+
end.parse!
|
80
|
+
|
81
|
+
%i[db_user db_password].each do |parameter|
|
82
|
+
next unless !options[parameter] ||
|
83
|
+
!options[parameter].is_a?(String) ||
|
84
|
+
options[parameter].empty?
|
85
|
+
|
86
|
+
puts "#{parameter.to_s.tr('_', '-')} required"
|
87
|
+
exit(1)
|
88
|
+
end
|
89
|
+
|
90
|
+
TacScribe::Daemon.new(
|
91
|
+
tacview_host: options[:tacview_host],
|
92
|
+
tacview_port: options[:tacview_port],
|
93
|
+
tacview_password: options[:tacview_password],
|
94
|
+
tacview_client_name: options[:tacview_client_name],
|
95
|
+
db_host: options[:db_host],
|
96
|
+
db_port: options[:db_port],
|
97
|
+
db_name: options[:db_name],
|
98
|
+
db_user: options[:db_user],
|
99
|
+
db_password: options[:db_password],
|
100
|
+
verbose_logging: options[:verbose],
|
101
|
+
thread_count: options[:threads].to_i,
|
102
|
+
populate_airfields: options[:populate_airfields]
|
103
|
+
).start_processing
|
@@ -0,0 +1,98 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require 'json'
|
4
|
+
require 'tacview_client'
|
5
|
+
require_relative 'datastore'
|
6
|
+
require_relative 'event_queue'
|
7
|
+
require_relative 'event_processor'
|
8
|
+
|
9
|
+
module TacScribe
|
10
|
+
# Main entry-point into Tac Scribe. Instantiate this class to start
|
11
|
+
# processing events.
|
12
|
+
class Daemon
|
13
|
+
def initialize(db_host:, db_port:, db_name:, db_user:, db_password:,
|
14
|
+
tacview_host:, tacview_port:, tacview_password:,
|
15
|
+
tacview_client_name:, verbose_logging:, thread_count:,
|
16
|
+
populate_airfields:)
|
17
|
+
Datastore.instance.configure do |config|
|
18
|
+
config.host = db_host
|
19
|
+
config.port = db_port
|
20
|
+
config.database = db_name
|
21
|
+
config.username = db_user
|
22
|
+
config.password = db_password
|
23
|
+
end
|
24
|
+
Datastore.instance.connect
|
25
|
+
|
26
|
+
@event_queue = EventQueue.new verbose_logging: verbose_logging
|
27
|
+
|
28
|
+
@populate_airfields = populate_airfields
|
29
|
+
@thread_count = thread_count
|
30
|
+
@processing_threads = []
|
31
|
+
|
32
|
+
@client = TacviewClient::Client.new(
|
33
|
+
host: tacview_host,
|
34
|
+
port: tacview_port,
|
35
|
+
password: tacview_password,
|
36
|
+
processor: @event_queue,
|
37
|
+
client_name: tacview_client_name
|
38
|
+
)
|
39
|
+
end
|
40
|
+
|
41
|
+
# Starts processing and reconnects if the client was disconnected.
|
42
|
+
# Because connecting to Tacview always gives an initial unit dump
|
43
|
+
# we truncate the table each time we reconnect. This will make sure
|
44
|
+
# there are no ghost units hanging around after server restart
|
45
|
+
# for example
|
46
|
+
def start_processing
|
47
|
+
loop do
|
48
|
+
kill_processing_threads
|
49
|
+
@event_queue.events.clear
|
50
|
+
Datastore.instance.truncate_table
|
51
|
+
start_processing_threads
|
52
|
+
populate_airfields if @populate_airfields
|
53
|
+
@client.connect
|
54
|
+
sleep 30
|
55
|
+
# Rescuing reliably from Net::HTTP is a complete bear so rescue
|
56
|
+
# StandardError. It ain't pretty but it is reliable. We will puts
|
57
|
+
# the exception just in case
|
58
|
+
# https://stackoverflow.com/questions/5370697/what-s-the-best-way-to-handle-exceptions-from-nethttp
|
59
|
+
rescue StandardError => e
|
60
|
+
puts e.message
|
61
|
+
puts e.backtrace
|
62
|
+
next
|
63
|
+
end
|
64
|
+
end
|
65
|
+
|
66
|
+
def kill_processing_threads
|
67
|
+
@processing_threads.each do |thr|
|
68
|
+
thr.kill
|
69
|
+
thr.join
|
70
|
+
end
|
71
|
+
end
|
72
|
+
|
73
|
+
def start_processing_threads
|
74
|
+
@thread_count.times do
|
75
|
+
@processing_threads << Thread.new do
|
76
|
+
EventProcessor.new(datastore: Datastore.instance,
|
77
|
+
event_queue: @event_queue).start
|
78
|
+
end
|
79
|
+
end
|
80
|
+
end
|
81
|
+
|
82
|
+
def populate_airfields
|
83
|
+
json = File.read(File.join(File.dirname(__FILE__), '../../data/airfields.json'))
|
84
|
+
airfields = JSON.parse(json)
|
85
|
+
airfields.each_with_index do |airfield, i|
|
86
|
+
@event_queue.update_object({
|
87
|
+
object_id: (45_000_000 + i).to_s,
|
88
|
+
latitude: BigDecimal(airfield['lat'].to_s),
|
89
|
+
longitude: BigDecimal(airfield['lon'].to_s),
|
90
|
+
altitude: BigDecimal(airfield['alt'].to_s),
|
91
|
+
type: 'Ground+Static+Aerodrome',
|
92
|
+
name: airfield['name'],
|
93
|
+
coalition: 2
|
94
|
+
})
|
95
|
+
end
|
96
|
+
end
|
97
|
+
end
|
98
|
+
end
|
@@ -0,0 +1,140 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require 'georuby'
|
4
|
+
require 'sequel'
|
5
|
+
require 'sequel-postgis-georuby'
|
6
|
+
require 'singleton'
|
7
|
+
|
8
|
+
module TacScribe
|
9
|
+
# Acts as the interface to the back-end datastore and hides all datastore
|
10
|
+
# implementation details from callers. Note that ruby does not support
|
11
|
+
# exception chaining we are not wrapping exceptions yet. This will happen
|
12
|
+
# when https://bugs.ruby-lang.org/issues/8257 is fixed.
|
13
|
+
class Datastore
|
14
|
+
include Singleton
|
15
|
+
include GeoRuby::SimpleFeatures
|
16
|
+
|
17
|
+
attr_accessor :db
|
18
|
+
|
19
|
+
@configuration = nil
|
20
|
+
@db = nil
|
21
|
+
|
22
|
+
def configuration
|
23
|
+
@configuration ||= Configuration.new
|
24
|
+
end
|
25
|
+
|
26
|
+
def configure
|
27
|
+
configuration
|
28
|
+
yield(@configuration) if block_given?
|
29
|
+
end
|
30
|
+
|
31
|
+
# Contains all the connection configuration required for connectin
|
32
|
+
# to a postgresql database. Defaults to localhost with ubuntu presets
|
33
|
+
class Configuration
|
34
|
+
attr_accessor :host, :port, :database, :username, :password
|
35
|
+
|
36
|
+
def initialize
|
37
|
+
@host = 'localhost'
|
38
|
+
@port = '5432'
|
39
|
+
@database = 'tac_scribe'
|
40
|
+
@username = 'tac_scribe'
|
41
|
+
@password = 'tac_scribe'
|
42
|
+
end
|
43
|
+
end
|
44
|
+
|
45
|
+
def connect
|
46
|
+
configure
|
47
|
+
@db = Sequel.connect(connection_string, max_connections: 49)
|
48
|
+
@db.extension :postgis_georuby
|
49
|
+
end
|
50
|
+
|
51
|
+
def truncate_table
|
52
|
+
@db[:units].truncate
|
53
|
+
end
|
54
|
+
|
55
|
+
def write_object(event, timestamp)
|
56
|
+
unit = get_unit(event[:object_id])
|
57
|
+
|
58
|
+
# Tacview omits values that don't change to save
|
59
|
+
# data bandwidth and storage. If something has not changed then
|
60
|
+
# use the old value
|
61
|
+
current_position = get_position(event, unit)
|
62
|
+
|
63
|
+
if unit
|
64
|
+
update_unit(event, unit, current_position, timestamp)
|
65
|
+
else
|
66
|
+
insert_unit(event, current_position, timestamp)
|
67
|
+
end
|
68
|
+
end
|
69
|
+
|
70
|
+
def delete_object(object_id)
|
71
|
+
count = @db[:units].where(id: object_id).delete
|
72
|
+
"Deleted #{object_id} #{object_id.class} - #{count}"
|
73
|
+
end
|
74
|
+
|
75
|
+
private
|
76
|
+
|
77
|
+
def connection_string
|
78
|
+
if RUBY_PLATFORM == 'java'
|
79
|
+
"jdbc:postgresql://#{@configuration.host}:#{@configuration.port}/" \
|
80
|
+
"#{@configuration.database}?user=#{@configuration.username}&" \
|
81
|
+
"password=#{configuration.password}"
|
82
|
+
else
|
83
|
+
"postgres://#{@configuration.username}:#{@configuration.password}@" \
|
84
|
+
"#{@configuration.host}:#{@configuration.port}" \
|
85
|
+
"/#{@configuration.database}"
|
86
|
+
end
|
87
|
+
end
|
88
|
+
|
89
|
+
def get_unit(id)
|
90
|
+
@db[:units].where(id: id).first
|
91
|
+
end
|
92
|
+
|
93
|
+
def update_unit(event, unit, current_position, timestamp)
|
94
|
+
heading = calculate_heading(unit[:position],
|
95
|
+
current_position,
|
96
|
+
unit[:heading])
|
97
|
+
|
98
|
+
@db[:units].where(id: event[:object_id]).update(
|
99
|
+
position: current_position[:lat_lon],
|
100
|
+
altitude: current_position[:altitude],
|
101
|
+
heading: heading,
|
102
|
+
updated_at: timestamp
|
103
|
+
)
|
104
|
+
end
|
105
|
+
|
106
|
+
def insert_unit(event, current_position, timestamp)
|
107
|
+
@db[:units].insert(id: event[:object_id],
|
108
|
+
position: current_position[:lat_lon],
|
109
|
+
altitude: current_position[:altitude],
|
110
|
+
type: event[:type],
|
111
|
+
name: event[:name],
|
112
|
+
group: event[:group],
|
113
|
+
pilot: event[:pilot],
|
114
|
+
coalition: event[:coalition] == 'Allies' ? 0 : 1,
|
115
|
+
updated_at: timestamp)
|
116
|
+
end
|
117
|
+
|
118
|
+
def get_position(event, unit)
|
119
|
+
{
|
120
|
+
lat_lon: Point.from_x_y(
|
121
|
+
event.fetch(:longitude) { unit ? unit[:position].y : nil },
|
122
|
+
event.fetch(:latitude) { unit ? unit[:position].x : nil }
|
123
|
+
),
|
124
|
+
altitude: event.fetch(:altitude) { unit ? unit[:altitude] : nil }
|
125
|
+
}
|
126
|
+
end
|
127
|
+
|
128
|
+
def calculate_heading(old_position, current_position, current_heading)
|
129
|
+
return current_heading if old_position == current_position[:lat_lon]
|
130
|
+
|
131
|
+
begin
|
132
|
+
old_position.bearing_to(current_position[:lat_lon]).to_i
|
133
|
+
rescue Math::DomainError => e
|
134
|
+
puts 'Could not calculate heading: ' + e.message
|
135
|
+
puts 'Old Position: ' + old_position.inspect
|
136
|
+
puts 'New Position: ' + current_position.inspect
|
137
|
+
end
|
138
|
+
end
|
139
|
+
end
|
140
|
+
end
|
@@ -0,0 +1,86 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require 'tacview_client/base_processor'
|
4
|
+
require 'time'
|
5
|
+
require_relative 'datastore'
|
6
|
+
|
7
|
+
module TacScribe
|
8
|
+
# Processes the events emitted by the Ruby Tacview Client
|
9
|
+
class EventProcessor
|
10
|
+
def initialize(datastore:, event_queue:)
|
11
|
+
@datastore = datastore
|
12
|
+
@event_queue = event_queue
|
13
|
+
end
|
14
|
+
|
15
|
+
def start
|
16
|
+
loop do
|
17
|
+
wrapped_event = @event_queue.events.shift
|
18
|
+
process_event(wrapped_event)
|
19
|
+
rescue StandardError => e
|
20
|
+
puts wrapped_event
|
21
|
+
puts e.inspect
|
22
|
+
next
|
23
|
+
end
|
24
|
+
end
|
25
|
+
|
26
|
+
def process_event(wrapped_event)
|
27
|
+
case wrapped_event[:type]
|
28
|
+
when :update_object
|
29
|
+
update_object(wrapped_event[:event], wrapped_event[:time])
|
30
|
+
when :delete_object
|
31
|
+
delete_object(wrapped_event[:object_id])
|
32
|
+
end
|
33
|
+
end
|
34
|
+
|
35
|
+
private
|
36
|
+
|
37
|
+
# Process an update event for an object
|
38
|
+
#
|
39
|
+
# On the first appearance of the object there are usually more fields
|
40
|
+
# including pilot names, object type etc.
|
41
|
+
#
|
42
|
+
# For existing objects these events are almost always lat/lon/alt updates
|
43
|
+
# only
|
44
|
+
#
|
45
|
+
# @param event [Hash] A parsed ACMI line. This hash will always include
|
46
|
+
# the fields listed below but may also include others depending on the
|
47
|
+
# source line.
|
48
|
+
# @option event [String] :object_id The hexadecimal format object ID.
|
49
|
+
# @option event [BigDecimal] :latitude The object latitude in WGS 84 format.
|
50
|
+
# @option event [BigDecimal] :longitude The object latitude in WGS 84
|
51
|
+
# format.
|
52
|
+
# @option event [BigDecimal] :altitude The object altitude above sea level
|
53
|
+
# in meters to 1 decimal place.
|
54
|
+
def update_object(event, time)
|
55
|
+
if @event_queue.reference_latitude || @event_queue.reference_longitude
|
56
|
+
localize_position(event)
|
57
|
+
end
|
58
|
+
|
59
|
+
@datastore.write_object(event, time)
|
60
|
+
end
|
61
|
+
|
62
|
+
# Process a delete event for an object
|
63
|
+
#
|
64
|
+
# @param object_id [String] A hexadecimal number representing the object
|
65
|
+
# ID
|
66
|
+
def delete_object(object_id)
|
67
|
+
@datastore.delete_object(object_id)
|
68
|
+
end
|
69
|
+
|
70
|
+
# If we have reference lat/long then use that as a base and update the event
|
71
|
+
# position
|
72
|
+
def localize_position(event)
|
73
|
+
# There is no combination of layouts for these lines that doesn' trip
|
74
|
+
# at least one rubocop cop so just ignore the guard clause. The best
|
75
|
+
# one is using an inline if but that breaks the line length.
|
76
|
+
# rubocop:disable Style/GuardClause
|
77
|
+
if event[:latitude]
|
78
|
+
event[:latitude] = @event_queue.reference_latitude + event[:latitude]
|
79
|
+
end
|
80
|
+
if event[:longitude]
|
81
|
+
event[:longitude] = @event_queue.reference_longitude + event[:longitude]
|
82
|
+
end
|
83
|
+
# rubocop:enable Style/GuardClause
|
84
|
+
end
|
85
|
+
end
|
86
|
+
end
|
@@ -0,0 +1,81 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require 'tacview_client/base_processor'
|
4
|
+
require 'time'
|
5
|
+
|
6
|
+
module TacScribe
|
7
|
+
# Processes the events emitted by the Ruby Tacview Client
|
8
|
+
class EventQueue < TacviewClient::BaseProcessor
|
9
|
+
attr_accessor :events, :reference_latitude, :reference_longitude
|
10
|
+
|
11
|
+
def initialize(verbose_logging:)
|
12
|
+
@verbose_logging = verbose_logging
|
13
|
+
@events = Queue.new
|
14
|
+
@event_count = 0
|
15
|
+
|
16
|
+
return unless verbose_logging == true
|
17
|
+
|
18
|
+
Thread.new do
|
19
|
+
loop do
|
20
|
+
puts "#{Time.now.strftime("%FT%T")} - Queue Size: #{@events.size} \t Events Processed: #{@event_count}"
|
21
|
+
@event_count = 0
|
22
|
+
sleep 1
|
23
|
+
end
|
24
|
+
end
|
25
|
+
end
|
26
|
+
|
27
|
+
# Process an update event for an object
|
28
|
+
#
|
29
|
+
# On the first appearance of the object there are usually more fields
|
30
|
+
# including pilot names, object type etc.
|
31
|
+
#
|
32
|
+
# For existing objects these events are almost always lat/lon/alt updates
|
33
|
+
# only
|
34
|
+
#
|
35
|
+
# @param event [Hash] A parsed ACMI line. This hash will always include
|
36
|
+
# the fields listed below but may also include others depending on the
|
37
|
+
# source line.
|
38
|
+
# @option event [String] :object_id The hexadecimal format object ID.
|
39
|
+
# @option event [BigDecimal] :latitude The object latitude in WGS 84 format.
|
40
|
+
# @option event [BigDecimal] :longitude The object latitude in WGS 84
|
41
|
+
# format.
|
42
|
+
# @option event [BigDecimal] :altitude The object altitude above sea level
|
43
|
+
# in meters to 1 decimal place.
|
44
|
+
def update_object(event)
|
45
|
+
@event_count += 1
|
46
|
+
events << { type: :update_object, event: event, time: @time }
|
47
|
+
end
|
48
|
+
|
49
|
+
# Process a delete event for an object
|
50
|
+
#
|
51
|
+
# @param object_id [String] A hexadecimal number representing the object
|
52
|
+
# ID
|
53
|
+
def delete_object(object_id)
|
54
|
+
@event_count += 1
|
55
|
+
events << { type: :delete_object, object_id: object_id }
|
56
|
+
end
|
57
|
+
|
58
|
+
# Process a time update event
|
59
|
+
#
|
60
|
+
# @param time [BigDecimal] A time update in seconds from the
|
61
|
+
# ReferenceTime to 2 decimal places
|
62
|
+
def update_time(time)
|
63
|
+
@time = @reference_time + time
|
64
|
+
end
|
65
|
+
|
66
|
+
# Set a property
|
67
|
+
#
|
68
|
+
# @param property [String] The name of the property to be set
|
69
|
+
# @param value [String] The value of the property to be set
|
70
|
+
def set_property(property:, value:)
|
71
|
+
case property
|
72
|
+
when 'ReferenceLatitude'
|
73
|
+
self.reference_latitude = BigDecimal(value)
|
74
|
+
when 'ReferenceLongitude'
|
75
|
+
self.reference_longitude = BigDecimal(value)
|
76
|
+
when 'ReferenceTime'
|
77
|
+
@reference_time = @time = Time.parse(value)
|
78
|
+
end
|
79
|
+
end
|
80
|
+
end
|
81
|
+
end
|
data/lib/tac_scribe.rb
ADDED
data/scratchpad
ADDED
@@ -0,0 +1,11 @@
|
|
1
|
+
# Request Bogey Dope
|
2
|
+
SELECT b.id, ST_AsText(b.position) AS human_position, b.position AS binary_position, b.altitude, b.heading, b.group, b.type, ST_DISTANCE(sub.position, b.position) as distance, degrees(ST_AZIMUTH(sub.position, b.position)) as bearing
|
3
|
+
FROM public.units AS b CROSS JOIN LATERAL
|
4
|
+
( SELECT s.position, s.coalition
|
5
|
+
FROM public.units AS s
|
6
|
+
WHERE ID = '102'
|
7
|
+
) as sub
|
8
|
+
WHERE NOT b.coalition = sub.coalition
|
9
|
+
AND b.type LIKE 'Air+%'
|
10
|
+
ORDER BY sub.position <-> b.position ASC
|
11
|
+
LIMIT 10
|
data/tac_scribe.gemspec
ADDED
@@ -0,0 +1,49 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
lib = File.expand_path('lib', __dir__)
|
4
|
+
$LOAD_PATH.unshift(lib) unless $LOAD_PATH.include?(lib)
|
5
|
+
require 'tac_scribe/version'
|
6
|
+
|
7
|
+
Gem::Specification.new do |spec|
|
8
|
+
spec.name = 'tac_scribe'
|
9
|
+
spec.version = TacScribe::VERSION
|
10
|
+
spec.authors = ['Jeffrey Jones']
|
11
|
+
spec.email = ['jeff@jones.be']
|
12
|
+
|
13
|
+
spec.summary = 'Write Tacview data to PostGIS database'
|
14
|
+
spec.description = 'Write Tacview data to PostGIS database'
|
15
|
+
spec.homepage = 'https://gitlab.com/overlord-bot/tac_scribe'
|
16
|
+
spec.license = 'AGPL-3.0-or-later'
|
17
|
+
|
18
|
+
spec.files = `git ls-files -z`.split("\x0").reject do |f|
|
19
|
+
f.match(%r{^(test|spec|features)/})
|
20
|
+
end
|
21
|
+
spec.bindir = 'exe'
|
22
|
+
spec.executables = spec.files.grep(%r{^exe/}) { |f| File.basename(f) }
|
23
|
+
spec.require_paths = ['lib']
|
24
|
+
|
25
|
+
spec.metadata['yard.run'] = 'yri'
|
26
|
+
|
27
|
+
if RUBY_PLATFORM == 'java'
|
28
|
+
# TODO: Specifying a verison chokes bundler for some reason
|
29
|
+
spec.platform = 'java'
|
30
|
+
spec.add_dependency 'activerecord-jdbcpostgresql-adapter'
|
31
|
+
else
|
32
|
+
spec.add_dependency 'pg', '~>1.1'
|
33
|
+
end
|
34
|
+
|
35
|
+
spec.add_dependency 'georuby', '~>2.5'
|
36
|
+
spec.add_dependency 'sequel', '~>5.22'
|
37
|
+
# The following gem is newish and not heavily updated so the chances
|
38
|
+
# of an API breaking change are higher then normal. Therefore lock the
|
39
|
+
# version
|
40
|
+
spec.add_dependency 'sequel-postgis-georuby', '0.1.2'
|
41
|
+
spec.add_dependency 'tacview_client', '~>0.1'
|
42
|
+
|
43
|
+
spec.add_development_dependency 'bundler', '~> 2.0'
|
44
|
+
spec.add_development_dependency 'rake', '~> 10.0'
|
45
|
+
spec.add_development_dependency 'rspec', '~> 3.8'
|
46
|
+
spec.add_development_dependency 'rubocop', '~>0.73'
|
47
|
+
spec.add_development_dependency 'simplecov', '~>0.17'
|
48
|
+
spec.add_development_dependency 'yard', '~>0.9'
|
49
|
+
end
|
metadata
ADDED
@@ -0,0 +1,226 @@
|
|
1
|
+
--- !ruby/object:Gem::Specification
|
2
|
+
name: tac_scribe
|
3
|
+
version: !ruby/object:Gem::Version
|
4
|
+
version: 0.2.0
|
5
|
+
platform: java
|
6
|
+
authors:
|
7
|
+
- Jeffrey Jones
|
8
|
+
autorequire:
|
9
|
+
bindir: exe
|
10
|
+
cert_chain: []
|
11
|
+
date: 2019-10-12 00:00:00.000000000 Z
|
12
|
+
dependencies:
|
13
|
+
- !ruby/object:Gem::Dependency
|
14
|
+
requirement: !ruby/object:Gem::Requirement
|
15
|
+
requirements:
|
16
|
+
- - ">="
|
17
|
+
- !ruby/object:Gem::Version
|
18
|
+
version: '0'
|
19
|
+
name: activerecord-jdbcpostgresql-adapter
|
20
|
+
prerelease: false
|
21
|
+
type: :runtime
|
22
|
+
version_requirements: !ruby/object:Gem::Requirement
|
23
|
+
requirements:
|
24
|
+
- - ">="
|
25
|
+
- !ruby/object:Gem::Version
|
26
|
+
version: '0'
|
27
|
+
- !ruby/object:Gem::Dependency
|
28
|
+
requirement: !ruby/object:Gem::Requirement
|
29
|
+
requirements:
|
30
|
+
- - "~>"
|
31
|
+
- !ruby/object:Gem::Version
|
32
|
+
version: '2.5'
|
33
|
+
name: georuby
|
34
|
+
prerelease: false
|
35
|
+
type: :runtime
|
36
|
+
version_requirements: !ruby/object:Gem::Requirement
|
37
|
+
requirements:
|
38
|
+
- - "~>"
|
39
|
+
- !ruby/object:Gem::Version
|
40
|
+
version: '2.5'
|
41
|
+
- !ruby/object:Gem::Dependency
|
42
|
+
requirement: !ruby/object:Gem::Requirement
|
43
|
+
requirements:
|
44
|
+
- - "~>"
|
45
|
+
- !ruby/object:Gem::Version
|
46
|
+
version: '5.22'
|
47
|
+
name: sequel
|
48
|
+
prerelease: false
|
49
|
+
type: :runtime
|
50
|
+
version_requirements: !ruby/object:Gem::Requirement
|
51
|
+
requirements:
|
52
|
+
- - "~>"
|
53
|
+
- !ruby/object:Gem::Version
|
54
|
+
version: '5.22'
|
55
|
+
- !ruby/object:Gem::Dependency
|
56
|
+
requirement: !ruby/object:Gem::Requirement
|
57
|
+
requirements:
|
58
|
+
- - '='
|
59
|
+
- !ruby/object:Gem::Version
|
60
|
+
version: 0.1.2
|
61
|
+
name: sequel-postgis-georuby
|
62
|
+
prerelease: false
|
63
|
+
type: :runtime
|
64
|
+
version_requirements: !ruby/object:Gem::Requirement
|
65
|
+
requirements:
|
66
|
+
- - '='
|
67
|
+
- !ruby/object:Gem::Version
|
68
|
+
version: 0.1.2
|
69
|
+
- !ruby/object:Gem::Dependency
|
70
|
+
requirement: !ruby/object:Gem::Requirement
|
71
|
+
requirements:
|
72
|
+
- - "~>"
|
73
|
+
- !ruby/object:Gem::Version
|
74
|
+
version: '0.1'
|
75
|
+
name: tacview_client
|
76
|
+
prerelease: false
|
77
|
+
type: :runtime
|
78
|
+
version_requirements: !ruby/object:Gem::Requirement
|
79
|
+
requirements:
|
80
|
+
- - "~>"
|
81
|
+
- !ruby/object:Gem::Version
|
82
|
+
version: '0.1'
|
83
|
+
- !ruby/object:Gem::Dependency
|
84
|
+
requirement: !ruby/object:Gem::Requirement
|
85
|
+
requirements:
|
86
|
+
- - "~>"
|
87
|
+
- !ruby/object:Gem::Version
|
88
|
+
version: '2.0'
|
89
|
+
name: bundler
|
90
|
+
prerelease: false
|
91
|
+
type: :development
|
92
|
+
version_requirements: !ruby/object:Gem::Requirement
|
93
|
+
requirements:
|
94
|
+
- - "~>"
|
95
|
+
- !ruby/object:Gem::Version
|
96
|
+
version: '2.0'
|
97
|
+
- !ruby/object:Gem::Dependency
|
98
|
+
requirement: !ruby/object:Gem::Requirement
|
99
|
+
requirements:
|
100
|
+
- - "~>"
|
101
|
+
- !ruby/object:Gem::Version
|
102
|
+
version: '10.0'
|
103
|
+
name: rake
|
104
|
+
prerelease: false
|
105
|
+
type: :development
|
106
|
+
version_requirements: !ruby/object:Gem::Requirement
|
107
|
+
requirements:
|
108
|
+
- - "~>"
|
109
|
+
- !ruby/object:Gem::Version
|
110
|
+
version: '10.0'
|
111
|
+
- !ruby/object:Gem::Dependency
|
112
|
+
requirement: !ruby/object:Gem::Requirement
|
113
|
+
requirements:
|
114
|
+
- - "~>"
|
115
|
+
- !ruby/object:Gem::Version
|
116
|
+
version: '3.8'
|
117
|
+
name: rspec
|
118
|
+
prerelease: false
|
119
|
+
type: :development
|
120
|
+
version_requirements: !ruby/object:Gem::Requirement
|
121
|
+
requirements:
|
122
|
+
- - "~>"
|
123
|
+
- !ruby/object:Gem::Version
|
124
|
+
version: '3.8'
|
125
|
+
- !ruby/object:Gem::Dependency
|
126
|
+
requirement: !ruby/object:Gem::Requirement
|
127
|
+
requirements:
|
128
|
+
- - "~>"
|
129
|
+
- !ruby/object:Gem::Version
|
130
|
+
version: '0.73'
|
131
|
+
name: rubocop
|
132
|
+
prerelease: false
|
133
|
+
type: :development
|
134
|
+
version_requirements: !ruby/object:Gem::Requirement
|
135
|
+
requirements:
|
136
|
+
- - "~>"
|
137
|
+
- !ruby/object:Gem::Version
|
138
|
+
version: '0.73'
|
139
|
+
- !ruby/object:Gem::Dependency
|
140
|
+
requirement: !ruby/object:Gem::Requirement
|
141
|
+
requirements:
|
142
|
+
- - "~>"
|
143
|
+
- !ruby/object:Gem::Version
|
144
|
+
version: '0.17'
|
145
|
+
name: simplecov
|
146
|
+
prerelease: false
|
147
|
+
type: :development
|
148
|
+
version_requirements: !ruby/object:Gem::Requirement
|
149
|
+
requirements:
|
150
|
+
- - "~>"
|
151
|
+
- !ruby/object:Gem::Version
|
152
|
+
version: '0.17'
|
153
|
+
- !ruby/object:Gem::Dependency
|
154
|
+
requirement: !ruby/object:Gem::Requirement
|
155
|
+
requirements:
|
156
|
+
- - "~>"
|
157
|
+
- !ruby/object:Gem::Version
|
158
|
+
version: '0.9'
|
159
|
+
name: yard
|
160
|
+
prerelease: false
|
161
|
+
type: :development
|
162
|
+
version_requirements: !ruby/object:Gem::Requirement
|
163
|
+
requirements:
|
164
|
+
- - "~>"
|
165
|
+
- !ruby/object:Gem::Version
|
166
|
+
version: '0.9'
|
167
|
+
description: Write Tacview data to PostGIS database
|
168
|
+
email:
|
169
|
+
- jeff@jones.be
|
170
|
+
executables:
|
171
|
+
- start_scribe
|
172
|
+
extensions: []
|
173
|
+
extra_rdoc_files: []
|
174
|
+
files:
|
175
|
+
- ".gitignore"
|
176
|
+
- ".gitlab-ci.yml"
|
177
|
+
- ".rspec"
|
178
|
+
- ".rubocop.yml"
|
179
|
+
- ".ruby-gemset"
|
180
|
+
- ".ruby-version"
|
181
|
+
- ".travis.yml"
|
182
|
+
- CHANGELOG.md
|
183
|
+
- Gemfile
|
184
|
+
- LICENSE.txt
|
185
|
+
- README.md
|
186
|
+
- Rakefile
|
187
|
+
- bin/console
|
188
|
+
- bin/setup
|
189
|
+
- data/airfields.json
|
190
|
+
- db/001_create_unit_table.rb
|
191
|
+
- db/README.md
|
192
|
+
- exe/start_scribe
|
193
|
+
- lib/tac_scribe.rb
|
194
|
+
- lib/tac_scribe/daemon.rb
|
195
|
+
- lib/tac_scribe/datastore.rb
|
196
|
+
- lib/tac_scribe/event_processor.rb
|
197
|
+
- lib/tac_scribe/event_queue.rb
|
198
|
+
- lib/tac_scribe/version.rb
|
199
|
+
- scratchpad
|
200
|
+
- tac_scribe.gemspec
|
201
|
+
homepage: https://gitlab.com/overlord-bot/tac_scribe
|
202
|
+
licenses:
|
203
|
+
- AGPL-3.0-or-later
|
204
|
+
metadata:
|
205
|
+
yard.run: yri
|
206
|
+
post_install_message:
|
207
|
+
rdoc_options: []
|
208
|
+
require_paths:
|
209
|
+
- lib
|
210
|
+
required_ruby_version: !ruby/object:Gem::Requirement
|
211
|
+
requirements:
|
212
|
+
- - ">="
|
213
|
+
- !ruby/object:Gem::Version
|
214
|
+
version: '0'
|
215
|
+
required_rubygems_version: !ruby/object:Gem::Requirement
|
216
|
+
requirements:
|
217
|
+
- - ">="
|
218
|
+
- !ruby/object:Gem::Version
|
219
|
+
version: '0'
|
220
|
+
requirements: []
|
221
|
+
rubyforge_project:
|
222
|
+
rubygems_version: 2.7.6
|
223
|
+
signing_key:
|
224
|
+
specification_version: 4
|
225
|
+
summary: Write Tacview data to PostGIS database
|
226
|
+
test_files: []
|