google-adwords-api 0.18.0 → 0.18.1

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.
Files changed (171) hide show
  1. checksums.yaml +4 -4
  2. data/ChangeLog +8 -0
  3. data/examples/v201509/basic_operations/add_ad_groups.rb +4 -2
  4. data/examples/v201509/remarketing/add_crm_based_user_list.rb +2 -4
  5. data/examples/v201509/{advanced_operations → remarketing}/upload_offline_conversions.rb +6 -21
  6. data/examples/v201601/account_management/create_account.rb +88 -0
  7. data/examples/v201601/account_management/get_account_changes.rb +139 -0
  8. data/examples/v201601/account_management/get_account_hierarchy.rb +132 -0
  9. data/examples/v201601/advanced_operations/add_ad_customizers.rb +184 -0
  10. data/examples/v201601/advanced_operations/add_ad_group_bid_modifier.rb +101 -0
  11. data/examples/v201601/advanced_operations/add_click_to_download_ad.rb +133 -0
  12. data/examples/v201601/advanced_operations/add_html5_ad.rb +137 -0
  13. data/examples/v201601/advanced_operations/add_keywords_using_incremental_batch_job.rb +179 -0
  14. data/examples/v201601/advanced_operations/add_text_ad_with_upgraded_urls.rb +134 -0
  15. data/examples/v201601/advanced_operations/create_and_attach_shared_keyword_set.rb +141 -0
  16. data/examples/v201601/advanced_operations/find_and_remove_criteria_from_shared_set.rb +174 -0
  17. data/examples/v201601/advanced_operations/get_ad_group_bid_modifiers.rb +102 -0
  18. data/examples/v201601/advanced_operations/use_shared_bidding_strategy.rb +147 -0
  19. data/examples/v201601/basic_operations/add_ad_groups.rb +142 -0
  20. data/examples/v201601/basic_operations/add_campaigns.rb +139 -0
  21. data/examples/v201601/basic_operations/add_keywords.rb +114 -0
  22. data/examples/v201601/basic_operations/add_text_ads.rb +109 -0
  23. data/examples/v201601/basic_operations/get_ad_groups.rb +102 -0
  24. data/examples/v201601/basic_operations/get_campaigns.rb +97 -0
  25. data/examples/v201601/basic_operations/get_campaigns_with_awql.rb +89 -0
  26. data/examples/v201601/basic_operations/get_keywords.rb +109 -0
  27. data/examples/v201601/basic_operations/get_text_ads.rb +110 -0
  28. data/examples/v201601/basic_operations/pause_ad.rb +88 -0
  29. data/examples/v201601/basic_operations/remove_ad.rb +89 -0
  30. data/examples/v201601/basic_operations/remove_ad_group.rb +85 -0
  31. data/examples/v201601/basic_operations/remove_campaign.rb +87 -0
  32. data/examples/v201601/basic_operations/remove_keyword.rb +94 -0
  33. data/examples/v201601/basic_operations/update_ad_group.rb +85 -0
  34. data/examples/v201601/basic_operations/update_campaign.rb +86 -0
  35. data/examples/v201601/basic_operations/update_keyword.rb +106 -0
  36. data/examples/v201601/campaign_management/add_campaign_labels.rb +82 -0
  37. data/examples/v201601/campaign_management/add_complete_campaigns_using_batch_job.rb +356 -0
  38. data/examples/v201601/campaign_management/add_experiment.rb +162 -0
  39. data/examples/v201601/campaign_management/add_keywords_in_bulk.rb +153 -0
  40. data/examples/v201601/campaign_management/get_all_disapproved_ads.rb +97 -0
  41. data/examples/v201601/campaign_management/get_all_disapproved_ads_with_awql.rb +89 -0
  42. data/examples/v201601/campaign_management/get_campaigns_by_label.rb +108 -0
  43. data/examples/v201601/campaign_management/promote_experiment.rb +81 -0
  44. data/examples/v201601/campaign_management/set_ad_parameters.rb +118 -0
  45. data/examples/v201601/campaign_management/set_criterion_bid_modifier.rb +104 -0
  46. data/examples/v201601/campaign_management/validate_text_ad.rb +110 -0
  47. data/examples/v201601/error_handling/handle_partial_failures.rb +130 -0
  48. data/examples/v201601/error_handling/handle_policy_violation_error.rb +141 -0
  49. data/examples/v201601/extensions/add_google_my_business_location_extensions.rb +193 -0
  50. data/examples/v201601/extensions/add_site_links.rb +164 -0
  51. data/examples/v201601/extensions/add_site_links_using_feeds.rb +281 -0
  52. data/examples/v201601/migration/migrate_to_extension_settings.rb +386 -0
  53. data/examples/v201601/migration/upgrade_ad_url.rb +93 -0
  54. data/examples/v201601/misc/get_all_images_and_videos.rb +104 -0
  55. data/examples/v201601/misc/setup_oauth2.rb +84 -0
  56. data/examples/v201601/misc/upload_image.rb +93 -0
  57. data/examples/v201601/misc/upload_media_bundle.rb +90 -0
  58. data/examples/v201601/misc/use_oauth2_jwt.rb +93 -0
  59. data/examples/v201601/misc/use_runtime_config.rb +92 -0
  60. data/examples/v201601/optimization/estimate_keyword_traffic.rb +146 -0
  61. data/examples/v201601/optimization/get_keyword_bid_simulations.rb +95 -0
  62. data/examples/v201601/optimization/get_keyword_ideas.rb +126 -0
  63. data/examples/v201601/remarketing/add_audience.rb +118 -0
  64. data/examples/v201601/remarketing/add_conversion_tracker.rb +97 -0
  65. data/examples/v201601/remarketing/add_crm_based_user_list.rb +119 -0
  66. data/examples/v201601/remarketing/add_rule_based_user_lists.rb +167 -0
  67. data/examples/v201601/remarketing/upload_offline_conversions.rb +98 -0
  68. data/examples/v201601/reporting/download_criteria_report.rb +92 -0
  69. data/examples/v201601/reporting/download_criteria_report_with_awql.rb +93 -0
  70. data/examples/v201601/reporting/get_report_fields.rb +75 -0
  71. data/examples/v201601/reporting/parallel_report_download.rb +166 -0
  72. data/examples/v201601/reporting/stream_criteria_report_results.rb +97 -0
  73. data/examples/v201601/shopping_campaigns/add_product_partition_tree.rb +267 -0
  74. data/examples/v201601/shopping_campaigns/add_product_scope.rb +129 -0
  75. data/examples/v201601/shopping_campaigns/add_shopping_campaign.rb +129 -0
  76. data/examples/v201601/shopping_campaigns/get_product_category_taxonomy.rb +115 -0
  77. data/examples/v201601/targeting/add_campaign_targeting_criteria.rb +180 -0
  78. data/examples/v201601/targeting/add_demographic_targeting_criteria.rb +112 -0
  79. data/examples/v201601/targeting/get_campaign_targeting_criteria.rb +106 -0
  80. data/examples/v201601/targeting/get_targetable_languages_and_carriers.rb +89 -0
  81. data/examples/v201601/targeting/lookup_location.rb +108 -0
  82. data/lib/adwords_api/api_config.rb +90 -4
  83. data/lib/adwords_api/batch_job_utils.rb +29 -17
  84. data/lib/adwords_api/incremental_upload_helper.rb +5 -1
  85. data/lib/adwords_api/report_stream.rb +7 -8
  86. data/lib/adwords_api/report_utils.rb +4 -4
  87. data/lib/adwords_api/v201601/account_label_service.rb +46 -0
  88. data/lib/adwords_api/v201601/account_label_service_registry.rb +46 -0
  89. data/lib/adwords_api/v201601/ad_customizer_feed_service.rb +46 -0
  90. data/lib/adwords_api/v201601/ad_customizer_feed_service_registry.rb +46 -0
  91. data/lib/adwords_api/v201601/ad_group_ad_service.rb +70 -0
  92. data/lib/adwords_api/v201601/ad_group_ad_service_registry.rb +46 -0
  93. data/lib/adwords_api/v201601/ad_group_bid_modifier_service.rb +54 -0
  94. data/lib/adwords_api/v201601/ad_group_bid_modifier_service_registry.rb +46 -0
  95. data/lib/adwords_api/v201601/ad_group_criterion_service.rb +62 -0
  96. data/lib/adwords_api/v201601/ad_group_criterion_service_registry.rb +46 -0
  97. data/lib/adwords_api/v201601/ad_group_extension_setting_service.rb +54 -0
  98. data/lib/adwords_api/v201601/ad_group_extension_setting_service_registry.rb +46 -0
  99. data/lib/adwords_api/v201601/ad_group_feed_service.rb +54 -0
  100. data/lib/adwords_api/v201601/ad_group_feed_service_registry.rb +46 -0
  101. data/lib/adwords_api/v201601/ad_group_service.rb +62 -0
  102. data/lib/adwords_api/v201601/ad_group_service_registry.rb +46 -0
  103. data/lib/adwords_api/v201601/ad_param_service.rb +46 -0
  104. data/lib/adwords_api/v201601/ad_param_service_registry.rb +46 -0
  105. data/lib/adwords_api/v201601/adwords_user_list_service.rb +54 -0
  106. data/lib/adwords_api/v201601/adwords_user_list_service_registry.rb +46 -0
  107. data/lib/adwords_api/v201601/batch_job_service.rb +54 -0
  108. data/lib/adwords_api/v201601/batch_job_service_registry.rb +46 -0
  109. data/lib/adwords_api/v201601/bidding_strategy_service.rb +54 -0
  110. data/lib/adwords_api/v201601/bidding_strategy_service_registry.rb +46 -0
  111. data/lib/adwords_api/v201601/budget_order_service.rb +54 -0
  112. data/lib/adwords_api/v201601/budget_order_service_registry.rb +46 -0
  113. data/lib/adwords_api/v201601/budget_service.rb +54 -0
  114. data/lib/adwords_api/v201601/budget_service_registry.rb +46 -0
  115. data/lib/adwords_api/v201601/campaign_criterion_service.rb +54 -0
  116. data/lib/adwords_api/v201601/campaign_criterion_service_registry.rb +46 -0
  117. data/lib/adwords_api/v201601/campaign_extension_setting_service.rb +54 -0
  118. data/lib/adwords_api/v201601/campaign_extension_setting_service_registry.rb +46 -0
  119. data/lib/adwords_api/v201601/campaign_feed_service.rb +54 -0
  120. data/lib/adwords_api/v201601/campaign_feed_service_registry.rb +46 -0
  121. data/lib/adwords_api/v201601/campaign_service.rb +62 -0
  122. data/lib/adwords_api/v201601/campaign_service_registry.rb +46 -0
  123. data/lib/adwords_api/v201601/campaign_shared_set_service.rb +54 -0
  124. data/lib/adwords_api/v201601/campaign_shared_set_service_registry.rb +46 -0
  125. data/lib/adwords_api/v201601/constant_data_service.rb +110 -0
  126. data/lib/adwords_api/v201601/constant_data_service_registry.rb +46 -0
  127. data/lib/adwords_api/v201601/conversion_tracker_service.rb +54 -0
  128. data/lib/adwords_api/v201601/conversion_tracker_service_registry.rb +46 -0
  129. data/lib/adwords_api/v201601/customer_extension_setting_service.rb +54 -0
  130. data/lib/adwords_api/v201601/customer_extension_setting_service_registry.rb +46 -0
  131. data/lib/adwords_api/v201601/customer_feed_service.rb +54 -0
  132. data/lib/adwords_api/v201601/customer_feed_service_registry.rb +46 -0
  133. data/lib/adwords_api/v201601/customer_service.rb +54 -0
  134. data/lib/adwords_api/v201601/customer_service_registry.rb +46 -0
  135. data/lib/adwords_api/v201601/customer_sync_service.rb +38 -0
  136. data/lib/adwords_api/v201601/customer_sync_service_registry.rb +47 -0
  137. data/lib/adwords_api/v201601/data_service.rb +78 -0
  138. data/lib/adwords_api/v201601/data_service_registry.rb +46 -0
  139. data/lib/adwords_api/v201601/experiment_service.rb +46 -0
  140. data/lib/adwords_api/v201601/experiment_service_registry.rb +46 -0
  141. data/lib/adwords_api/v201601/feed_item_service.rb +54 -0
  142. data/lib/adwords_api/v201601/feed_item_service_registry.rb +46 -0
  143. data/lib/adwords_api/v201601/feed_mapping_service.rb +54 -0
  144. data/lib/adwords_api/v201601/feed_mapping_service_registry.rb +46 -0
  145. data/lib/adwords_api/v201601/feed_service.rb +54 -0
  146. data/lib/adwords_api/v201601/feed_service_registry.rb +46 -0
  147. data/lib/adwords_api/v201601/label_service.rb +54 -0
  148. data/lib/adwords_api/v201601/label_service_registry.rb +46 -0
  149. data/lib/adwords_api/v201601/location_criterion_service.rb +46 -0
  150. data/lib/adwords_api/v201601/location_criterion_service_registry.rb +46 -0
  151. data/lib/adwords_api/v201601/managed_customer_service.rb +78 -0
  152. data/lib/adwords_api/v201601/managed_customer_service_registry.rb +46 -0
  153. data/lib/adwords_api/v201601/media_service.rb +54 -0
  154. data/lib/adwords_api/v201601/media_service_registry.rb +46 -0
  155. data/lib/adwords_api/v201601/mutate_job_service.rb +54 -0
  156. data/lib/adwords_api/v201601/mutate_job_service_registry.rb +46 -0
  157. data/lib/adwords_api/v201601/offline_conversion_feed_service.rb +38 -0
  158. data/lib/adwords_api/v201601/offline_conversion_feed_service_registry.rb +46 -0
  159. data/lib/adwords_api/v201601/report_definition_service.rb +38 -0
  160. data/lib/adwords_api/v201601/report_definition_service_registry.rb +46 -0
  161. data/lib/adwords_api/v201601/shared_criterion_service.rb +54 -0
  162. data/lib/adwords_api/v201601/shared_criterion_service_registry.rb +46 -0
  163. data/lib/adwords_api/v201601/shared_set_service.rb +54 -0
  164. data/lib/adwords_api/v201601/shared_set_service_registry.rb +46 -0
  165. data/lib/adwords_api/v201601/targeting_idea_service.rb +38 -0
  166. data/lib/adwords_api/v201601/targeting_idea_service_registry.rb +46 -0
  167. data/lib/adwords_api/v201601/traffic_estimator_service.rb +38 -0
  168. data/lib/adwords_api/v201601/traffic_estimator_service_registry.rb +46 -0
  169. data/lib/adwords_api/version.rb +1 -1
  170. data/test/adwords_api/test_batch_job_utils.rb +11 -2
  171. metadata +163 -5
