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.

Sales & Marketing

Tasks & Ticketing

Files & Knowledge

Accounting

Popular

Sales & Marketing

Tasks & Ticketing

Files & Knowledge

Accounting

Popular

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

Production