fat_table 0.9.7 → 0.9.8

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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 0eecda446a24945c159963cc45088fdd281fa8e67acc995eacea8b0daa9f428b
4
- data.tar.gz: 497feefa41678e0b95c257953c72d7a435cb692d70b4e59eabc681c69a758747
3
+ metadata.gz: baa77411122a8531eae63a312545846cb1a496a808e57aa8e2660dc720950144
4
+ data.tar.gz: d4000cda7c6b01dc18bf0baf8a9b6150882ae2189f2e3a1ccfcc750cc8f952d2
5
5
  SHA512:
6
- metadata.gz: 601e15e198b25d248c049e48484b4fe4c43fb83ee19fa8a7d477c5b1b91d3c99d4f5b5b6a9887bf0e532f93a37ee1616b301154dc0eeac01541bc2ae568d9c66
7
- data.tar.gz: 10168ba22085112da7abe6cfb013554566920fbd0d3e166f3a0ff306b1d460cc611d4e96dd57e473431128a3027e8d33191c692e283c34f0771472938a2f9faa
6
+ metadata.gz: 56752060d0e589fb35b71d31f046a01fd307f6958f6383d6ae564dfcd12981f7ed41223b0266f693df281ef04fb412bc748bb448d7b65fceb4331135a6d87816
7
+ data.tar.gz: f54634a20445d7715cb8608bd4eef5851229e2a2085dd1924bb8279c6f2779db879aec281de57d810715fee388e116abba70ff7f0956d70a8d2f8de9fa76cf01
data/.envrc ADDED
@@ -0,0 +1 @@
1
+ PATH_add .bundle/bin
data/.gitignore CHANGED
@@ -26,3 +26,4 @@
26
26
  /GTAGS
27
27
  /TAGS
28
28
  /README.pyg
29
+ /.ruby-version
data/.rubocop.yml CHANGED
@@ -2,6 +2,9 @@ inherit_from:
2
2
  - ~/.rubocop.yml
3
3
 
4
4
  AllCops:
5
+ Include:
6
+ - 'lib/**/*'
7
+ - 'spec/**/*'
5
8
  Exclude:
6
9
  - 'test/tmp/**/*'
7
10
  - 'vendor/bundle/**/*'
@@ -14,3 +17,9 @@ Style/MethodCallWithArgsParentheses:
14
17
  Style/ClassAndModuleChildren:
15
18
  Exclude:
16
19
  - 'test/**/*'
20
+
21
+ Naming/VariableName:
22
+ Enabled: false
23
+
24
+ RSpec/VariableName:
25
+ Enabled: false
data/Gemfile CHANGED
@@ -14,6 +14,8 @@ gem 'redcarpet'
14
14
  gem 'pg'
15
15
  gem 'sqlite3'
16
16
  gem 'rspec', '~> 3.0'
17
+ gem 'rubocop'
17
18
  gem 'rubocop-rspec'
19
+ gem 'rubocop-rake'
18
20
  gem 'rubocop-performance'
19
21
  gem 'simplecov'
data/Rakefile CHANGED
@@ -2,6 +2,10 @@ require 'bundler/gem_tasks'
2
2
  require 'rspec/core/rake_task'
3
3
  require 'rdoc/task'
4
4
 
5
+ require 'rubocop/rake_task'
6
+
7
+ RuboCop::RakeTask.new
8
+
5
9
  RDoc::Task.new do |rdoc|
6
10
  rdoc.main = 'README.rdoc'
7
11
  rdoc.rdoc_files.include('README.rdoc', 'lib')
@@ -9,4 +13,4 @@ end
9
13
 
10
14
  RSpec::Core::RakeTask.new(:spec)
11
15
 
12
- task default: :spec
16
+ task :default => [:spec, :rubocop]
data/fat_table.gemspec CHANGED
@@ -55,7 +55,7 @@ Gem::Specification.new do |spec|
55
55
  'public gem pushes.'
56
56
  end
57
57
 
58
- spec.files = `git ls-files -z`.split("\x0").reject do |f|
58
+ spec.files = %x(git ls-files -z).split("\x0").reject do |f|
59
59
  f.match(%r{^(test|spec|features)/})
60
60
  end
61
61
  spec.bindir = 'bin'
@@ -63,9 +63,9 @@ Gem::Specification.new do |spec|
63
63
  spec.require_paths = ['lib']
64
64
  spec.metadata['yard.run'] = 'yri' # use "yard" to build full HTML docs.
