Merge pull request #681 from automatisch/salesforce/updated-field-in-object
feat(salesforce): add updated field in objects trigger
This commit is contained in:
7
packages/backend/src/apps/salesforce/data/index.ts
Normal file
7
packages/backend/src/apps/salesforce/data/index.ts
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
import listObjects from './list-objects';
|
||||||
|
import listFields from './list-fields';
|
||||||
|
|
||||||
|
export default [
|
||||||
|
listObjects,
|
||||||
|
listFields,
|
||||||
|
];
|
@@ -0,0 +1,34 @@
|
|||||||
|
import { IGlobalVariable } from '@automatisch/types';
|
||||||
|
|
||||||
|
type TResponse = {
|
||||||
|
fields: TField[];
|
||||||
|
};
|
||||||
|
|
||||||
|
type TField = {
|
||||||
|
name: string;
|
||||||
|
label: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: 'List fields',
|
||||||
|
key: 'listFields',
|
||||||
|
|
||||||
|
async run($: IGlobalVariable) {
|
||||||
|
const { object } = $.step.parameters;
|
||||||
|
|
||||||
|
if (!object) return { data: [] };
|
||||||
|
|
||||||
|
const response = await $.http.get<TResponse>(
|
||||||
|
`/services/data/v56.0/sobjects/${object}/describe`
|
||||||
|
);
|
||||||
|
|
||||||
|
const fields = response.data.fields.map((field) => {
|
||||||
|
return {
|
||||||
|
value: field.name,
|
||||||
|
name: field.label,
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
return { data: fields };
|
||||||
|
},
|
||||||
|
};
|
@@ -0,0 +1,28 @@
|
|||||||
|
import { IGlobalVariable } from '@automatisch/types';
|
||||||
|
|
||||||
|
type TResponse = {
|
||||||
|
sobjects: TObject[];
|
||||||
|
}
|
||||||
|
|
||||||
|
type TObject = {
|
||||||
|
name: string;
|
||||||
|
label: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: 'List objects',
|
||||||
|
key: 'listObjects',
|
||||||
|
|
||||||
|
async run($: IGlobalVariable) {
|
||||||
|
const response = await $.http.get<TResponse>('/services/data/v56.0/sobjects');
|
||||||
|
|
||||||
|
const objects = response.data.sobjects.map((object) => {
|
||||||
|
return {
|
||||||
|
value: object.name,
|
||||||
|
name: object.label,
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
return { data: objects };
|
||||||
|
},
|
||||||
|
};
|
@@ -1,6 +1,8 @@
|
|||||||
import defineApp from '../../helpers/define-app';
|
import defineApp from '../../helpers/define-app';
|
||||||
import addAuthHeader from './common/add-auth-header';
|
import addAuthHeader from './common/add-auth-header';
|
||||||
import auth from './auth';
|
import auth from './auth';
|
||||||
|
import triggers from './triggers';
|
||||||
|
import data from './data';
|
||||||
|
|
||||||
export default defineApp({
|
export default defineApp({
|
||||||
name: 'Salesforce',
|
name: 'Salesforce',
|
||||||
@@ -13,4 +15,6 @@ export default defineApp({
|
|||||||
primaryColor: '00A1E0',
|
primaryColor: '00A1E0',
|
||||||
beforeRequest: [addAuthHeader],
|
beforeRequest: [addAuthHeader],
|
||||||
auth,
|
auth,
|
||||||
|
triggers,
|
||||||
|
data,
|
||||||
});
|
});
|
||||||
|
5
packages/backend/src/apps/salesforce/triggers/index.ts
Normal file
5
packages/backend/src/apps/salesforce/triggers/index.ts
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
import updatedFieldInObjects from "./updated-field-in-objects";
|
||||||
|
|
||||||
|
export default [
|
||||||
|
updatedFieldInObjects
|
||||||
|
];
|
@@ -0,0 +1,55 @@
|
|||||||
|
import defineTrigger from '../../../../helpers/define-trigger';
|
||||||
|
import updatedFieldInObjects from './updated-field-in-objects';
|
||||||
|
|
||||||
|
export default defineTrigger({
|
||||||
|
name: 'Updated field in objects',
|
||||||
|
key: 'updatedFieldInObjects',
|
||||||
|
pollInterval: 15,
|
||||||
|
description: 'Triggers when a field is updated in an object.',
|
||||||
|
arguments: [
|
||||||
|
{
|
||||||
|
label: 'Object',
|
||||||
|
key: 'object',
|
||||||
|
type: 'dropdown' as const,
|
||||||
|
required: true,
|
||||||
|
variables: false,
|
||||||
|
source: {
|
||||||
|
type: 'query',
|
||||||
|
name: 'getData',
|
||||||
|
arguments: [
|
||||||
|
{
|
||||||
|
name: 'key',
|
||||||
|
value: 'listObjects',
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Field',
|
||||||
|
key: 'field',
|
||||||
|
type: 'dropdown' as const,
|
||||||
|
description: 'Track updates by this field',
|
||||||
|
required: true,
|
||||||
|
variables: false,
|
||||||
|
dependsOn: ['parameters.object'],
|
||||||
|
source: {
|
||||||
|
type: 'query',
|
||||||
|
name: 'getData',
|
||||||
|
arguments: [
|
||||||
|
{
|
||||||
|
name: 'key',
|
||||||
|
value: 'listFields',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'parameters.object',
|
||||||
|
value: '{parameters.object}',
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
|
||||||
|
async run($) {
|
||||||
|
await updatedFieldInObjects($);
|
||||||
|
},
|
||||||
|
});
|
@@ -0,0 +1,45 @@
|
|||||||
|
import { IGlobalVariable } from '@automatisch/types';
|
||||||
|
|
||||||
|
function getQuery(object: string, limit: number, offset: number) {
|
||||||
|
return `
|
||||||
|
SELECT
|
||||||
|
FIELDS(ALL)
|
||||||
|
FROM
|
||||||
|
${object}
|
||||||
|
ORDER BY LastModifiedDate DESC
|
||||||
|
LIMIT ${limit}
|
||||||
|
OFFSET ${offset}
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
const updatedFieldInObjects = async ($: IGlobalVariable): Promise<void> => {
|
||||||
|
const limit = 200;
|
||||||
|
const field = $.step.parameters.field as string;
|
||||||
|
const object = $.step.parameters.object as string;
|
||||||
|
|
||||||
|
let response;
|
||||||
|
let offset = 0;
|
||||||
|
do {
|
||||||
|
const options = {
|
||||||
|
params: {
|
||||||
|
q: getQuery(object, limit, offset),
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
response = await $.http.get('/services/data/v56.0/query', options);
|
||||||
|
const records = response.data.records;
|
||||||
|
|
||||||
|
for (const record of records) {
|
||||||
|
$.pushTriggerItem({
|
||||||
|
raw: record,
|
||||||
|
meta: {
|
||||||
|
internalId: `${record.Id}-${record[field]}`,
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
offset = offset + limit;
|
||||||
|
} while (response.data.records?.length === limit);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default updatedFieldInObjects;
|
@@ -93,7 +93,7 @@ function TestSubstep(props: TestSubstepProps): React.ReactElement {
|
|||||||
/>
|
/>
|
||||||
<Collapse in={expanded} timeout="auto" unmountOnExit>
|
<Collapse in={expanded} timeout="auto" unmountOnExit>
|
||||||
<ListItem sx={{ pt: 2, pb: 3, flexDirection: 'column', alignItems: 'flex-start' }}>
|
<ListItem sx={{ pt: 2, pb: 3, flexDirection: 'column', alignItems: 'flex-start' }}>
|
||||||
{error?.graphQLErrors?.length && <Alert severity="error" sx={{ mb: 1, fontWeight: 500, width: '100%' }}>
|
{!!error?.graphQLErrors?.length && <Alert severity="error" sx={{ mb: 1, fontWeight: 500, width: '100%' }}>
|
||||||
{serializeErrors(error.graphQLErrors).map((error: any) => (<div>{error.message}</div>))}
|
{serializeErrors(error.graphQLErrors).map((error: any) => (<div>{error.message}</div>))}
|
||||||
</Alert>}
|
</Alert>}
|
||||||
|
|
||||||
|
@@ -22,6 +22,10 @@ export const GET_APPS = gql`
|
|||||||
description
|
description
|
||||||
docUrl
|
docUrl
|
||||||
clickToCopy
|
clickToCopy
|
||||||
|
options {
|
||||||
|
label
|
||||||
|
value
|
||||||
|
}
|
||||||
}
|
}
|
||||||
authenticationSteps {
|
authenticationSteps {
|
||||||
type
|
type
|
||||||
|
Reference in New Issue
Block a user