Ship every SaaS integration your users need
Add dozens of integrations to your app, quickly and reliably, with Paragon's embedded iPaaS for developers.
What your users see
How your developers build
Paragraph
Workflow Builder
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
define(
integration: IGoogledriveIntegration,
context: IContext<InputResultMap>,
connectUser: IConnectUser<IPersona<typeof personaMeta>>,
) {
const triggerStep = new IntegrationEnabledStep();
const getAllFilesStep = integration.actions.googleDriveSearchFolders(
{
parentId: `${context.getInput(sharedInputs.folder_to_share)}`,
},
{
autoRetry: true,
description: 'Get all files',
},
);
const fanOutFilesStep = new FanOutStep({
description: 'Fan out files',
iterator: getAllFilesStep.output.result,
});
const integrationRequestStep = new IntegrationRequestStep({
description: 'Get file permissions',
method: 'GET',
url: `/files/${fanOutFilesStep.output.instance.id}/permissions`,
params: { ['']: '' },
headers: {},
});
const mapStep = new FanOutStep({
description: 'Fan out permissions',
iterator: integrationRequestStep.output.response.body.permissions,
});
const ifelseStep = new ConditionalStep({
if: Operators.StringContains(mapStep.output.instance.type, 'domain'),
description: 'Is the permission a group/domain?',
});
const integrationRequestStep1 = new IntegrationRequestStep({
autoRetry: false,
continueWorkflowOnError: false,
description: 'List users in domain',
method: 'GET',
url: `admin/directory/v1/users`,
params: { ['']: '' },
headers: {},
});
const requestStep = new RequestStep({
autoRetry: false,
continueWorkflowOnError: false,
description: 'Send each file and associated permissions to my API',
url: `https://yourapp.com/files`,
method: 'GET',
params: { ['']: '' },
headers: {},
});
triggerStep
.nextStep(getAllFilesStep)
.nextStep(
fanOutFilesStep.branch(
integrationRequestStep
.nextStep(
mapStep.branch(ifelseStep.whenTrue(integrationRequestStep1)),
)
.nextStep(requestStep),
),
);
/**
* Pass all steps used in the workflow to the `.register()`
* function. The keys used in this function must remain stable.
*/
return this.register({
triggerStep,
getAllFilesStep,
fanOutFilesStep,
integrationRequestStep,
mapStep,
ifelseStep,
integrationRequestStep1,
requestStep,
});
}
Paragraph
Workflow Builder
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
define(
integration: IGoogledriveIntegration,
context: IContext<InputResultMap>,
connectUser: IConnectUser<IPersona<typeof personaMeta>>,
) {
const triggerStep = new IntegrationEnabledStep();
const getAllFilesStep = integration.actions.googleDriveSearchFolders(
{
parentId: `${context.getInput(sharedInputs.folder_to_share)}`,
},
{
autoRetry: true,
description: 'Get all files',
},
);
const fanOutFilesStep = new FanOutStep({
description: 'Fan out files',
iterator: getAllFilesStep.output.result,
});
const integrationRequestStep = new IntegrationRequestStep({
description: 'Get file permissions',
method: 'GET',
url: `/files/${fanOutFilesStep.output.instance.id}/permissions`,
params: { ['']: '' },
headers: {},
});
const mapStep = new FanOutStep({
description: 'Fan out permissions',
iterator: integrationRequestStep.output.response.body.permissions,
});
const ifelseStep = new ConditionalStep({
if: Operators.StringContains(mapStep.output.instance.type, 'domain'),
description: 'Is the permission a group/domain?',
});
const integrationRequestStep1 = new IntegrationRequestStep({
autoRetry: false,
continueWorkflowOnError: false,
description: 'List users in domain',
method: 'GET',
url: `admin/directory/v1/users`,
params: { ['']: '' },
headers: {},
});
const requestStep = new RequestStep({
autoRetry: false,
continueWorkflowOnError: false,
description: 'Send each file and associated permissions to my API',
url: `https://yourapp.com/files`,
method: 'GET',
params: { ['']: '' },
headers: {},
});
triggerStep
.nextStep(getAllFilesStep)
.nextStep(
fanOutFilesStep.branch(
integrationRequestStep
.nextStep(
mapStep.branch(ifelseStep.whenTrue(integrationRequestStep1)),
)
.nextStep(requestStep),
),
);
/**
* Pass all steps used in the workflow to the `.register()`
* function. The keys used in this function must remain stable.
*/
return this.register({
triggerStep,
getAllFilesStep,
fanOutFilesStep,
integrationRequestStep,
mapStep,
ifelseStep,
integrationRequestStep1,
requestStep,
});
}
Paragraph
Workflow Builder
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
define(
integration: IGoogledriveIntegration,
context: IContext<InputResultMap>,
connectUser: IConnectUser<IPersona<typeof personaMeta>>,
) {
const triggerStep = new IntegrationEnabledStep();
const getAllFilesStep = integration.actions.googleDriveSearchFolders(
{
parentId: `${context.getInput(sharedInputs.folder_to_share)}`,
},
{
autoRetry: true,
description: 'Get all files',
},
);
const fanOutFilesStep = new FanOutStep({
description: 'Fan out files',
iterator: getAllFilesStep.output.result,
});
const integrationRequestStep = new IntegrationRequestStep({
description: 'Get file permissions',
method: 'GET',
url: `/files/${fanOutFilesStep.output.instance.id}/permissions`,
params: { ['']: '' },
headers: {},
});
const mapStep = new FanOutStep({
description: 'Fan out permissions',
iterator: integrationRequestStep.output.response.body.permissions,
});
const ifelseStep = new ConditionalStep({
if: Operators.StringContains(mapStep.output.instance.type, 'domain'),
description: 'Is the permission a group/domain?',
});
const integrationRequestStep1 = new IntegrationRequestStep({
autoRetry: false,
continueWorkflowOnError: false,
description: 'List users in domain',
method: 'GET',
url: `admin/directory/v1/users`,
params: { ['']: '' },
headers: {},
});
const requestStep = new RequestStep({
autoRetry: false,
continueWorkflowOnError: false,
description: 'Send each file and associated permissions to my API',
url: `https://yourapp.com/files`,
method: 'GET',
params: { ['']: '' },
headers: {},
});
triggerStep
.nextStep(getAllFilesStep)
.nextStep(
fanOutFilesStep.branch(
integrationRequestStep
.nextStep(
mapStep.branch(ifelseStep.whenTrue(integrationRequestStep1)),
)
.nextStep(requestStep),
),
);
/**
* Pass all steps used in the workflow to the `.register()`
* function. The keys used in this function must remain stable.
*/
return this.register({
triggerStep,
getAllFilesStep,
fanOutFilesStep,
integrationRequestStep,
mapStep,
ifelseStep,
integrationRequestStep1,
requestStep,
});
}
Paragraph
Workflow Builder
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
define(
integration: IGoogledriveIntegration,
context: IContext<InputResultMap>,
connectUser: IConnectUser<IPersona<typeof personaMeta>>,
) {
const triggerStep = new IntegrationEnabledStep();
const getAllFilesStep = integration.actions.googleDriveSearchFolders(
{
parentId: `${context.getInput(sharedInputs.folder_to_share)}`,
},
{
autoRetry: true,
description: 'Get all files',
},
);
const fanOutFilesStep = new FanOutStep({
description: 'Fan out files',
iterator: getAllFilesStep.output.result,
});
const integrationRequestStep = new IntegrationRequestStep({
description: 'Get file permissions',
method: 'GET',
url: `/files/${fanOutFilesStep.output.instance.id}/permissions`,
params: { ['']: '' },
headers: {},
});
const mapStep = new FanOutStep({
description: 'Fan out permissions',
iterator: integrationRequestStep.output.response.body.permissions,
});
const ifelseStep = new ConditionalStep({
if: Operators.StringContains(mapStep.output.instance.type, 'domain'),
description: 'Is the permission a group/domain?',
});
const integrationRequestStep1 = new IntegrationRequestStep({
autoRetry: false,
continueWorkflowOnError: false,
description: 'List users in domain',
method: 'GET',
url: `admin/directory/v1/users`,
params: { ['']: '' },
headers: {},
});
const requestStep = new RequestStep({
autoRetry: false,
continueWorkflowOnError: false,
description: 'Send each file and associated permissions to my API',
url: `https://yourapp.com/files`,
method: 'GET',
params: { ['']: '' },
headers: {},
});
triggerStep
.nextStep(getAllFilesStep)
.nextStep(
fanOutFilesStep.branch(
integrationRequestStep
.nextStep(
mapStep.branch(ifelseStep.whenTrue(integrationRequestStep1)),
)
.nextStep(requestStep),
),
);
/**
* Pass all steps used in the workflow to the `.register()`
* function. The keys used in this function must remain stable.
*/
return this.register({
triggerStep,
getAllFilesStep,
fanOutFilesStep,
integrationRequestStep,
mapStep,
ifelseStep,
integrationRequestStep1,
requestStep,
});
}
Trusted by 100+ B2B SaaS engineering teams
Trusted by 100+ fast-growing B2B SaaS companies
Trusted by 100+ B2B SaaS engineering teams
Ship integrations faster
On average, engineering teams ship integrations 7x faster with Paragon.
Managed integration auth
Paragon securely handles auth for every integration and keeps your users' OAuth tokens refreshed.
White-labeled and native
With the SDK, you can surface integrations in your app with our pre-built UI, or build your own headless UI.
Streamlined maintenance
Avoid breaking changes on 100+ APIs and get fine-grained observability with Paragon's mointoring tools.
Author durable, scalable jobs for integration logic
Author durable, scalable jobs for integration logic
Author durable, scalable jobs for integration logic
Build background jobs that can extract data, sync records bi-directionally, or automate activities in your users' accounts, and deploy them to all your users.
Build background jobs that can extract data, sync records bi-directionally, or automate activities in your users' accounts, and deploy them to all your users.
Build background jobs that can extract data, sync records bi-directionally, or automate activities in your users' accounts, and deploy them to all your users.
Enable users to connect in a few clicks
Embed a white-labeled, prebuilt UI for your users to connect their 3rd party accounts, enable the Workflows you've built, and configure any user settings that you surface.
Frictionless connection experience
Fully managed authentication for any integration
Out-of-the-box user settings
paragon.connect(’salesforce’)
Keep your integrations running smoothly
Guarantee reliability to your users with end-to-end monitoring & observability across their integrations. Errors are inevitable when working with 3rd party APIs - Paragon enables you to debug issues faster by easily pin pointing when and why an integration isn’t working, for any user.
Task history
Customer dashboards
Error notifications
Release diff
Explore the platform
Experience Paragon with our collection of self-serve product tours
Ship 100+ native integrations
Integrate your app with our growing list of pre-built connectors - or build your own custom connector with any API.
Paragon for AI applications
See how enterprise AI SaaS companies use Paragon as their ingestion engine for multi-tenant RAG and integration layer for agentic AI workflows.
Built for developers
Easy to pick up, highly extensible, and extremely scalable, Paragon provides all the tools developers need to build integrations to spec.
Git sync
Use Git to version-control and introduce code review to your workflows and integration config.
Git sync
Use Git to version-control and introduce code review to your workflows and integration config.
const functionStep = new FunctionStep({
- code: function yourFunction(parameters, libraries) {
- return "Task created by TaskLab"
+ code: function generateDescription(parameters, libraries) {
+ return "${parameters.description}`\n\n- Created by TaskLab"
},
description: 'Generate Description',
parameters: { description: triggerStep.output.description }
});
+ const createTaskStep = clickup.createTask({
+ listId: context.getInput(sharedInputs.list),
+ name: triggerStep.output.title,
+ description: functionStep.output.result,
+ });
Code review
Paragon monitors API changes for all integrations in our catalog, so you can rely on prebuilt steps in Workflows without worrying about breaking changes.
const functionStep = new FunctionStep({
- code: function yourFunction(parameters, libraries) {
- return "Task created by TaskLab"
+ code: function generateDescription(parameters, libraries) {
+ return "${parameters.description}`\n\n- Created by TaskLab"
},
description: 'Generate Description',
parameters: { description: triggerStep.output.description }
});
+ const createTaskStep = clickup.createTask({
+ listId: context.getInput(sharedInputs.list),
+ name: triggerStep.output.title,
+ description: functionStep.output.result,
+ });
Code review
Paragon monitors API changes for all integrations in our catalog, so you can rely on prebuilt steps in Workflows without worrying about breaking changes.
1,600
Requests per second
2TB
Data per day
Workflow engine
Run syncs and automations at scale on our Workflow Engine. Workflows can retry from errors automatically, replay from the original request payload, and show input and output for every running step.
1,600
Requests per second
2TB
Data per day
Workflow engine
Run syncs and automations at scale on our Workflow Engine. Workflows can retry from errors automatically, replay from the original request payload, and show input and output for every running step.
await paragon.request('slack', '/chat.postMessage', {
method: 'POST',
body: {
channel: 'CXXXXXXX0' // Channel ID,
text: 'This message was sent with Paragon Connect 🤯'
}
});
APIs for everything
All workflow executions, integrations, and user accounts are available over APIs.
Users API
Task History API
Connect API
await paragon.request('slack', '/chat.postMessage', {
method: 'POST',
body: {
channel: 'CXXXXXXX0' // Channel ID,
text: 'This message was sent with Paragon Connect 🤯'
}
});
APIs for everything
All workflow executions, integrations, and user accounts are available over APIs.
Users API
Task History API
Connect API
//authenticate users
await paragon.authenticate();
//show Connect Portal
paragon.connect('netsuite');
//send App Event to trigger workflows
paragon.event('contact_updated');
Powerful SDKs methods
Install the Paragon SDK from npm and embed any integration into your app. Fully compatible with Next.js, React, and Vue apps.
//authenticate users
await paragon.authenticate();
//show Connect Portal
paragon.connect('netsuite');
//send App Event to trigger workflows
paragon.event('contact_updated');
Powerful SDKs methods
Install the Paragon SDK from npm and embed any integration into your app. Fully compatible with Next.js, React, and Vue apps.
Development
Staging
1.2
Production
1.1
Release environments
Paragon monitors API changes for all integrations in our catalog, so you can rely on prebuilt steps in Workflows without worrying about breaking changes.
Development
Staging
1.2