object_table 0.3.0 → 0.3.2

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.
@@ -95,6 +95,10 @@ describe ObjectTable::BasicGrid do
95
95
  expect(grid[:col1]).to eql columns[:col1]
96
96
  expect(grid[:col2]).to eql columns[:col2]
97
97
  end
98
+
99
+ it 'should return the number of rows' do
100
+ expect(subject).to eql 3
101
+ end
98
102
  end
99
103
 
100
104
  context 'with columns of differing length' do
@@ -110,15 +114,23 @@ describe ObjectTable::BasicGrid do
110
114
  subject
111
115
  expect(grid[:col3]).to eql [6] * 3
112
116
  end
117
+
118
+ it 'should return the number of rows' do
119
+ expect(subject).to eql 3
120
+ end
113
121
  end
114
122
 
115
123
  context 'with scalars only' do
116
124
  let(:columns){ {col1: 1, col2: 2} }
117
- it 'should assume there is one column' do
125
+ it 'should assume there is one row' do
118
126
  subject
119
127
  expect(grid[:col1]).to eql [columns[:col1]]
120
128
  expect(grid[:col2]).to eql [columns[:col2]]
121
129
  end
130
+
131
+ it 'should return the number of rows' do
132
+ expect(subject).to eql 1
133
+ end
122
134
  end
123
135
 
124
136
  context 'with ranges' do
@@ -126,6 +138,10 @@ describe ObjectTable::BasicGrid do
126
138
  it 'should succeed' do
127
139
  expect{subject}.to_not raise_error
128
140
  end
141
+
142
+ it 'should return the number of rows' do
143
+ expect(subject).to eql 3
144
+ end
129
145
  end
130
146
 
131
147
  context 'with multi dimensional narrays' do
@@ -136,6 +152,10 @@ describe ObjectTable::BasicGrid do
136
152
  subject
137
153
  expect(grid).to eql columns
138
154
  end
155
+
156
+ it 'should return the number of rows' do
157
+ expect(subject).to eql 3
158
+ end
139
159
  end
140
160
 
141
161
  context 'with an incorrect last dimension' do
@@ -155,6 +175,10 @@ describe ObjectTable::BasicGrid do
155
175
  it 'should succeed' do
156
176
  expect{subject}.to_not raise_error
157
177
  end
178
+
179
+ it 'should return the number of rows' do
180
+ expect(subject).to eql 0
181
+ end
158
182
  end
159
183
 
160
184
 
@@ -2,6 +2,24 @@ require 'object_table/column'
2
2
 
3
3
  describe ObjectTable::Column do
4
4
 
5
+ describe '.length_of' do
6
+ it 'should return the length of narrays' do
7
+ expect(described_class.length_of NArray.float(10, 20, 30)).to eql 30
8
+ end
9
+
10
+ it 'should return the length of arrays' do
11
+ expect(described_class.length_of [ [0] * 20 ] * 30).to eql 30
12
+ end
13
+
14
+ it 'should return the length of empty narrays' do
15
+ expect(described_class.length_of NArray.float(0)).to eql 0
16
+ end
17
+
18
+ it 'should fail on other inputs' do
19
+ expect{described_class.length_of 123456}.to raise_error
20
+ end
21
+ end
22
+
5
23
  describe '.stack' do
6
24
  let(:columns) do
