Merge pull request #1257 from cantino/google_cal_agent_accepts_liquid

The GoogleCalendarPublishAgent now accepts Liquid in the calendar_id option

Andrew Cantino vor 9 Jahren
Ursprung
Commit
3077aefa52

+ 5 - 6
app/models/agents/google_calendar_publish_agent.rb

@@ -25,17 +25,16 @@ module Agents
25 25
 
26 26
       Agent Configuration:
27 27
 
28
-      `calendar_id` - The id the calendar you want to publish to. Typically your google account email address.
28
+      `calendar_id` - The id the calendar you want to publish to. Typically your google account email address.  Liquid formatting (e.g. `{{ cal_id }}`) is allowed here in order to extract the calendar_id from the incoming event.
29 29
 
30 30
       `google` A hash of configuration options for the agent.
31 31
 
32 32
       `google` `service_account_email` - The authorised service account.
33 33
 
34
-      `google` `key_file` - The path to the key file.
34
+      `google` `key_file` OR `google` `key` - The path to the key file or the key itself.  Liquid formatting is supported if you want to use a Credential.  (E.g., `{% credential google_key %}`)
35 35
 
36 36
       `google` `key_secret` - The secret for the key, typically 'notasecret'
37 37
 
38
-      
39 38
 
40 39
       Set `expected_update_period_in_days` to the maximum amount of time that you'd expect to pass between Events being created by this Agent.
41 40
 
@@ -92,10 +91,10 @@ module Agents
92 91
 
93 92
     def receive(incoming_events)
94 93
      incoming_events.each do |event|
95
-        calendar = GoogleCalendar.new(options, Rails.logger)
94
+        calendar = GoogleCalendar.new(interpolate_options(options, event), Rails.logger)
95
+
96
+        calendar_event = JSON.parse(calendar.publish_as(interpolated(event)['calendar_id'], event.payload["message"]).response.body)
96 97
 
