-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathoperator.rb
860 lines (783 loc) · 29.3 KB
/
operator.rb
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
#!/usr/bin/ruby
# OSDV Tabulator - TTV Tabulator Operator
# Author: Jeff Cook
# Date: 1/20/2011
#
# License Version: OSDV Public License 1.2
#
# The contents of this file are subject to the OSDV Public License
# Version 1.2 (the "License"); you may not use this file except in
# compliance with the License. You may obtain a copy of the License at
# http://www.osdv.org/license/
# Software distributed under the License is distributed on an "AS IS"
# basis, WITHOUT WARRANTY OF ANY KIND, either express or implied. See the
# License for the specific language governing rights and limitations
# under the License.
# The Original Code is: TTV Tabulator
# The Initial Developer of the Original Code is Open Source Digital Voting Foundation.
# Portions created by Open Source Digital Voting Foundation are Copyright (C) 2010, 2011.
# All Rights Reserved.
# Contributors: Jeff Cook
require "yaml"
require "lib/syntax_checker"
require "lib/tabulator"
# The Operator class contains the Tabulator Operator. There is only one
# public method, operator_command, which implements the command-line interface
# to the Operator. Methods beginning with "opc_" implement Operator commands.
# Methods beginning with "opx_" handle the system calls, and all are protected
# from exceptions by rescue clauses that generate either a File error, in the
# case the exception was raised while processing an Operator-supplied file, or
# a Fatal error, in the case that the operation being performed was internal
# to the Tabulator and/or Operator and should never cause problems. There is
# one other type of error, a Command error, which occurs when an Operator
# command is syntactically invalid.
#
# The Operator handles errors by raising its locally-defined TabOpError
# exception, passing it the error message. This exception is caught only by
# operator_command, the Operator's single public method and thus its only
# interface, which displays the error message and terminates.
class TabulatorOperator
TABULATOR_DATA_FILE = "TABULATOR_DATA.yml"
TABULATOR_BACKUP_FILE = "TABULATOR_BACKUP.yml"
TABULATOR_SPREADSHEET_FILE = "TABULATOR_SPREADSHEET.csv"
# The TabOpError exception is used internally by the Operator for
# error-handling.
class TabOpError < Exception
end
# Arguments:
# * <i>args</i> (<i>Array</i>) command-line arguments
#
# Returns: <i>String</i>
#
# Implements the command-line interface for the Tabulator Operator. Checks
# all commands for syntactic correctness, and generates Command errors when
# violations occur. Rescues TabOpError exceptions, by displaying their
# error messages and any messages from previously caught System exceptions,
# and then terminating. Finally, any unhandled exceptions are rescued to
# permit a graceful failure in this unlikely event.
def operator_command(args)
if (args.length == 0)
opc_help()
else
cmd = "\"#{args[0]}\""
opx_err("Command #{cmd} has no arguments") if (args.length > 1) &&
["help", "reset", "data", "state", "total"].include?(args[0])
case args[0]
when "help"
opc_help(true)
when "reset"
opc_reset()
when "data"
opc_data()
when "state"
opc_state()
when "total"
opc_total()
when "load"
if (args.length == 3)
opc_load(args[1], args[2])
elsif ((args.length == 4) && (args[3] == "OK"))
opc_load(args[1], args[2], true)
else
opx_err("Command #{cmd} has 2 arguments (file names)")
end
when "add"
if (args.length == 2)
opc_add(args[1])
else
opx_err("Command #{cmd} has 1 argument (file name)")
end
when "check"
if (args.length <= 2)
opc_check((args.length == 1) ? false : args[1])
else
opx_err("Command #{cmd} has 1 optional argument (file name)")
end
else
opx_err("Command #{cmd} not recognized")
end
end
return ""
rescue TabOpError => e
opx_print("** TABOP ERROR ** #{e.message}\n")
return e.message
rescue => e
message = "Fatal Unrecognized Error\n#{e.message}\n"
opx_print("** TABOP ERROR ** #{message}\n")
return message
end
# Arguments:
# * <i>detail</i> (<i>Boolean</i>) whether to print detailed help information (optional)
#
# Returns: N/A
#
# Implements the "help" command. Prints simple or detailed help information
# concerning the Operator commands.
private
def opc_help(detail = false)
opx_print("\n")
if (detail)
opx_print("Note: [DEBUG] indicates a temporary/prototype command option.
")
end
opx_print("Commands:
ruby operator.rb # basic command help information
ruby operator.rb help # detailed command help information
ruby operator.rb reset # reset Tabulator to EMPTY state
ruby operator.rb data # print Tabulator data
ruby operator.rb state # print Tabulator state, one of:
EMPTY, INITIAL, ACCUMULATING, or DONE
ruby operator.rb total # print Tabulator state, show missing counts, and
print spreadsheet with current voting results
ruby operator.rb load <Jurisdiction_Def_File> <Election_Def_File>
")
if (detail)
opx_print("
# The two files must contain, respectively, a Jurisdiction Definition
# and an Election Definition. Each is checked for proper syntax and
# then validated, after which a zero-initialized Tabulator Count is
# constructed and saved to file. This command moves the state of the
# Tabulator from EMPTY to INITIAL.
")
else
opx_print("
# Process two Jurisdiction and Election Definition files to initialize
# the Tabulator, moving its state from EMPTY to INITIAL.
")
end
opx_print("
ruby operator.rb add <Counter_Count_File>
")
if (detail)
opx_print("
# The file contains a Counter Count, rejected if the state of the
# Tabulator is EMPTY. First the Tabulator is re-instantiated using the
# current Tabulator data file. Then the Counter Count is checked for
# proper syntax, validated, and incorporated into the Tabulator dataset,
# which is saved to file. This command allows the Tabulator to
# accumulate votes, and enter the ACCUMULATING state. When the last
# expected count is accumulated, the Tabulator enters the DONE state.
")
else
opx_print("
# Process a Counter Count file to accumulate votes. The Tabulator state
# becomes ACCUMULATING, unless this is the last expected count, in which
# case the Tabulator state is DONE.
")
end
if (detail)
opx_print("
ruby operator.rb check [<Tabulator_Count_File>]
# [DEBUG] The file contains a Tabulator
# Count (the default Tabulator dataset is used if the file is left
# unspecified). It is checked for proper syntax and validated. This
# command is informational only and may be used to check the consistency
# of the current Tabulator dataset.
")
end
opx_print("
Tabulator data file: #{TABULATOR_DATA_FILE}
")
end
# No Arguments
#
# Returns: N/A
#
# Implements the "reset" command. Resets the Tabulator state to EMPTY, by
# backing up the contents of the Tabulator data file from
# <tt><b>TABULATOR_DATA_FILE</b></tt> to
# <tt><b>TABULATOR_BACKUP_FILE</b></tt>.
def opc_reset()
opx_file_backup() unless opx_empty_state?()
opx_print("Tabulator reset to EMPTY state.\n")
end
# Arguments:
# * <i>tab</i> (<i>Class Object</i>) Tabulator object (optional)
#
# Returns: N/A
#
# Implements the "data" command, which is a no-op if the Tabulator is in the
# EMPTY state. Prints Tabulator data held either by the optional Tabulator
# object (<i>tab</i>) provided as input or by re-instantiating the Tabulator
# from its persistent data file <tt><b>TABULATOR_DATA_FILE</b></tt>.
def opc_data(tab = false)
if (tab != false)
opx_print_tabulator_count(tab)
elsif (opx_empty_state?())
opx_err("Command \"data\" ignored, Tabulator state: EMPTY")
else
tab = opx_instantiate_tabulator()
opx_print_tabulator_count(tab)
opx_dump_tabulator_data(tab)
end
end
# Arguments:
# * <i>tab</i> (<i>Class Object</i>) Tabulator object (optional)
# * <i>detail</i> (<i>Boolean</i>) whether to print details about the state (optional)
#
# Returns: N/A
#
# Implements the "state" command. Prints the Tabulator state either using the
# optional Tabulator object (<i>tab</i>) provided as input or by
# re-instantiating the Tabulator from its persistent data file
# <tt><b>TABULATOR_DATA_FILE</b></tt>. If indicated, print detailed
# information about the missing counts (which is only applicable if Expected
# Counts were previously defined).
def opc_state(tab = false, detail = false)
if (opx_empty_state?)
opx_print("Tabulator State: EMPTY\n")
return
end
tab = opx_instantiate_tabulator() unless tab
state, missing, finished, expected = tab.tabulator_state
opx_print("Tabulator State: #{state}\n")
state = state.split(/ /)[0]
if (detail && (state == "ACCUMULATING"))
if (expected == 0)
opx_print("No Missing Counts, because no Expected Counts\n")
return state
end
opx_print("Missing Counts: Counter UID, Precinct UID, Reporting Group\n")
missing.each { |cid, rg, pid| opx_print(" #{cid}, #{pid}, #{rg}\n") }
opx_print("Precincts Finished Reporting: ")
if (finished.length == 0)
opx_print("NONE\n")
else
opx_print("#{finished.inspect.gsub(/[\")\[\]]/,"")}\n")
end
end
state
end
# No Arguments
#
# Returns: N/A
#
# Implements the "total" command. Re-instantiates the Tabulator from its
# persistent data file <tt><b>TABULATOR_DATA_FILE</b></tt>, prints out
# detailed state information, then writes out a Tabulator spreadsheet, in CSV
# (Comma-Separated Value) format, that contains the current vote count. (For
# testing purposes, it also prints the CSV data to STDOUT.)
def opc_total()
opx_err("Command \"total\" ignored, Tabulator state: EMPTY") if opx_empty_state?()
tab = opx_instantiate_tabulator()
state = opc_state(tab, true)
if (["ACCUMULATING", "DONE"].include?(state))
if (state == "DONE")
opx_print("Writing Tabulator Spreadsheet: " +
"#{TABULATOR_SPREADSHEET_FILE}\n")
lines = opx_file_write_spreadsheet(tab, true)
#opx_print("\nFinal Vote Count Data (CSV Spreadsheet Format):\n\n")
#opx_print(lines)
else
lines = opx_file_write_spreadsheet(tab, false)
#opx_print("\nPartial Vote Count Data (CSV Spreadsheet Format):\n\n")
#opx_print(lines)
opx_print("\n")
opx_err("Command \"total\" REJECTED, Counts MISSING (See Above)")
end
end
end
# Arguments:
# * <i>jd_file</i> (<i>String</i>) Jurisdiction Definition (JD) File
# * <i>ed_file</i> (<i>String</i>) Election Definition (JD) File
# * <i>proceed</i> (<i>Boolean</i>) indicates when to proceed without asking the operator (optional, for testing)
#
# Returns: N/A
#
# Implements the "load" command, which is a no-op unless the Tabulator is in
# the EMPTY state. Processes the JD and ED files, checking them for
# existence, correct syntax, and validity. Any Tabulator validation errors
# cause them to be rejected, otherwise they may be used to create a new
# initial Tabulator dataset. In testing mode, <i>proceed</i> is <i>true</i>,
# and they are used without question. Otherwise, the operator is shown a
# Tabulator data summary, asked for confirmation, and they are used only if
# approved, rejected otherwise.
def opc_load(jd_file, ed_file, proceed = false)
opx_err("Command \"load\" ignored, Tabulator state: not EMPTY") unless
opx_empty_state?()
opx_print("Reading Jurisdiction Definition: #{jd_file}\n")
jd = opx_file_process(jd_file, "Jurisdiction Definition",
"jurisdiction_definition")
opx_print("Reading Election Definition: #{ed_file}\n")
ed = opx_file_process(ed_file, "Election Definition", "election_definition")
tab = opx_new_tabulator_jd_ed(jd, ed, TABULATOR_DATA_FILE)
if (tab.validation_errors.length > 0)
opx_print("Jurisdiction and Election Definitions: REJECTED\n")
elsif (proceed)
opx_print("Jurisdiction and Election Definitions: ACCEPTED\n")
opx_file_write_tabulator(tab)
opc_state(tab)
else
opx_print("\n")
opc_data(tab)
opx_warn(tab)
opx_print("** ATTENTION ** ATTENTION **
Carefully examine the data above, then confirm approval to continue [y/n]: ")
answer = STDIN.gets.chomp
opx_print("\n")
if (answer =~ /^[Yy]/)
opx_print("Jurisdiction and Election Definitions: ACCEPTED\n")
opx_file_write_tabulator(tab)
opc_state(tab)
else
opx_print("Jurisdiction and Election Definitions: REJECTED\n")
end
end
end
# Arguments:
# * <i>cc_file</i>: (<i>String</i>) Counter Count file name
#
# Returns: N/A
#
# Implements the "add" command, which is a no-op if the Tabulator is in the
# EMPTY state. Processes the Counter Count file, checking for existence,
# correct syntax, and validity. Any Tabulator validation errors cause the
# Counter Count to be rejected, that is, its votes are not counted.
# Regardless, the Counter Count is incorporated into the Tabulator dataset,
# which is written to the <tt><b>TABULATOR_DATA_FILE</b></tt>.
def opc_add(cc_file)
opx_err("Command \"add\" ignored, Tabulator State: EMPTY\n") if opx_empty_state?()
tab = opx_instantiate_tabulator()
opx_print("Reading Counter Count: #{cc_file}\n")
cc = opx_file_process(cc_file, "Counter Count", "counter_count")
tab.validate_counter_count(cc)
tab.update_tabulator_count(cc)
if (tab.validation_errors.length > 0)
opx_print("Counter Count: REJECTED\n")
else
opx_print("Counter Count: ACCUMULATED\n")
end
opx_file_write_tabulator(tab)
opx_warn(tab)
opc_state(tab)
end
# Arguments:
# * <i>tc_file</i>: (<i>String</i>) Tabulator Count file name (optional)
#
# Returns: N/A
#
# (Temporary, for DEBUG purposes only.) Implements the "check" command, which
# is a no-op if the Tabulator is in the EMPTY state. Processes the Tabulator
# Count file (using <tt><b>TABULATOR_DATA_FILE</b></tt> if <i>tc_file</i> is
# not specified), checking for existence, correct syntax, and validity.
def opc_check(tc_file = false)
fatal = false
if (tc_file == false)
opx_err("Command \"check\" ignored, Tabulator state: EMPTY") if
opx_empty_state?()
tc_file = TABULATOR_DATA_FILE
fatal = true
end
opx_print("Reading Tabulator Count: #{tc_file}\n")
tc = opx_file_process(tc_file, "Tabulator Count", "tabulator_count", fatal)
tab = opx_new_tabulator_tc(tc)
opx_print("Validating Tabulator Count: OK\n") unless
(tab.validation_errors.length > 0 || tab.validation_warnings.length > 0)
opc_state(tab)
end
# Arguments:
# * <i>message</i> (<i>String</i>) error message
# * <i>message</i> (<i>Exception Object</i> (optional) appears when the error was caused by rescuing a System-level exception
#
# Returns: N/A
#
# Handles Operator-detected errors by raising the TabOpError exception.
# Passes the error message to the TabOpError exception, and if the error
# was produced while rescuing another exception, appends the exception's
# message to the end of the error message.
def opx_err(message, e = false)
raise TabOpError.new((e == false ? message : "#{message}\n#{e.message}"))
end
# Arguments:
# * <i>text</i> (<i>String</i>) text to print
#
# Returns: N/A
#
# Prints the <i>text</i> to STDOUT. Generates a Fatal error if problems occur.
def opx_print(text)
print text
rescue => e
opx_err("Fatal error while printing to STDOUT: #{text}", e)
end
# Arguments:
# * <i>tab</i> (<i>Class Object</i>) Tabulator object
#
# Returns: N/A
#
# Prints the YAML dump of the Tabulator Count. Generates a Fatal error if
# problems occur.
def opx_print_tabulator_count(tab)
opx_print("Tabulator Count\n")
opx_print(YAML::dump(tab.tabulator_count))
opx_print("\n")
rescue => e
opx_err("Fatal failure during YAML::dump of Tabulator Count", e)
end
# Arguments:
# * <i>tab</i> (<i>Class Object</i>) Tabulator object
#
# Returns: Tabulator Object
#
# Prints out any error and/or warning messages held by the Tabulator object,
# and then returns it.
def opx_warn(tab)
tab.validation_errors.each {|message| opx_print("** ERROR ** #{message}\n")}
tab.validation_warnings.each {|message| opx_print("** WARNING ** #{message}\n")}
tab
rescue => e
opx_err("Fatal error while fetching Tabulator errors/warnings")
end
# No Arguments
#
# Returns: <i>Boolean</i>
#
# Returns <i>true</i> iff the Tabulator state is EMPTY, as evidenced by the
# non-existence of the <tt><b>TABULATOR_DATA_FILE</b></tt> .
def opx_empty_state?()
! opx_file_exist?(TABULATOR_DATA_FILE)
rescue => e
opx_err("Fatal error while checking Tabulator for EMPTY state")
end
# Arguments:
# * <i>file</i> (<i>String</i>) file name
# * <i>type</i> (<i>String</i>) file type
# * <i>key</i> (<i>String</i>) Hash key expected to appear in file
# * <i>fatal</i>: (<i>Boolean</i>) indicates when errors are Fatal (optional)
#
# Returns: <i>Hash</i>
#
# Processes Tabulator election data (Jurisdiction Definition, Election
# Definition, Counter Count, or Tabulator Count) stored in <i>file</i> and
# returns the resulting datum. The datum undergoes a syntax check against a
# built-in schema before being returned. Generates a File or Fatal error
# (depending on <i>fatal</i>, which is <i>true</i> only when processing the
# <tt><b>TABULATOR_DATA_FILE</b></tt>) if any problems occur.
def opx_file_process(file, type, key, fatal = false)
datum = opx_file_read(file, fatal)
if (!datum.is_a?(Hash))
opx_err((fatal ? "Fatal" : "File") + " contents error, not a Hash: #{file}")
elsif (!datum.keys.include?(key))
opx_err((fatal ? "Fatal" : "File") + " contents error, improper Hash, Key #{key} missing: #{file}")
elsif (opx_check_syntax(key, datum))
datum
else
opx_err((fatal ? "Fatal" : "File") + " syntax error in #{type}: #{file}")
end
rescue => e
opx_err("Fatal failure processing file: #{file}", e)
end
# Arguments:
# * <i>file</i>: (<i>String</i>) file name
#
# Returns: <i>Boolean</i>
#
# Returns <t>true</i> iff the <i>file</i> exists, by returning the result of
# calling File.exist? on the <i>file</i>. Generates a Fatal error if
# any problems occur.
def opx_file_exist?(file)
File.exist?(file)
rescue => e
opx_err("Fatal failure of File.exist? for file: #{file}", e)
end
# No Arguments
#
# Returns: N/A
#
# Backs up the Tabulator data in <tt><b>TABULATOR_DATA_FILE</b></tt> to
# <tt><b>TABULATOR_BACKUP_FILE</b></tt>, by renaming the first file to the
# second. Generates a Fatal error if any problems occur.
def opx_file_backup()
File.rename(TABULATOR_DATA_FILE, TABULATOR_BACKUP_FILE)
rescue => e
opx_err("Fatal failure of File.rename from #{TABULATOR_DATA_FILE} " +
"to #{TABULATOR_BACKUP_FILE}", e)
end
# Arguments:
# * <i>file</i>: (<i>String</i>) file name
# * <i>fatal</i>: (<i>Boolean</i>) indicates when errors are Fatal (optional)
#
# Returns: <i>Hash</i>
#
# Reads Tabulator election data (Jurisdiction Definition, Election Definition,
# Counter Count, or Tabulator Count) from an existing <i>file</i> and returns
# the Hash loaded by the YAML module. Generates a File or Fatal error
# (depending on <i>fatal</i>, which is <i>true</i> only when reading the
# <tt><b>TABULATOR_DATA_FILE</b></tt> or a built-in Schema file) if a problem
# occurs while reading from the file.
def opx_file_read(file, fatal = false)
if (opx_file_exist?(file))
infile = opx_file_open_read(file, fatal)
datum = YAML::load(infile)
opx_file_close(infile)
datum
else
(fatal ?
opx_err("Fatal failure, non-existent file: #{file}") :
opx_err("File non-existent: #{file}"))
end
rescue => e
(fatal ?
opx_err("Fatal failure during YAML::load for file: #{file}", e) :
opx_err("File read error during YAML::load for file: #{file}", e))
end
# Arguments:
# * <i>file</i>: (<i>String</i>) file name
# * <i>fatal</i>: (<i>Boolean</i>) indicates when errors are Fatal (optional)
#
# Returns: <i>File Handle</i>
#
# Opens the <i>file</i> for reading and returns the resulting file handle.
# Generates a File or Fatal error (depending on <i>fatal</i>, which is
# <i>true</i> only when opening the <tt><b>TABULATOR_DATA_FILE</b></tt> or a
# built-in Schema file) if a problem occurs while opening the file.
def opx_file_open_read(file, fatal = false)
File.open(file, "r")
rescue => e
(fatal ?
opx_err("Fatal failure of File.open (for read): #{file}", e) :
opx_err("File open error: #{file}", e))
end
# Arguments:
# * <i>file</i>: (<i>File Handle</i>) file handle
#
# Returns: N/A
#
# Closes the named <i>file</i>. Generates a Fatal error if a problem occurs
# while closing the file.
def opx_file_close(file)
file.close()
rescue => e
opx_err("Fatal failure while closing file: #{file.inspect}", e)
end
# Arguments:
# * <i>file</i>: (<i>String</i>) file name
#
# Returns: <i>File Handle</i>
#
# Opens the <i>file</i> for writing and returns the resulting file handle.
# Generates a Fatal error if a problem occurs while opening the file.
def opx_file_open_write(file)
File.open(file, "w")
rescue => e
opx_err("Fatal failure of File.open for writing: #{file}", e)
end
# Arguments:
# * <i>tab</i>: (<i>Class Object</i>) Tabulator object
# * <i>output</i>: (<i>Boolean</i>) whether or not to write the data to file
#
# Returns: <i>Array</i> of <i>String</i>
#
# Calls the Tabulator to produce the lines of textual data for a CSV
# spreadsheet representing the current set of voting results, which it writes
# to <tt><b>TABULATOR_SPREADSHEET_FILE</b></tt>, and then returns. Generates
# a Fatal error if problems occur during the write.
def opx_file_write_spreadsheet(tab, output)
file = TABULATOR_SPREADSHEET_FILE
lines = tab.tabulator_spreadsheet()
if (output)
outfile = opx_file_open_write(file)
outfile.puts lines
opx_file_close(outfile)
end
lines
rescue => e
opx_err("Fatal failure writing to spreadsheet file: #{file}", e)
end
# Arguments:
# * <i>tab</i>: (<i>Class Object</i>) Tabulator object
#
# Returns: N/A
#
# Writes the current Tabulator data to <tt><b>TABULATOR_DATA_FILE</b></tt>.
# Generates a Fatal error if problems occur while writing the file.
def opx_file_write_tabulator(tab)
outfile = opx_file_open_write(TABULATOR_DATA_FILE)
YAML::dump(tab.tabulator_count, outfile)
opx_file_close(outfile)
rescue => e
opx_err("Fatal failure of YAML::dump for file: #{TABULATOR_DATA_FILE}", e)
end
# No Arguments
#
# Returns: Tabulator Object
#
# Use the contents of the <tt><b>TABULATOR_DATA_FILE</b></tt> to instantiate
# a new Tabulator object, which is returned. Generates a Fatal error if there
# are any Tabulator validation errors or warnings, because this implies that
# Tabulator is in an inconsistent state.
def opx_instantiate_tabulator()
file = TABULATOR_DATA_FILE
tc = opx_file_process(file, "Tabulator Count", "tabulator_count", true)
tab = opx_new_tabulator_tc(tc)
opx_err("Fatal failure during Tabulator validation: #{file}") if
(tab.validation_errors.length > 0 || tab.validation_warnings.length > 0)
tab
rescue => e
opx_err("Fatal failure during Tabulator instantiation", e)
end
# Arguments:
# * <i>jd</i> (<i>Hash</i>) Jurisdiction Definition
# * <i>ed</i> (<i>Hash</i>) Election Definition
# * <i>file</i> (<i>String</i>) Tabulator data file name
#
# Returns: Tabulator Object
#
# Returns the Tabulator Object which results from calling Tabulator.new with
# Jurisdiction and Election Definitions. Generates a Fatal error if any
# problems occur.
def opx_new_tabulator_jd_ed(jd, ed, file)
opx_warn(Tabulator.new(jd, ed, file, false))
rescue => e
opx_err("Fatal failure of Tabulator.new for Jurisdiction/Election Definitions", e)
end
# Arguments:
# * <i>tc</i> (<i>Hash</i>) Tabulator Count
#
# Returns: Tabulator Object
#
# Returns the Tabulator Object which results from calling Tabulator.new with a
# Tabulator Count, re-instantiating the Tabulator. Generates a Fatal error if
# any problems occur.
def opx_new_tabulator_tc(tc)
opx_warn(Tabulator.new(false, false, false, tc))
rescue => e
opx_err("Fatal failure of Tabulator.new for Tabulator Count", e)
end
# Arguments:
# * <i>key</i> (<i>String</i>) Hash key expected in datum and used to identify the schema
# * <i>datum</i> (<i>Hash</i>) Tabulator election data to be syntax-checked against a built-in schema
#
# Returns: <i>Boolean</i>
#
# Returns <i>true</i> iff there are no errors resulting from syntax-checking
# the <i>datum</i> against the schema identified by the <i>key</i>. Generates
# a Fatal error if any problems occur.
def opx_check_syntax(key, datum)
schema_file = "data/Schemas/#{key}_schema.yml"
schema = opx_file_read(schema_file, true)
errors, messages = SyntaxCheckerYaml.new.check_syntax(schema, datum, true)
errors.length == 0
rescue => e
opx_err("Fatal failure of SyntaxCheckerYaml.new.check_syntax(...)", e)
end
end
# Arguments:
# * <i>tab</i>: (<i>Class Object</i>) Tabulator object
#
# Returns: N/A
#
# Prints a summary of the data held by the Tabulator.
def opx_dump_tabulator_data(tab)
print "Tabulator Data Summary\n"
print " Jurisdiction UID: #{tab.uids['jurisdiction'][0]}\n"
print " Election UID: #{tab.uids['election'][0]}\n"
['district','precinct','contest','candidate','question','counter',
'file','reporting group'].each do |k|
length = tab.uids[k].length.to_s
myuids = tab.uids[k].sort
type = (k =~ /^report/ ? "Reporting Groups" : "#{k.capitalize} UIDs")
if (myuids.length == 0)
prefix = " #{type} (NONE)\n"
else
prefix = " #{type} (#{myuids.length.to_s}): "
print prefix
if (myuids.length > 10)
prefix = " "
print "\n#{prefix}"
end
opx_pp(myuids, prefix.length, prefix.length, 78)
end
end
missing = (tab.counts_expected - tab.counts_accumulated)
count = missing.length
total = tab.counts_expected.length
print " Expected Counts "
if (total == 0)
print "(NONE)\n"
else
print "(#{total}): Counter UID, Reporting Group, Precinct UIDs\n"
tab.uids["counter"].sort.each do |cid|
if (tab.counts_expected.any? {|ce| ce[0] == cid})
rgs = tab.uids["reporting group"].sort.select { |rg|
tab.counts_expected.any? {|ce| ce[0] == cid && ce[1] == rg }}
rgs.each { |rg|
pids = tab.uids["precinct"].sort.select { |pid|
tab.counts_expected.any? {|ce| ce == [cid, rg, pid]}}
print " #{cid}, #{rg}, #{pids.uniq.inspect.gsub(/\"/,"")}\n"}
end
end
end
print " Missing Counts "
if (count == 0)
print "(NONE)\n"
else
print "(#{count}): Counter UID, Precinct UID, Reporting Group\n"
missing.each do |cid, rg, pid|
print " #{cid}, #{pid}, #{rg}\n"
end
end
if (tab.counts_contests.keys.length == 0)
print " Contests (NONE)\n"
else
print " Contests (",tab.counts_contests.keys.length.to_s,"):"
print " Contest UID: overvote, undervote, write-in, Candidate UIDs\n"
end
tab.counts_contests.keys.sort.each do |k|
v = tab.counts_contests[k]
print " #{k}: "
print "overvote = #{v["overvote_count"]}, "
print "undervote = #{v["undervote_count"]}, "
print "writeins = #{v["writein_count"]}\n"
v["candidate_count_list"].each do |item|
print " #{item["candidate_ident"]} = #{item["count"]}\n"
end
end
if (tab.counts_questions.keys.length == 0)
print " Questions (NONE)\n"
else
print " Questions (",tab.counts_questions.keys.length.to_s,"):"
print " Question UID: overvote, undervote, Answers\n"
end
tab.counts_questions.keys.sort.each do |k|
v = tab.counts_questions[k]
print " #{k}: "
print "overvote = #{v["overvote_count"]}, "
print "undervote = #{v["undervote_count"]}\n"
v["answer_count_list"].each do |item|
print " #{item["answer"]} = #{item["count"]}\n"
end
end
print "\n"
rescue => e
opx_err("Fatal failure while dumping Tabulator data", e)
end
# Arguments:
# * <i>item</i> (<i>String</i>) typically a UID
# * <i>col</i> (<i>Integer</i>) current output column
# * <i>start_col</i> (<i>Integer</i>) output start column
# * <i>fill_col</i> (<i>Integer</i>) output end column, fill up to this column
#
# Returns: N/A
#
# Pretty-prints the <i>items</i> as a comma-separated list, trying to keep
# them withing the boundaries of <i>start_col</i> and <i>fill_col</i>.
def opx_pp(items, col, start_col, fill_col)
return print("\n") if (items.length == 0)
item = items.shift
print (item = item + (items.length == 0 ? "" : ", "))
col += item.length
if ((items.length > 0) && (fill_col <= (col + items[0].length)))
print "\n"
start_col.times { |x| print(" ") }
col = start_col
end
opx_pp(items, col, start_col, fill_col)
rescue => e
opx_err("Fatal failure while pretty-printing Tabulator data", e)
end
# Command-line interface for the Tabulator Operator.
begin
TabulatorOperator.new.operator_command(ARGV)
rescue => e
print "** TABOP ERROR ** Fatal Unrecognized Error\n#{e.message}\n"
end