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 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
@@ -0,0 +1,11 @@
1
+ /.bundle/
2
+ /.yardoc
3
+ /_yardoc/
4
+ /coverage/
5
+ /doc/
6
+ /pkg/
7
+ /spec/reports/
8
+ /tmp/
9
+ Gemfile.lock
10
+ *.swp
11
+ *.gem
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
@@ -0,0 +1,5 @@
1
+ sudo: false
2
+ language: ruby
3
+ rvm:
4
+ - 2.3.6
5
+ before_install: gem install bundler -v 1.16.1
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
@@ -0,0 +1,7 @@
1
+ # frozen_string_literal: true
2
+
3
+ source 'https://rubygems.org'
4
+
5
+ git_source(:github) { |repo_name| "https://github.com/#{repo_name}" }
6
+
7
+ gemspec
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
@@ -0,0 +1,11 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'bundler/gem_tasks'
4
+ require 'rubocop/rake_task'
5
+ require 'rspec/core/rake_task'
6
+
7
+ RSpec::Core::RakeTask.new(:spec)
8
+ RuboCop::RakeTask.new(:rubocop)
9
+
10
+ task default: :rubocop
11
+ task default: :spec
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
@@ -0,0 +1,8 @@
1
+ #!/usr/bin/env bash
2
+ set -euo pipefail
3
+ IFS=$'\n\t'
4
+ set -vx
5
+
6
+ bundle install
7
+
8
+ # Do any other automated setup that you need to do here
@@ -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
@@ -0,0 +1,5 @@
1
+ # frozen_string_literal: true
2
+
3
+ module TacScribe
4
+ VERSION = '0.2.0'
5
+ end
data/lib/tac_scribe.rb ADDED
@@ -0,0 +1,8 @@
1
+ # frozen_string_literal: true
2
+
3
+ require_relative 'tac_scribe/version'
4
+ require_relative 'tac_scribe/daemon'
5
+
6
+ # Top-level name space for all TacScribe clases
7
+ module TacScribe
8
+ end
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
@@ -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: []