@@ -0,0 +1,130 @@
1
+ #!/usr/bin/env ruby
2
+ # Encoding: utf-8
3
+ #
4
+ # Copyright:: Copyright 2011, Google Inc. All Rights Reserved.
5
+ #
6
+ # License:: Licensed under the Apache License, Version 2.0 (the "License");
7
+ # you may not use this file except in compliance with the License.
8
+ # You may obtain a copy of the License at
9
+ #
10
+ # http://www.apache.org/licenses/LICENSE-2.0
11
+ #
12
+ # Unless required by applicable law or agreed to in writing, software
13
+ # distributed under the License is distributed on an "AS IS" BASIS,
14
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
15
+ # implied.
16
+ # See the License for the specific language governing permissions and
17
+ # limitations under the License.
18
+ #
19
+ # This example demonstrates how to handle partial failures.
20
+
21
+ require 'adwords_api'
22
+ require 'adwords_api/utils'
23
+
24
+ def handle_partial_failures(ad_group_id)
25
+ # AdwordsApi::Api will read a config file from ENV['HOME']/adwords_api.yml
26
+ # when called without parameters.
27
+ adwords = AdwordsApi::Api.new
28
+
29
+ # To enable logging of SOAP requests, set the log_level value to 'DEBUG' in
30
+ # the configuration file or provide your own logger:
31
+ # adwords.logger = Logger.new('adwords_xml.log')
32
+
33
+ ad_group_criterion_srv =
34
+ adwords.service(:AdGroupCriterionService, API_VERSION)
35
+
36
+ # Set partial failures flag.
37
+ adwords.partial_failure = true
38
+
39
+ # Create keywords.
40
+ keyword_text = ['mars cruise', 'inv@lid cruise', 'venus cruise',
41
+ 'b(a)d keyword cruise']
42
+ keywords = []
43
+ keyword_text.each do |text|
44
+ keyword = {
45
+ # The 'xsi_type' field allows you to specify the xsi:type of the object
46
+ # being created. It's only necessary when you must provide an explicit
47
+ # type that the client library can't infer.
48
+ :xsi_type => 'Keyword',
49
+ :match_type => 'BROAD',
50
+ :text => text
51
+ }
52
+ keywords << keyword
53
+ end
54
+
55
+ # Create biddable ad group criteria and operations.
56
+ operations = []
57
+ keywords.each do |kwd|
58
+ operation = {
59
+ :operator => 'ADD',
60
+ :operand => {
61
+ :xsi_type => 'BiddableAdGroupCriterion',
62
+ :ad_group_id => ad_group_id,
63
+ :criterion => kwd
64
+ }
65
+ }
66
+ operations << operation
67
+ end
68
+
69
+ # Add criteria.
70
+ response = ad_group_criterion_srv.mutate(operations)
71
+ if response and response[:value]
72
+ ad_group_criteria = response[:value]
73
+ ad_group_criteria.each do |ad_group_criterion|
74
+ if ad_group_criterion[:criterion]
75
+ puts "Ad group criterion with ad group id " +
76
+ "#{ad_group_criterion[:ad_group_id]}, criterion id "+
77
+ "#{ad_group_criterion[:criterion][:id]} and keyword \"" +
78
+ "#{ad_group_criterion[:criterion][:text]}\" was added."
79
+ end
80
+ end
81
+ else
82
+ puts "No criteria were added."
83
+ end
84
+
85
+ # Check partial failures.
86
+ if response and response[:partial_failure_errors]
87
+ response[:partial_failure_errors].each do |error|
88
+ operation_index = AdwordsApi::Utils.operation_index_for_error(error)
89
+ if operation_index
90
+ ad_group_criterion = operations[operation_index][:operand]
91
+ puts "Ad group criterion with ad group id " +
92
+ "#{ad_group_criterion[:ad_group_id]} and keyword \"" +
93
+ "#{ad_group_criterion[:criterion][:text]}\" triggered an error " +
94
+ "for the following reason: \"#{error[:error_string]}\"."
95
+ end
96
+ end
97
+ end
98
+ end
99
+
100
+ if __FILE__ == $0
101
+ API_VERSION = :v201601
102
+
103
+ begin
104
+ ad_group_id = 'INSERT_AD_GROUP_ID_HERE'.to_i
105
+ handle_partial_failures(ad_group_id)
106
+
107
+ # Authorization error.
108
+ rescue AdsCommon::Errors::OAuth2VerificationRequired => e
109
+ puts "Authorization credentials are not valid. Edit adwords_api.yml for " +
110
+ "OAuth2 client ID and secret and run misc/setup_oauth2.rb example " +
111
+ "to retrieve and store OAuth2 tokens."
112
+ puts "See this wiki page for more details:\n\n " +
113
+ 'https://github.com/googleads/google-api-ads-ruby/wiki/OAuth2'
114
+
115
+ # HTTP errors.
116
+ rescue AdsCommon::Errors::HttpError => e
117
+ puts "HTTP Error: %s" % e
118
+
119
+ # API errors.
120
+ rescue AdwordsApi::Errors::ApiException => e
121
+ puts "Message: %s" % e.message
122
+ puts 'Errors:'
123
+ e.errors.each_with_index do |error, index|
124
+ puts "\tError [%d]:" % (index + 1)
125
+ error.each do |field, value|
126
+ puts "\t\t%s: %s" % [field, value]
127
+ end
128
+ end
129
+ end
130
+ end
@@ -0,0 +1,141 @@
1
+ #!/usr/bin/env ruby
2
+ # Encoding: utf-8
3
+ #
4
+ # Copyright:: Copyright 2011, Google Inc. All Rights Reserved.
5
+ #
6
+ # License:: Licensed under the Apache License, Version 2.0 (the "License");
7
+ # you may not use this file except in compliance with the License.
8
+ # You may obtain a copy of the License at
9
+ #
10
+ # http://www.apache.org/licenses/LICENSE-2.0
11
+ #
12
+ # Unless required by applicable law or agreed to in writing, software
13
+ # distributed under the License is distributed on an "AS IS" BASIS,
14
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
15
+ # implied.
16
+ # See the License for the specific language governing permissions and
17
+ # limitations under the License.
18
+ #
19
+ # This example demonstrates how to handle policy violation errors. To create
20
+ # an ad group, run add_ad_group.rb.
21
+
22
+ require 'adwords_api'
23
+ require 'adwords_api/utils'
24
+
25
+ def handle_policy_violation_error(ad_group_id)
26
+ # AdwordsApi::Api will read a config file from ENV['HOME']/adwords_api.yml
27
+ # when called without parameters.
28
+ adwords = AdwordsApi::Api.new
29
+
30
+ # To enable logging of SOAP requests, set the log_level value to 'DEBUG' in
31
+ # the configuration file or provide your own logger:
32
+ # adwords.logger = Logger.new('adwords_xml.log')
33
+
34
+ ad_group_ad_srv = adwords.service(:AdGroupAdService, API_VERSION)
35
+
36
+ # Create text ad.
37
+ text_ad_operation = {
38
+ :operator => 'ADD',
39
+ :operand => {
40
+ :ad_group_id => ad_group_id,
41
+ :ad => {
42
+ # The 'xsi_type' field allows you to specify the xsi:type of the object
43
+ # being created. It's only necessary when you must provide an explicit
44
+ # type that the client library can't infer.
45
+ :xsi_type => 'TextAd',
46
+ :headline => 'Mars Cruise!!!',
47
+ :description1 => 'Visit the Red Planet in style.',
48
+ :description2 => 'Low-gravity fun for everyone!',
49
+ :final_urls => ['http://www.example.com'],
50
+ :display_url => 'www.example.com',
51
+ }
52
+ }
53
+ }
54
+
55
+ operations = [text_ad_operation]
56
+
57
+ # Validate ad.
58
+ begin
59
+ # Enable "validate only" for the length of this block
60
+ adwords.validate_only do
61
+ ad_group_ad_srv.mutate(operations)
62
+ end
63
+ puts 'Validation successful, no errors returned.'
64
+ rescue AdwordsApi::Errors::ApiException => e
65
+ e.errors.each do |error|
66
+ if error[:xsi_type] == 'PolicyViolationError'
67
+ operation_index = AdwordsApi::Utils.operation_index_for_error(error)
68
+ operation = operations[operation_index]
69
+ puts "Ad with headline '%s' violated %s policy '%s'." %
70
+ [operation[:operand][:ad][:headline],
71
+ error[:is_exemptable] ? 'exemptable' : 'non-exemptable',
72
+ error[:external_policy_name]]
73
+ if error[:is_exemptable]
74
+ # Add exemption request to the operation.
75
+ puts "Adding exemption request for policy name '%s' on text '%s'." %
76
+ [error[:key][:policy_name], error[:key][:violating_text]]
77
+ unless operation[:exemption_requests]
78
+ operation[:exemption_requests] = []
79
+ end
80
+ operation[:exemption_requests] << {
81
+ :key => error[:key]
82
+ }
83
+ else
84
+ # Remove non-exemptable operation
85
+ puts "Removing the operation from the request."
86
+ operations.delete(operation)
87
+ end
88
+ else
89
+ # Non-policy error returned, re-throw exception.
90
+ raise e
91
+ end
92
+ end
93
+ end
94
+
95
+ # Add ads.
96
+ if operations.size > 0
97
+ response = ad_group_ad_srv.mutate(operations)
98
+ if response and response[:value]
99
+ ads = response[:value]
100
+ puts "Added #{ads.length} ad(s) to ad group #{ad_group_id}."
101
+ ads.each do |ad|
102
+ puts " Ad id is #{ad[:ad][:id]}, type is #{ad[:ad][:xsi_type]} and " +
103
+ "status is \"#{ad[:status]}\"."
104
+ end
105
+ else
106
+ puts "No ads were added."
107
+ end
108
+ end
109
+ end
110
+
111
+ if __FILE__ == $0
112
+ API_VERSION = :v201601
113
+
114
+ begin
115
+ ad_group_id = 'INSERT_AD_GROUP_ID_HERE'.to_i
116
+ handle_policy_violation_error(ad_group_id)
117
+
118
+ # Authorization error.
119
+ rescue AdsCommon::Errors::OAuth2VerificationRequired => e
120
+ puts "Authorization credentials are not valid. Edit adwords_api.yml for " +
121
+ "OAuth2 client ID and secret and run misc/setup_oauth2.rb example " +
122
+ "to retrieve and store OAuth2 tokens."
123
+ puts "See this wiki page for more details:\n\n " +
124
+ 'https://github.com/googleads/google-api-ads-ruby/wiki/OAuth2'
125
+
126
+ # HTTP errors.
127
+ rescue AdsCommon::Errors::HttpError => e
128
+ puts "HTTP Error: %s" % e
129
+
130
+ # API errors.
131
+ rescue AdwordsApi::Errors::ApiException => e
132
+ puts "Message: %s" % e.message
133
+ puts 'Errors:'
134
+ e.errors.each_with_index do |error, index|
135
+ puts "\tError [%d]:" % (index + 1)
136
+ error.each do |field, value|
137
+ puts "\t\t%s: %s" % [field, value]
138
+ end
139
+ end
140
+ end
141
+ end
@@ -0,0 +1,193 @@
1
+ #!/usr/bin/env ruby
2
+ # Encoding: utf-8
3
+ #
4
+ # Copyright:: Copyright 2014, Google Inc. All Rights Reserved.
5
+ #
6
+ # License:: Licensed under the Apache License, Version 2.0 (the "License");
7
+ # you may not use this file except in compliance with the License.
8
+ # You may obtain a copy of the License at
9
+ #
10
+ # http://www.apache.org/licenses/LICENSE-2.0
11
+ #
12
+ # Unless required by applicable law or agreed to in writing, software
13
+ # distributed under the License is distributed on an "AS IS" BASIS,
14
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
15
+ # implied.
16
+ # See the License for the specific language governing permissions and
17
+ # limitations under the License.
18
+ #
19
+ # This example adds a feed that syncs feed items from a Google My Business (GMB)
20
+ # account and associates the feed with a customer.
21
+
22
+ require 'adwords_api'
23
+ require 'date'
24
+
25
+ def add_gmb_location_extensions(gmb_email_address, gmb_access_token,
26
+ business_account_identifier)
27
+ # AdwordsApi::Api will read a config file from ENV['HOME']/adwords_api.yml
28
+ # when called without parameters.
29
+ adwords = AdwordsApi::Api.new
30
+
31
+ # To enable logging of SOAP requests, set the log_level value to 'DEBUG' in
32
+ # the configuration file or provide your own logger:
33
+ # adwords.logger = Logger.new('adwords_xml.log')
34
+
35
+ feed_srv = adwords.service(:FeedService, API_VERSION)
36
+ customer_feed_srv = adwords.service(:CustomerFeedService, API_VERSION)
37
+
38
+ if gmb_access_token.nil?
39
+ gmb_access_token = adwords.get_auth_handler.get_token(
40
+ adwords.credential_handler.credentials)[:access_token]
41
+ end
42
+
43
+ # Create a feed that will sync to the Google My Business account specified
44
+ # by gmb_email_address. Do not add FeedAttributes to this object, as AdWords
45
+ # will add them automatically because this will be a system generated feed.
46
+ gmb_feed = {
47
+ :name => "GMB feed #%d" % (Time.new.to_f * 1000).to_i,
48
+ :system_feed_generation_data => {
49
+ :xsi_type => 'PlacesLocationFeedData',
50
+ :o_auth_info => {
51
+ :http_method => 'GET',
52
+ :http_request_url => 'https://www.googleapis.com/auth/adwords',
53
+ :http_authorization_header => "Bearer %s" % gmb_access_token
54
+ },
55
+ :email_address => gmb_email_address
56
+ },
57
+ # Since this feed's feed items will be managed by AdWords, you must set
58
+ # its origin to ADWORDS.
59
+ :origin => 'ADWORDS'
60
+ }
61
+
62
+ # Optional: specify labels to filter Google My Business listings. If
63
+ # specified, only listings that have any of the labels set are synchronized
64
+ # into FeedItems.
65
+ gmb_feed[:system_feed_generation_data][:label_filters] =
66
+ ['Stores in New York City']
67
+
68
+ # Only include the business_account_identifier if it's specified.
69
+ # A nil value will cause an invalid request.
70
+ unless business_account_identifier.nil?
71
+ gmb_feed[:system_feed_generation_data][:business_account_identifier] =
72
+ business_account_identifier
73
+ end
74
+
75
+ gmb_operation = {
76
+ :operator => 'ADD',
77
+ :operand => gmb_feed
78
+ }
79
+
80
+ result = feed_srv.mutate([gmb_operation])
81
+ added_feed = result[:value].first
82
+ puts "Added GMB feed with ID %d" % added_feed[:id]
83
+
84
+ # Add a CustomerFeed that associates the feed with this customer for the
85
+ # LOCATION placeholder type.
86
+ customer_feed = {
87
+ :feed_id => added_feed[:id],
88
+ :placeholder_types => [PLACEHOLDER_TYPE_LOCATION],
89
+ :matching_function => {
90
+ :operator => 'IDENTITY',
91
+ :lhs_operand => {
92
+ :xsi_type => 'ConstantOperand',
93
+ :type => 'BOOLEAN',
94
+ :boolean_value => true
95
+ }
96
+ }
97
+ }
98
+
99
+ customer_feed_operation = {
100
+ :xsi_type => 'CustomerFeedOperation',
101
+ :operator => 'ADD',
102
+ :operand => customer_feed
103
+ }
104
+
105
+ added_customer_feed = nil
106
+ number_of_attempts = 0
107
+ while i < MAX_CUSTOMER_FEED_ADD_ATTEMPTS && !added_customer_feed
108
+ number_of_attempts += 1
109
+ begin
110
+ result = customer_feed_srv.mutate([customer_feed_operation])
111
+ added_customer_feed = result[:value].first
112
+ puts "Attempt #%d to add the CustomerFeed was successful" %
113
+ number_of_attempts
114
+ rescue
115
+ sleep_seconds = 5 * (2 ** number_of_attempts)
116
+ puts ("Attempt #%d to add the CustomerFeed was not succeessful. " +
117
+ "Waiting %d seconds before trying again.") %
118
+ [number_of_attempts, sleep_seconds]
119
+ sleep(sleep_seconds)
120
+ end
121
+ end
122
+
123
+ unless added_customer_feed
124
+ raise StandardError, ("Could not create the CustomerFeed after %d " +
125
+ "attempts. Please retry the CustomerFeed ADD operation later.") %
126
+ MAX_CUSTOMER_FEED_ADD_ATTEMPTS
127
+ end
128
+
129
+ puts "Added CustomerFeed for feed ID %d and placeholder type %d" %
130
+ [added_customer_feed[:id], added_customer_feed[:placeholder_types].first]
131
+
132
+ # OPTIONAL: Create a CampaignFeed to specify which FeedItems to use at the
133
+ # Campaign level. This will be similar to the CampaignFeed in the
134
+ # add_site_links example, except you can filter based on the business name
135
+ # and category of each FeedItem by using a FeedAttributeOperand in your
136
+ # matching function.
137
+
138
+ # OPTIONAL: Create an AdGroupFeed for even more fine grained control over
139
+ # which feed items are used at the AdGroup level.
140
+ end
141
+
142
+ if __FILE__ == $0
143
+ API_VERSION = :v201601
144
+ PLACEHOLDER_TYPE_LOCATION = 7
145
+ MAX_CUSTOMER_FEED_ADD_ATTEMPTS = 10
146
+
147
+ begin
148
+ # The email address of either an owner or a manager of the GMB account.
149
+ gmb_email_address = 'INSERT_GMB_EMAIL_ADDRESS_HERE'
150
+
151
+ # If the gmbEmailAddress above is the same as you used to generate your
152
+ # AdWords API refresh token, leave the value below as nil.
153
+ # Otherwise, to obtain an access token for your GMB account, generate a
154
+ # refresh token as you did for AdWords, but make sure you are logged in as
155
+ # the same user as gmb_email_address above when you follow the link, then
156
+ # capture the generated access token
157
+ gmb_access_token = nil
158
+
159
+ # If the gmb_email_address above is for a GMB manager instead of
160
+ # the GMB account owner, then set business_account_identifier to the
161
+ # +Page ID of a location for which the manager has access. See the
162
+ # location extensions guide at
163
+ # https://developers.google.com/adwords/api/docs/guides/feed-services-locations
164
+ # for details.
165
+ business_account_identifier = nil
166
+
167
+ add_gmb_location_extensions(gmb_email_address, gmb_access_token,
168
+ business_account_identifier)
169
+
170
+ # Authorization error.
171
+ rescue AdsCommon::Errors::OAuth2VerificationRequired => e
172
+ puts "Authorization credentials are not valid. Edit adwords_api.yml for " +
173
+ "OAuth2 client ID and secret and run misc/setup_oauth2.rb example " +
174
+ "to retrieve and store OAuth2 tokens."
175
+ puts "See this wiki page for more details:\n\n " +
176
+ 'https://github.com/googleads/google-api-ads-ruby/wiki/OAuth2'
177
+
178
+ # HTTP errors.
179
+ rescue AdsCommon::Errors::HttpError => e
180
+ puts "HTTP Error: %s" % e
181
+
182
+ # API errors.
183
+ rescue AdwordsApi::Errors::ApiException => e
184
+ puts "Message: %s" % e.message
185
+ puts 'Errors:'
186
+ e.errors.each_with_index do |error, index|
187
+ puts "\tError [%d]:" % (index + 1)
188
+ error.each do |field, value|
189
+ puts "\t\t%s: %s" % [field, value]
190
+ end
191
+ end
192
+ end
193
+ end