65
65
 
66
- spec.add_runtime_dependency 'fat_core', '>= 4.9.0'
67
- spec.add_runtime_dependency 'csv'
68
- spec.add_runtime_dependency 'rainbow'
69
- spec.add_runtime_dependency 'sequel'
70
- spec.add_runtime_dependency 'gem-path'
66
+ spec.add_dependency 'csv'
67
+ spec.add_dependency 'fat_core', '>= 4.9.0'
68
+ spec.add_dependency 'gem-path'
69
+ spec.add_dependency 'rainbow'
70
+ spec.add_dependency 'sequel'
71
71
  end
data/lib/ext/array.rb CHANGED
@@ -18,11 +18,11 @@ class Array
18
18
  compact.select { |i| i.is_a?(TrueClass) || i.is_a?(FalseClass) }
19
19
  elsif typ == 'DateTime'
20
20
  compact.select { |i| i.is_a?(Date) || i.is_a?(DateTime) || i.is_a?(Time) }
21
- .map { |i| i.to_datetime }
21
+ .map(&:to_datetime)
22
22
  elsif typ == 'Numeric'
23
23
  compact.select { |i| i.is_a?(Numeric) }
24
24
  elsif typ == 'String'
25
- map { |i| i.to_s }
25
+ map(&:to_s)
26
26
  elsif typ == 'NilClass'
27
27
  self
28
28
  else
@@ -174,9 +174,9 @@ module FatTable
174
174
  # Yield each item in the Column in the order in which they appear in the
175
175
  # Column. This makes Columns Enumerable, so all the Enumerable methods are
176
176
  # available on a Column.
177
- def each
178
- if block_given?
179
- items.each { |itm| yield itm }
177
+ def each(&block)
178
+ if block
179
+ items.each(&block)
180
180
  self
181
181
  else
182
182
  to_enum(:each)
@@ -201,7 +201,7 @@ module FatTable
201
201
  # Return the first non-nil item in the Column, or nil if all items are
202
202
  # nil. Works with any Column type.
203
203
  def first
204
- return nil if items.all?(&:nil?)
204
+ return if items.all?(&:nil?)
205
205
 
206
206
  if type == 'String'
207
207
  items.reject(&:blank?).first
@@ -214,7 +214,7 @@ module FatTable
214
214
 
215
215
  # Return the last non-nil item in the Column. Works with any Column type.
216
216
  def last
217
- return nil if items.all?(&:nil?)
217
+ return if items.all?(&:nil?)
218
218
 
219
219
  if type == 'String'
220
220
  items.reject(&:blank?).last
@@ -231,7 +231,7 @@ module FatTable
231
231
  return items.size if items.all?(&:nil?)
232
232
 
233
233
  if type == 'String'
234
- items.reject(&:blank?).count.to_d
234
+ items.count { |i| !i.blank? }
235
235
  else
236
236
  items.filter_to_type(type).count.to_d
237
237
  end
@@ -270,7 +270,7 @@ module FatTable
270
270
  # Columns.
271
271
  def range
272
272
  only_with('range', 'NilClass', 'Numeric', 'String', 'DateTime')
273
- return nil if items.all?(&:nil?)
273
+ return if items.all?(&:nil?)
274
274
 
275
275
  Range.new(min, max)
276
276
  end
@@ -304,7 +304,7 @@ module FatTable
304
304
  itms = items.filter_to_type(type)
305
305
  size = itms.size.to_d
306
306
  if type == 'DateTime'
307
- avg_jd = itms.map(&:jd).sum / size
307
+ avg_jd = itms.sum(&:jd) / size
308
308
  DateTime.jd(avg_jd)
309
309
  else
310
310
  itms.sum / size
@@ -331,6 +331,7 @@ module FatTable
331
331
  end
332
332
  n = count
333
333
  return BigDecimal('0.0') if n <= 1
334
+
334
335
  mu = Column.new(header: :mu, items: all_items).avg
335
336
  sq_dev = BigDecimal('0.0')
336
337
  all_items.each do |itm|
@@ -353,6 +354,7 @@ module FatTable
353
354
  only_with('var', 'DateTime', 'Numeric')
354
355
  n = items.filter_to_type(type).size.to_d
355
356
  return BigDecimal('0.0') if n <= 1
357
+
356
358
  var * ((n - 1) / n)
357
359
  end
358
360
 
@@ -471,6 +473,7 @@ module FatTable
471
473
  def +(other)