7
25
  [
@@ -36,6 +54,22 @@ describe ObjectTable::Column do
36
54
  expect(subject).to eq NArray[]
37
55
  end
38
56
  end
57
+
58
+ context 'with empty narrays' do
59
+ let(:columns) do
60
+ [
61
+ NArray.float(10, 10).random!,
62
+ NArray.float(10, 30).random!,
63
+ NArray[],
64
+ NArray.to_na([[100] * 10] * 5),
65
+ ]
66
+ end
67
+
68
+ it 'should skip empty narrays' do
69
+ expect(subject).to eq ObjectTable::Column.stack(columns[0], columns[1], columns[3])
70
+ end
71
+ end
72
+
39
73
  end
40
74
 
41
75
  end
@@ -144,18 +144,26 @@ describe ObjectTable::Grouped do
144
144
 
145
145
  it 'should give access to the keys' do
146
146
  keys = []
147
+ grouped.each{ keys << Hash[@K.each_pair.to_a] }
148
+ expect(keys).to match_array [{parity: 0}, {parity: 1}]
149
+ end
150
+
151
+ it 'should give access to the correct key' do
152
+ keys = []
153
+ correct_keys = []
147
154
  grouped.each do
148
- keys << @K
155
+ keys << [@K[:parity]]
156
+ correct_keys << (self.col1 % 2).to_a.uniq
149
157
  end
150
158
 
151
- expect(keys).to match_array [{parity: 0}, {parity: 1}]
159
+ expect(keys).to match_array(correct_keys)
152
160
  end
153
161
 
154
162
  it 'should give access to the correct key' do
155
163
  keys = []
156
164
  correct_keys = []
157
165
  grouped.each do
158
- keys << [@K[:parity]]
166
+ keys << [@K.parity]
159
167
  correct_keys << (self.col1 % 2).to_a.uniq
160
168
  end
161
169
 
@@ -226,6 +234,51 @@ describe ObjectTable::Grouped do
226
234
  end
227
235
  end
228
236
 
237
+ context 'with results that are tables' do
238
+ subject{ grouped.apply{ ObjectTable.new(sum: col1.sum, mean: col2.mean) } }
239
+
240
+ it 'should return a table with the group keys' do
241
+ expect(subject).to be_a ObjectTable
242
+ expect(subject.colnames).to include :parity
243
+ end
244
+
245
+ it 'should stack the grids' do
246
+ expect(subject.sort_by(subject.parity)).to eql ObjectTable.new(
247
+ parity: [0, 1],
248
+ sum: [even_group.col1.sum, odd_group.col1.sum],
249
+ mean: [even_group.col2.mean, odd_group.col2.mean],
250
+ )
251
+ end
252
+ end
253
+
254
+ context 'with results that are arrays' do
255
+ subject{ grouped.apply{ [col1[0], col1[-1]] } }
256
+
257
+ it 'should return a table with the group keys' do
258
+ expect(subject).to be_a ObjectTable
259
+ expect(subject.colnames).to include :parity
260
+ end
261
+
262
+ it 'should stack the grids' do
263
+ expect(subject.where{parity.eq 0}.v_0).to eq even_group.col1[[0, -1]]
264
+ expect(subject.where{parity.eq 1}.v_0).to eq odd_group.col1[[0, -1]]
265
+ end
266
+ end
267
+
268
+ context 'with results that are narrays' do
269
+ subject{ grouped.apply{ col1 < 2 } }
270
+
271
+ it 'should return a table with the group keys' do
272
+ expect(subject).to be_a ObjectTable
273
+ expect(subject.colnames).to include :parity
274
+ end
275
+
276
+ it 'should stack the grids' do
277
+ expect(subject.where{parity.eq 0}.v_0).to eq (even_group.col1 < 2)
278
+ expect(subject.where{parity.eq 1}.v_0).to eq (odd_group.col1 < 2)
279
+ end
280
+ end
281
+
229
282
  context 'when the block takes an argument' do
230
283
  it 'should not evaluate in the context of the group' do
231
284
  rspec_context = self
@@ -135,19 +135,19 @@ describe ObjectTable::MaskedColumn do
135
135
  end
136
136
  end
137
137
 
138
- include_examples 'destructive methods', 'indgen!'
139
- include_examples 'destructive methods', 'indgen'
140
- include_examples 'destructive methods', 'fill!', 100
141
- include_examples 'destructive methods', 'random!'
142
- include_examples 'destructive methods', 'mod!', 2
143
- include_examples 'destructive methods', 'add!', 56
144
- include_examples 'destructive methods', 'sbt!', 56
145
- include_examples 'destructive methods', 'mul!', 56
146
- include_examples 'destructive methods', 'div!', 56
147
- include_examples 'destructive methods', 'map!' do
138
+ it_behaves_like 'destructive methods', 'indgen!'
139
+ it_behaves_like 'destructive methods', 'indgen'
140
+ it_behaves_like 'destructive methods', 'fill!', 100
141
+ it_behaves_like 'destructive methods', 'random!'
142
+ it_behaves_like 'destructive methods', 'mod!', 2
143
+ it_behaves_like 'destructive methods', 'add!', 56
144
+ it_behaves_like 'destructive methods', 'sbt!', 56
145
+ it_behaves_like 'destructive methods', 'mul!', 56
146
+ it_behaves_like 'destructive methods', 'div!', 56
147
+ it_behaves_like 'destructive methods', 'map!' do
148
148
  let(:block) { proc{|x| x + 1} }
149
149
  end
150
- include_examples 'destructive methods', 'collect!' do
150
+ it_behaves_like 'destructive methods', 'collect!' do
151
151
  let(:block) { proc{|x| x + 1} }
152
152
  end
153
153
 
@@ -92,4 +92,32 @@ describe ObjectTable::View do
92
92
  end
93
93
  end
94
94
 
95
+ describe '#cache_indices' do
96
+ let(:table) { ObjectTable.new(col1: [1, 2, 3], col2: 5) }
97
+ let(:view) { ObjectTable::View.new(table){ col1 > 2 } }
98
+
99
+ let(:block) { Proc.new{'success!'} }
100
+
101
+ subject{ view.cache_indices(&block) }
102
+
103
+ it 'should call the block' do
104
+ expect(block).to receive(:call)
105
+ subject
106
+ end
107
+ end
108
+
109
+ describe '#cache_columns' do
110
+ let(:table) { ObjectTable.new(col1: [1, 2, 3], col2: 5) }
111
+ let(:view) { ObjectTable::View.new(table){ col1 > 2 } }
112
+
113
+ let(:block) { Proc.new{'success!'} }
114
+
115
+ subject{ view.cache_columns(&block) }
116
+
117
+ it 'should call the block' do
118
+ expect(block).to receive(:call)
119
+ subject
120
+ end
121
+ end
122
+
95
123
  end
@@ -1,9 +1,11 @@
1
1
  require 'object_table'
2
2
 
3
3
  require 'support/object_table_example'
4
+ require 'support/stacker_example'
4
5
 
5
6
  describe ObjectTable do
6
7
  it_behaves_like 'an object table', ObjectTable
8
+ it_behaves_like 'a table stacker'
7
9
 
8
10
  describe '#initialize' do
9
11
  let(:columns){ {col1: [1, 2, 3], col2: NArray[4, 5, 6], col3: 7..9, col4: 10} }
@@ -72,46 +74,70 @@ describe ObjectTable do
72
74
  let(:args) { [] }
73
75
  let(:table){ ObjectTable.new(col1: [1, 2, 3], col2: 5) }
74
76
 
75
- let(:column) { table.colnames[0] }
76
-
77
77
  subject{ table.set_column(column, value, *args) }
78
78
 
79
- it 'should allow assigning columns' do
80
- subject
81
- expect(table.columns[column].to_a).to eql value
82
- end
79
+ shared_examples 'a column setter' do
80
+ it 'should allow assigning columns' do
81
+ subject
82
+ expect(table.columns[column].to_a).to eql value
83
+ end
83
84
 
84
- it 'should coerce the value to a narray' do
85
- subject
86
- expect(table.columns[column]).to be_a NArray
87
- end
85
+ it 'should coerce the value to a narray' do
86
+ subject
87
+ expect(table.columns[column]).to be_a NArray
88
+ end
88
89
 
89
- context 'with the wrong length' do
90
- let(:value) { [1, 2] }
91
- it 'should fail' do
92
- expect{subject}.to raise_error
90
+ context 'with the wrong length' do
91
+ let(:value) { [1, 2] }
92
+ it 'should fail' do
93
+ expect{subject}.to raise_error
94
+ end
93
95
  end
94
- end
95
96
 
96
- context 'with a scalar' do
97
- let(:value){ 10 }
98
- it 'should fill the column with that value' do
99
- subject
100
- expect(table.columns[column].to_a).to eql ([value] * table.nrows)
97
+ context 'with a scalar' do
98
+ let(:value){ 10 }
99
+ it 'should fill the column with that value' do
100
+ subject
101
+ expect(table.columns[column].to_a).to eql ([value] * table.nrows)
102
+ end
101
103
  end
102
- end
103
104
 
104
- context 'with a range' do
105
- let(:value){ 0...3 }
106
- it 'should assign the range values' do
107
- subject
108
- expect(table.columns[column].to_a).to eql value.to_a
105
+ context 'with a range' do
106
+ let(:value){ 0...3 }
107
+ it 'should assign the range values' do
108
+ subject
109
+ expect(table.columns[column].to_a).to eql value.to_a
110
+ end
109
111
  end
112
+
113
+ context 'with an empty table' do
114
+ let(:table) { ObjectTable.new }
115
+ let(:value) { 3 }
116
+
117
+ context 'adding an empty column' do
118
+ it 'should add the column' do
119
+ subject
120
+ expect(table.columns[column]).to eq NArray[]
121
+ end
122
+
123
+ context 'and setting an empty array to the column' do
124
+ it 'should work' do
125
+ subject
126
+ expect{table[column] = []}.to_not raise_error
127
+ expect(table[column]).to be_empty
128
+ end
129
+ end
130
+
131
+ end
132
+ end
133
+
110
134
  end
111
135
 
112
136
  context 'for a new column' do
113
137
  let(:column) { :col3 }
114
138
 
139
+ it_behaves_like 'a column setter'
140
+
115
141
  it 'should create a new column' do
116
142
  subject
117
143
  expect(table.columns).to include column
@@ -136,26 +162,23 @@ describe ObjectTable do
136
162
  end
137
163
 
138
164
  context 'when failed to add column' do
139
- let(:value){ NArray[1, 2, 3] }
165
+ let(:value) { 'a' }
166
+ let(:args) { ['int'] }
140
167
 
141
- it 'should not have that column' do
142
- expect(table).to receive(:add_column).with(column, value.typecode) do
143
- table.columns[column] = 12345
144
- end
168
+ it 'should fail' do
169
+ expect{subject}.to raise_error
170
+ end
145
171
 
172
+ it 'should not have that column' do
146
173
  # the assignment is going to chuck an error
147
174
  subject rescue nil
148
175
  expect(table.columns).to_not include column
149
176
  end
150
177
  end
151
- end
152
-
153
- context 'with narray args' do
154
- let(:args) { ['int', 3, 4] }
155
- let(:value){ NArray.float(3, 4, table.nrows) }
156
178
 
157
- context 'for a new column' do
158
- let(:column) { :col3 }
179
+ context 'with narray args' do
180
+ let(:args) { ['int', 3, 4] }
181
+ let(:value){ NArray.float(3, 4, table.nrows) }
159
182
 
160
183
  it 'should create a column with the typecode' do
161
184
  subject
@@ -171,24 +194,29 @@ describe ObjectTable do
171
194
 
172
195
  end
173
196
 
174
- context 'with an empty table' do
175
- let(:table) { ObjectTable.new }
176
- let(:value) { [] }
197
+ context 'on an existing column' do
198
+ let(:column) { table.colnames[0] }
199
+ it_behaves_like 'a column setter'
177
200
 
178
- context 'adding an empty column' do
179
- it 'should add the column' do
180
- subject
181
- expect(table.columns[column].to_a).to eq value
201
+ context 'when failed to set column' do
202
+ let(:value) { 'a' }
203
+
204
+ it 'should fail' do
205
+ expect{subject}.to raise_error
182
206
  end
183
207
 
184
- context 'and setting an empty array to the column' do
185
- it 'should work' do
186
- subject
187
- expect{table[column] = []}.to_not raise_error
188
- expect(table[column]).to be_empty
189
- end
208
+ it 'should still have the column' do
209
+ # the assignment is going to chuck an error
210
+ subject rescue nil
211
+ expect(table.columns).to include column
190
212
  end
191
213
 
214
+ it 'should make no changes' do
215
+ original = table.clone
216
+ subject rescue nil
217
+ # the assignment is going to chuck an error
218
+ expect(table).to eql original
219
+ end
192
220
  end
193
221
  end
194
222
 
@@ -212,82 +240,6 @@ describe ObjectTable do
212
240
  end
213
241
  end
214
242
 
215
- describe '.stack' do
216
- let(:others) do
217
- [
218
- ObjectTable.new(col1: [1, 2, 3], col2: 5),
219
- ObjectTable.new(col1: 10, col2: 50),
220
- ObjectTable.new(col2: [10, 30], col1: 15).where{col2.eq 10},
221
- ObjectTable::BasicGrid[col2: [1, 2], col1: 3..4],
222
- ]
223
- end
224
-
225
- subject{ ObjectTable.stack *others }
226
-
227
- it 'should join the tables and grids together' do
228
- expect(subject).to be_a ObjectTable
229
- expect(subject).to eql ObjectTable.new(
230
- col1: others.flat_map{|x| x[:col1].to_a},
231
- col2: others.flat_map{|x| x[:col2].to_a},
232
- )
233
- end
234
-
235
- it 'should duplicate the contents' do
236
- others.each do |chunk|
237
- expect(subject).to_not be chunk
238
- end
239
- end
240
-
241
- context 'with non grids/tables' do
242
- let(:others){ [ObjectTable.new(col1: 10, col2: 50), 'not a table'] }
243
-
244
- it 'should fail' do
245
- expect{subject}.to raise_error
246
- end
247
- end
248
-
249
- context 'with extra column names' do
250
- let(:others){ [ObjectTable.new(col1: 10, col2: 50), ObjectTable.new(col1: 10, col2: 30, col3: 50)] }
251
-
252
- it 'should fail' do
253
- expect{subject}.to raise_error
254
- end
255
- end
256
-
257
- context 'with missing column names' do
258
- let(:others){ [ObjectTable.new(col1: 10, col2: 50), ObjectTable.new(col1: 10)] }
259
-
260
- it 'should fail' do
261
- expect{subject}.to raise_error
262
- end
263
- end
264
-
265
- context 'with empty tables' do
266
- let(:others) { [ ObjectTable.new(col1: [1, 2, 3], col2: 5), ObjectTable.new ] }
267
-
268
- it 'should ignore empty tables' do
269
- expect(subject).to eql others[0]
270
- end
271
- end
272
-
273
- context 'with tables with empty rows' do
274
- let(:others) { [ ObjectTable.new(col1: [1, 2, 3], col2: 5), ObjectTable.new(col1: [], col2: []) ] }
275
-
276
- it 'should ignore empty tables' do
277
- expect(subject).to eql others[0]
278
- end
279
- end
280
-
281
- context 'with empty grids' do
282
- let(:others) { [ ObjectTable.new(col1: [1, 2, 3], col2: 5), ObjectTable::BasicGrid.new ] }
283
-
284
- it 'should ignore empty grids' do
285
- expect(subject).to eql others[0]
286
- end
287
- end
288
-
289
- end
290
-
291
243
  describe '#sort_by!' do
292
244
  let(:table){ ObjectTable.new(col1: [2, 2, 1, 1], col2: [0, 1, 0, 1], col3: [5, 6, 7, 8]) }
293
245
  subject{ table.sort_by!(table.col1, table.col2) }
@@ -301,85 +253,4 @@ describe ObjectTable do
301
253
  end
302
254
  end
303
255
 
304
- describe '#join' do
305
- let(:left) do
306
- ObjectTable.new(
307
- key1: [ "a", "b", "a", "b", "c"],
308
- lvalue: [ 0, 1, 2, 3, 4],
309
- )
310
- end
311
-
312
- let(:right) do
313
- ObjectTable.new(
314
- key1: [ "d", "c", "b"],
315
- rvalue: [ 0, 1, 2],
316
- )
317
- end
318
-
319
- context 'inner join' do
320
- subject{ left.join(right, key=:key1, type: 'inner') }
321
-
322
- let(:result) do
323
- ObjectTable.new(
324
- key1: [ "b", "b", "c"],
325
- lvalue: [ 1, 3, 4],
326
- rvalue: [ 2, 2, 1],
327
- )
328
- end
329
-
330
- it 'should return the joined result' do
331
- expect(subject).to eql result
332
- end
333
- end
334
-
335
- context 'left join' do
336
- subject{ left.join(right, key=:key1, type: 'left') }
337
-
338
- let(:result) do
339
- ObjectTable.new(
340
- key1: [ "b", "b", "c", "a", "a"],
341
- lvalue: [ 1, 3, 4, 0, 2],
342
- rvalue: [ 2, 2, 1, 0, 0],
343
- )
344
- end
345
-
346
- it 'should return the joined result' do
347
- expect(subject).to eql result
348
- end
349
- end
350
-
351
- context 'right join' do
352
- subject{ left.join(right, key=:key1, type: 'right') }
353
-
354
- let(:result) do
355
- ObjectTable.new(
356
- key1: [ "b", "b", "c", "d"],
357
- lvalue: [ 1, 3, 4, 0],
358
- rvalue: [ 2, 2, 1, 0],
359
- )
360
- end
361
-
362
- it 'should return the joined result' do
363
- expect(subject).to eql result
364
- end
365
- end
366
-
367
- context 'outer join' do
368
- subject{ left.join(right, key=:key1, type: 'outer') }
369
-
370
- let(:result) do
371
- ObjectTable.new(
372
- key1: [ "b", "b", "c", "a", "a", "d"],
373
- lvalue: [ 1, 3, 4, 0, 2, 0],
374
- rvalue: [ 2, 2, 1, 0, 0, 0],
375
- )
376
- end
377
-
378
- it 'should return the joined result' do
379
- expect(subject).to eql result
380
- end
381
- end
382
-
383
- end
384
-
385
256
  end