-
Notifications
You must be signed in to change notification settings - Fork 8
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #65 from psrebniak/60-salesforce-data-transfer-api…
…-for-source feat: #60 add SalesforceDataTransferApi option to source
- Loading branch information
Showing
8 changed files
with
732 additions
and
16 deletions.
There are no files selected for viewing
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,13 @@ | ||
/** | ||
* The default. Amazon AppFlow selects which API to use based on the number of records that your flow transfers to Salesforce. If your flow transfers fewer than 1,000 records, Amazon AppFlow uses Salesforce REST API. If your flow transfers 1,000 records or more, Amazon AppFlow uses Salesforce Bulk API 2.0. | ||
* | ||
* Each of these Salesforce APIs structures data differently. If Amazon AppFlow selects the API automatically, be aware that, for recurring flows, the data output might vary from one flow run to the next. For example, if a flow runs daily, it might use REST API on one day to transfer 900 records, and it might use Bulk API 2.0 on the next day to transfer 1,100 records. For each of these flow runs, the respective Salesforce API formats the data differently. Some of the differences include how dates are formatted and null values are represented. Also, Bulk API 2.0 doesn't transfer Salesforce compound fields. | ||
* | ||
* By choosing this option, you optimize flow performance for both small and large data transfers, but the tradeoff is inconsistent formatting in the output. | ||
*/ | ||
export enum SalesforceDataTransferApi { | ||
|
||
AUTOMATIC = 'AUTOMATIC', | ||
BULKV2 = 'BULKV2', | ||
REST_SYNC = 'REST_SYNC' | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,243 @@ | ||
/* | ||
Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved. | ||
SPDX-License-Identifier: Apache-2.0 | ||
*/ | ||
import { Stack } from 'aws-cdk-lib'; | ||
import { Template } from 'aws-cdk-lib/assertions'; | ||
import { Bucket } from 'aws-cdk-lib/aws-s3'; | ||
|
||
import { Secret } from 'aws-cdk-lib/aws-secretsmanager'; | ||
import { | ||
Mapping, | ||
OnDemandFlow, | ||
S3InputFileType, | ||
S3Source, | ||
SalesforceConnectorProfile, | ||
SalesforceConnectorType, | ||
SalesforceDataTransferApi, | ||
SalesforceDestination, | ||
WriteOperation, | ||
} from '../../src'; | ||
|
||
describe('SalesforceDestination', () => { | ||
|
||
test('Destination with only connector name', () => { | ||
const stack = new Stack(undefined, 'TestStack'); | ||
const destination = new SalesforceDestination({ | ||
profile: SalesforceConnectorProfile.fromConnectionProfileName(stack, 'TestProfile', 'dummy-profile'), | ||
object: 'Account', | ||
operation: WriteOperation.insert(), | ||
}); | ||
|
||
const expectedConnectorType = SalesforceConnectorType.instance; | ||
expect(destination.connectorType.asProfileConnectorLabel).toEqual(expectedConnectorType.asProfileConnectorLabel); | ||
expect(destination.connectorType.asProfileConnectorType).toEqual(expectedConnectorType.asProfileConnectorType); | ||
expect(destination.connectorType.asTaskConnectorOperatorOrigin).toEqual(expectedConnectorType.asTaskConnectorOperatorOrigin); | ||
expect(destination.connectorType.isCustom).toEqual(expectedConnectorType.isCustom); | ||
}); | ||
|
||
test('Destination in a Flow is in the stack', () => { | ||
const stack = new Stack(undefined, 'TestStack'); | ||
|
||
const s3Bucket = new Bucket(stack, 'TestBucket', {}); | ||
const source = new S3Source({ | ||
bucket: s3Bucket, | ||
prefix: '', | ||
format: { | ||
type: S3InputFileType.JSON, | ||
}, | ||
}); | ||
|
||
const destination = new SalesforceDestination({ | ||
profile: SalesforceConnectorProfile.fromConnectionProfileName(stack, 'TestProfile', 'dummy-profile'), | ||
dataTransferApi: SalesforceDataTransferApi.REST_SYNC, | ||
object: 'Account', | ||
operation: WriteOperation.insert(), | ||
}); | ||
|
||
new OnDemandFlow(stack, 'TestFlow', { | ||
source: source, | ||
destination: destination, | ||
mappings: [Mapping.mapAll()], | ||
}); | ||
|
||
Template.fromStack(stack).hasResourceProperties('AWS::AppFlow::Flow', { | ||
DestinationFlowConfigList: [ | ||
{ | ||
ConnectorProfileName: 'dummy-profile', | ||
ConnectorType: 'Salesforce', | ||
DestinationConnectorProperties: { | ||
Salesforce: { | ||
DataTransferApi: 'REST_SYNC', | ||
Object: 'Account', | ||
WriteOperationType: 'INSERT', | ||
}, | ||
}, | ||
}, | ||
], | ||
FlowName: 'TestFlow', | ||
SourceFlowConfig: { | ||
ConnectorType: 'S3', | ||
SourceConnectorProperties: { | ||
S3: { | ||
BucketName: { | ||
Ref: 'TestBucket560B80BC', | ||
}, | ||
BucketPrefix: '', | ||
S3InputFormatConfig: { | ||
S3InputFileType: 'JSON', | ||
}, | ||
}, | ||
}, | ||
}, | ||
Tasks: [ | ||
{ | ||
ConnectorOperator: { | ||
S3: 'NO_OP', | ||
}, | ||
SourceFields: [], | ||
TaskProperties: [ | ||
{ | ||
Key: 'EXCLUDE_SOURCE_FIELDS_LIST', | ||
Value: '[]', | ||
}, | ||
], | ||
TaskType: 'Map_all', | ||
}, | ||
], | ||
TriggerConfig: { | ||
TriggerType: 'OnDemand', | ||
}, | ||
}); | ||
}); | ||
|
||
test('Destination for dummy-profile in a Flow is in the stack', () => { | ||
const stack = new Stack(undefined, 'TestStack'); | ||
|
||
const secret = Secret.fromSecretNameV2(stack, 'TestSecret', 'appflow/salesforce/client'); | ||
const profile = new SalesforceConnectorProfile(stack, 'TestProfile', { | ||
oAuth: { | ||
accessToken: 'accessToken', | ||
flow: { | ||
refreshTokenGrant: { | ||
refreshToken: 'refreshToken', | ||
client: secret, | ||
}, | ||
}, | ||
}, | ||
instanceUrl: 'https://instance-id.develop.my.salesforce.com', | ||
}); | ||
|
||
|
||
const s3Bucket = new Bucket(stack, 'TestBucket', {}); | ||
const source = new S3Source({ | ||
bucket: s3Bucket, | ||
prefix: '', | ||
format: { | ||
type: S3InputFileType.JSON, | ||
}, | ||
}); | ||
|
||
const destination = new SalesforceDestination({ | ||
profile: profile, | ||
dataTransferApi: SalesforceDataTransferApi.REST_SYNC, | ||
object: 'Account', | ||
operation: WriteOperation.insert(), | ||
}); | ||
|
||
new OnDemandFlow(stack, 'TestFlow', { | ||
source: source, | ||
destination: destination, | ||
mappings: [Mapping.mapAll()], | ||
}); | ||
|
||
const template = Template.fromStack(stack); | ||
template.hasResourceProperties('AWS::AppFlow::ConnectorProfile', { | ||
ConnectionMode: 'Public', | ||
ConnectorProfileConfig: { | ||
ConnectorProfileCredentials: { | ||
Salesforce: { | ||
AccessToken: 'accessToken', | ||
ClientCredentialsArn: { | ||
'Fn::Join': [ | ||
'', | ||
[ | ||
'arn:', | ||
{ | ||
Ref: 'AWS::Partition', | ||
}, | ||
':secretsmanager:', | ||
{ | ||
Ref: 'AWS::Region', | ||
}, | ||
':', | ||
{ | ||
Ref: 'AWS::AccountId', | ||
}, | ||
':secret:appflow/salesforce/client', | ||
], | ||
], | ||
}, | ||
RefreshToken: 'refreshToken', | ||
}, | ||
}, | ||
ConnectorProfileProperties: { | ||
Salesforce: { | ||
InstanceUrl: 'https://instance-id.develop.my.salesforce.com', | ||
}, | ||
}, | ||
}, | ||
ConnectorProfileName: 'TestProfile', | ||
ConnectorType: 'Salesforce', | ||
}); | ||
|
||
template.hasResourceProperties('AWS::AppFlow::Flow', { | ||
DestinationFlowConfigList: [ | ||
{ | ||
ConnectorProfileName: 'TestProfile', | ||
ConnectorType: 'Salesforce', | ||
DestinationConnectorProperties: { | ||
Salesforce: { | ||
DataTransferApi: 'REST_SYNC', | ||
Object: 'Account', | ||
WriteOperationType: 'INSERT', | ||
}, | ||
}, | ||
}, | ||
], | ||
FlowName: 'TestFlow', | ||
SourceFlowConfig: { | ||
ConnectorType: 'S3', | ||
SourceConnectorProperties: { | ||
S3: { | ||
BucketName: { | ||
Ref: 'TestBucket560B80BC', | ||
}, | ||
BucketPrefix: '', | ||
S3InputFormatConfig: { | ||
S3InputFileType: 'JSON', | ||
}, | ||
}, | ||
}, | ||
}, | ||
Tasks: [ | ||
{ | ||
ConnectorOperator: { | ||
S3: 'NO_OP', | ||
}, | ||
SourceFields: [], | ||
TaskProperties: [ | ||
{ | ||
Key: 'EXCLUDE_SOURCE_FIELDS_LIST', | ||
Value: '[]', | ||
}, | ||
], | ||
TaskType: 'Map_all', | ||
}, | ||
], | ||
TriggerConfig: { | ||
TriggerType: 'OnDemand', | ||
}, | ||
}); | ||
}); | ||
}); |
Oops, something went wrong.