upcoming 0.0.1 → 0.1.0
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 +4 -4
- data/README.md +37 -0
- data/lib/upcoming/factory.rb +32 -16
- data/lib/upcoming/generators/generator.rb +24 -0
- data/lib/upcoming/generators/last_day_of_month_generator.rb +7 -0
- data/lib/upcoming/generators/working_day_generator.rb +3 -5
- data/lib/upcoming/upcoming.rb +0 -3
- data/lib/upcoming/version.rb +1 -1
- data/lib/upcoming.rb +1 -0
- data/spec/factory_spec.rb +58 -41
- data/spec/generators/generator_spec.rb +69 -0
- data/spec/generators/last_day_of_month_generator_spec.rb +23 -0
- data/spec/generators/working_day_generator_spec.rb +8 -10
- metadata +8 -5
- data/lib/upcoming/generators/day_generator.rb +0 -7
- data/spec/generators/day_generator_spec.rb +0 -12
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA1:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: ccef27184948b16da3feecded92350ac1fe53983
|
4
|
+
data.tar.gz: 00efad8af30f96ffe666be793e581af3d02c40f0
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 27976606835d81a76dfcc4204fe186d880e693db5e056d4e8b86e536ab70dea2d284110544e2d67b03149f54a7c498f1de18fa4da9c7ec0b55185aa7bf3aedc0
|
7
|
+
data.tar.gz: 67c27a1f8a237a102aef6f45250726285633a51d84c0da81f957e855c230f062cd42415883a9706233754a5a5980fa6689c08a6a8e5368180fc85316c7aec9d6
|
data/README.md
CHANGED
@@ -1,9 +1,46 @@
|
|
1
1
|
# upcoming
|
2
2
|
|
3
|
+
[](http://badge.fury.io/rb/upcoming)
|
3
4
|
[](https://travis-ci.org/sldblog/upcoming)
|
5
|
+
[](https://codeclimate.com/github/sldblog/upcoming)
|
6
|
+
[](https://gemnasium.com/sldblog/upcoming)
|
4
7
|
|
5
8
|
Recurring date sequence generator.
|
6
9
|
|
10
|
+
## Examples
|
11
|
+
|
12
|
+
`Upcoming::Factory.every` will generate sequences using the given method. This uses an enumerator so any number of dates can be queried.
|
13
|
+
|
14
|
+
```ruby
|
15
|
+
# running on 20th of June, 2014
|
16
|
+
> factory = Upcoming::Factory.every(:last_day_of_month)
|
17
|
+
=> #<Upcoming::Factory:0xb82fb490 @options={:from=>#<Date: 2014-06-20 ((2456829j,0s,0n),+0s,2299161j)>}, @chain=[#<Upcoming::LastDayOfMonthGenerator:0xb82fb094>]>
|
18
|
+
|
19
|
+
> factory.first
|
20
|
+
=> #<Date: 2014-06-30 ((2456839j,0s,0n),+0s,2299161j)>
|
21
|
+
|
22
|
+
> factory.take(12).map(&:iso8601)
|
23
|
+
=> ["2014-06-30", "2014-07-31", "2014-08-31", "2014-09-30", "2014-10-31", "2014-11-30", "2014-12-31", "2015-01-31", "2015-02-28", "2015-03-31", "2015-04-30", "2015-05-31"]
|
24
|
+
```
|
25
|
+
|
26
|
+
It is possible to chain methods together. Running sequentially, the methods will first test whether the date given for them is valid and if it is not, alter the previous result. Any number of chains can be added.
|
27
|
+
|
28
|
+
```ruby
|
29
|
+
> factory.then_find_first(:working_day).take(12).map(&:iso8601)
|
30
|
+
=> ["2014-06-30", "2014-07-31", "2014-09-01", "2014-09-30", "2014-10-31", "2014-12-01", "2014-12-31", "2015-02-02", "2015-03-02", "2015-03-31", "2015-04-30", "2015-06-01"]
|
31
|
+
```
|
32
|
+
|
33
|
+
Chaining backwards:
|
34
|
+
|
35
|
+
```ruby
|
36
|
+
> Upcoming::Factory.every(:last_day_of_month, from: '2014-08-20').then_find_latest(:working_day).first
|
37
|
+
=> 2014-08-29
|
38
|
+
```
|
39
|
+
|
40
|
+
## Generators
|
41
|
+
|
42
|
+
The available generators are in `lib/upcoming/generators`. They are mapped to the symbol by converting snake case to camel case and postfixing `Generator`. They can be anywhere in the load path.
|
43
|
+
|
7
44
|
## License
|
8
45
|
|
9
46
|
MIT
|
data/lib/upcoming/factory.rb
CHANGED
@@ -4,36 +4,52 @@ require 'active_support/core_ext/string'
|
|
4
4
|
module Upcoming
|
5
5
|
class Factory
|
6
6
|
include Enumerable
|
7
|
-
attr_reader :options
|
8
7
|
|
9
8
|
def initialize(options = {})
|
10
9
|
@options = parse(options)
|
10
|
+
@chain = []
|
11
|
+
end
|
12
|
+
|
13
|
+
def self.every(method, options = {})
|
14
|
+
new(options).then_find_first(method)
|
15
|
+
end
|
16
|
+
|
17
|
+
def then_find_first(method)
|
18
|
+
@chain << create_generator(method, :first)
|
19
|
+
self
|
20
|
+
end
|
21
|
+
|
22
|
+
def then_find_latest(method)
|
23
|
+
@chain << create_generator(method, :latest)
|
24
|
+
self
|
11
25
|
end
|
12
26
|
|
13
27
|
def each
|
14
28
|
from = @options[:from]
|
15
|
-
|
16
|
-
|
17
|
-
|
18
|
-
yield date
|
19
|
-
from =
|
29
|
+
while true do
|
30
|
+
from += 1
|
31
|
+
next_date = @chain.first.step(from)
|
32
|
+
yield @chain[1..-1].inject(next_date) { |date, generator| generator.step(date) }
|
33
|
+
from = next_date
|
20
34
|
end
|
21
35
|
end
|
22
36
|
|
23
37
|
private
|
24
38
|
|
25
39
|
def parse(options)
|
26
|
-
options
|
27
|
-
|
28
|
-
|
29
|
-
|
30
|
-
if result[:from].is_a? String
|
31
|
-
iso = result[:from] =~ /\d{4}-\d{2}-\d{2}/
|
32
|
-
raise ArgumentError, 'Please use ISO dates (YYYY-MM-DD) as those are not ambigious.' unless iso
|
33
|
-
result[:from] = Date.parse(result[:from])
|
34
|
-
end
|
35
|
-
result[:from] = Date.today if result[:from] == :today
|
40
|
+
options[:from] ||= Date.today
|
41
|
+
if options[:from].is_a? String
|
42
|
+
iso = options[:from] =~ /\d{4}-\d{2}-\d{2}/
|
43
|
+
raise ArgumentError, 'Please use ISO dates (YYYY-MM-DD) as those are not ambigious.' unless iso
|
36
44
|
end
|
45
|
+
options[:from] = options[:from].to_date if options[:from].respond_to? :to_date
|
46
|
+
options
|
47
|
+
end
|
48
|
+
|
49
|
+
def create_generator(name, direction)
|
50
|
+
class_name = name.to_s.classify + 'Generator'
|
51
|
+
generator_class = Upcoming.const_get class_name
|
52
|
+
generator_class.new(choose: direction)
|
37
53
|
end
|
38
54
|
|
39
55
|
end
|
@@ -0,0 +1,24 @@
|
|
1
|
+
require 'active_support/core_ext/string'
|
2
|
+
|
3
|
+
module Upcoming
|
4
|
+
class Generator
|
5
|
+
|
6
|
+
attr_reader :choose
|
7
|
+
|
8
|
+
def initialize(options = {})
|
9
|
+
@choose = options.fetch(:choose, :first)
|
10
|
+
end
|
11
|
+
|
12
|
+
def step(from)
|
13
|
+
date_range(from).find { |date| valid?(date) }
|
14
|
+
end
|
15
|
+
|
16
|
+
private
|
17
|
+
|
18
|
+
def date_range(date)
|
19
|
+
return date.downto(date.prev_year) if choose == :latest
|
20
|
+
date.upto(date.next_year)
|
21
|
+
end
|
22
|
+
|
23
|
+
end
|
24
|
+
end
|
@@ -1,11 +1,9 @@
|
|
1
1
|
module Upcoming
|
2
|
-
class WorkingDayGenerator
|
2
|
+
class WorkingDayGenerator < Generator
|
3
3
|
WEEKDAYS = (1..5)
|
4
4
|
|
5
|
-
def
|
6
|
-
date
|
7
|
-
date += 1 until WEEKDAYS.include?(date.wday)
|
8
|
-
date
|
5
|
+
def valid?(date)
|
6
|
+
WEEKDAYS.include? date.wday
|
9
7
|
end
|
10
8
|
end
|
11
9
|
end
|
data/lib/upcoming/upcoming.rb
CHANGED
data/lib/upcoming/version.rb
CHANGED
data/lib/upcoming.rb
CHANGED
data/spec/factory_spec.rb
CHANGED
@@ -2,70 +2,87 @@ require 'spec_helper'
|
|
2
2
|
|
3
3
|
describe Upcoming::Factory do
|
4
4
|
|
5
|
-
|
6
|
-
|
7
|
-
|
8
|
-
|
9
|
-
Then { upcoming.take(2).size == 2 }
|
10
|
-
And { upcoming.take(20).size == 20 }
|
5
|
+
class Upcoming::FizzGenerator < Upcoming::Generator
|
6
|
+
def valid?(date)
|
7
|
+
date.day % 3 == 0
|
8
|
+
end
|
11
9
|
end
|
12
10
|
|
13
|
-
|
14
|
-
|
15
|
-
|
11
|
+
class Upcoming::BuzzGenerator < Upcoming::Generator
|
12
|
+
def valid?(date)
|
13
|
+
date.day % 5 == 0
|
14
|
+
end
|
15
|
+
end
|
16
16
|
|
17
|
-
|
18
|
-
|
19
|
-
|
17
|
+
Given(:fixed_date) { Date.parse('2014-06-15') }
|
18
|
+
When(:result) do
|
19
|
+
Date.stub :today, fixed_date do
|
20
|
+
subject.take(3).map(&:iso8601)
|
20
21
|
end
|
22
|
+
end
|
21
23
|
|
22
|
-
|
23
|
-
|
24
|
-
|
24
|
+
context 'must be enumerable' do
|
25
|
+
Given(:subject) { Upcoming::Factory.every(:fizz) }
|
26
|
+
Then { subject.class.include? Enumerable }
|
27
|
+
Then { subject.respond_to? :each }
|
28
|
+
end
|
29
|
+
|
30
|
+
context '#every' do
|
31
|
+
context 'generates a sequence of days matching method given' do
|
32
|
+
Given(:subject) { Upcoming::Factory.every(:fizz) }
|
33
|
+
Then { result == %w(2014-06-18 2014-06-21 2014-06-24) }
|
25
34
|
end
|
26
35
|
|
27
|
-
context '
|
28
|
-
|
29
|
-
|
30
|
-
|
36
|
+
context 'generates a sequence from the given start date' do
|
37
|
+
Given(:subject) { Upcoming::Factory.every(:fizz, from: date) }
|
38
|
+
|
39
|
+
context 'given as ISO date' do
|
40
|
+
Given(:date) { '2014-06-20' }
|
41
|
+
Then { result == %w(2014-06-21 2014-06-24 2014-06-27) }
|
31
42
|
end
|
32
43
|
|
33
|
-
context '
|
34
|
-
Given(:
|
35
|
-
Then {
|
44
|
+
context 'given as Date object' do
|
45
|
+
Given(:date) { Date.parse('2014-05-01') }
|
46
|
+
Then { result == %w(2014-05-03 2014-05-06 2014-05-09) }
|
36
47
|
end
|
37
48
|
|
38
|
-
context '
|
39
|
-
Given(:
|
40
|
-
Then {
|
49
|
+
context 'given something responding to :to_date' do
|
50
|
+
Given(:date) { OpenStruct.new(to_date: Date.parse('2014-08-17')) }
|
51
|
+
Then { result == %w(2014-08-18 2014-08-21 2014-08-24) }
|
41
52
|
end
|
42
53
|
|
43
|
-
context 'non-ISO
|
44
|
-
Given(:
|
45
|
-
Then {
|
54
|
+
context 'generates error if given as non-ISO date' do
|
55
|
+
Given(:date) { '01/05/2014' }
|
56
|
+
Then { result == Failure(ArgumentError, /Please use ISO dates \(YYYY-MM-DD\) as those are not ambigious/) }
|
46
57
|
end
|
47
58
|
end
|
48
59
|
end
|
49
60
|
|
50
|
-
context '
|
51
|
-
|
52
|
-
|
53
|
-
|
54
|
-
|
55
|
-
end
|
61
|
+
context 'chained generators' do
|
62
|
+
class Upcoming::MonthAgoIfTwentyFifthGenerator < Upcoming::Generator
|
63
|
+
def step(date)
|
64
|
+
return date.prev_month if date.day == 25
|
65
|
+
date
|
56
66
|
end
|
57
67
|
end
|
58
68
|
|
59
|
-
|
69
|
+
context 'chains do not alter main sequence' do
|
70
|
+
Given(:subject) { Upcoming::Factory.every(:buzz).then_find_latest(:month_ago_if_twenty_fifth) }
|
71
|
+
Then { result == %w(2014-06-20 2014-05-25 2014-06-30) }
|
72
|
+
end
|
60
73
|
|
61
|
-
context '
|
62
|
-
|
63
|
-
|
74
|
+
context '#then_find_first' do
|
75
|
+
context 'modifies date found by moving to the next date that is a match' do
|
76
|
+
Given(:subject) { Upcoming::Factory.every(:buzz).then_find_first(:fizz) }
|
77
|
+
Then { result == %w(2014-06-21 2014-06-27 2014-06-30) }
|
78
|
+
end
|
64
79
|
end
|
65
80
|
|
66
|
-
context '
|
67
|
-
|
68
|
-
|
81
|
+
context '#then_find_latest' do
|
82
|
+
context 'modifies date found by moving to the previous date that is a match' do
|
83
|
+
Given(:subject) { Upcoming::Factory.every(:buzz).then_find_latest(:fizz) }
|
84
|
+
Then { result == %w(2014-06-18 2014-06-24 2014-06-30) }
|
85
|
+
end
|
69
86
|
end
|
70
87
|
end
|
71
88
|
|
@@ -0,0 +1,69 @@
|
|
1
|
+
require 'spec_helper'
|
2
|
+
|
3
|
+
describe Upcoming::Generator do
|
4
|
+
|
5
|
+
class Upcoming::FakeGenerator < Upcoming::Generator
|
6
|
+
def initialize(options = {})
|
7
|
+
super
|
8
|
+
@valid_dates = {}
|
9
|
+
end
|
10
|
+
|
11
|
+
def make_valid(date)
|
12
|
+
@valid_dates[Date.parse(date)] = true
|
13
|
+
end
|
14
|
+
|
15
|
+
def valid?(date)
|
16
|
+
@valid_dates[date]
|
17
|
+
end
|
18
|
+
end
|
19
|
+
|
20
|
+
def step(date)
|
21
|
+
result = subject.step(Date.parse(date))
|
22
|
+
result and result.iso8601
|
23
|
+
end
|
24
|
+
|
25
|
+
Given(:subject) { Upcoming::FakeGenerator.new }
|
26
|
+
|
27
|
+
context 'returns the same date if it is valid' do
|
28
|
+
Given(:start_date) { '2014-05-19' }
|
29
|
+
When { subject.make_valid(start_date) }
|
30
|
+
Then { step(start_date) == start_date }
|
31
|
+
end
|
32
|
+
|
33
|
+
context 'forward in time' do
|
34
|
+
context 'returns the first valid date, checking all dates in sequence' do
|
35
|
+
When { subject.make_valid('2014-05-20') }
|
36
|
+
When { subject.make_valid('2014-06-20') }
|
37
|
+
Then { step('2014-05-19') == '2014-05-20' }
|
38
|
+
Then { step('2014-05-21') == '2014-06-20' }
|
39
|
+
Then { step('2014-06-10') == '2014-06-20' }
|
40
|
+
end
|
41
|
+
|
42
|
+
context 'returns nil if there is no valid date within 1 year' do
|
43
|
+
# implement a custom +step+ method in your generator if you need this
|
44
|
+
When { subject.make_valid('2020-01-01') }
|
45
|
+
Then { step('2018-12-31') == nil }
|
46
|
+
Then { step('2019-01-01') == '2020-01-01' }
|
47
|
+
end
|
48
|
+
end
|
49
|
+
|
50
|
+
context 'backward in time' do
|
51
|
+
Given(:subject) { Upcoming::FakeGenerator.new(choose: :latest) }
|
52
|
+
|
53
|
+
context 'returns the closest past date, checking all dates in sequence' do
|
54
|
+
When { subject.make_valid('2010-03-04') }
|
55
|
+
When { subject.make_valid('2010-09-21') }
|
56
|
+
Then { step('2010-09-22') == '2010-09-21' }
|
57
|
+
Then { step('2010-09-20') == '2010-03-04' }
|
58
|
+
Then { step('2010-06-01') == '2010-03-04' }
|
59
|
+
end
|
60
|
+
|
61
|
+
context 'returns nil if there is no valid date within -1 year' do
|
62
|
+
# implement a custom +step+ method in your generator if you need this
|
63
|
+
When { subject.make_valid('2010-01-01') }
|
64
|
+
Then { step('2011-01-02') == nil }
|
65
|
+
Then { step('2011-01-01') == '2010-01-01' }
|
66
|
+
end
|
67
|
+
end
|
68
|
+
|
69
|
+
end
|
@@ -0,0 +1,23 @@
|
|
1
|
+
require 'spec_helper'
|
2
|
+
|
3
|
+
describe Upcoming::LastDayOfMonthGenerator do
|
4
|
+
|
5
|
+
Given(:subject) { Upcoming::LastDayOfMonthGenerator.new }
|
6
|
+
When(:valid) { subject.valid?(date) }
|
7
|
+
|
8
|
+
context 'not valid on not last day of month' do
|
9
|
+
Given(:date) { Date.parse('2014-06-15') }
|
10
|
+
Then { !valid }
|
11
|
+
end
|
12
|
+
|
13
|
+
context 'valid on last day of month' do
|
14
|
+
Given(:date) { Date.parse('2014-05-31') }
|
15
|
+
Then { valid }
|
16
|
+
end
|
17
|
+
|
18
|
+
context 'valid on leap day' do
|
19
|
+
Given(:date) { Date.parse('2012-02-29') }
|
20
|
+
Then { valid }
|
21
|
+
end
|
22
|
+
|
23
|
+
end
|
@@ -4,15 +4,12 @@ describe Upcoming::WorkingDayGenerator do
|
|
4
4
|
|
5
5
|
Given(:subject) { Upcoming::WorkingDayGenerator.new }
|
6
6
|
|
7
|
-
context 'returns
|
8
|
-
Given(:friday) { Date.parse('2014-06-13') }
|
7
|
+
context 'returns invalid for weekends' do
|
9
8
|
Given(:saturday) { Date.parse('2014-06-14') }
|
10
9
|
Given(:sunday) { Date.parse('2014-06-15') }
|
11
|
-
Given(:monday) { Date.parse('2014-06-16') }
|
12
10
|
|
13
|
-
Then { subject.
|
14
|
-
Then { subject.
|
15
|
-
Then { subject.next(sunday) == monday }
|
11
|
+
Then { !subject.valid?(saturday) }
|
12
|
+
Then { !subject.valid?(sunday) }
|
16
13
|
end
|
17
14
|
|
18
15
|
context 'returns next workday when given workday' do
|
@@ -22,10 +19,11 @@ describe Upcoming::WorkingDayGenerator do
|
|
22
19
|
Given(:thursday) { wednesday + 1 }
|
23
20
|
Given(:friday) { thursday + 1 }
|
24
21
|
|
25
|
-
Then { subject.
|
26
|
-
Then { subject.
|
27
|
-
Then { subject.
|
28
|
-
Then { subject.
|
22
|
+
Then { subject.valid?(monday) }
|
23
|
+
Then { subject.valid?(tuesday) }
|
24
|
+
Then { subject.valid?(wednesday) }
|
25
|
+
Then { subject.valid?(thursday) }
|
26
|
+
Then { subject.valid?(friday) }
|
29
27
|
end
|
30
28
|
|
31
29
|
end
|
metadata
CHANGED
@@ -1,14 +1,14 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: upcoming
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 0.0
|
4
|
+
version: 0.1.0
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- David Lantos
|
8
8
|
autorequire:
|
9
9
|
bindir: bin
|
10
10
|
cert_chain: []
|
11
|
-
date: 2014-06-
|
11
|
+
date: 2014-06-26 00:00:00.000000000 Z
|
12
12
|
dependencies:
|
13
13
|
- !ruby/object:Gem::Dependency
|
14
14
|
name: activesupport
|
@@ -65,12 +65,14 @@ files:
|
|
65
65
|
- Rakefile
|
66
66
|
- lib/upcoming.rb
|
67
67
|
- lib/upcoming/factory.rb
|
68
|
-
- lib/upcoming/generators/
|
68
|
+
- lib/upcoming/generators/generator.rb
|
69
|
+
- lib/upcoming/generators/last_day_of_month_generator.rb
|
69
70
|
- lib/upcoming/generators/working_day_generator.rb
|
70
71
|
- lib/upcoming/upcoming.rb
|
71
72
|
- lib/upcoming/version.rb
|
72
73
|
- spec/factory_spec.rb
|
73
|
-
- spec/generators/
|
74
|
+
- spec/generators/generator_spec.rb
|
75
|
+
- spec/generators/last_day_of_month_generator_spec.rb
|
74
76
|
- spec/generators/working_day_generator_spec.rb
|
75
77
|
- spec/spec_helper.rb
|
76
78
|
homepage: https://github.com/sldblog/upcoming
|
@@ -99,6 +101,7 @@ specification_version: 4
|
|
99
101
|
summary: Recurring date generator
|
100
102
|
test_files:
|
101
103
|
- spec/factory_spec.rb
|
102
|
-
- spec/generators/
|
104
|
+
- spec/generators/generator_spec.rb
|
105
|
+
- spec/generators/last_day_of_month_generator_spec.rb
|
103
106
|
- spec/generators/working_day_generator_spec.rb
|
104
107
|
- spec/spec_helper.rb
|
@@ -1,12 +0,0 @@
|
|
1
|
-
require 'spec_helper'
|
2
|
-
|
3
|
-
describe Upcoming::DayGenerator do
|
4
|
-
|
5
|
-
Given(:subject) { Upcoming::DayGenerator.new }
|
6
|
-
|
7
|
-
context 'returns the following day' do
|
8
|
-
When(:result) { subject.next(Date.parse('2001-12-31')) }
|
9
|
-
Then { result == Date.parse('2002-01-01') }
|
10
|
-
end
|
11
|
-
|
12
|
-
end
|