diff --git a/test/fixtures/api_namespaces.yml b/test/fixtures/api_namespaces.yml
index f5c4b7c4c..f94424649 100755
--- a/test/fixtures/api_namespaces.yml
+++ b/test/fixtures/api_namespaces.yml
@@ -81,4 +81,31 @@ namespace_with_all_types:
"null": null
}
requires_authentication: false
+ namespace_type: create-read-update-delete
+
+mailchimp:
+ name: mailchimp_namespace
+ slug: mailchimp_namespace
+ version: 1
+ properties: {
+ 'first_name': 'Violet',
+ 'last_name': 'Rails',
+ 'email': 'violet@rails.com',
+ 'contact': '',
+ 'synced_to_mailchimp': false
+ }
+ requires_authentication: false
+ namespace_type: create-read-update-delete
+
+mailchimp_logger:
+ name: mailchimp/logger
+ slug: mailchimp_logger
+ version: 1
+ properties: {
+ 'status': '',
+ 'response': {},
+ 'timstamp': '',
+ 'api_resource': ''
+ }
+ requires_authentication: false
namespace_type: create-read-update-delete
\ No newline at end of file
diff --git a/test/fixtures/api_resources.yml b/test/fixtures/api_resources.yml
index 1bca4c93e..87d904cee 100755
--- a/test/fixtures/api_resources.yml
+++ b/test/fixtures/api_resources.yml
@@ -33,3 +33,12 @@ plugin_subdomain_message_event:
"body": "
Hello
"
}
}
+mailchimp:
+ api_namespace: mailchimp
+ properties: {
+ 'first_name': 'Violet',
+ 'last_name': 'Rails',
+ 'email': 'violet@rails.com',
+ 'contact': '555-777',
+ 'synced_to_mailchimp': false
+ }
diff --git a/test/fixtures/external_api_clients.yml b/test/fixtures/external_api_clients.yml
index 7877d1510..46a260008 100644
--- a/test/fixtures/external_api_clients.yml
+++ b/test/fixtures/external_api_clients.yml
@@ -238,4 +238,80 @@ vacuum_job:
end
end
# at the end of the file we have to implicitly return the class
- VacuumJob
\ No newline at end of file
+ VacuumJob
+
+mailchimp_plugin:
+ api_namespace: mailchimp
+ slug: mailchimp-plugin
+ label: MailChimp
+ enabled: true
+ metadata: {
+ 'API_KEY': 'testkey',
+ 'SERVER_PREFIX': 'us9',
+ 'LIST_ID': 'valid_list_id',
+ }
+ model_definition: |
+ class SyncToMailchimp
+ def initialize(parameters)
+ @external_api_client = parameters[:external_api_client]
+ @api_key = @external_api_client.metadata["API_KEY"]
+ @unsynced_api_resources = @external_api_client.api_namespace.api_resources.where("properties @> ?", {synced_to_mailchimp: false}.to_json)
+ @mailchimp_uri = "https://#{@external_api_client.metadata['SERVER_PREFIX']}.api.mailchimp.com/3.0/lists/#{@external_api_client.metadata['LIST_ID']}/members?skip_merge_validation=true"
+ @custom_merge_fields_map = @external_api_client.metadata['CUSTOM_MERGE_FIELDS_MAP'] || {}
+ @attr_to_exclude = (@external_api_client.metadata['ATTR_TO_EXCLUDE'] || []) + @custom_merge_fields_map.keys + ['synced_to_mailchimp']
+ @logger_namespace = ApiNamespace.find_by(slug: @external_api_client.metadata["LOGGER_NAMESPACE"]) if @external_api_client.metadata["LOGGER_NAMESPACE"]
+ end
+
+ def start
+ @unsynced_api_resources.each do |api_resource|
+ begin
+ merge_fields = api_resource.properties.except(*@attr_to_exclude).transform_keys(&:upcase).transform_values(&:to_s)
+
+ @custom_merge_fields_map.each do |key, value|
+ merge_fields[value.upcase] = api_resource.properties[key].to_s if value
+ end
+
+ response = HTTParty.post(@mailchimp_uri,
+ body: {
+ email_address: api_resource.properties["email"],
+ status: "subscribed",
+ merge_fields: merge_fields,
+ tags: @external_api_client.metadata['TAGS'] || []
+ }.to_json,
+
+ headers: {
+ 'Content-Type': 'application/json',
+ 'Authorization': "Basic #{@api_key}"
+ }
+ )
+
+ if response.success?
+ api_resource.properties["synced_to_mailchimp"] = true
+ api_resource.save
+ end
+
+ @logger_namespace.api_resources.create!(
+ properties: {
+ api_resource: api_resource.id,
+ status: response.success? ? "success" : "error",
+ response: JSON.parse(response.body),
+ timestamp: Time.zone.now
+ }
+ ) if @logger_namespace
+
+ rescue StandardError => e
+ @logger_namespace.api_resources.create!(
+ properties: {
+ api_resource: api_resource.id,
+ status: "error",
+ response: { detail: e.message},
+ timestamp: Time.zone.now
+ }
+ ) if @logger_namespace
+ end
+ end
+ end
+ end
+
+ # at the end of the file we have to implicitly return the class
+ SyncToMailchimp
\ No newline at end of file
diff --git a/test/plugins/mailchimp_plugin_test.rb b/test/plugins/mailchimp_plugin_test.rb
new file mode 100644
index 000000000..013320475
--- /dev/null
+++ b/test/plugins/mailchimp_plugin_test.rb
@@ -0,0 +1,139 @@
+require "test_helper"
+
+class MailchimpPluginTest < ActiveSupport::TestCase
+ setup do
+ @mailchimp_plugin = external_api_clients(:mailchimp_plugin)
+ @api_namespace = api_namespaces(:mailchimp)
+ @api_resource = api_resources(:mailchimp)
+ @logger_namespace = api_namespaces(:mailchimp_logger)
+ @mailchimp_api_uri = "https://#{@mailchimp_plugin.metadata['SERVER_PREFIX']}.api.mailchimp.com/3.0/lists/#{@mailchimp_plugin.metadata['LIST_ID']}/members?skip_merge_validation=true"
+ @request_headers = { 'Authorization': "Basic #{@mailchimp_plugin.metadata['API_KEY']}", 'Content-Type': 'application/json' }
+ Sidekiq::Testing.fake!
+ end
+
+ test "should send api request to mailchimp" do
+ metadata = @mailchimp_plugin.metadata
+
+ request_body = {
+ "email_address": @api_resource.properties['email'],
+ "status": "subscribed",
+ # synced_to_mailchimp property should be excluded by deafult
+ # merge fields keys should be uppercase properties keys
+ # tags should be an empty array if TAGS metadata is not defined
+ "merge_fields": {"EMAIL": @api_resource.properties['email'], "FIRST_NAME": @api_resource.properties['first_name'], "LAST_NAME": @api_resource.properties['last_name'], "CONTACT": @api_resource.properties['contact']},
+ "tags": []
+ }
+
+ response_body = {
+ "id" => "123adc",
+ "email" => "violet@rails.com"
+ }
+
+ mailchimp_request = stub_request(:post, @mailchimp_api_uri)
+ .with(body: request_body, headers: @request_headers)
+ .to_return(status: 200, body: response_body.to_json)
+
+ assert_changes -> { @api_resource.reload.properties['synced_to_mailchimp'] }, from: false, to: true do
+ # should not create log if LOGGER_NAMESPACE metadata is not defined
+ assert_no_difference "@logger_namespace.api_resources.count" do
+ perform_enqueued_jobs do
+ @mailchimp_plugin.run
+ Sidekiq::Worker.drain_all
+ end
+ end
+ end
+
+ assert_requested mailchimp_request
+ end
+
+ test "should send api request to mailchimp for all unsynced api resources" do
+ ApiResource.create(api_namespace_id: @api_namespace.id, properties: {'first_name': 'first_name', 'email': 'some_email@some_domain.com', 'synced_to_mailchimp': false})
+ ApiResource.create(api_namespace_id: @api_namespace.id, properties: {'first_name': 'some_name', 'email': 'second@some_domain.com', 'synced_to_mailchimp': true})
+
+ mailchimp_request = stub_request(:post, @mailchimp_api_uri).to_return(status: 200, body: {id: 'some_id'}.to_json)
+ expected_count = @api_namespace.api_resources.where("properties @> ?", {synced_to_mailchimp: false}.to_json).count
+
+ perform_enqueued_jobs do
+ @mailchimp_plugin.run
+ Sidekiq::Worker.drain_all
+ end
+
+ assert_requested mailchimp_request, times: expected_count
+ end
+
+ test "optional metadata" do
+ metadata = @mailchimp_plugin.metadata
+
+ # optional metadata
+ metadata['LOGGER_NAMESPACE'] = 'mailchimp_logger'
+ metadata['ATTR_TO_EXCLUDE'] = ['first_name']
+ metadata['TAGS'] = ['some_tag']
+ metadata['CUSTOM_MERGE_FIELDS_MAP'] = {'contact': 'PHONE'}
+
+ @mailchimp_plugin.update(metadata: metadata)
+
+ request_body = {
+ "email_address": @api_resource.properties['email'],
+ "status": "subscribed",
+ # synced_to_mailchimp property should be excluded by deafult
+ # excluded attributes [ATTR_TO_EXCLUDE] should not be present in the merge fields
+ # custom_merge_fields should replace default merge field name if CUSTOM_MERGE_FIELDS_MAP is defined eg. CONTACT is replaced by PHONE
+ "merge_fields": {"EMAIL": @api_resource.properties['email'], "LAST_NAME": @api_resource.properties['last_name'], "PHONE": @api_resource.properties['contact']},
+ "tags": ['some_tag']
+ }
+
+ response_body = {
+ "id" => "123adc",
+ "email" => "violet@rails.com"
+ }
+
+ mailchimp_request = stub_request(:post, @mailchimp_api_uri).with(body: request_body, headers: @request_headers).to_return(status: 200, body: response_body.to_json)
+
+ assert_changes -> { @api_resource.reload.properties['synced_to_mailchimp'] }, from: false, to: true do
+ # should create log if LOGGER_NAMESPACE metadata is defined
+ assert_difference "@logger_namespace.api_resources.count", +1 do
+ perform_enqueued_jobs do
+ @mailchimp_plugin.run
+ Sidekiq::Worker.drain_all
+ end
+ end
+ end
+
+ log = @logger_namespace.api_resources.last.properties
+
+ assert_equal log['status'], 'success'
+ assert_equal log['api_resource'], @api_resource.id
+ assert_equal log['response'], response_body
+
+ assert_requested mailchimp_request
+ end
+
+ test "error response: should create log with error status" do
+ metadata = @mailchimp_plugin.metadata
+ metadata['LOGGER_NAMESPACE'] = 'mailchimp_logger'
+
+ @mailchimp_plugin.update(metadata: metadata)
+
+ response_body = { "message" => "error message" }
+
+ mailchimp_request = stub_request(:post, @mailchimp_api_uri).to_return(status: 400, body: response_body.to_json)
+
+ assert_no_changes @api_resource.reload.properties['synced_to_mailchimp'] do
+ # should create log if LOGGER_NAMESPACE metadata is defined
+ assert_difference "@logger_namespace.api_resources.count", +1 do
+ perform_enqueued_jobs do
+ @mailchimp_plugin.run
+ Sidekiq::Worker.drain_all
+ end
+ end
+ end
+
+ log = @logger_namespace.api_resources.last.properties
+
+ assert_equal log['status'], 'error'
+ assert_equal log['api_resource'], @api_resource.id
+ assert_equal log['response'], response_body
+
+ assert_requested mailchimp_request
+ end
+end
\ No newline at end of file