472
474
  msg = 'cannot combine columns with different types'
473
475
  raise UserError, msg unless type == other.type
476
+
474
477
  Column.new(header: header, items: items + other.items)
475
478
  end
476
479
 
@@ -38,6 +38,7 @@ module FatTable
38
38
  if new_val.nil?
39
39
  raise IncompatibleTypeError
40
40
  end
41
+
41
42
  new_val
42
43
  end
43
44
  when 'DateTime'
@@ -48,6 +49,7 @@ module FatTable
48
49
  if new_val.nil?
49
50
  raise IncompatibleTypeError
50
51
  end
52
+
51
53
  new_val
52
54
  end
53
55
  when 'Numeric'
@@ -58,6 +60,7 @@ module FatTable
58
60
  if new_val.nil?
59
61
  raise IncompatibleTypeError
60
62
  end
63
+
61
64
  new_val
62
65
  end
63
66
  when 'String'
@@ -68,6 +71,7 @@ module FatTable
68
71
  if new_val.nil?
69
72
  raise IncompatibleTypeError
70
73
  end
74
+
71
75
  new_val
72
76
  end
73
77
  else
@@ -80,8 +84,10 @@ module FatTable
80
84
  # of case is assumed to be a boolean.
81
85
  def self.convert_to_boolean(val)
82
86
  return val if val.is_a?(TrueClass) || val.is_a?(FalseClass)
87
+
83
88
  val = val.to_s.clean
84
- return nil if val.blank?
89
+ return if val.blank?
90
+
85
91
  if val.match?(/\A(false|f|n|no)\z/i)
86
92
  false
87
93
  elsif val.match?(/\A(true|t|y|yes)\z/i)
@@ -94,7 +100,7 @@ module FatTable
94
100
  (?<dy>\d\d?)\s*
95
101
  (T?\s*\d\d:\d\d(:\d\d)?\s*
96
102
  ([-+](\d\d?)(:\d\d?))?)?\s*
97
- ((mon|tue|wed|thu|fri|sat|sun)[a-zA-z]*)?\s*
103
+ ((mon|tue|wed|thu|fri|sat|sun)[a-zA-Z]*)?\s*
98
104
  \z}xi
99
105
 
