@rudderstack/integrations-lib 0.2.23 → 0.2.25
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.
- package/build/sdks/googleAdsRestAPI/config.d.ts +2 -0
- package/build/sdks/googleAdsRestAPI/config.d.ts.map +1 -0
- package/build/sdks/googleAdsRestAPI/config.js +5 -0
- package/build/sdks/googleAdsRestAPI/googleAds.d.ts +19 -0
- package/build/sdks/googleAdsRestAPI/googleAds.d.ts.map +1 -0
- package/build/sdks/googleAdsRestAPI/googleAds.js +119 -0
- package/build/sdks/googleAdsRestAPI/googleAds.test.d.ts +2 -0
- package/build/sdks/googleAdsRestAPI/googleAds.test.d.ts.map +1 -0
- package/build/sdks/googleAdsRestAPI/googleAds.test.js +757 -0
- package/build/sdks/googleAdsRestAPI/index.d.ts +3 -0
- package/build/sdks/googleAdsRestAPI/index.d.ts.map +1 -0
- package/build/sdks/googleAdsRestAPI/index.js +9 -0
- package/build/sdks/googleAdsRestAPI/types.d.ts +243 -0
- package/build/sdks/googleAdsRestAPI/types.d.ts.map +1 -0
- package/build/sdks/googleAdsRestAPI/types.js +3 -0
- package/build/sdks/index.d.ts +4 -4
- package/build/sdks/index.d.ts.map +1 -1
- package/build/sdks/index.js +8 -8
- package/build/utils/misc.d.ts +10 -0
- package/build/utils/misc.d.ts.map +1 -1
- package/build/utils/misc.js +24 -2
- package/build/utils/misc.test.js +144 -1
- package/package.json +2 -1
|
@@ -0,0 +1,757 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
14
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
15
|
+
}) : function(o, v) {
|
|
16
|
+
o["default"] = v;
|
|
17
|
+
});
|
|
18
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
19
|
+
var ownKeys = function(o) {
|
|
20
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
21
|
+
var ar = [];
|
|
22
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
23
|
+
return ar;
|
|
24
|
+
};
|
|
25
|
+
return ownKeys(o);
|
|
26
|
+
};
|
|
27
|
+
return function (mod) {
|
|
28
|
+
if (mod && mod.__esModule) return mod;
|
|
29
|
+
var result = {};
|
|
30
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
31
|
+
__setModuleDefault(result, mod);
|
|
32
|
+
return result;
|
|
33
|
+
};
|
|
34
|
+
})();
|
|
35
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
36
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
37
|
+
};
|
|
38
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
39
|
+
const axios_mock_adapter_1 = __importDefault(require("axios-mock-adapter"));
|
|
40
|
+
const axios_1 = __importStar(require("axios"));
|
|
41
|
+
const googleAds_1 = __importDefault(require("./googleAds"));
|
|
42
|
+
describe('Google ads SDK', () => {
|
|
43
|
+
let mock;
|
|
44
|
+
beforeEach(() => {
|
|
45
|
+
mock = new axios_mock_adapter_1.default(axios_1.default, { onNoMatch: 'throwException' });
|
|
46
|
+
});
|
|
47
|
+
afterEach(() => {
|
|
48
|
+
mock.restore();
|
|
49
|
+
});
|
|
50
|
+
describe('Test getConversionAction function', () => {
|
|
51
|
+
const testCases = [
|
|
52
|
+
{
|
|
53
|
+
name: 'should return conversion action ID when API returns success response',
|
|
54
|
+
authObject: {
|
|
55
|
+
accessToken: 'test-access-token',
|
|
56
|
+
customerId: '123456789',
|
|
57
|
+
developerToken: 'test-developer-token',
|
|
58
|
+
},
|
|
59
|
+
mockResponse: [
|
|
60
|
+
{
|
|
61
|
+
results: [
|
|
62
|
+
{
|
|
63
|
+
conversionAction: {
|
|
64
|
+
resourceName: 'customers/123456789/conversionActions/9876543',
|
|
65
|
+
id: '9876543',
|
|
66
|
+
},
|
|
67
|
+
},
|
|
68
|
+
],
|
|
69
|
+
},
|
|
70
|
+
],
|
|
71
|
+
conversionName: 'test_conversion',
|
|
72
|
+
status: 200,
|
|
73
|
+
expectedResult: 'customers/123456789/conversionActions/9876543',
|
|
74
|
+
expectedRequest: {
|
|
75
|
+
url: '/123456789/googleAds:searchStream',
|
|
76
|
+
data: JSON.stringify({
|
|
77
|
+
query: "SELECT conversion_action.id FROM conversion_action WHERE conversion_action.name = 'test_conversion'",
|
|
78
|
+
}),
|
|
79
|
+
},
|
|
80
|
+
},
|
|
81
|
+
{
|
|
82
|
+
name: 'handle failure scenario when accessToken is wrong',
|
|
83
|
+
authObject: {
|
|
84
|
+
customerId: '123456789',
|
|
85
|
+
developerToken: 'test-developer-token',
|
|
86
|
+
accessToken: 'wrong-access-token',
|
|
87
|
+
},
|
|
88
|
+
mockResponse: [
|
|
89
|
+
{
|
|
90
|
+
error: {
|
|
91
|
+
code: 401,
|
|
92
|
+
message: 'Request had invalid authentication credentials. Expected OAuth 2 access token, login cookie or other valid authentication credential. See https://developers.google.com/identity/sign-in/web/devconsole-project.',
|
|
93
|
+
status: 'UNAUTHENTICATED',
|
|
94
|
+
},
|
|
95
|
+
},
|
|
96
|
+
],
|
|
97
|
+
mockResponseHeader: new axios_1.AxiosHeaders({
|
|
98
|
+
'transfer-encoding': 'chunked',
|
|
99
|
+
}),
|
|
100
|
+
conversionName: 'test_conversion',
|
|
101
|
+
status: 401,
|
|
102
|
+
expectedResult: {
|
|
103
|
+
headers: new axios_1.AxiosHeaders({
|
|
104
|
+
'transfer-encoding': 'chunked',
|
|
105
|
+
}),
|
|
106
|
+
message: undefined,
|
|
107
|
+
responseBody: [
|
|
108
|
+
{
|
|
109
|
+
error: {
|
|
110
|
+
code: 401,
|
|
111
|
+
message: 'Request had invalid authentication credentials. Expected OAuth 2 access token, login cookie or other valid authentication credential. See https://developers.google.com/identity/sign-in/web/devconsole-project.',
|
|
112
|
+
status: 'UNAUTHENTICATED',
|
|
113
|
+
},
|
|
114
|
+
},
|
|
115
|
+
],
|
|
116
|
+
statusCode: 401,
|
|
117
|
+
type: 'application-error',
|
|
118
|
+
},
|
|
119
|
+
expectedRequest: {
|
|
120
|
+
url: '/123456789/googleAds:searchStream',
|
|
121
|
+
data: JSON.stringify({
|
|
122
|
+
query: "SELECT conversion_action.id FROM conversion_action WHERE conversion_action.name = 'test_conversion'",
|
|
123
|
+
}),
|
|
124
|
+
},
|
|
125
|
+
},
|
|
126
|
+
{
|
|
127
|
+
name: 'handle the situation when the conversion is not available',
|
|
128
|
+
authObject: {
|
|
129
|
+
accessToken: 'test-access-token',
|
|
130
|
+
customerId: '123456789',
|
|
131
|
+
developerToken: 'test-developer-token',
|
|
132
|
+
},
|
|
133
|
+
mockResponse: [],
|
|
134
|
+
conversionName: 'non-existing-conversion',
|
|
135
|
+
status: 200,
|
|
136
|
+
expectedResult: null,
|
|
137
|
+
expectedRequest: {
|
|
138
|
+
url: '/123456789/googleAds:searchStream',
|
|
139
|
+
data: JSON.stringify({
|
|
140
|
+
query: "SELECT conversion_action.id FROM conversion_action WHERE conversion_action.name = 'non-existing-conversion'",
|
|
141
|
+
}),
|
|
142
|
+
},
|
|
143
|
+
},
|
|
144
|
+
];
|
|
145
|
+
testCases.forEach(({ name, authObject, mockResponse, conversionName, expectedRequest, expectedResult, status, mockResponseHeader, }) => {
|
|
146
|
+
it(name, async () => {
|
|
147
|
+
const googleAds = new googleAds_1.default(authObject);
|
|
148
|
+
mock
|
|
149
|
+
.onPost('https://googleads.googleapis.com/v19/customers/123456789/googleAds:searchStream')
|
|
150
|
+
.reply(status, mockResponse, mockResponseHeader);
|
|
151
|
+
const result = await googleAds.getConversionActionId(conversionName);
|
|
152
|
+
expect(mock.history.post[0]).toMatchObject(expectedRequest);
|
|
153
|
+
expect(result).toEqual(expectedResult);
|
|
154
|
+
});
|
|
155
|
+
});
|
|
156
|
+
});
|
|
157
|
+
describe('Test getCustomVariable function', () => {
|
|
158
|
+
const testCases = [
|
|
159
|
+
{
|
|
160
|
+
name: 'should return an array of CustomVariableResults when API call succeeds',
|
|
161
|
+
authObject: {
|
|
162
|
+
accessToken: 'test-access-token',
|
|
163
|
+
customerId: '123456789',
|
|
164
|
+
developerToken: 'test-developer-token',
|
|
165
|
+
},
|
|
166
|
+
mockResponse: [
|
|
167
|
+
{
|
|
168
|
+
results: [
|
|
169
|
+
{
|
|
170
|
+
conversionCustomVariable: {
|
|
171
|
+
resourceName: 'customers/123/conversionCustomVariables/456',
|
|
172
|
+
name: 'test_variable',
|
|
173
|
+
},
|
|
174
|
+
},
|
|
175
|
+
],
|
|
176
|
+
fieldMask: 'conversion_custom_variable.name',
|
|
177
|
+
requestId: '123456',
|
|
178
|
+
queryResourceConsumption: '100',
|
|
179
|
+
},
|
|
180
|
+
],
|
|
181
|
+
status: 200,
|
|
182
|
+
expectedResult: [
|
|
183
|
+
{
|
|
184
|
+
conversionCustomVariable: {
|
|
185
|
+
resourceName: 'customers/123/conversionCustomVariables/456',
|
|
186
|
+
name: 'test_variable',
|
|
187
|
+
},
|
|
188
|
+
},
|
|
189
|
+
],
|
|
190
|
+
expectedRequest: {
|
|
191
|
+
url: '/123456789/googleAds:searchStream',
|
|
192
|
+
data: JSON.stringify({
|
|
193
|
+
query: 'SELECT conversion_custom_variable.name FROM conversion_custom_variable',
|
|
194
|
+
}),
|
|
195
|
+
},
|
|
196
|
+
},
|
|
197
|
+
{
|
|
198
|
+
name: 'should handle the scenario when there is not any customVariables',
|
|
199
|
+
authObject: {
|
|
200
|
+
accessToken: 'test-access-token',
|
|
201
|
+
customerId: '123456789',
|
|
202
|
+
developerToken: 'test-developer-token',
|
|
203
|
+
},
|
|
204
|
+
status: 200,
|
|
205
|
+
mockResponse: [
|
|
206
|
+
{
|
|
207
|
+
results: [],
|
|
208
|
+
fieldMask: 'conversion_custom_variable.name',
|
|
209
|
+
requestId: '123456',
|
|
210
|
+
queryResourceConsumption: '100',
|
|
211
|
+
},
|
|
212
|
+
],
|
|
213
|
+
expectedResult: [],
|
|
214
|
+
expectedRequest: {
|
|
215
|
+
url: '/123456789/googleAds:searchStream',
|
|
216
|
+
data: JSON.stringify({
|
|
217
|
+
query: 'SELECT conversion_custom_variable.name FROM conversion_custom_variable',
|
|
218
|
+
}),
|
|
219
|
+
},
|
|
220
|
+
},
|
|
221
|
+
{
|
|
222
|
+
name: 'should return an array of CustomVariableResults when API call succeeds',
|
|
223
|
+
authObject: {
|
|
224
|
+
accessToken: 'wrong-access-token',
|
|
225
|
+
customerId: '123456789',
|
|
226
|
+
developerToken: 'test-developer-token',
|
|
227
|
+
},
|
|
228
|
+
mockResponse: [
|
|
229
|
+
{
|
|
230
|
+
error: {
|
|
231
|
+
code: 401,
|
|
232
|
+
message: 'Request had invalid authentication credentials. Expected OAuth 2 access token, login cookie or other valid authentication credential. See https://developers.google.com/identity/sign-in/web/devconsole-project.',
|
|
233
|
+
status: 'UNAUTHENTICATED',
|
|
234
|
+
},
|
|
235
|
+
},
|
|
236
|
+
],
|
|
237
|
+
status: 401,
|
|
238
|
+
expectedResult: {
|
|
239
|
+
headers: new axios_1.AxiosHeaders({
|
|
240
|
+
'transfer-encoding': 'chunked',
|
|
241
|
+
}),
|
|
242
|
+
message: undefined,
|
|
243
|
+
responseBody: [
|
|
244
|
+
{
|
|
245
|
+
error: {
|
|
246
|
+
code: 401,
|
|
247
|
+
message: 'Request had invalid authentication credentials. Expected OAuth 2 access token, login cookie or other valid authentication credential. See https://developers.google.com/identity/sign-in/web/devconsole-project.',
|
|
248
|
+
status: 'UNAUTHENTICATED',
|
|
249
|
+
},
|
|
250
|
+
},
|
|
251
|
+
],
|
|
252
|
+
statusCode: 401,
|
|
253
|
+
type: 'application-error',
|
|
254
|
+
},
|
|
255
|
+
expectedRequest: {
|
|
256
|
+
url: '/123456789/googleAds:searchStream',
|
|
257
|
+
data: JSON.stringify({
|
|
258
|
+
query: 'SELECT conversion_custom_variable.name FROM conversion_custom_variable',
|
|
259
|
+
}),
|
|
260
|
+
},
|
|
261
|
+
mockResponseHeader: new axios_1.AxiosHeaders({
|
|
262
|
+
'transfer-encoding': 'chunked',
|
|
263
|
+
}),
|
|
264
|
+
},
|
|
265
|
+
];
|
|
266
|
+
testCases.forEach(({ name, authObject, mockResponse, expectedRequest, expectedResult, status, mockResponseHeader, }) => {
|
|
267
|
+
it(name, async () => {
|
|
268
|
+
const googleAds = new googleAds_1.default(authObject);
|
|
269
|
+
mock
|
|
270
|
+
.onPost('https://googleads.googleapis.com/v19/customers/123456789/googleAds:searchStream')
|
|
271
|
+
.reply(status, mockResponse, mockResponseHeader);
|
|
272
|
+
const result = await googleAds.getCustomVariable();
|
|
273
|
+
expect(mock.history.post[0]).toMatchObject(expectedRequest);
|
|
274
|
+
expect(result).toEqual(expectedResult);
|
|
275
|
+
});
|
|
276
|
+
});
|
|
277
|
+
});
|
|
278
|
+
describe('Test addConversionAdjustMent function', () => {
|
|
279
|
+
const TestCases = [
|
|
280
|
+
{
|
|
281
|
+
name: 'should successfully upload conversion adjustments with valid data',
|
|
282
|
+
authObject: {
|
|
283
|
+
accessToken: 'valid-access-token',
|
|
284
|
+
customerId: '123456789',
|
|
285
|
+
developerToken: 'test-developer-token',
|
|
286
|
+
},
|
|
287
|
+
requestData: {
|
|
288
|
+
conversionAdjustments: [
|
|
289
|
+
{
|
|
290
|
+
gclidDateTimePair: {
|
|
291
|
+
gclid: 'test-gclid',
|
|
292
|
+
conversionDateTime: '2023-01-01T00:00:00Z',
|
|
293
|
+
},
|
|
294
|
+
adjustmentType: 'RESTATEMENT',
|
|
295
|
+
restatementValue: {
|
|
296
|
+
adjustedValue: 100,
|
|
297
|
+
currencyCode: 'USD',
|
|
298
|
+
},
|
|
299
|
+
userIdentifiers: [
|
|
300
|
+
{
|
|
301
|
+
userIdentifierSource: 'FIRST_PARTY',
|
|
302
|
+
hashedEmail: '389205904d6c7bb83fc676513911226f2be25bf1465616bb9b29587100ab1414',
|
|
303
|
+
},
|
|
304
|
+
],
|
|
305
|
+
},
|
|
306
|
+
],
|
|
307
|
+
partialFailure: false,
|
|
308
|
+
},
|
|
309
|
+
status: 200,
|
|
310
|
+
mockResponse: {
|
|
311
|
+
conversionAdjustments: [
|
|
312
|
+
{
|
|
313
|
+
gclidDateTimePair: {
|
|
314
|
+
gclid: 'test-gclid',
|
|
315
|
+
conversionDateTime: '2023-01-01T00:00:00Z',
|
|
316
|
+
},
|
|
317
|
+
adjustmentType: 'RESTATEMENT',
|
|
318
|
+
restatementValue: {
|
|
319
|
+
adjustedValue: 100,
|
|
320
|
+
currencyCode: 'USD',
|
|
321
|
+
},
|
|
322
|
+
userIdentifiers: [],
|
|
323
|
+
},
|
|
324
|
+
],
|
|
325
|
+
partialFailure: false,
|
|
326
|
+
},
|
|
327
|
+
expectedResult: {
|
|
328
|
+
type: 'success',
|
|
329
|
+
statusCode: 200,
|
|
330
|
+
responseBody: {
|
|
331
|
+
conversionAdjustments: [
|
|
332
|
+
{
|
|
333
|
+
gclidDateTimePair: {
|
|
334
|
+
gclid: 'test-gclid',
|
|
335
|
+
conversionDateTime: '2023-01-01T00:00:00Z',
|
|
336
|
+
},
|
|
337
|
+
adjustmentType: 'RESTATEMENT',
|
|
338
|
+
restatementValue: { adjustedValue: 100, currencyCode: 'USD' },
|
|
339
|
+
userIdentifiers: [],
|
|
340
|
+
},
|
|
341
|
+
],
|
|
342
|
+
partialFailure: false,
|
|
343
|
+
},
|
|
344
|
+
headers: new axios_1.AxiosHeaders({ 'Content-Type': 'application/json' }),
|
|
345
|
+
},
|
|
346
|
+
mockResponseHeader: new axios_1.AxiosHeaders({ 'Content-Type': 'application/json' }),
|
|
347
|
+
},
|
|
348
|
+
{
|
|
349
|
+
name: 'should successfully handle failure scenario',
|
|
350
|
+
authObject: {
|
|
351
|
+
accessToken: 'wrong-access-token',
|
|
352
|
+
customerId: '123456789',
|
|
353
|
+
developerToken: 'test-developer-token',
|
|
354
|
+
},
|
|
355
|
+
requestData: {
|
|
356
|
+
conversionAdjustments: [
|
|
357
|
+
{
|
|
358
|
+
gclidDateTimePair: {
|
|
359
|
+
gclid: 'test-gclid',
|
|
360
|
+
conversionDateTime: '2023-01-01T00:00:00Z',
|
|
361
|
+
},
|
|
362
|
+
adjustmentType: 'RESTATEMENT',
|
|
363
|
+
restatementValue: {
|
|
364
|
+
adjustedValue: 100,
|
|
365
|
+
currencyCode: 'USD',
|
|
366
|
+
},
|
|
367
|
+
userIdentifiers: [
|
|
368
|
+
{
|
|
369
|
+
userIdentifierSource: 'FIRST_PARTY',
|
|
370
|
+
hashedEmail: '389205904d6c7bb83fc676513911226f2be25bf1465616bb9b29587100ab1414',
|
|
371
|
+
},
|
|
372
|
+
],
|
|
373
|
+
},
|
|
374
|
+
],
|
|
375
|
+
partialFailure: false,
|
|
376
|
+
},
|
|
377
|
+
status: 401,
|
|
378
|
+
mockResponse: {
|
|
379
|
+
headers: new axios_1.AxiosHeaders({
|
|
380
|
+
'transfer-encoding': 'chunked',
|
|
381
|
+
}),
|
|
382
|
+
message: undefined,
|
|
383
|
+
responseBody: [
|
|
384
|
+
{
|
|
385
|
+
error: {
|
|
386
|
+
code: 401,
|
|
387
|
+
message: 'Request had invalid authentication credentials. Expected OAuth 2 access token, login cookie or other valid authentication credential. See https://developers.google.com/identity/sign-in/web/devconsole-project.',
|
|
388
|
+
status: 'UNAUTHENTICATED',
|
|
389
|
+
},
|
|
390
|
+
},
|
|
391
|
+
],
|
|
392
|
+
statusCode: 401,
|
|
393
|
+
type: 'application-error',
|
|
394
|
+
},
|
|
395
|
+
expectedResult: {
|
|
396
|
+
headers: new axios_1.AxiosHeaders({
|
|
397
|
+
'Content-Type': 'application/json',
|
|
398
|
+
}),
|
|
399
|
+
message: undefined,
|
|
400
|
+
responseBody: {
|
|
401
|
+
headers: { 'transfer-encoding': 'chunked' },
|
|
402
|
+
responseBody: [
|
|
403
|
+
{
|
|
404
|
+
error: {
|
|
405
|
+
code: 401,
|
|
406
|
+
message: 'Request had invalid authentication credentials. Expected OAuth 2 access token, login cookie or other valid authentication credential. See https://developers.google.com/identity/sign-in/web/devconsole-project.',
|
|
407
|
+
status: 'UNAUTHENTICATED',
|
|
408
|
+
},
|
|
409
|
+
},
|
|
410
|
+
],
|
|
411
|
+
statusCode: 401,
|
|
412
|
+
type: 'application-error',
|
|
413
|
+
},
|
|
414
|
+
statusCode: 401,
|
|
415
|
+
type: 'application-error',
|
|
416
|
+
},
|
|
417
|
+
mockResponseHeader: new axios_1.AxiosHeaders({ 'Content-Type': 'application/json' }),
|
|
418
|
+
},
|
|
419
|
+
];
|
|
420
|
+
TestCases.forEach(({ name, authObject, mockResponse, requestData, expectedResult, mockResponseHeader, status, }) => {
|
|
421
|
+
it(name, async () => {
|
|
422
|
+
const googleAds = new googleAds_1.default(authObject);
|
|
423
|
+
mock
|
|
424
|
+
.onPost('https://googleads.googleapis.com/v19/customers/123456789:uploadConversionAdjustments')
|
|
425
|
+
.reply(status, mockResponse, mockResponseHeader);
|
|
426
|
+
const result = await googleAds.addConversionAdjustMent(requestData);
|
|
427
|
+
expect(result).toEqual(expectedResult);
|
|
428
|
+
});
|
|
429
|
+
});
|
|
430
|
+
});
|
|
431
|
+
describe('Test createOfflineUserDataJob function', () => {
|
|
432
|
+
const TestCases = [
|
|
433
|
+
{
|
|
434
|
+
name: 'should return job ID when response type is success for customerMatchUserList job',
|
|
435
|
+
authObject: {
|
|
436
|
+
accessToken: 'valid-access-token',
|
|
437
|
+
customerId: '123456789',
|
|
438
|
+
developerToken: 'test-developer-token',
|
|
439
|
+
loginCustomerId: '987654321',
|
|
440
|
+
},
|
|
441
|
+
requestData: {
|
|
442
|
+
job: {
|
|
443
|
+
type: 'CUSTOMER_MATCH_USER_LIST',
|
|
444
|
+
customerMatchUserListMetadata: {
|
|
445
|
+
userList: 'customers/123456789/userLists/1234',
|
|
446
|
+
},
|
|
447
|
+
},
|
|
448
|
+
},
|
|
449
|
+
status: 200,
|
|
450
|
+
mockResponse: {
|
|
451
|
+
resourceName: 'customers/123456789/offlineUserDataJobs/5678',
|
|
452
|
+
},
|
|
453
|
+
expectedResult: '5678',
|
|
454
|
+
mockResponseHeader: new axios_1.AxiosHeaders({ 'Content-Type': 'application/json' }),
|
|
455
|
+
},
|
|
456
|
+
{
|
|
457
|
+
name: 'should return job ID when response type is success for store sales job',
|
|
458
|
+
authObject: {
|
|
459
|
+
accessToken: 'valid-access-token',
|
|
460
|
+
customerId: '123456789',
|
|
461
|
+
developerToken: 'test-developer-token',
|
|
462
|
+
},
|
|
463
|
+
requestData: {
|
|
464
|
+
job: {
|
|
465
|
+
storeSalesMetadata: {
|
|
466
|
+
custom_key: 'CUSTOM_KEY',
|
|
467
|
+
loyaltyFraction: 1,
|
|
468
|
+
transaction_upload_fraction: '1',
|
|
469
|
+
},
|
|
470
|
+
type: 'STORE_SALES_UPLOAD_FIRST_PARTY',
|
|
471
|
+
},
|
|
472
|
+
},
|
|
473
|
+
status: 200,
|
|
474
|
+
mockResponse: {
|
|
475
|
+
resourceName: 'customers/123456789/offlineUserDataJobs/5678',
|
|
476
|
+
},
|
|
477
|
+
expectedResult: '5678',
|
|
478
|
+
mockResponseHeader: new axios_1.AxiosHeaders({ 'Content-Type': 'application/json' }),
|
|
479
|
+
},
|
|
480
|
+
];
|
|
481
|
+
TestCases.forEach(({ name, authObject, mockResponse, status, requestData, expectedResult, mockResponseHeader, }) => {
|
|
482
|
+
it(name, async () => {
|
|
483
|
+
const googleAds = new googleAds_1.default(authObject);
|
|
484
|
+
mock
|
|
485
|
+
.onPost('https://googleads.googleapis.com/v19/customers/123456789/offlineUserDataJobs:create')
|
|
486
|
+
.reply(status, mockResponse, mockResponseHeader);
|
|
487
|
+
const result = await googleAds.createOfflineUserDataJob(requestData);
|
|
488
|
+
expect(result).toEqual(expectedResult);
|
|
489
|
+
});
|
|
490
|
+
});
|
|
491
|
+
});
|
|
492
|
+
describe('Test addUserToOfflineUserDataJob function', () => {
|
|
493
|
+
const TestCases = [
|
|
494
|
+
{
|
|
495
|
+
name: 'should successfully add user data to an offline user data job',
|
|
496
|
+
authObject: {
|
|
497
|
+
accessToken: 'valid-access-token',
|
|
498
|
+
customerId: '123456789',
|
|
499
|
+
developerToken: 'test-developer-token',
|
|
500
|
+
loginCustomerId: '987654321',
|
|
501
|
+
},
|
|
502
|
+
jobId: '1234',
|
|
503
|
+
requestData: {
|
|
504
|
+
operations: [
|
|
505
|
+
{
|
|
506
|
+
create: {
|
|
507
|
+
userIdentifiers: [{ hashedEmail: 'hashed_email_value' }],
|
|
508
|
+
},
|
|
509
|
+
},
|
|
510
|
+
],
|
|
511
|
+
enablePartialFailure: false,
|
|
512
|
+
},
|
|
513
|
+
status: 200,
|
|
514
|
+
mockResponse: {},
|
|
515
|
+
expectedResult: {
|
|
516
|
+
headers: new axios_1.AxiosHeaders({ 'Content-Type': 'application/json' }),
|
|
517
|
+
responseBody: {},
|
|
518
|
+
statusCode: 200,
|
|
519
|
+
type: 'success',
|
|
520
|
+
},
|
|
521
|
+
mockResponseHeader: new axios_1.AxiosHeaders({ 'Content-Type': 'application/json' }),
|
|
522
|
+
},
|
|
523
|
+
];
|
|
524
|
+
TestCases.forEach(({ name, authObject, mockResponse, status, requestData, expectedResult, mockResponseHeader, jobId, }) => {
|
|
525
|
+
it(name, async () => {
|
|
526
|
+
const googleAds = new googleAds_1.default(authObject);
|
|
527
|
+
mock
|
|
528
|
+
.onPost('https://googleads.googleapis.com/v19/customers/123456789/offlineUserDataJobs/1234:addOperations')
|
|
529
|
+
.reply(status, mockResponse, mockResponseHeader);
|
|
530
|
+
const result = await googleAds.addUserToOfflineUserDataJob(jobId, requestData);
|
|
531
|
+
expect(result).toEqual(expectedResult);
|
|
532
|
+
});
|
|
533
|
+
});
|
|
534
|
+
});
|
|
535
|
+
describe('Test addConversionsToOfflineUserDataJob function', () => {
|
|
536
|
+
const TestCases = [
|
|
537
|
+
{
|
|
538
|
+
name: 'should successfully add store sales conversion data to an offline user data job',
|
|
539
|
+
authObject: {
|
|
540
|
+
accessToken: 'valid-access-token',
|
|
541
|
+
customerId: '123456789',
|
|
542
|
+
developerToken: 'test-developer-token',
|
|
543
|
+
loginCustomerId: '987654321',
|
|
544
|
+
},
|
|
545
|
+
jobId: '1234',
|
|
546
|
+
requestData: {
|
|
547
|
+
operations: [
|
|
548
|
+
{
|
|
549
|
+
create: {
|
|
550
|
+
userIdentifiers: [{ hashedEmail: 'hashed-email-value' }],
|
|
551
|
+
transactionInfo: {
|
|
552
|
+
conversionAction: 'customers/123456789/conversionActions/987',
|
|
553
|
+
conversionDateTime: '2023-01-01 12:00:00',
|
|
554
|
+
conversionValue: 10.99,
|
|
555
|
+
currencyCode: 'USD',
|
|
556
|
+
},
|
|
557
|
+
},
|
|
558
|
+
},
|
|
559
|
+
],
|
|
560
|
+
enablePartialFailure: true,
|
|
561
|
+
},
|
|
562
|
+
status: 200,
|
|
563
|
+
mockResponse: {},
|
|
564
|
+
expectedResult: {
|
|
565
|
+
headers: new axios_1.AxiosHeaders({ 'Content-Type': 'application/json' }),
|
|
566
|
+
responseBody: {},
|
|
567
|
+
statusCode: 200,
|
|
568
|
+
type: 'success',
|
|
569
|
+
},
|
|
570
|
+
mockResponseHeader: new axios_1.AxiosHeaders({ 'Content-Type': 'application/json' }),
|
|
571
|
+
},
|
|
572
|
+
];
|
|
573
|
+
TestCases.forEach(({ name, authObject, mockResponse, status, requestData, expectedResult, mockResponseHeader, jobId, }) => {
|
|
574
|
+
it(name, async () => {
|
|
575
|
+
const googleAds = new googleAds_1.default(authObject);
|
|
576
|
+
mock
|
|
577
|
+
.onPost('https://googleads.googleapis.com/v19/customers/123456789/offlineUserDataJobs/1234:addOperations')
|
|
578
|
+
.reply(status, mockResponse, mockResponseHeader);
|
|
579
|
+
const result = await googleAds.addConversionsToOfflineUserDataJob(jobId, requestData);
|
|
580
|
+
expect(result).toEqual(expectedResult);
|
|
581
|
+
});
|
|
582
|
+
});
|
|
583
|
+
});
|
|
584
|
+
describe('Test runOfflineUserDataJob function', () => {
|
|
585
|
+
const TestCases = [
|
|
586
|
+
{
|
|
587
|
+
name: 'should successfully run an offline user data job with valid jobId',
|
|
588
|
+
authObject: {
|
|
589
|
+
accessToken: 'valid-access-token',
|
|
590
|
+
customerId: '123456789',
|
|
591
|
+
developerToken: 'test-developer-token',
|
|
592
|
+
loginCustomerId: '987654321',
|
|
593
|
+
},
|
|
594
|
+
jobId: '1234',
|
|
595
|
+
requestData: {},
|
|
596
|
+
status: 200,
|
|
597
|
+
mockResponse: {
|
|
598
|
+
success: true,
|
|
599
|
+
data: { name: 'test-job', done: true },
|
|
600
|
+
},
|
|
601
|
+
expectedResult: {
|
|
602
|
+
headers: new axios_1.AxiosHeaders({ 'Content-Type': 'application/json' }),
|
|
603
|
+
responseBody: {
|
|
604
|
+
success: true,
|
|
605
|
+
data: { name: 'test-job', done: true },
|
|
606
|
+
},
|
|
607
|
+
statusCode: 200,
|
|
608
|
+
type: 'success',
|
|
609
|
+
},
|
|
610
|
+
mockResponseHeader: new axios_1.AxiosHeaders({ 'Content-Type': 'application/json' }),
|
|
611
|
+
},
|
|
612
|
+
];
|
|
613
|
+
TestCases.forEach(({ name, authObject, mockResponse, status, expectedResult, mockResponseHeader, jobId }) => {
|
|
614
|
+
it(name, async () => {
|
|
615
|
+
const googleAds = new googleAds_1.default(authObject);
|
|
616
|
+
mock
|
|
617
|
+
.onPost('https://googleads.googleapis.com/v19/customers/123456789/offlineUserDataJobs/1234:run')
|
|
618
|
+
.reply(status, mockResponse, mockResponseHeader);
|
|
619
|
+
const result = await googleAds.runOfflineUserDataJob(jobId);
|
|
620
|
+
expect(result).toEqual(expectedResult);
|
|
621
|
+
});
|
|
622
|
+
});
|
|
623
|
+
});
|
|
624
|
+
describe('Test uploadClickConversion function', () => {
|
|
625
|
+
const TestCases = [
|
|
626
|
+
{
|
|
627
|
+
name: 'should successfully add user data to an offline user data job',
|
|
628
|
+
authObject: {
|
|
629
|
+
accessToken: 'valid-access-token',
|
|
630
|
+
customerId: '123456789',
|
|
631
|
+
developerToken: 'test-developer-token',
|
|
632
|
+
loginCustomerId: '987654321',
|
|
633
|
+
},
|
|
634
|
+
requestData: {
|
|
635
|
+
conversions: [
|
|
636
|
+
{
|
|
637
|
+
gclid: 'test-gclid',
|
|
638
|
+
conversionAction: 'test-action',
|
|
639
|
+
conversionDateTime: '2023-01-01T00:00:00Z',
|
|
640
|
+
conversionValue: 100,
|
|
641
|
+
currencyCode: 'USD',
|
|
642
|
+
},
|
|
643
|
+
],
|
|
644
|
+
},
|
|
645
|
+
status: 200,
|
|
646
|
+
mockResponse: {
|
|
647
|
+
success: true,
|
|
648
|
+
data: {
|
|
649
|
+
results: [
|
|
650
|
+
{
|
|
651
|
+
gclid: 'test-gclid',
|
|
652
|
+
conversionAction: 'test-action',
|
|
653
|
+
conversionDateTime: '2023-01-01T00:00:00Z',
|
|
654
|
+
},
|
|
655
|
+
],
|
|
656
|
+
},
|
|
657
|
+
},
|
|
658
|
+
expectedResult: {
|
|
659
|
+
headers: new axios_1.AxiosHeaders({ 'Content-Type': 'application/json' }),
|
|
660
|
+
responseBody: {
|
|
661
|
+
data: {
|
|
662
|
+
results: [
|
|
663
|
+
{
|
|
664
|
+
conversionAction: 'test-action',
|
|
665
|
+
conversionDateTime: '2023-01-01T00:00:00Z',
|
|
666
|
+
gclid: 'test-gclid',
|
|
667
|
+
},
|
|
668
|
+
],
|
|
669
|
+
},
|
|
670
|
+
success: true,
|
|
671
|
+
},
|
|
672
|
+
statusCode: 200,
|
|
673
|
+
type: 'success',
|
|
674
|
+
},
|
|
675
|
+
mockResponseHeader: new axios_1.AxiosHeaders({ 'Content-Type': 'application/json' }),
|
|
676
|
+
},
|
|
677
|
+
];
|
|
678
|
+
TestCases.forEach(({ name, authObject, mockResponse, status, expectedResult, mockResponseHeader, requestData, }) => {
|
|
679
|
+
it(name, async () => {
|
|
680
|
+
const googleAds = new googleAds_1.default(authObject);
|
|
681
|
+
mock
|
|
682
|
+
.onPost('https://googleads.googleapis.com/v19/customers/123456789:uploadClickConversions')
|
|
683
|
+
.reply(status, mockResponse, mockResponseHeader);
|
|
684
|
+
const result = await googleAds.uploadClickConversion(requestData);
|
|
685
|
+
expect(result).toEqual(expectedResult);
|
|
686
|
+
});
|
|
687
|
+
});
|
|
688
|
+
});
|
|
689
|
+
describe('Test uploadCallConversion function', () => {
|
|
690
|
+
const TestCases = [
|
|
691
|
+
{
|
|
692
|
+
name: 'should successfully upload call conversion data and return a successful response',
|
|
693
|
+
authObject: {
|
|
694
|
+
accessToken: 'valid-access-token',
|
|
695
|
+
customerId: '123456789',
|
|
696
|
+
developerToken: 'test-developer-token',
|
|
697
|
+
loginCustomerId: '987654321',
|
|
698
|
+
},
|
|
699
|
+
requestData: {
|
|
700
|
+
conversions: [
|
|
701
|
+
{
|
|
702
|
+
callerId: '1234567890',
|
|
703
|
+
callStartDateTime: '2023-01-01T12:00:00Z',
|
|
704
|
+
conversionAction: 'action1',
|
|
705
|
+
conversionDateTime: '2023-01-01T12:05:00Z',
|
|
706
|
+
},
|
|
707
|
+
],
|
|
708
|
+
partialFailure: false,
|
|
709
|
+
},
|
|
710
|
+
status: 200,
|
|
711
|
+
mockResponse: {
|
|
712
|
+
success: true,
|
|
713
|
+
data: {
|
|
714
|
+
results: [
|
|
715
|
+
{
|
|
716
|
+
callerId: '1234567890',
|
|
717
|
+
callStartDateTime: '2023-01-01T12:00:00Z',
|
|
718
|
+
conversionAction: 'action1',
|
|
719
|
+
conversionDateTime: '2023-01-01T12:05:00Z',
|
|
720
|
+
},
|
|
721
|
+
],
|
|
722
|
+
},
|
|
723
|
+
},
|
|
724
|
+
expectedResult: {
|
|
725
|
+
headers: new axios_1.AxiosHeaders({ 'Content-Type': 'application/json' }),
|
|
726
|
+
responseBody: {
|
|
727
|
+
data: {
|
|
728
|
+
results: [
|
|
729
|
+
{
|
|
730
|
+
callStartDateTime: '2023-01-01T12:00:00Z',
|
|
731
|
+
callerId: '1234567890',
|
|
732
|
+
conversionAction: 'action1',
|
|
733
|
+
conversionDateTime: '2023-01-01T12:05:00Z',
|
|
734
|
+
},
|
|
735
|
+
],
|
|
736
|
+
},
|
|
737
|
+
success: true,
|
|
738
|
+
},
|
|
739
|
+
statusCode: 200,
|
|
740
|
+
type: 'success',
|
|
741
|
+
},
|
|
742
|
+
mockResponseHeader: new axios_1.AxiosHeaders({ 'Content-Type': 'application/json' }),
|
|
743
|
+
},
|
|
744
|
+
];
|
|
745
|
+
TestCases.forEach(({ name, authObject, mockResponse, status, expectedResult, mockResponseHeader, requestData, }) => {
|
|
746
|
+
it(name, async () => {
|
|
747
|
+
const googleAds = new googleAds_1.default(authObject);
|
|
748
|
+
mock
|
|
749
|
+
.onPost('https://googleads.googleapis.com/v19/customers/123456789:uploadClickConversions')
|
|
750
|
+
.reply(status, mockResponse, mockResponseHeader);
|
|
751
|
+
const result = await googleAds.uploadClickConversion(requestData);
|
|
752
|
+
expect(result).toEqual(expectedResult);
|
|
753
|
+
});
|
|
754
|
+
});
|
|
755
|
+
});
|
|
756
|
+
});
|
|
757
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"googleAds.test.js","sourceRoot":"","sources":["../../../src/sdks/googleAdsRestAPI/googleAds.test.ts"],"names":[],"mappings":"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","sourcesContent":["import AxiosMockAdapter from 'axios-mock-adapter';\nimport axios, { AxiosHeaders } from 'axios';\nimport GoogleAds from './googleAds';\n\ndescribe('Google ads SDK', () => {\n  let mock: AxiosMockAdapter;\n\n  beforeEach(() => {\n    mock = new AxiosMockAdapter(axios, { onNoMatch: 'throwException' });\n  });\n\n  afterEach(() => {\n    mock.restore();\n  });\n  describe('Test getConversionAction function', () => {\n    const testCases = [\n      {\n        name: 'should return conversion action ID when API returns success response',\n        authObject: {\n          accessToken: 'test-access-token',\n          customerId: '123456789',\n          developerToken: 'test-developer-token',\n        },\n        mockResponse: [\n          {\n            results: [\n              {\n                conversionAction: {\n                  resourceName: 'customers/123456789/conversionActions/9876543',\n                  id: '9876543',\n                },\n              },\n            ],\n          },\n        ],\n        conversionName: 'test_conversion',\n        status: 200,\n        expectedResult: 'customers/123456789/conversionActions/9876543',\n        expectedRequest: {\n          url: '/123456789/googleAds:searchStream',\n          data: JSON.stringify({\n            query:\n              \"SELECT conversion_action.id FROM conversion_action WHERE conversion_action.name = 'test_conversion'\",\n          }),\n        },\n      },\n      {\n        name: 'handle failure scenario when accessToken is wrong',\n        authObject: {\n          customerId: '123456789',\n          developerToken: 'test-developer-token',\n          accessToken: 'wrong-access-token',\n        },\n        mockResponse: [\n          {\n            error: {\n              code: 401,\n              message:\n                'Request had invalid authentication credentials. Expected OAuth 2 access token, login cookie or other valid authentication credential. See https://developers.google.com/identity/sign-in/web/devconsole-project.',\n              status: 'UNAUTHENTICATED',\n            },\n          },\n        ],\n        mockResponseHeader: new AxiosHeaders({\n          'transfer-encoding': 'chunked',\n        }),\n        conversionName: 'test_conversion',\n        status: 401,\n        expectedResult: {\n          headers: new AxiosHeaders({\n            'transfer-encoding': 'chunked',\n          }),\n          message: undefined,\n          responseBody: [\n            {\n              error: {\n                code: 401,\n                message:\n                  'Request had invalid authentication credentials. Expected OAuth 2 access token, login cookie or other valid authentication credential. See https://developers.google.com/identity/sign-in/web/devconsole-project.',\n                status: 'UNAUTHENTICATED',\n              },\n            },\n          ],\n          statusCode: 401,\n          type: 'application-error',\n        },\n        expectedRequest: {\n          url: '/123456789/googleAds:searchStream',\n          data: JSON.stringify({\n            query:\n              \"SELECT conversion_action.id FROM conversion_action WHERE conversion_action.name = 'test_conversion'\",\n          }),\n        },\n      },\n      {\n        name: 'handle the situation when the conversion is not available',\n        authObject: {\n          accessToken: 'test-access-token',\n          customerId: '123456789',\n          developerToken: 'test-developer-token',\n        },\n        mockResponse: [],\n        conversionName: 'non-existing-conversion',\n        status: 200,\n        expectedResult: null,\n        expectedRequest: {\n          url: '/123456789/googleAds:searchStream',\n          data: JSON.stringify({\n            query:\n              \"SELECT conversion_action.id FROM conversion_action WHERE conversion_action.name = 'non-existing-conversion'\",\n          }),\n        },\n      },\n    ];\n    testCases.forEach(\n      ({\n        name,\n        authObject,\n        mockResponse,\n        conversionName,\n        expectedRequest,\n        expectedResult,\n        status,\n        mockResponseHeader,\n      }) => {\n        it(name, async () => {\n          const googleAds = new GoogleAds(authObject);\n          mock\n            .onPost(\n              'https://googleads.googleapis.com/v19/customers/123456789/googleAds:searchStream',\n            )\n            .reply(status, mockResponse, mockResponseHeader);\n          const result = await googleAds.getConversionActionId(conversionName);\n          expect(mock.history.post[0]).toMatchObject(expectedRequest);\n          expect(result).toEqual(expectedResult);\n        });\n      },\n    );\n  });\n\n  describe('Test getCustomVariable function', () => {\n    const testCases = [\n      {\n        name: 'should return an array of CustomVariableResults when API call succeeds',\n        authObject: {\n          accessToken: 'test-access-token',\n          customerId: '123456789',\n          developerToken: 'test-developer-token',\n        },\n        mockResponse: [\n          {\n            results: [\n              {\n                conversionCustomVariable: {\n                  resourceName: 'customers/123/conversionCustomVariables/456',\n                  name: 'test_variable',\n                },\n              },\n            ],\n            fieldMask: 'conversion_custom_variable.name',\n            requestId: '123456',\n            queryResourceConsumption: '100',\n          },\n        ],\n        status: 200,\n        expectedResult: [\n          {\n            conversionCustomVariable: {\n              resourceName: 'customers/123/conversionCustomVariables/456',\n              name: 'test_variable',\n            },\n          },\n        ],\n        expectedRequest: {\n          url: '/123456789/googleAds:searchStream',\n          data: JSON.stringify({\n            query: 'SELECT conversion_custom_variable.name FROM conversion_custom_variable',\n          }),\n        },\n      },\n      {\n        name: 'should handle the scenario when there is not any customVariables',\n        authObject: {\n          accessToken: 'test-access-token',\n          customerId: '123456789',\n          developerToken: 'test-developer-token',\n        },\n        status: 200,\n        mockResponse: [\n          {\n            results: [],\n            fieldMask: 'conversion_custom_variable.name',\n            requestId: '123456',\n            queryResourceConsumption: '100',\n          },\n        ],\n        expectedResult: [],\n        expectedRequest: {\n          url: '/123456789/googleAds:searchStream',\n          data: JSON.stringify({\n            query: 'SELECT conversion_custom_variable.name FROM conversion_custom_variable',\n          }),\n        },\n      },\n      {\n        name: 'should return an array of CustomVariableResults when API call succeeds',\n        authObject: {\n          accessToken: 'wrong-access-token',\n          customerId: '123456789',\n          developerToken: 'test-developer-token',\n        },\n        mockResponse: [\n          {\n            error: {\n              code: 401,\n              message:\n                'Request had invalid authentication credentials. Expected OAuth 2 access token, login cookie or other valid authentication credential. See https://developers.google.com/identity/sign-in/web/devconsole-project.',\n              status: 'UNAUTHENTICATED',\n            },\n          },\n        ],\n        status: 401,\n        expectedResult: {\n          headers: new AxiosHeaders({\n            'transfer-encoding': 'chunked',\n          }),\n          message: undefined,\n          responseBody: [\n            {\n              error: {\n                code: 401,\n                message:\n                  'Request had invalid authentication credentials. Expected OAuth 2 access token, login cookie or other valid authentication credential. See https://developers.google.com/identity/sign-in/web/devconsole-project.',\n                status: 'UNAUTHENTICATED',\n              },\n            },\n          ],\n          statusCode: 401,\n          type: 'application-error',\n        },\n        expectedRequest: {\n          url: '/123456789/googleAds:searchStream',\n          data: JSON.stringify({\n            query: 'SELECT conversion_custom_variable.name FROM conversion_custom_variable',\n          }),\n        },\n        mockResponseHeader: new AxiosHeaders({\n          'transfer-encoding': 'chunked',\n        }),\n      },\n    ];\n\n    testCases.forEach(\n      ({\n        name,\n        authObject,\n        mockResponse,\n        expectedRequest,\n        expectedResult,\n        status,\n        mockResponseHeader,\n      }) => {\n        it(name, async () => {\n          const googleAds = new GoogleAds(authObject);\n          mock\n            .onPost(\n              'https://googleads.googleapis.com/v19/customers/123456789/googleAds:searchStream',\n            )\n            .reply(status, mockResponse, mockResponseHeader);\n          const result = await googleAds.getCustomVariable();\n          expect(mock.history.post[0]).toMatchObject(expectedRequest);\n          expect(result).toEqual(expectedResult);\n        });\n      },\n    );\n  });\n\n  describe('Test addConversionAdjustMent function', () => {\n    const TestCases = [\n      {\n        name: 'should successfully upload conversion adjustments with valid data',\n        authObject: {\n          accessToken: 'valid-access-token',\n          customerId: '123456789',\n          developerToken: 'test-developer-token',\n        },\n        requestData: {\n          conversionAdjustments: [\n            {\n              gclidDateTimePair: {\n                gclid: 'test-gclid',\n                conversionDateTime: '2023-01-01T00:00:00Z',\n              },\n              adjustmentType: 'RESTATEMENT' as const,\n              restatementValue: {\n                adjustedValue: 100,\n                currencyCode: 'USD',\n              },\n              userIdentifiers: [\n                {\n                  userIdentifierSource: 'FIRST_PARTY' as const,\n                  hashedEmail: '389205904d6c7bb83fc676513911226f2be25bf1465616bb9b29587100ab1414',\n                },\n              ],\n            },\n          ],\n          partialFailure: false,\n        },\n        status: 200,\n        mockResponse: {\n          conversionAdjustments: [\n            {\n              gclidDateTimePair: {\n                gclid: 'test-gclid',\n                conversionDateTime: '2023-01-01T00:00:00Z',\n              },\n              adjustmentType: 'RESTATEMENT',\n              restatementValue: {\n                adjustedValue: 100,\n                currencyCode: 'USD',\n              },\n              userIdentifiers: [],\n            },\n          ],\n          partialFailure: false,\n        },\n        expectedResult: {\n          type: 'success',\n          statusCode: 200,\n          responseBody: {\n            conversionAdjustments: [\n              {\n                gclidDateTimePair: {\n                  gclid: 'test-gclid',\n                  conversionDateTime: '2023-01-01T00:00:00Z',\n                },\n                adjustmentType: 'RESTATEMENT',\n                restatementValue: { adjustedValue: 100, currencyCode: 'USD' },\n                userIdentifiers: [],\n              },\n            ],\n            partialFailure: false,\n          },\n          headers: new AxiosHeaders({ 'Content-Type': 'application/json' }),\n        },\n        mockResponseHeader: new AxiosHeaders({ 'Content-Type': 'application/json' }),\n      },\n      {\n        name: 'should successfully handle failure scenario',\n        authObject: {\n          accessToken: 'wrong-access-token',\n          customerId: '123456789',\n          developerToken: 'test-developer-token',\n        },\n        requestData: {\n          conversionAdjustments: [\n            {\n              gclidDateTimePair: {\n                gclid: 'test-gclid',\n                conversionDateTime: '2023-01-01T00:00:00Z',\n              },\n              adjustmentType: 'RESTATEMENT' as const,\n              restatementValue: {\n                adjustedValue: 100,\n                currencyCode: 'USD',\n              },\n              userIdentifiers: [\n                {\n                  userIdentifierSource: 'FIRST_PARTY' as const,\n                  hashedEmail: '389205904d6c7bb83fc676513911226f2be25bf1465616bb9b29587100ab1414',\n                },\n              ],\n            },\n          ],\n          partialFailure: false,\n        },\n        status: 401,\n        mockResponse: {\n          headers: new AxiosHeaders({\n            'transfer-encoding': 'chunked',\n          }),\n          message: undefined,\n          responseBody: [\n            {\n              error: {\n                code: 401,\n                message:\n                  'Request had invalid authentication credentials. Expected OAuth 2 access token, login cookie or other valid authentication credential. See https://developers.google.com/identity/sign-in/web/devconsole-project.',\n                status: 'UNAUTHENTICATED',\n              },\n            },\n          ],\n          statusCode: 401,\n          type: 'application-error',\n        },\n        expectedResult: {\n          headers: new AxiosHeaders({\n            'Content-Type': 'application/json',\n          }),\n          message: undefined,\n          responseBody: {\n            headers: { 'transfer-encoding': 'chunked' },\n            responseBody: [\n              {\n                error: {\n                  code: 401,\n                  message:\n                    'Request had invalid authentication credentials. Expected OAuth 2 access token, login cookie or other valid authentication credential. See https://developers.google.com/identity/sign-in/web/devconsole-project.',\n                  status: 'UNAUTHENTICATED',\n                },\n              },\n            ],\n            statusCode: 401,\n            type: 'application-error',\n          },\n          statusCode: 401,\n          type: 'application-error',\n        },\n        mockResponseHeader: new AxiosHeaders({ 'Content-Type': 'application/json' }),\n      },\n    ];\n    TestCases.forEach(\n      ({\n        name,\n        authObject,\n        mockResponse,\n        requestData,\n        expectedResult,\n        mockResponseHeader,\n        status,\n      }) => {\n        it(name, async () => {\n          const googleAds = new GoogleAds(authObject);\n          mock\n            .onPost(\n              'https://googleads.googleapis.com/v19/customers/123456789:uploadConversionAdjustments',\n            )\n            .reply(status, mockResponse, mockResponseHeader);\n          const result = await googleAds.addConversionAdjustMent(requestData);\n          expect(result).toEqual(expectedResult);\n        });\n      },\n    );\n  });\n\n  describe('Test createOfflineUserDataJob function', () => {\n    const TestCases = [\n      {\n        name: 'should return job ID when response type is success for customerMatchUserList job',\n        authObject: {\n          accessToken: 'valid-access-token',\n          customerId: '123456789',\n          developerToken: 'test-developer-token',\n          loginCustomerId: '987654321',\n        },\n        requestData: {\n          job: {\n            type: 'CUSTOMER_MATCH_USER_LIST' as const,\n            customerMatchUserListMetadata: {\n              userList: 'customers/123456789/userLists/1234',\n            },\n          },\n        },\n        status: 200,\n        mockResponse: {\n          resourceName: 'customers/123456789/offlineUserDataJobs/5678',\n        },\n        expectedResult: '5678',\n        mockResponseHeader: new AxiosHeaders({ 'Content-Type': 'application/json' }),\n      },\n      {\n        name: 'should return job ID when response type is success for store sales job',\n        authObject: {\n          accessToken: 'valid-access-token',\n          customerId: '123456789',\n          developerToken: 'test-developer-token',\n        },\n        requestData: {\n          job: {\n            storeSalesMetadata: {\n              custom_key: 'CUSTOM_KEY',\n              loyaltyFraction: 1,\n              transaction_upload_fraction: '1',\n            },\n            type: 'STORE_SALES_UPLOAD_FIRST_PARTY' as const,\n          },\n        },\n        status: 200,\n        mockResponse: {\n          resourceName: 'customers/123456789/offlineUserDataJobs/5678',\n        },\n        expectedResult: '5678',\n        mockResponseHeader: new AxiosHeaders({ 'Content-Type': 'application/json' }),\n      },\n    ];\n    TestCases.forEach(\n      ({\n        name,\n        authObject,\n        mockResponse,\n        status,\n        requestData,\n        expectedResult,\n        mockResponseHeader,\n      }) => {\n        it(name, async () => {\n          const googleAds = new GoogleAds(authObject);\n          mock\n            .onPost(\n              'https://googleads.googleapis.com/v19/customers/123456789/offlineUserDataJobs:create',\n            )\n            .reply(status, mockResponse, mockResponseHeader);\n          const result = await googleAds.createOfflineUserDataJob(requestData);\n          expect(result).toEqual(expectedResult);\n        });\n      },\n    );\n  });\n\n  describe('Test addUserToOfflineUserDataJob function', () => {\n    const TestCases = [\n      {\n        name: 'should successfully add user data to an offline user data job',\n        authObject: {\n          accessToken: 'valid-access-token',\n          customerId: '123456789',\n          developerToken: 'test-developer-token',\n          loginCustomerId: '987654321',\n        },\n        jobId: '1234',\n        requestData: {\n          operations: [\n            {\n              create: {\n                userIdentifiers: [{ hashedEmail: 'hashed_email_value' }],\n              },\n            },\n          ],\n          enablePartialFailure: false,\n        },\n        status: 200,\n        mockResponse: {},\n        expectedResult: {\n          headers: new AxiosHeaders({ 'Content-Type': 'application/json' }),\n          responseBody: {},\n          statusCode: 200,\n          type: 'success',\n        },\n        mockResponseHeader: new AxiosHeaders({ 'Content-Type': 'application/json' }),\n      },\n    ];\n    TestCases.forEach(\n      ({\n        name,\n        authObject,\n        mockResponse,\n        status,\n        requestData,\n        expectedResult,\n        mockResponseHeader,\n        jobId,\n      }) => {\n        it(name, async () => {\n          const googleAds = new GoogleAds(authObject);\n          mock\n            .onPost(\n              'https://googleads.googleapis.com/v19/customers/123456789/offlineUserDataJobs/1234:addOperations',\n            )\n            .reply(status, mockResponse, mockResponseHeader);\n          const result = await googleAds.addUserToOfflineUserDataJob(jobId, requestData);\n          expect(result).toEqual(expectedResult);\n        });\n      },\n    );\n  });\n\n  describe('Test addConversionsToOfflineUserDataJob function', () => {\n    const TestCases = [\n      {\n        name: 'should successfully add store sales conversion data to an offline user data job',\n        authObject: {\n          accessToken: 'valid-access-token',\n          customerId: '123456789',\n          developerToken: 'test-developer-token',\n          loginCustomerId: '987654321',\n        },\n        jobId: '1234',\n        requestData: {\n          operations: [\n            {\n              create: {\n                userIdentifiers: [{ hashedEmail: 'hashed-email-value' }],\n                transactionInfo: {\n                  conversionAction: 'customers/123456789/conversionActions/987',\n                  conversionDateTime: '2023-01-01 12:00:00',\n                  conversionValue: 10.99,\n                  currencyCode: 'USD',\n                },\n              },\n            },\n          ],\n          enablePartialFailure: true,\n        },\n        status: 200,\n        mockResponse: {},\n        expectedResult: {\n          headers: new AxiosHeaders({ 'Content-Type': 'application/json' }),\n          responseBody: {},\n          statusCode: 200,\n          type: 'success',\n        },\n        mockResponseHeader: new AxiosHeaders({ 'Content-Type': 'application/json' }),\n      },\n    ];\n    TestCases.forEach(\n      ({\n        name,\n        authObject,\n        mockResponse,\n        status,\n        requestData,\n        expectedResult,\n        mockResponseHeader,\n        jobId,\n      }) => {\n        it(name, async () => {\n          const googleAds = new GoogleAds(authObject);\n          mock\n            .onPost(\n              'https://googleads.googleapis.com/v19/customers/123456789/offlineUserDataJobs/1234:addOperations',\n            )\n            .reply(status, mockResponse, mockResponseHeader);\n          const result = await googleAds.addConversionsToOfflineUserDataJob(jobId, requestData);\n          expect(result).toEqual(expectedResult);\n        });\n      },\n    );\n  });\n\n  describe('Test runOfflineUserDataJob function', () => {\n    const TestCases = [\n      {\n        name: 'should successfully run an offline user data job with valid jobId',\n        authObject: {\n          accessToken: 'valid-access-token',\n          customerId: '123456789',\n          developerToken: 'test-developer-token',\n          loginCustomerId: '987654321',\n        },\n        jobId: '1234',\n        requestData: {},\n        status: 200,\n        mockResponse: {\n          success: true,\n          data: { name: 'test-job', done: true },\n        },\n        expectedResult: {\n          headers: new AxiosHeaders({ 'Content-Type': 'application/json' }),\n          responseBody: {\n            success: true,\n            data: { name: 'test-job', done: true },\n          },\n          statusCode: 200,\n          type: 'success',\n        },\n        mockResponseHeader: new AxiosHeaders({ 'Content-Type': 'application/json' }),\n      },\n    ];\n    TestCases.forEach(\n      ({ name, authObject, mockResponse, status, expectedResult, mockResponseHeader, jobId }) => {\n        it(name, async () => {\n          const googleAds = new GoogleAds(authObject);\n          mock\n            .onPost(\n              'https://googleads.googleapis.com/v19/customers/123456789/offlineUserDataJobs/1234:run',\n            )\n            .reply(status, mockResponse, mockResponseHeader);\n          const result = await googleAds.runOfflineUserDataJob(jobId);\n          expect(result).toEqual(expectedResult);\n        });\n      },\n    );\n  });\n\n  describe('Test uploadClickConversion function', () => {\n    const TestCases = [\n      {\n        name: 'should successfully add user data to an offline user data job',\n        authObject: {\n          accessToken: 'valid-access-token',\n          customerId: '123456789',\n          developerToken: 'test-developer-token',\n          loginCustomerId: '987654321',\n        },\n        requestData: {\n          conversions: [\n            {\n              gclid: 'test-gclid',\n              conversionAction: 'test-action',\n              conversionDateTime: '2023-01-01T00:00:00Z',\n              conversionValue: 100,\n              currencyCode: 'USD',\n            },\n          ],\n        },\n        status: 200,\n        mockResponse: {\n          success: true,\n          data: {\n            results: [\n              {\n                gclid: 'test-gclid',\n                conversionAction: 'test-action',\n                conversionDateTime: '2023-01-01T00:00:00Z',\n              },\n            ],\n          },\n        },\n        expectedResult: {\n          headers: new AxiosHeaders({ 'Content-Type': 'application/json' }),\n          responseBody: {\n            data: {\n              results: [\n                {\n                  conversionAction: 'test-action',\n                  conversionDateTime: '2023-01-01T00:00:00Z',\n                  gclid: 'test-gclid',\n                },\n              ],\n            },\n            success: true,\n          },\n          statusCode: 200,\n          type: 'success',\n        },\n        mockResponseHeader: new AxiosHeaders({ 'Content-Type': 'application/json' }),\n      },\n    ];\n    TestCases.forEach(\n      ({\n        name,\n        authObject,\n        mockResponse,\n        status,\n        expectedResult,\n        mockResponseHeader,\n        requestData,\n      }) => {\n        it(name, async () => {\n          const googleAds = new GoogleAds(authObject);\n          mock\n            .onPost(\n              'https://googleads.googleapis.com/v19/customers/123456789:uploadClickConversions',\n            )\n            .reply(status, mockResponse, mockResponseHeader);\n          const result = await googleAds.uploadClickConversion(requestData);\n          expect(result).toEqual(expectedResult);\n        });\n      },\n    );\n  });\n\n  describe('Test uploadCallConversion function', () => {\n    const TestCases = [\n      {\n        name: 'should successfully upload call conversion data and return a successful response',\n        authObject: {\n          accessToken: 'valid-access-token',\n          customerId: '123456789',\n          developerToken: 'test-developer-token',\n          loginCustomerId: '987654321',\n        },\n        requestData: {\n          conversions: [\n            {\n              callerId: '1234567890',\n              callStartDateTime: '2023-01-01T12:00:00Z',\n              conversionAction: 'action1',\n              conversionDateTime: '2023-01-01T12:05:00Z',\n            },\n          ],\n          partialFailure: false,\n        },\n        status: 200,\n        mockResponse: {\n          success: true,\n          data: {\n            results: [\n              {\n                callerId: '1234567890',\n                callStartDateTime: '2023-01-01T12:00:00Z',\n                conversionAction: 'action1',\n                conversionDateTime: '2023-01-01T12:05:00Z',\n              },\n            ],\n          },\n        },\n        expectedResult: {\n          headers: new AxiosHeaders({ 'Content-Type': 'application/json' }),\n          responseBody: {\n            data: {\n              results: [\n                {\n                  callStartDateTime: '2023-01-01T12:00:00Z',\n                  callerId: '1234567890',\n                  conversionAction: 'action1',\n                  conversionDateTime: '2023-01-01T12:05:00Z',\n                },\n              ],\n            },\n            success: true,\n          },\n          statusCode: 200,\n          type: 'success',\n        },\n        mockResponseHeader: new AxiosHeaders({ 'Content-Type': 'application/json' }),\n      },\n    ];\n    TestCases.forEach(\n      ({\n        name,\n        authObject,\n        mockResponse,\n        status,\n        expectedResult,\n        mockResponseHeader,\n        requestData,\n      }) => {\n        it(name, async () => {\n          const googleAds = new GoogleAds(authObject);\n          mock\n            .onPost(\n              'https://googleads.googleapis.com/v19/customers/123456789:uploadClickConversions',\n            )\n            .reply(status, mockResponse, mockResponseHeader);\n          const result = await googleAds.uploadClickConversion(requestData);\n          expect(result).toEqual(expectedResult);\n        });\n      },\n    );\n  });\n});\n"]}
|