97
-        calendar_event = JSON.parse(calendar.publish_as(options['calendar_id'], event.payload["message"]).response.body)
98
-  
99 98
         create_event :payload => {
100 99
           'success' => true,
101 100
           'published_calendar_event' => calendar_event,

+ 8 - 2
lib/google_calendar.rb

@@ -1,7 +1,13 @@
1 1
 class GoogleCalendar
2 2
   def initialize(config, logger)
3 3
     @config = config
4
-    @key = Google::APIClient::PKCS12.load_key(@config['google']['key_file'], @config['google']['key_secret'])
4
+
5
+    if @config['google']['key'].present?
6
+      @key = OpenSSL::PKCS12.new(@config['google']['key'], @config['google']['key_secret']).key
7
+    else
8
+      @key = Google::APIClient::PKCS12.load_key(@config['google']['key_file'], @config['google']['key_secret'])
9
+    end
10
+
5 11
     @client = Google::APIClient.new(application_name: "Huginn", application_version: "0.0.1")
6 12
     @client.retries = 2
7 13
     @logger ||= logger
@@ -60,4 +66,4 @@ class GoogleCalendar
60 66
     @logger.debug ret.to_yaml
61 67
     ret    
62 68
   end
63
-end
69
+end

+ 110 - 29
spec/models/agents/google_calendar_publish_agent_spec.rb

@@ -1,43 +1,124 @@
1 1
 require 'rails_helper'
2 2
 
3 3
 describe Agents::GoogleCalendarPublishAgent, :vcr do
4
-  before do
5
-    @valid_params = {
6
-        'expected_update_period_in_days' => "10",
7
-        'calendar_id' => 'sqv39gj35tc837gdns1g4d81cg@group.calendar.google.com',
8
-        'google' => {
9
-          'key_file' => File.dirname(__FILE__) + '/../../data_fixtures/private.key',
10
-          'key_secret' => 'notasecret',
11
-          'service_account_email' => '1029936966326-ncjd7776pcspc98hsg82gsb56t3217ef@developer.gserviceaccount.com'
12
-        }
4
+  let(:valid_params) do
5
+    {
6
+      'expected_update_period_in_days' => "10",
7
+      'calendar_id' => calendar_id,
8
+      'google' => {
9
+        'key_file' => File.dirname(__FILE__) + '/../../data_fixtures/private.key',
10
+        'key_secret' => 'notasecret',
11
+        'service_account_email' => '1029936966326-ncjd7776pcspc98hsg82gsb56t3217ef@developer.gserviceaccount.com'
13 12
       }
14
-    @checker = Agents::GoogleCalendarPublishAgent.new(:name => "somename", :options => @valid_params)
15
-    @checker.user = users(:jane)
16
-    @checker.save!
13
+    }
14
+  end
15
+
16
+  let(:agent) do
17
+    _agent = Agents::GoogleCalendarPublishAgent.new(name: "somename", options: valid_params)
18
+    _agent.user = users(:jane)
19
+    _agent.save!
20
+    _agent
17 21
   end
18 22
 
19 23
   describe '#receive' do
20
-    it 'should publish any payload it receives' do
21
-      event1 = Event.new
22
-      event1.agent = agents(:bob_manual_event_agent)
23
-      event1.payload = {
24
-        'message' => { 
25
-          'visibility' => 'default',
26
-          'summary' => "Awesome event",
27
-          'description' => "An example event with text. Pro tip: DateTimes are in RFC3339",
28
-          'end' => {
29
-            'dateTime' => '2014-10-02T11:00:00-05:00'
30
-          },
31
-          'start' => {
32
-            'dateTime' => '2014-10-02T10:00:00-05:00'
33
-          }
24
+    let(:event) do
25
+      _event = Event.new
26
+      _event.agent = agents(:bob_manual_event_agent)
27
+      _event.payload = { 'message' => message }
28
+      _event.save!
29
+      _event
30
+    end
31
+
32
+    let(:calendar_id) { 'sqv39gj35tc837gdns1g4d81cg@group.calendar.google.com' }
33
+    let(:message) do
34
+      {
35
+        'visibility' => 'default',
36
+        'summary' => "Awesome event",
37
+        'description' => "An example event with text. Pro tip: DateTimes are in RFC3339",
38
+        'end' => {
39
+          'dateTime' => '2014-10-02T11:00:00-05:00'
40
+        },
41
+        'start' => {
42
+          'dateTime' => '2014-10-02T10:00:00-05:00'
34 43
         }
35 44
       }
36
-      event1.save!
45
+    end
46
+    let(:response_body) do
47
+      {"kind"=>"calendar#event",
48
+        "etag"=>"\"2908684044040000\"",
49
+        "id"=>"baz",
50
+        "status"=>"confirmed",
51
+        "htmlLink"=>
52
+          "https://calendar.google.com/calendar/event?eid=foobar",
53
+        "created"=>"2016-02-01T15:53:41.000Z",
54
+        "updated"=>"2016-02-01T15:53:42.020Z",
55
+        "summary"=>"Awesome event",
56
+        "description"=>
57
+          "An example event with text. Pro tip: DateTimes are in RFC3339",
58
+        "creator"=>
59
+          {"email"=>
60
+            "blah-foobar@developer.gserviceaccount.com"},
61
+        "organizer"=>
62
+          {"email"=>calendar_id,
63
+            "displayName"=>"Huginn Location Log",
64
+            "self"=>true},
65
+        "start"=>{"dateTime"=>"2014-10-03T00:30:00+09:30"},
66
+        "end"=>{"dateTime"=>"2014-10-03T01:30:00+09:30"},
67
+        "iCalUID"=>"blah@google.com",
68
+        "sequence"=>0,
69
+        "reminders"=>{"useDefault"=>true}
70
+      }.to_json
71
+    end
72
+
73
+    def setup_mock!
74
+      fake_interface = Object.new
75
+      mock(GoogleCalendar).new(agent.interpolate_options(agent.options), Rails.logger) { fake_interface }
76
+      mock(fake_interface).publish_as(calendar_id, message) { stub!.response.stub!.body { response_body } }
77
+    end
78
+
79
+    describe 'when the calendar_id is in the options' do
80
+      it 'should publish any payload it receives' do
81
+        setup_mock!
82
+
83
+        expect {
84
+          agent.receive([event])
85
+        }.to change { agent.events.count }.by(1)
86
+
87
+        expect(agent.events.last.payload).to eq({ "success" => true, "published_calendar_event" => JSON.parse(response_body), "agent_id" => event.agent_id, "event_id" => event.id })
88
+      end
89
+    end
90
+
91
+    describe 'with Liquid templating' do
92
+      it 'should allow Liquid in the calendar_id' do
93
+        setup_mock!
94
+
95
+        agent.options['calendar_id'] = '{{ cal_id }}'
96
+        agent.save!
97
+
98
+        event.payload['cal_id'] = calendar_id
99
+        event.save!
100
+
101
+        agent.receive([event])
102
+
103
+        expect(agent.events.count).to eq(1)
104
+        expect(agent.events.last.payload).to eq({ "success" => true, "published_calendar_event" => JSON.parse(response_body), "agent_id" => event.agent_id, "event_id" => event.id })
105
+      end
106
+
107
+      it 'should allow Liquid in the key' do
108
+        agent.options['google'].delete('key_file')
109
+        agent.options['google']['key'] = '{% credential google_key %}'
110
+        agent.save!
111
+
112
+        users(:jane).user_credentials.create! credential_name: 'google_key', credential_value: 'something'
113
+
114
+        agent.reload
115
+
116
+        setup_mock!
37 117
 
38
-      @checker.receive([event1])
118
+        agent.receive([event])
39 119
 
40
-      expect(@checker.events.count).to eq(1)
120
+        expect(agent.events.count).to eq(1)
121
+      end
41 122
     end
42 123
   end
43 124
 end