aipp 2.1.0 → 2.1.2

Sign up to get free protection for your applications and to get access to all the features.
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: d141c73e057eda545578568c2817d648c998e56154f1d68ecd40c59589a04738
4
- data.tar.gz: 70986dde2f1a7120f49061a65ee1b07c980469ff5f9ea6da4527f0160f613ed8
3
+ metadata.gz: cb4a7ae75e5b180c62d60b63a28ef1e28de3d23e0fe008defc4cc7bc0cf61f14
4
+ data.tar.gz: 6705dbfab25adfb9c71b1781c8ded9106ab1adc6906ec67f3c1330beb73de3e7
5
5
  SHA512:
6
- metadata.gz: 1e97fc5731a4356836ce60703cddd076d33d99d94cfaa667a8f1cf203f28003710d3c27263efd8e638e902f6ee9a076c46fb36881575425bf994202842b2a3d3
7
- data.tar.gz: '08648422f96c2dd925873ec4153b3420754323704df5ec037042bed1f5a4611929d01ae2ee6ab02b718e6e28f9ea57ef76481eaf7ab4abe0a6dd0d89a547fa90'
6
+ metadata.gz: f394783a21b6aaac134155d06c1e3f5c082dcfdd835d0cb3abf708bc1a417fcc2a4bebf382fa1cc70b82c8fe8fa440cacb6a2f73eb73a1efbd0d3b72c437d5e4
7
+ data.tar.gz: 6c9d845076a9c2f894c2d4e586f66a32616d1a1903beb54423844b26d12e930bf85089b72494abe51a47c2701450044ef35508b0221c994d9ed512e3a821b6be
checksums.yaml.gz.sig CHANGED
@@ -1,2 +1,2 @@
1
- 1�ǵm��Gy
2
- Z��ѳ�y@�'�i��:��x��� 琇�� �����%�\
1
+ M{��T��8(��3;p =[PR��oS�WoE���ʦ�q�3���U~���
2
+ \�ܚ)��o�>�g���
data/CHANGELOG.md CHANGED
@@ -2,6 +2,19 @@
2
2
 
3
3
  Nothing so far
4
4
 
5
+ ## 2.1.2
6
+
7
+ #### Changes
8
+ * Improve timesheet calculation for shooting grounds in LS
9
+ * Improve safety margins for shooting grounds in LS
10
+ * Switch to unzipped OFMX for now due to issues upstream
11
+
12
+ ## 2.1.1
13
+
14
+ #### Additions
15
+ * Improve help when no scope is given
16
+ * Add `-0` to write empty OFMX in case of no upstream data
17
+
5
18
  ## 2.1.0
6
19
 
7
20
  #### Breaking Changes
@@ -5,30 +5,35 @@ module AIPP
5
5
  include AIPP::Debugger
6
6
 
7
7
  def initialize(exe_file)
8
+ @exe_file = exe_file
9
+ help if ARGV.none? || ARGV.first == '--help'
8
10
  require_scope
9
11
  AIPP.options.replace(
10
12
  scope: scope,
11
- schema: exe_file.split('2').last.to_sym,
13
+ schema: schema,
12
14
  storage: Pathname(Dir.home).join('.aipp'),
15
+ check_links: false,
13
16
  clean: false,
14
17
  force: false,
15
18
  mid: false,
19
+ write_empty: false,
16
20
  quiet: false,
17
21
  verbose: false,
18
22
  debug_on_warning: false,
19
23
  debug_on_error: false
20
24
  )
21
- options
25
+ options if respond_to? :options
22
26
  OptionParser.new do |o|
23
27
  o.on('-r', '--region STRING', String, 'region (e.g. "LF")') { AIPP.options.region = _1.upcase }
24
28
  o.on('-s', '--section STRING', String, 'process this section only') { AIPP.options.section = _1.classify }
25
29
  o.on('-d', '--storage DIR', String, 'storage directory (default: "~/.aipp")') { AIPP.options.storage = Pathname(_1) }
26
30
  o.on('-o', '--output FILE', String, 'output file') { AIPP.options.output_file = _1 }
27
31
  option_parser(o)
28
- if AIPP.options.schema == :ofmx
32
+ if schema == :ofmx
29
33
  o.on('-m', '--[no-]mid', 'insert mid attributes into all Uid elements (default: false)') { AIPP.options.mid = _1 }
34
+ o.on('-0', '--[no-]empty', 'write empty OFMX files in case of no upstream data (default: false)') { AIPP.options.write_empty = _1 }
30
35
  end
31
- o.on('-h', '--[no-]check-links', 'check all links with HEAD requests') { AIPP.options.check_links = _1 }
36
+ o.on('-h', '--[no-]check-links', 'check all links with HEAD requests (default: false)') { AIPP.options.check_links = _1 }
32
37
  o.on('-c', '--[no-]clean', 'clean cache and download from sources anew (default: false)') { AIPP.options.clean = _1 }
33
38
  o.on('-f', '--[no-]force', 'continue on non-fatal errors (default: false)') { AIPP.options.force = _1 }
34
39
  o.on('-q', '--[no-]quiet', 'suppress all informational output (default: false)') { AIPP.options.quiet = _1 }
@@ -55,6 +60,14 @@ module AIPP
55
60
 
56
61
  private
57
62
 
63
+ def help
64
+ puts <<~END
65
+ Download online aeronautical data and convert it to #{schema.upcase}.
66
+ Usage: #{File.basename($0)} [aip|notam|shoot] --help
67
+ END
68
+ exit
69
+ end
70
+
58
71
  def about
59
72
  puts 'Written by Sven Schwyn (bitcetera.com) and distributed under MIT license.'
60
73
  exit
@@ -90,6 +103,10 @@ module AIPP
90
103
  Pathname(__FILE__).dirname
91
104
  end
92
105
 
106
+ def schema
107
+ @schema ||= @exe_file.split('2').last.to_sym
108
+ end
109
+
93
110
  def scope
94
111
  @scope ||= ARGV.first.match?(/^-/) ? 'AIP' : ARGV.shift.upcase
95
112
  end
@@ -92,6 +92,14 @@ TOT | 0 | Row | text(3) | yes | "TOT" (aka: Total)
92
92
  TOT | 1 | Created at | datetime(yyyymmddhhmmss) | yes | Date and time of CSV creation
93
93
  TOT | 2 | SPL count | number | yes | Number of SPL records
94
94
 
95
+ ### Safety Margins
96
+
97
+ The max height (BSZ col 15) has to be treated as an advisory value, not a guarantee:
98
+
99
+ If no value is given, the ammunition *should* not exceed 250m above ground. However, regulations for certain ammunition types allow for higher peaks. As of March 2023, 6cm mortars *may* reach up to 500m and future weapon systems *may* even go beyond that.
100
+
101
+ To account for this, the two constants `DEFAULT_Z` and `SAFETY` should be revisited from time to time.
102
+
95
103
  ## Asynchronous Use
96
104
 
97
105
  ### Command Line Arguments
@@ -48,9 +48,12 @@ module AIPP
48
48
  fail "not yet implemented"
49
49
  when 'AIP'
50
50
  AIPP::Downloader::HTTP.new(
51
- archive: "https://snapshots.openflightmaps.org/live/#{AIRAC::Cycle.new.id}/ofmx/lsas/latest/ofmx_ls.zip",
52
- file: "ofmx_ls/isolated/ofmx_ls.ofmx"
51
+ file: "https://storage.googleapis.com/snapshots.openflightmaps.org/live/#{AIRAC::Cycle.new.id}/ofmx/lsas/latest/isolated/ofmx_ls.xml"
53
52
  )
53
+ # AIPP::Downloader::HTTP.new(
54
+ # archive: "https://snapshots.openflightmaps.org/live/#{AIRAC::Cycle.new.id}/ofmx/lsas/latest/ofmx_ls.zip",
55
+ # file: "ofmx_ls/isolated/ofmx_ls.ofmx"
56
+ # )
54
57
  when 'DABS'
55
58
  if aixm.effective_at.to_date == Date.today # DABS cross check works reliably for today only
56
59
  AIPP::Downloader::HTTP.new(
@@ -5,33 +5,40 @@ module AIPP::LS::SHOOT
5
5
 
6
6
  include AIPP::LS::Helpers::Base
7
7
 
8
- DEFAULT_Z = AIXM.z(1000, :qfe) # height 300m unless present in publication
8
+ DEFAULT_Z = AIXM.z(2000, :qfe) # fallback if no max height is defined
9
+ SAFETY = 100 # safety margin in meters added to max height
9
10
 
10
11
  def parse
11
- effective_date = aixm.effective_at.strftime('%Y%m%d')
12
+ effective_date = AIPP.options.local_effective_at.strftime('%Y%m%d')
12
13
  airac_date = AIRAC::Cycle.new(aixm.effective_at).to_s('%Y-%m-%d')
13
14
  shooting_grounds = {}
14
15
  read.each_with_index do |row, line|
15
16
  type, id, date, no_shooting = row[0], row[1], row[2], (row[17] == "1")
16
- if type == 'BSZ' && !no_shooting && date == effective_date
17
- shooting_grounds[id] ||= read("shooting_grounds-#{id}")
18
- .fetch(:feature)
19
- .merge(
20
- csv_line: line,
21
- location_codes: row[5].split(/ *, */), # TODO: currently ignored - not available as separate geometries
22
- details: row[6].blank_to_nil,
23
- url: row[10].blank_to_nil,
24
- upper_z: (AIXM.z(AIXM.d(row[15].to_i, :m).to_ft.dim.round, :qfe) if row[15]),
25
- dabs: (row[16] == '1'),
26
- schedules: []
27
- )
28
- shooting_grounds[id][:schedules] << schedule_for(row)
17
+ next unless type == 'BSZ'
18
+ next if no_shooting || date != effective_date
19
+ next if AIPP.options.id && AIPP.options.id != id
20
+ # TODO: Several BSZ lines may exist for the same shooting area with
21
+ # different location codes (aka: partial activations). The geometries
22
+ # of those location codes are not currently available, we therefore
23
+ # have to merge the data into one record. The geometries should become
24
+ # available by the end of 2023 which will make it possible to map
25
+ # each line to one geometry and remove the merging logic.
26
+ upper_z = row[15] ? AIXM.z(AIXM.d(row[15].to_i + SAFETY, :m).to_ft.dim.round, :qfe) : DEFAULT_Z
27
+ (shooting_grounds[id] ||= { schedules: [], upper_z: AIXM::GROUND }).then do |s|
28
+ s[:feature] ||= read("shooting_grounds-#{id}").fetch(:feature)
29
+ s[:csv_line] ||= line
30
+ s[:url] ||= row[10].blank_to_nil
31
+ s[:details] = [s[:details], row[6].blank_to_nil].compact.join("\n")
32
+ s[:dabs] ||= (row[16] == '1')
33
+ s[:upper_z] = upper_z if upper_z.alt > s[:upper_z].alt
34
+ s[:schedules] += schedules_for(row)
29
35
  end
30
36
  end
31
37
  shooting_grounds.each do |id, data|
32
- data in csv_line:, location_codes:, details:, url:, upper_z:, schedules:, properties: { bezeichnung: name, infotelefonnr: phone, infoemail: email }
33
- if schedules.compact.any?
34
- geometries = geometries_for data[:geometry]
38
+ data in csv_line:, details:, url:, upper_z:, schedules:, dabs:, feature: { geometry: polygons, properties: { bezeichnung: name, infotelefonnr: phone, infoemail: email } }
39
+ schedules = consolidate(schedules)
40
+ if schedules.any?
41
+ geometries = geometries_for polygons
35
42
  indexed = geometries.count > 1
36
43
  geometries.each_with_index do |geometry, index|
37
44
  remarks = {
@@ -49,7 +56,7 @@ module AIPP::LS::SHOOT
49
56
  ).tap do |airspace|
50
57
  airspace.add_layer layer_for(upper_z, schedules, remarks)
51
58
  airspace.geometry = geometry
52
- airspace.comment = "DABS: marked for publication"
59
+ airspace.comment = "DABS: marked for publication" if dabs
53
60
  end
54
61
  )
55
62
  end
@@ -59,18 +66,34 @@ module AIPP::LS::SHOOT
59
66
 
60
67
  private
61
68
 
62
- # Returns +nil+ if neither beginning nor ending time is declared which
63
- # has to be treated as "no shooting".
64
- def schedule_for(row)
65
- from = AIXM.time("#{row[3]} #{AIPP.options.time_zone}") if row[3]
66
- to = AIXM.time("#{row[4]} #{AIPP.options.time_zone}") if row[4]
67
- case
68
- when from && to then (from..to)
69
- when from then (from..AIXM::END_OF_DAY)
70
- when to then (AIXM::BEGINNING_OF_DAY..to)
69
+ def schedules_for(row)
70
+ from, to = time_for(row[3]), time_for(row[4], ending: true)
71
+ if from.to_date == to.to_date
72
+ [
73
+ [AIXM.date(from), (AIXM.time(from)..AIXM.time(to))]
74
+ ]
75
+ else
76
+ [
77
+ [AIXM.date(from), (AIXM.time(from)..AIXM::END_OF_DAY)],
78
+ [AIXM.date(to), (AIXM::BEGINNING_OF_DAY..AIXM.time(to))]
79
+ ]
71
80
  end
72
81
  end
73
82
 
83
+ def time_for(string, ending: false)
84
+ hour, min = case string.strip
85
+ when /(\d{2})(\d{2})/
86
+ [$1.to_i, $2.to_i]
87
+ when '', '0'
88
+ [0, 0]
89
+ else
90
+ warn("ignoring malformed time `#{string}'")
91
+ [0, 0]
92
+ end
93
+ hour = 24 if hour.zero? && min.zero? && ending
94
+ AIPP.options.local_effective_at.change(hour: hour, min: min).utc
95
+ end
96
+
74
97
  def geometries_for(polygons)
75
98
  fail "only type MultiPolygon supported" unless polygons[:type] == 'MultiPolygon'
76
99
  fail "polygon coordinates missing" unless polygons[:coordinates]
@@ -97,20 +120,30 @@ module AIPP::LS::SHOOT
97
120
 
98
121
  def timetable_for(schedules)
99
122
  AIXM.timetable.tap do |timetable|
100
- schedules.each do |schedule|
101
- timetable.add_timesheet(
102
- AIXM.timesheet(
103
- adjust_to_dst: true,
104
- dates: (AIXM.date(aixm.effective_at)..AIXM.date(aixm.effective_at))
105
- # TODO: transform to...
106
- # dates: AIXM.date(aixm.effective_at)
107
- ).tap do |timesheet|
108
- timesheet.times = schedule
109
- end
110
- )
123
+ schedules.each do |date, times_array|
124
+ times_array.each do |times|
125
+ timetable.add_timesheet(
126
+ AIXM.timesheet(
127
+ adjust_to_dst: true,
128
+ dates: (date..date)
129
+ # TODO: transform to...
130
+ # dates: AIXM.date(date)
131
+ ).tap do |timesheet|
132
+ timesheet.times = times
133
+ end
134
+ )
135
+ end
111
136
  end
112
137
  end
113
138
  end
114
139
 
140
+ # TODO: Consolidate will become obsolete once location code geometries
141
+ # are available.
142
+ def consolidate(schedules)
143
+ schedules
144
+ .group_by(&:first)
145
+ .transform_values { _1.map(&:last).consolidate_ranges(:time) }
146
+ end
147
+
115
148
  end
116
149
  end
data/lib/aipp/runner.rb CHANGED
@@ -128,9 +128,13 @@ module AIPP
128
128
 
129
129
  # Write the AIXM document.
130
130
  def write_aixm(file)
131
- info("writing #{file}")
132
- AIXM.config.mid = AIPP.options.mid
133
- File.write(file, aixm.to_xml)
131
+ if aixm.features.any? || AIPP.options.write_empty
132
+ info("writing #{file}")
133
+ AIXM.config.mid = AIPP.options.mid
134
+ File.write(file, aixm.to_xml)
135
+ else
136
+ info("no features to write")
137
+ end
134
138
  end
135
139
 
136
140
  # Write build information.
@@ -14,7 +14,7 @@ module AIPP
14
14
  def option_parser(o)
15
15
  o.banner = <<~END
16
16
  Download online AIP and convert it to #{AIPP.options.schema.upcase}.
17
- Usage: #{File.basename($0)} [options]
17
+ Usage: #{File.basename($0)} [aip] [options]
18
18
  END
19
19
  o.on('-a', '--airac (DATE|INTEGER)', String, %Q[AIRAC date or delta e.g. "+1" (default: "#{AIPP.options.airac.date.xmlschema}")]) { AIPP.options.airac = airac_for(_1) }
20
20
  if AIPP.options.schema == :ofmx
@@ -20,10 +20,8 @@ module AIPP
20
20
  if aixm.features.any?
21
21
  validate_aixm
22
22
  write_build
23
- write_aixm(AIPP.options.output_file || output_file)
24
- else
25
- warn("no features to write")
26
23
  end
24
+ write_aixm(AIPP.options.output_file || output_file)
27
25
  write_config
28
26
  end
29
27
 
@@ -19,10 +19,8 @@ module AIPP
19
19
  parse_sections
20
20
  if aixm.features.any?
21
21
  validate_aixm
22
- write_aixm(AIPP.options.output_file || output_file)
23
- else
24
- warn("no features to write")
25
22
  end
23
+ write_aixm(AIPP.options.output_file || output_file)
26
24
  write_config
27
25
  end
28
26
 
@@ -6,7 +6,8 @@ module AIPP
6
6
  def options
7
7
  AIPP.options.merge(
8
8
  module: 'Shoot',
9
- effective_at: Time.now.at_midnight
9
+ local_effective_at: Time.now.at_midnight,
10
+ id: nil
10
11
  )
11
12
  end
12
13
 
@@ -15,12 +16,13 @@ module AIPP
15
16
  Download online shooting activities and convert them to #{AIPP.options.schema.upcase}.
16
17
  Usage: #{File.basename($0)} shoot [options]
17
18
  END
18
- o.on('-t', '--effective (DATE)', String, %Q[effective on this date (default: "#{AIPP.options.effective_at.to_date}")]) { AIPP.options.effective_at = Time.parse("#{_1} CET") }
19
+ o.on('-t', '--effective (DATE)', String, %Q[effective on this date (default: "#{AIPP.options.local_effective_at.to_date}")]) { AIPP.options.local_effective_at = Time.parse("#{_1} CET") }
20
+ o.on('-i', '--id ID', String, %Q[process shooting ground with this ID only]) { AIPP.options.id = _1 }
19
21
  end
20
22
 
21
23
  def guard
22
- AIPP.options.time_zone = AIPP.options.effective_at.at_noon.strftime('%z')
23
- AIPP.options.effective_at = AIPP.options.effective_at.at_midnight.utc
24
+ AIPP.options.time_zone = AIPP.options.local_effective_at.at_noon.strftime('%z')
25
+ AIPP.options.effective_at = AIPP.options.local_effective_at.at_midnight.utc
24
26
  end
25
27
 
26
28
  end
@@ -8,7 +8,7 @@ module AIPP
8
8
  end
9
9
 
10
10
  def expiration_at
11
- effective_at.end_of_day.round - 1
11
+ effective_at + 86399
12
12
  end
13
13
 
14
14
  def run
@@ -19,10 +19,8 @@ module AIPP
19
19
  parse_sections
20
20
  if aixm.features.any?
21
21
  validate_aixm
22
- write_aixm(AIPP.options.output_file || output_file)
23
- else
24
- warn("no features to write")
25
22
  end
23
+ write_aixm(AIPP.options.output_file || output_file)
26
24
  write_config
27
25
  end
28
26
 
data/lib/aipp/version.rb CHANGED
@@ -1,3 +1,3 @@
1
1
  module AIPP
2
- VERSION = "2.1.0".freeze
2
+ VERSION = "2.1.2".freeze
3
3
  end
@@ -10,4 +10,27 @@ class Array
10
10
  map(&:to_s).join('::').constantize
11
11
  end
12
12
 
13
+ # Consolidate array of possibly overlapping ranges.
14
+ #
15
+ # @example
16
+ # [15..17, 7..11, 12..13, 8..12, 12..13].consolidate_ranges
17
+ # # => [7..13, 15..17]
18
+ #
19
+ # @param [Symbol, nil] method to call on range members for comparison
20
+ # @return [Array] consolidated array
21
+ def consolidate_ranges(method=:itself)
22
+ uniq.sort_by { [_1.begin, _1.end] }.then do |ranges|
23
+ consolidated, a = [], ranges.first
24
+ Array(ranges[1..]).each do |b|
25
+ if a.end.send(method) >= b.begin.send(method) # overlapping
26
+ a = (a.begin..(a.end.send(method) > b.end.send(method) ? a.end : b.end))
27
+ else # not overlapping
28
+ consolidated << a
29
+ a = b
30
+ end
31
+ end
32
+ consolidated << a
33
+ end.compact
34
+ end
35
+
13
36
  end
data.tar.gz.sig CHANGED
Binary file
metadata CHANGED
@@ -1,7 +1,7 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: aipp
3
3
  version: !ruby/object:Gem::Version
4
- version: 2.1.0
4
+ version: 2.1.2
5
5
  platform: ruby
6
6
  authors:
7
7
  - Sven Schwyn
@@ -29,7 +29,7 @@ cert_chain:
29
29
  kAyiRqgxF4dJviwtqI7mZIomWL63+kXLgjOjMe1SHxfIPo/0ji6+r1p4KYa7o41v
30
30
  fwIwU1MKlFBdsjkd
31
31
  -----END CERTIFICATE-----
32
- date: 2023-02-27 00:00:00.000000000 Z
32
+ date: 2023-03-23 00:00:00.000000000 Z
33
33
  dependencies:
34
34
  - !ruby/object:Gem::Dependency
35
35
  name: airac
@@ -60,7 +60,7 @@ dependencies:
60
60
  version: '1'
61
61
  - - ">="
62
62
  - !ruby/object:Gem::Version
63
- version: 1.4.0
63
+ version: 1.4.2
64
64
  type: :runtime
65
65
  prerelease: false
66
66
  version_requirements: !ruby/object:Gem::Requirement
@@ -70,7 +70,7 @@ dependencies:
70
70
  version: '1'
71
71
  - - ">="
72
72
  - !ruby/object:Gem::Version
73
- version: 1.4.0
73
+ version: 1.4.2
74
74
  - !ruby/object:Gem::Dependency
75
75
  name: notam
76
76
  requirement: !ruby/object:Gem::Requirement
@@ -442,7 +442,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
442
442
  - !ruby/object:Gem::Version
443
443
  version: '0'
444
444
  requirements: []
445
- rubygems_version: 3.4.7
445
+ rubygems_version: 3.4.9
446
446
  signing_key:
447
447
  specification_version: 4
448
448
  summary: Parser for aeronautical information publications
metadata.gz.sig CHANGED
Binary file