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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA1:
3
- metadata.gz: 250bbc397078aab7189913788bfce89c84195389
4
- data.tar.gz: 5cdce22ca4d56607b031dddecd961b9e7db3e26b
3
+ metadata.gz: ccef27184948b16da3feecded92350ac1fe53983
4
+ data.tar.gz: 00efad8af30f96ffe666be793e581af3d02c40f0
5
5
  SHA512:
6
- metadata.gz: a40d327bda39041f5f4bc56b5ffedbc11a4fbd690e32074d89df1783ea80063776d10d5c9e935be165a0174cb0456055a645746e32d4b54ed21762463c85a523
7
- data.tar.gz: 6b1918388fc8cb9d481780dda93f0bd2e99d0beee033035cd9f9f623a50dbb4b3253cdf182a9ddd7564f4fb7a646803c55c96455a872466514fe35597a86bf16
6
+ metadata.gz: 27976606835d81a76dfcc4204fe186d880e693db5e056d4e8b86e536ab70dea2d284110544e2d67b03149f54a7c498f1de18fa4da9c7ec0b55185aa7bf3aedc0
7
+ data.tar.gz: 67c27a1f8a237a102aef6f45250726285633a51d84c0da81f957e855c230f062cd42415883a9706233754a5a5980fa6689c08a6a8e5368180fc85316c7aec9d6
data/README.md CHANGED
@@ -1,9 +1,46 @@
1
1
  # upcoming
2
2
 
3
+ [![Gem Version](https://badge.fury.io/rb/upcoming.png)](http://badge.fury.io/rb/upcoming)
3
4
  [![Build Status](https://travis-ci.org/sldblog/upcoming.svg)](https://travis-ci.org/sldblog/upcoming)
5
+ [![Code Climate](https://codeclimate.com/github/sldblog/upcoming.png)](https://codeclimate.com/github/sldblog/upcoming)
6
+ [![Dependency Status](https://gemnasium.com/sldblog/upcoming.svg)](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
@@ -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
- generator = Upcoming.const_get("#{@options[:every].to_s.classify}Generator").new
16
- (1..Float::INFINITY).each do |n|
17
- date = generator.next(from)
18
- yield date
19
- from = date
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.dup.tap do |result|
27
- result[:every] ||= :day
28
- result[:from] ||= :today
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
@@ -0,0 +1,7 @@
1
+ module Upcoming
2
+ class LastDayOfMonthGenerator < Generator
3
+ def valid?(date)
4
+ (date + 1).month != date.month
5
+ end
6
+ end
7
+ 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 next(from)
6
- date = from + 1
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
@@ -1,5 +1,2 @@
1
1
  module Upcoming
2
- def self.for(config = {})
3
- Upcoming::Factory.new(config)
4
- end
5
2
  end
@@ -1,3 +1,3 @@
1
1
  module Upcoming
2
- VERSION = '0.0.1'
2
+ VERSION = '0.1.0'
3
3
  end
data/lib/upcoming.rb CHANGED
@@ -2,6 +2,7 @@ require 'upcoming/version'
2
2
  require 'upcoming/factory'
3
3
  require 'upcoming/upcoming'
4
4
 
5
+ require 'upcoming/generators/generator'
5
6
  Dir[File.join(File.dirname(__FILE__), 'upcoming', 'generators', '**/*.rb')].each do |generator|
6
7
  require generator
7
8
  end
data/spec/factory_spec.rb CHANGED
@@ -2,70 +2,87 @@ require 'spec_helper'
2
2
 
3
3
  describe Upcoming::Factory do
4
4
 
5
- let(:today) { Date.today }
6
-
7
- context 'generates any number of dates' do
8
- Given(:upcoming) { Upcoming::Factory.new }
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
- context 'configuration' do
14
- When(:upcoming) { Upcoming::Factory.new(config) }
15
- When(:options) { upcoming.options }
11
+ class Upcoming::BuzzGenerator < Upcoming::Generator
12
+ def valid?(date)
13
+ date.day % 5 == 0
14
+ end
15
+ end
16
16
 
17
- context 'defaults to daily recurrence from today' do
18
- Given(:config) { {} }
19
- Then { options == {every: :day, from: today} }
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
- context 'options is configured through constructor' do
23
- Given(:config) { {every: :bazooka, from: :other} }
24
- Then { options == config }
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 'dates' do
28
- context 'from dates other than strings are copied as-is' do
29
- Given(:config) { {from: 42} }
30
- Then { options[:from] == 42 }
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 'from date can be given as Date' do
34
- Given(:config) { {from: Date.parse('2001-10-10')} }
35
- Then { options[:from] == config[:from] }
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 'string ISO dates are converted automatically' do
39
- Given(:config) { {from: '2000-01-01'} }
40
- Then { options[:from] == Date.parse('2000-01-01') }
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 strings raise ambiguity error' do
44
- Given(:config) { {from: '12/01/2000'} }
45
- Then { upcoming.must_raise ArgumentError, /Please use ISO dates (YYYY-MM-DD) as those are not ambigious./ }
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 'execution' do
51
- module Upcoming
52
- class DeepThoughtGenerator
53
- def next(from)
54
- from + 42
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
- Given(:upcoming) { Upcoming::Factory.new(every: :deep_thought) }
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 'invokes generator based on the name of "every" parameter' do
62
- When(:result) { upcoming.first }
63
- Then { result == today + 42 }
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 'invokes generator with previous result iteratively' do
67
- When(:result) { upcoming.take(3) }
68
- Then { result == [today + 42, today + 84, today + 126] }
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 upcoming Monday when Friday, Saturday or Sunday are given' do
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.next(friday) == monday }
14
- Then { subject.next(saturday) == monday }
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.next(monday) == tuesday }
26
- Then { subject.next(tuesday) == wednesday }
27
- Then { subject.next(wednesday) == thursday }
28
- Then { subject.next(thursday) == friday }
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.1
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-15 00:00:00.000000000 Z
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/day_generator.rb
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/day_generator_spec.rb
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/day_generator_spec.rb
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,7 +0,0 @@
1
- module Upcoming
2
- class DayGenerator
3
- def next(from)
4
- from + 1
5
- end
6
- end
7
- end
@@ -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