100
106
  AMR_DATE_RE = %r{\A(?<dy>\d\d?)[-/](?<mo>\d\d?)[-/](?<yr>\d\d\d\d)\s*
@@ -103,7 +109,7 @@ module FatTable
103
109
  # A Date like 'Tue, 01 Nov 2016' or 'Tue 01 Nov 2016' or '01 Nov 2016'.
104
110
  # These are emitted by Postgresql, so it makes from_sql constructor
105
111
  # possible without special formatting of the dates.
106
- INV_DATE_RE = %r{\A((mon|tue|wed|thu|fri|sat|sun)[a-zA-z]*,?)?\s+ # looks like dow
112
+ INV_DATE_RE = %r{\A((mon|tue|wed|thu|fri|sat|sun)[a-zA-Za-z]*,?)?\s+ # looks like dow
107
113
  (?<dy>\d\d?)\s+ # one or two-digit day
108
114
  (?<mo_name>[jfmasondJFMASOND][A-Za-z]{2,})\s+ # looks like a month name
109
115
  (?<yr>\d\d\d\d) # and a 4-digit year
@@ -122,20 +128,25 @@ module FatTable
122
128
 
123
129
  begin
124
130
  str = val.to_s.clean.sub(/\A[\[\(\{\<]\s*/, '').sub(/[\]\)\}\>]\s*\z/, '')
125
- return nil if str.blank?
131
+ return if str.blank?
126
132
 
127
133
  if str.match(ISO_DATE_RE)
128
134
  date = DateTime.parse(val)
129
135
  elsif str =~ AMR_DATE_RE
130
- date = DateTime.new(Regexp.last_match[:yr].to_i,
131
- Regexp.last_match[:mo].to_i,
132
- Regexp.last_match[:dy].to_i)
136
+ date = DateTime.new(
137
+ Regexp.last_match[:yr].to_i,
138
+ Regexp.last_match[:mo].to_i,
139
+ Regexp.last_match[:dy].to_i,
140
+ )
133
141
  elsif str =~ INV_DATE_RE
134
142
  mo = Date.mo_name_to_num(last_match[:mo_name])
135
- date = DateTime.new(Regexp.last_match[:yr].to_i, mo,
136
- Regexp.last_match[:dy].to_i)
143
+ date = DateTime.new(
144
+ Regexp.last_match[:yr].to_i,
145
+ mo,
146
+ Regexp.last_match[:dy].to_i,
147
+ )
137
148
  else
138
- return nil
149
+ return
139
150
  end
140
151
  # val = val.to_date if
141
152
  date.seconds_since_midnight.zero? ? date.to_date : date
@@ -155,7 +166,7 @@ module FatTable
155
166
  cursym = Regexp.quote(FatTable.currency_symbol)
156
167
  clean_re = /[,_#{cursym}]/
157
168
  val = val.to_s.clean.gsub(clean_re, '')
158
- return nil if val.blank?
169
+ return if val.blank?
159
170
 
160
171
  case val
161
172
  when /\A[-+]?\d+\.\z/
@@ -50,11 +50,14 @@ module FatTable
50
50
  # errors simply return nil as the result. This can happen, for example,
51
51
  # when a string gets into an otherwise numeric column because the column
52
52
  # is set to tolerant.
53
+ #
54
+ # rubocop:disable Security/Eval
53
55
  def evaluate(expr = '', locals: {})
54
56
  eval(expr, local_vars(binding, locals))
55
- rescue NoMethodError, TypeError => ex
57
+ rescue NoMethodError, TypeError
56
58
  nil
57
59
  end
60
+ # rubocop:enable Security/Eval
58
61
 
59
62
  private
60
63
 
@@ -13,18 +13,20 @@ module FatTable
13
13
  # Initialize a labeled footer, optionally specifying a column for the
14
14
  # label and whether the footer is to be a group footer. One or more values
15
15
  # for the footer are added later with the #add_value method.
16
- def initialize(label = 'Total', table, label_col: nil, group: false)
16
+ def initialize(table, label: 'Total', label_col: nil, group: false)
17
17
  @label = label
18
18
 
19
19
  unless table.is_a?(Table)
20
20
  raise ArgumentError, 'Footer.new needs a table argument'
21
21
  end
22
+
22
23
  if label_col.nil?
23
24
  @label_col = table.headers.first
24
25
  else
25
26
  unless table.headers.include?(label_col.as_sym)
26
27
  raise ArgumentError, "Footer.new label column '#{label_col}' not a header of table."
27
28
  end
29
+
28
30
  @label_col = label_col.as_sym
29
31
  end
30
32
  @table = table
@@ -77,7 +79,7 @@ module FatTable
77
79
  # header.
78
80
  def [](key)
79
81
  key = key.as_sym
80
- if values.keys.include?(key)
82
+ if values.key?(key)
81
83
  if group
82
84
  values[key]
83
85
  else
@@ -140,8 +142,6 @@ module FatTable
140
142
  hsh[h] =
141
143
  if values[h]
142
144
  values[h].first
143
- else
144
- nil
145
145
  end
146
146
  end
147
147
  end
@@ -243,9 +243,12 @@ module FatTable
243
243
  end
244
244
  else
245
245
  if k
246
- raise ArgumentError, "group footer label proc may only have 0, 1, or 2 arguments for group number k and containing footer f"
246
+ m = "group footer label proc may only have 0, 1, or 2 arguments for group number k and containing footer f"
247
+ raise ArgumentError, m
248
+
247
249
  else
248
- raise ArgumentError, "a non-group footer label proc may only have 0 or 1 arguments for the containing footer f"
250
+ raise ArgumentError,
251
+ "a non-group footer label proc may only have 0 or 1 arguments for the containing footer f"
249
252
  end
250
253
  end
251
254
  else
@@ -40,7 +40,7 @@ module FatTable
40
40
  # eval'ed, we need to escape any single-quotes (') that appear in the
41
41
  # string.
42
42
  def quote_cell(val)
43
- if val.match?(/'/)
43
+ if val.include?("'")
44
44
  # Use a negative look-behind to only quote single-quotes that are not
45
45
  # already preceded by a backslash
46
46
  val.gsub(/(?<!\\)'/, "'" => "\\'")
@@ -40,7 +40,7 @@ module FatTable
40
40
  # eval'ed, we need to escape any single-quotes (') that appear in the
41
41
  # string.
42
42
  def quote_cell(val)
43
- if val.match?(/'/)
43
+ if val.include?("'")
44
44
  # Use a negative look-behind to only quote single-quotes that are not
45
45
  # already preceded by a backslash
46
46
  val.gsub(/(?<!\\)'/, "'" => "\\'")