diff --git a/bundle/default-bundle/pom.xml b/bundle/default-bundle/pom.xml
index 2732e047b7..43c871371b 100644
--- a/bundle/default-bundle/pom.xml
+++ b/bundle/default-bundle/pom.xml
@@ -33,6 +33,10 @@
io.camunda.connector
connector-aws-lambda
+
+ io.camunda.connector
+ connector-aws-bedrock
+
io.camunda.connector
connector-google-drive
diff --git a/bundle/pom.xml b/bundle/pom.xml
index 563d961086..af08c6cd8a 100644
--- a/bundle/pom.xml
+++ b/bundle/pom.xml
@@ -59,6 +59,11 @@
connector-sendgrid
${project.version}
+
+ io.camunda.connector
+ connector-aws-bedrock
+ ${project.version}
+
io.camunda.connector
connector-aws-sns
diff --git a/connectors/aws/aws-base/pom.xml b/connectors/aws/aws-base/pom.xml
index 5eaa3f4f4d..91dd145e1d 100644
--- a/connectors/aws/aws-base/pom.xml
+++ b/connectors/aws/aws-base/pom.xml
@@ -18,7 +18,9 @@
Camunda Self-Managed Free Edition license
- https://camunda.com/legal/terms/cloud-terms-and-conditions/camunda-cloud-self-managed-free-edition-terms/
+
+ https://camunda.com/legal/terms/cloud-terms-and-conditions/camunda-cloud-self-managed-free-edition-terms/
+
Camunda Self-Managed Enterprise Edition license
@@ -31,7 +33,13 @@
aws-java-sdk-core
${version.aws-java-sdk}
-
+
+
+ software.amazon.awssdk
+ auth
+ ${version.software-aws-java-sdk}
+
+
com.amazonaws
aws-java-sdk-sagemakerruntime
diff --git a/connectors/aws/aws-base/src/main/java/io/camunda/connector/aws/CredentialsProviderSupportV2.java b/connectors/aws/aws-base/src/main/java/io/camunda/connector/aws/CredentialsProviderSupportV2.java
new file mode 100644
index 0000000000..4a8e79fa8f
--- /dev/null
+++ b/connectors/aws/aws-base/src/main/java/io/camunda/connector/aws/CredentialsProviderSupportV2.java
@@ -0,0 +1,29 @@
+/*
+ * Copyright Camunda Services GmbH and/or licensed to Camunda Services GmbH
+ * under one or more contributor license agreements. Licensed under a proprietary license.
+ * See the License.txt file for more information. You may not use this file
+ * except in compliance with the proprietary license.
+ */
+package io.camunda.connector.aws;
+
+import io.camunda.connector.aws.model.impl.AwsAuthentication;
+import io.camunda.connector.aws.model.impl.AwsBaseRequest;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+import software.amazon.awssdk.auth.credentials.*;
+
+public class CredentialsProviderSupportV2 {
+
+ private static final Logger LOGGER = LoggerFactory.getLogger(CredentialsProviderSupportV2.class);
+
+ public static AwsCredentialsProvider credentialsProvider(AwsBaseRequest request) {
+ AwsAuthentication authentication = request.getAuthentication();
+ if (authentication instanceof AwsAuthentication.AwsStaticCredentialsAuthentication sca) {
+ LOGGER.debug("Using StaticCredentialsProvider for AWS authentication (using aws sdk v2)");
+ return StaticCredentialsProvider.create(
+ AwsBasicCredentials.create(sca.accessKey(), sca.secretKey()));
+ }
+ LOGGER.debug("Falling to DefaultCredentialsProvider for AWS authentication (using aws sdk v2)");
+ return DefaultCredentialsProvider.create();
+ }
+}
diff --git a/connectors/aws/aws-bedrock/LICENSE.txt b/connectors/aws/aws-bedrock/LICENSE.txt
new file mode 100644
index 0000000000..85fdd16e79
--- /dev/null
+++ b/connectors/aws/aws-bedrock/LICENSE.txt
@@ -0,0 +1,5 @@
+Copyright Camunda Services GmbH and/or licensed to Camunda Services GmbH under one or more contributor license agreements and licensed to you under a proprietary license.
+You may not use this file except in compliance with the proprietary license.
+The proprietary license can be either the Camunda Self-Managed Free Edition license (available on Camunda’s website) or the Camunda Self-Managed Enterprise Edition license (a copy you obtain when you contact Camunda).
+The Camunda Self-Managed Free Edition comes for free but only allows for usage of the software (file) in non-production environments.
+If you want to use the software (file) in production, you need to purchase the Camunda Self-Managed Enterprise Edition.
\ No newline at end of file
diff --git a/connectors/aws/aws-bedrock/element-templates/aws-bedrock-outbound-connector.json b/connectors/aws/aws-bedrock/element-templates/aws-bedrock-outbound-connector.json
new file mode 100644
index 0000000000..26c2ba8411
--- /dev/null
+++ b/connectors/aws/aws-bedrock/element-templates/aws-bedrock-outbound-connector.json
@@ -0,0 +1,349 @@
+{
+ "$schema" : "https://unpkg.com/@camunda/zeebe-element-templates-json-schema/resources/schema.json",
+ "name" : "AWS BedRock Outbound Connector",
+ "id" : "io.camunda.connectors.aws.bedrock.v1",
+ "description" : "Execute bedrock requests",
+ "documentationRef" : "https://docs.camunda.io/docs/",
+ "version" : 1,
+ "category" : {
+ "id" : "connectors",
+ "name" : "Connectors"
+ },
+ "appliesTo" : [ "bpmn:Task" ],
+ "elementType" : {
+ "value" : "bpmn:ServiceTask"
+ },
+ "groups" : [ {
+ "id" : "authentication",
+ "label" : "Authentication"
+ }, {
+ "id" : "configuration",
+ "label" : "Configuration"
+ }, {
+ "id" : "action",
+ "label" : "Action"
+ }, {
+ "id" : "invokeModel",
+ "label" : "Invoke Model"
+ }, {
+ "id" : "converse",
+ "label" : "Converse"
+ }, {
+ "id" : "output",
+ "label" : "Output mapping"
+ }, {
+ "id" : "error",
+ "label" : "Error handling"
+ }, {
+ "id" : "retries",
+ "label" : "Retries"
+ } ],
+ "properties" : [ {
+ "value" : "io.camunda:aws-bedrock:1",
+ "binding" : {
+ "property" : "type",
+ "type" : "zeebe:taskDefinition"
+ },
+ "type" : "Hidden"
+ }, {
+ "id" : "authentication.type",
+ "label" : "Authentication",
+ "description" : "Specify AWS authentication strategy. Learn more at the documentation page",
+ "value" : "credentials",
+ "group" : "authentication",
+ "binding" : {
+ "name" : "authentication.type",
+ "type" : "zeebe:input"
+ },
+ "type" : "Dropdown",
+ "choices" : [ {
+ "name" : "Default Credentials Chain (Hybrid/Self-Managed only)",
+ "value" : "defaultCredentialsChain"
+ }, {
+ "name" : "Credentials",
+ "value" : "credentials"
+ } ]
+ }, {
+ "id" : "authentication.accessKey",
+ "label" : "Access key",
+ "description" : "Provide an IAM access key tailored to a user, equipped with the necessary permissions",
+ "optional" : false,
+ "constraints" : {
+ "notEmpty" : true
+ },
+ "feel" : "optional",
+ "group" : "authentication",
+ "binding" : {
+ "name" : "authentication.accessKey",
+ "type" : "zeebe:input"
+ },
+ "condition" : {
+ "property" : "authentication.type",
+ "equals" : "credentials",
+ "type" : "simple"
+ },
+ "type" : "String"
+ }, {
+ "id" : "authentication.secretKey",
+ "label" : "Secret key",
+ "description" : "Provide a secret key of a user with permissions to invoke specified AWS Lambda function",
+ "optional" : false,
+ "constraints" : {
+ "notEmpty" : true
+ },
+ "feel" : "optional",
+ "group" : "authentication",
+ "binding" : {
+ "name" : "authentication.secretKey",
+ "type" : "zeebe:input"
+ },
+ "condition" : {
+ "property" : "authentication.type",
+ "equals" : "credentials",
+ "type" : "simple"
+ },
+ "type" : "String"
+ }, {
+ "id" : "configuration.region",
+ "label" : "Region",
+ "description" : "Specify the AWS region",
+ "optional" : false,
+ "constraints" : {
+ "notEmpty" : true
+ },
+ "feel" : "optional",
+ "group" : "configuration",
+ "binding" : {
+ "name" : "configuration.region",
+ "type" : "zeebe:input"
+ },
+ "type" : "String"
+ }, {
+ "id" : "configuration.endpoint",
+ "label" : "Endpoint",
+ "description" : "Specify endpoint if need to use custom endpoint",
+ "optional" : true,
+ "group" : "configuration",
+ "binding" : {
+ "name" : "configuration.endpoint",
+ "type" : "zeebe:input"
+ },
+ "type" : "Hidden"
+ }, {
+ "id" : "action",
+ "label" : "Action",
+ "value" : "invokeModel",
+ "group" : "action",
+ "binding" : {
+ "name" : "action",
+ "type" : "zeebe:input"
+ },
+ "type" : "Dropdown",
+ "choices" : [ {
+ "name" : "Invoke Model",
+ "value" : "invokeModel"
+ }, {
+ "name" : "Converse",
+ "value" : "converse"
+ } ]
+ }, {
+ "id" : "data.modelId0",
+ "label" : "Model id",
+ "optional" : false,
+ "constraints" : {
+ "notEmpty" : true
+ },
+ "feel" : "optional",
+ "group" : "invokeModel",
+ "binding" : {
+ "name" : "data.modelId",
+ "type" : "zeebe:input"
+ },
+ "condition" : {
+ "property" : "action",
+ "equals" : "invokeModel",
+ "type" : "simple"
+ },
+ "type" : "String"
+ }, {
+ "id" : "data.payload",
+ "label" : "Payload",
+ "optional" : false,
+ "constraints" : {
+ "notEmpty" : true
+ },
+ "feel" : "required",
+ "group" : "invokeModel",
+ "binding" : {
+ "name" : "data.payload",
+ "type" : "zeebe:input"
+ },
+ "condition" : {
+ "property" : "action",
+ "equals" : "invokeModel",
+ "type" : "simple"
+ },
+ "type" : "String"
+ }, {
+ "id" : "data.modelId1",
+ "label" : "Model id",
+ "optional" : false,
+ "constraints" : {
+ "notEmpty" : true
+ },
+ "feel" : "optional",
+ "group" : "converse",
+ "binding" : {
+ "name" : "data.modelId",
+ "type" : "zeebe:input"
+ },
+ "condition" : {
+ "property" : "action",
+ "equals" : "converse",
+ "type" : "simple"
+ },
+ "type" : "String"
+ }, {
+ "id" : "data.nextMessage",
+ "label" : "New Message",
+ "optional" : false,
+ "constraints" : {
+ "notEmpty" : true
+ },
+ "feel" : "optional",
+ "group" : "converse",
+ "binding" : {
+ "name" : "data.nextMessage",
+ "type" : "zeebe:input"
+ },
+ "condition" : {
+ "property" : "action",
+ "equals" : "converse",
+ "type" : "simple"
+ },
+ "type" : "String"
+ }, {
+ "id" : "data.messages",
+ "label" : "Messages History",
+ "optional" : true,
+ "feel" : "optional",
+ "group" : "converse",
+ "binding" : {
+ "name" : "data.messages",
+ "type" : "zeebe:input"
+ },
+ "condition" : {
+ "property" : "action",
+ "equals" : "converse",
+ "type" : "simple"
+ },
+ "type" : "String"
+ }, {
+ "id" : "data.maxTokens",
+ "label" : "Max token returned",
+ "optional" : true,
+ "feel" : "optional",
+ "group" : "converse",
+ "binding" : {
+ "name" : "data.maxTokens",
+ "type" : "zeebe:input"
+ },
+ "condition" : {
+ "property" : "action",
+ "equals" : "converse",
+ "type" : "simple"
+ },
+ "type" : "String"
+ }, {
+ "id" : "data.temperature",
+ "label" : "Temperature",
+ "optional" : true,
+ "feel" : "optional",
+ "group" : "converse",
+ "binding" : {
+ "name" : "data.temperature",
+ "type" : "zeebe:input"
+ },
+ "condition" : {
+ "property" : "action",
+ "equals" : "converse",
+ "type" : "simple"
+ },
+ "type" : "String"
+ }, {
+ "id" : "data.topP",
+ "label" : "top P",
+ "optional" : true,
+ "feel" : "optional",
+ "group" : "converse",
+ "binding" : {
+ "name" : "data.topP",
+ "type" : "zeebe:input"
+ },
+ "condition" : {
+ "property" : "action",
+ "equals" : "converse",
+ "type" : "simple"
+ },
+ "type" : "String"
+ }, {
+ "id" : "resultVariable",
+ "label" : "Result variable",
+ "description" : "Name of variable to store the response in",
+ "group" : "output",
+ "binding" : {
+ "key" : "resultVariable",
+ "type" : "zeebe:taskHeader"
+ },
+ "type" : "String"
+ }, {
+ "id" : "resultExpression",
+ "label" : "Result expression",
+ "description" : "Expression to map the response into process variables",
+ "feel" : "required",
+ "group" : "output",
+ "binding" : {
+ "key" : "resultExpression",
+ "type" : "zeebe:taskHeader"
+ },
+ "type" : "Text"
+ }, {
+ "id" : "errorExpression",
+ "label" : "Error expression",
+ "description" : "Expression to handle errors. Details in the documentation.",
+ "feel" : "required",
+ "group" : "error",
+ "binding" : {
+ "key" : "errorExpression",
+ "type" : "zeebe:taskHeader"
+ },
+ "type" : "Text"
+ }, {
+ "id" : "retryCount",
+ "label" : "Retries",
+ "description" : "Number of retries",
+ "value" : "3",
+ "feel" : "optional",
+ "group" : "retries",
+ "binding" : {
+ "property" : "retries",
+ "type" : "zeebe:taskDefinition"
+ },
+ "type" : "String"
+ }, {
+ "id" : "retryBackoff",
+ "label" : "Retry backoff",
+ "description" : "ISO-8601 duration to wait between retries",
+ "value" : "PT0S",
+ "feel" : "optional",
+ "group" : "retries",
+ "binding" : {
+ "key" : "retryBackoff",
+ "type" : "zeebe:taskHeader"
+ },
+ "type" : "String"
+ } ],
+ "icon" : {
+ "contents" : ""
+ }
+}
\ No newline at end of file
diff --git a/connectors/aws/aws-bedrock/element-templates/hybrid/aws-bedrock-outbound-connector-hybrid.json b/connectors/aws/aws-bedrock/element-templates/hybrid/aws-bedrock-outbound-connector-hybrid.json
new file mode 100644
index 0000000000..e4e504572f
--- /dev/null
+++ b/connectors/aws/aws-bedrock/element-templates/hybrid/aws-bedrock-outbound-connector-hybrid.json
@@ -0,0 +1,354 @@
+{
+ "$schema" : "https://unpkg.com/@camunda/zeebe-element-templates-json-schema/resources/schema.json",
+ "name" : "Hybrid AWS BedRock Outbound Connector",
+ "id" : "io.camunda.connectors.aws.bedrock.v1-hybrid",
+ "description" : "Execute bedrock requests",
+ "documentationRef" : "https://docs.camunda.io/docs/",
+ "version" : 1,
+ "category" : {
+ "id" : "connectors",
+ "name" : "Connectors"
+ },
+ "appliesTo" : [ "bpmn:Task" ],
+ "elementType" : {
+ "value" : "bpmn:ServiceTask"
+ },
+ "groups" : [ {
+ "id" : "taskDefinitionType",
+ "label" : "Task definition type"
+ }, {
+ "id" : "authentication",
+ "label" : "Authentication"
+ }, {
+ "id" : "configuration",
+ "label" : "Configuration"
+ }, {
+ "id" : "action",
+ "label" : "Action"
+ }, {
+ "id" : "invokeModel",
+ "label" : "Invoke Model"
+ }, {
+ "id" : "converse",
+ "label" : "Converse"
+ }, {
+ "id" : "output",
+ "label" : "Output mapping"
+ }, {
+ "id" : "error",
+ "label" : "Error handling"
+ }, {
+ "id" : "retries",
+ "label" : "Retries"
+ } ],
+ "properties" : [ {
+ "id" : "taskDefinitionType",
+ "value" : "io.camunda:aws-bedrock:1",
+ "group" : "taskDefinitionType",
+ "binding" : {
+ "property" : "type",
+ "type" : "zeebe:taskDefinition"
+ },
+ "type" : "String"
+ }, {
+ "id" : "authentication.type",
+ "label" : "Authentication",
+ "description" : "Specify AWS authentication strategy. Learn more at the documentation page",
+ "value" : "credentials",
+ "group" : "authentication",
+ "binding" : {
+ "name" : "authentication.type",
+ "type" : "zeebe:input"
+ },
+ "type" : "Dropdown",
+ "choices" : [ {
+ "name" : "Default Credentials Chain (Hybrid/Self-Managed only)",
+ "value" : "defaultCredentialsChain"
+ }, {
+ "name" : "Credentials",
+ "value" : "credentials"
+ } ]
+ }, {
+ "id" : "authentication.accessKey",
+ "label" : "Access key",
+ "description" : "Provide an IAM access key tailored to a user, equipped with the necessary permissions",
+ "optional" : false,
+ "constraints" : {
+ "notEmpty" : true
+ },
+ "feel" : "optional",
+ "group" : "authentication",
+ "binding" : {
+ "name" : "authentication.accessKey",
+ "type" : "zeebe:input"
+ },
+ "condition" : {
+ "property" : "authentication.type",
+ "equals" : "credentials",
+ "type" : "simple"
+ },
+ "type" : "String"
+ }, {
+ "id" : "authentication.secretKey",
+ "label" : "Secret key",
+ "description" : "Provide a secret key of a user with permissions to invoke specified AWS Lambda function",
+ "optional" : false,
+ "constraints" : {
+ "notEmpty" : true
+ },
+ "feel" : "optional",
+ "group" : "authentication",
+ "binding" : {
+ "name" : "authentication.secretKey",
+ "type" : "zeebe:input"
+ },
+ "condition" : {
+ "property" : "authentication.type",
+ "equals" : "credentials",
+ "type" : "simple"
+ },
+ "type" : "String"
+ }, {
+ "id" : "configuration.region",
+ "label" : "Region",
+ "description" : "Specify the AWS region",
+ "optional" : false,
+ "constraints" : {
+ "notEmpty" : true
+ },
+ "feel" : "optional",
+ "group" : "configuration",
+ "binding" : {
+ "name" : "configuration.region",
+ "type" : "zeebe:input"
+ },
+ "type" : "String"
+ }, {
+ "id" : "configuration.endpoint",
+ "label" : "Endpoint",
+ "description" : "Specify endpoint if need to use custom endpoint",
+ "optional" : true,
+ "group" : "configuration",
+ "binding" : {
+ "name" : "configuration.endpoint",
+ "type" : "zeebe:input"
+ },
+ "type" : "Hidden"
+ }, {
+ "id" : "action",
+ "label" : "Action",
+ "value" : "invokeModel",
+ "group" : "action",
+ "binding" : {
+ "name" : "action",
+ "type" : "zeebe:input"
+ },
+ "type" : "Dropdown",
+ "choices" : [ {
+ "name" : "Invoke Model",
+ "value" : "invokeModel"
+ }, {
+ "name" : "Converse",
+ "value" : "converse"
+ } ]
+ }, {
+ "id" : "data.modelId0",
+ "label" : "Model id",
+ "optional" : false,
+ "constraints" : {
+ "notEmpty" : true
+ },
+ "feel" : "optional",
+ "group" : "invokeModel",
+ "binding" : {
+ "name" : "data.modelId",
+ "type" : "zeebe:input"
+ },
+ "condition" : {
+ "property" : "action",
+ "equals" : "invokeModel",
+ "type" : "simple"
+ },
+ "type" : "String"
+ }, {
+ "id" : "data.payload",
+ "label" : "Payload",
+ "optional" : false,
+ "constraints" : {
+ "notEmpty" : true
+ },
+ "feel" : "required",
+ "group" : "invokeModel",
+ "binding" : {
+ "name" : "data.payload",
+ "type" : "zeebe:input"
+ },
+ "condition" : {
+ "property" : "action",
+ "equals" : "invokeModel",
+ "type" : "simple"
+ },
+ "type" : "String"
+ }, {
+ "id" : "data.modelId1",
+ "label" : "Model id",
+ "optional" : false,
+ "constraints" : {
+ "notEmpty" : true
+ },
+ "feel" : "optional",
+ "group" : "converse",
+ "binding" : {
+ "name" : "data.modelId",
+ "type" : "zeebe:input"
+ },
+ "condition" : {
+ "property" : "action",
+ "equals" : "converse",
+ "type" : "simple"
+ },
+ "type" : "String"
+ }, {
+ "id" : "data.nextMessage",
+ "label" : "New Message",
+ "optional" : false,
+ "constraints" : {
+ "notEmpty" : true
+ },
+ "feel" : "optional",
+ "group" : "converse",
+ "binding" : {
+ "name" : "data.nextMessage",
+ "type" : "zeebe:input"
+ },
+ "condition" : {
+ "property" : "action",
+ "equals" : "converse",
+ "type" : "simple"
+ },
+ "type" : "String"
+ }, {
+ "id" : "data.messages",
+ "label" : "Messages History",
+ "optional" : true,
+ "feel" : "optional",
+ "group" : "converse",
+ "binding" : {
+ "name" : "data.messages",
+ "type" : "zeebe:input"
+ },
+ "condition" : {
+ "property" : "action",
+ "equals" : "converse",
+ "type" : "simple"
+ },
+ "type" : "String"
+ }, {
+ "id" : "data.maxTokens",
+ "label" : "Max token returned",
+ "optional" : true,
+ "feel" : "optional",
+ "group" : "converse",
+ "binding" : {
+ "name" : "data.maxTokens",
+ "type" : "zeebe:input"
+ },
+ "condition" : {
+ "property" : "action",
+ "equals" : "converse",
+ "type" : "simple"
+ },
+ "type" : "String"
+ }, {
+ "id" : "data.temperature",
+ "label" : "Temperature",
+ "optional" : true,
+ "feel" : "optional",
+ "group" : "converse",
+ "binding" : {
+ "name" : "data.temperature",
+ "type" : "zeebe:input"
+ },
+ "condition" : {
+ "property" : "action",
+ "equals" : "converse",
+ "type" : "simple"
+ },
+ "type" : "String"
+ }, {
+ "id" : "data.topP",
+ "label" : "top P",
+ "optional" : true,
+ "feel" : "optional",
+ "group" : "converse",
+ "binding" : {
+ "name" : "data.topP",
+ "type" : "zeebe:input"
+ },
+ "condition" : {
+ "property" : "action",
+ "equals" : "converse",
+ "type" : "simple"
+ },
+ "type" : "String"
+ }, {
+ "id" : "resultVariable",
+ "label" : "Result variable",
+ "description" : "Name of variable to store the response in",
+ "group" : "output",
+ "binding" : {
+ "key" : "resultVariable",
+ "type" : "zeebe:taskHeader"
+ },
+ "type" : "String"
+ }, {
+ "id" : "resultExpression",
+ "label" : "Result expression",
+ "description" : "Expression to map the response into process variables",
+ "feel" : "required",
+ "group" : "output",
+ "binding" : {
+ "key" : "resultExpression",
+ "type" : "zeebe:taskHeader"
+ },
+ "type" : "Text"
+ }, {
+ "id" : "errorExpression",
+ "label" : "Error expression",
+ "description" : "Expression to handle errors. Details in the documentation.",
+ "feel" : "required",
+ "group" : "error",
+ "binding" : {
+ "key" : "errorExpression",
+ "type" : "zeebe:taskHeader"
+ },
+ "type" : "Text"
+ }, {
+ "id" : "retryCount",
+ "label" : "Retries",
+ "description" : "Number of retries",
+ "value" : "3",
+ "feel" : "optional",
+ "group" : "retries",
+ "binding" : {
+ "property" : "retries",
+ "type" : "zeebe:taskDefinition"
+ },
+ "type" : "String"
+ }, {
+ "id" : "retryBackoff",
+ "label" : "Retry backoff",
+ "description" : "ISO-8601 duration to wait between retries",
+ "value" : "PT0S",
+ "feel" : "optional",
+ "group" : "retries",
+ "binding" : {
+ "key" : "retryBackoff",
+ "type" : "zeebe:taskHeader"
+ },
+ "type" : "String"
+ } ],
+ "icon" : {
+ "contents" : ""
+ }
+}
\ No newline at end of file
diff --git a/connectors/aws/aws-bedrock/pom.xml b/connectors/aws/aws-bedrock/pom.xml
new file mode 100644
index 0000000000..60d512493f
--- /dev/null
+++ b/connectors/aws/aws-bedrock/pom.xml
@@ -0,0 +1,70 @@
+
+
+ 4.0.0
+
+ io.camunda.connector
+ connector-aws-parent
+ 8.6.0-SNAPSHOT
+
+
+
+ connector-aws-bedrock
+ Camunda AWS Bedrock Connector
+ connector-aws-bedrock
+ jar
+
+
+
+ Camunda Self-Managed Free Edition license
+ https://camunda.com/legal/terms/cloud-terms-and-conditions/camunda-cloud-self-managed-free-edition-terms/
+
+
+ Camunda Self-Managed Enterprise Edition license
+
+
+
+
+
+ io.camunda.connector
+ connector-aws-base
+ ${project.version}
+
+
+
+ software.amazon.awssdk
+ bedrockruntime
+ ${version.software-aws-java-sdk}
+
+
+
+
+
+
+ io.camunda.connector
+ element-template-generator-maven-plugin
+ ${project.version}
+
+
+
+ io.camunda.connector.aws.bedrock.BedrockConnectorFunction
+
+
+ io.camunda.connectors.aws.bedrock.v1
+ aws-bedrock-outbound-connector.json
+
+
+ true
+
+
+
+ io.camunda.connector:connector-aws-base
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/connectors/aws/aws-bedrock/src/main/java/io/camunda/connector/aws/bedrock/BedrockConnectorFunction.java b/connectors/aws/aws-bedrock/src/main/java/io/camunda/connector/aws/bedrock/BedrockConnectorFunction.java
new file mode 100644
index 0000000000..5991241b51
--- /dev/null
+++ b/connectors/aws/aws-bedrock/src/main/java/io/camunda/connector/aws/bedrock/BedrockConnectorFunction.java
@@ -0,0 +1,44 @@
+/*
+ * Copyright Camunda Services GmbH and/or licensed to Camunda Services GmbH
+ * under one or more contributor license agreements. Licensed under a proprietary license.
+ * See the License.txt file for more information. You may not use this file
+ * except in compliance with the proprietary license.
+ */
+package io.camunda.connector.aws.bedrock;
+
+import io.camunda.connector.api.annotation.OutboundConnector;
+import io.camunda.connector.api.outbound.OutboundConnectorContext;
+import io.camunda.connector.api.outbound.OutboundConnectorFunction;
+import io.camunda.connector.aws.bedrock.core.BedrockExecutor;
+import io.camunda.connector.aws.bedrock.model.BedrockRequest;
+import io.camunda.connector.generator.java.annotation.ElementTemplate;
+
+@OutboundConnector(
+ name = "AWS BedRock",
+ inputVariables = {"authentication", "configuration", "action", "data"},
+ type = "io.camunda:aws-bedrock:1")
+@ElementTemplate(
+ id = "io.camunda.connectors.aws.bedrock.v1",
+ name = "AWS BedRock Outbound Connector",
+ description = "Execute bedrock requests",
+ inputDataClass = BedrockRequest.class,
+ version = 1,
+ propertyGroups = {
+ @ElementTemplate.PropertyGroup(id = "authentication", label = "Authentication"),
+ @ElementTemplate.PropertyGroup(id = "configuration", label = "Configuration"),
+ @ElementTemplate.PropertyGroup(id = "action", label = "Action"),
+ @ElementTemplate.PropertyGroup(id = "invokeModel", label = "Invoke Model"),
+ @ElementTemplate.PropertyGroup(id = "converse", label = "Converse"),
+ },
+ documentationRef = "https://docs.camunda.io/docs/",
+ icon = "icon.svg")
+public class BedrockConnectorFunction implements OutboundConnectorFunction {
+
+ public BedrockConnectorFunction() {}
+
+ @Override
+ public Object execute(OutboundConnectorContext context) {
+ BedrockRequest bedrockRequest = context.bindVariables(BedrockRequest.class);
+ return BedrockExecutor.create(bedrockRequest).execute();
+ }
+}
diff --git a/connectors/aws/aws-bedrock/src/main/java/io/camunda/connector/aws/bedrock/core/BedrockExecutor.java b/connectors/aws/aws-bedrock/src/main/java/io/camunda/connector/aws/bedrock/core/BedrockExecutor.java
new file mode 100644
index 0000000000..ddf4d4f64c
--- /dev/null
+++ b/connectors/aws/aws-bedrock/src/main/java/io/camunda/connector/aws/bedrock/core/BedrockExecutor.java
@@ -0,0 +1,39 @@
+/*
+ * Copyright Camunda Services GmbH and/or licensed to Camunda Services GmbH
+ * under one or more contributor license agreements. Licensed under a proprietary license.
+ * See the License.txt file for more information. You may not use this file
+ * except in compliance with the proprietary license.
+ */
+package io.camunda.connector.aws.bedrock.core;
+
+import io.camunda.connector.aws.CredentialsProviderSupportV2;
+import io.camunda.connector.aws.ObjectMapperSupplier;
+import io.camunda.connector.aws.bedrock.model.BedrockRequest;
+import io.camunda.connector.aws.bedrock.model.BedrockResponse;
+import io.camunda.connector.aws.bedrock.model.RequestData;
+import software.amazon.awssdk.regions.Region;
+import software.amazon.awssdk.services.bedrockruntime.BedrockRuntimeClient;
+
+public class BedrockExecutor {
+
+ private final BedrockRuntimeClient bedrockRuntimeClient;
+ private final RequestData requestData;
+
+ public BedrockExecutor(BedrockRuntimeClient bedrockRuntimeClient, RequestData requestData) {
+ this.bedrockRuntimeClient = bedrockRuntimeClient;
+ this.requestData = requestData;
+ }
+
+ public static BedrockExecutor create(BedrockRequest bedrockRequest) {
+ return new BedrockExecutor(
+ BedrockRuntimeClient.builder()
+ .credentialsProvider(CredentialsProviderSupportV2.credentialsProvider(bedrockRequest))
+ .region(Region.of(bedrockRequest.getConfiguration().region()))
+ .build(),
+ bedrockRequest.getData());
+ }
+
+ public BedrockResponse execute() {
+ return this.requestData.execute(bedrockRuntimeClient, ObjectMapperSupplier.getMapperInstance());
+ }
+}
diff --git a/connectors/aws/aws-bedrock/src/main/java/io/camunda/connector/aws/bedrock/model/BedrockRequest.java b/connectors/aws/aws-bedrock/src/main/java/io/camunda/connector/aws/bedrock/model/BedrockRequest.java
new file mode 100644
index 0000000000..7f7197d4db
--- /dev/null
+++ b/connectors/aws/aws-bedrock/src/main/java/io/camunda/connector/aws/bedrock/model/BedrockRequest.java
@@ -0,0 +1,38 @@
+/*
+ * Copyright Camunda Services GmbH and/or licensed to Camunda Services GmbH
+ * under one or more contributor license agreements. Licensed under a proprietary license.
+ * See the License.txt file for more information. You may not use this file
+ * except in compliance with the proprietary license.
+ */
+package io.camunda.connector.aws.bedrock.model;
+
+import com.fasterxml.jackson.annotation.JsonSubTypes;
+import com.fasterxml.jackson.annotation.JsonTypeInfo;
+import io.camunda.connector.aws.model.impl.AwsBaseRequest;
+import io.camunda.connector.generator.java.annotation.NestedProperties;
+import jakarta.validation.Valid;
+import jakarta.validation.constraints.NotNull;
+
+public class BedrockRequest extends AwsBaseRequest {
+ @JsonTypeInfo(
+ use = JsonTypeInfo.Id.NAME,
+ include = JsonTypeInfo.As.EXTERNAL_PROPERTY,
+ property = "action")
+ @JsonSubTypes(
+ value = {
+ @JsonSubTypes.Type(value = ConverseData.class, name = "converse"),
+ @JsonSubTypes.Type(value = InvokeModelData.class, name = "invokeModel"),
+ })
+ @Valid
+ @NotNull
+ @NestedProperties(addNestedPath = false)
+ private RequestData data;
+
+ public BedrockRequest() {}
+
+ @Valid
+ @NotNull
+ public RequestData getData() {
+ return data;
+ }
+}
diff --git a/connectors/aws/aws-bedrock/src/main/java/io/camunda/connector/aws/bedrock/model/BedrockResponse.java b/connectors/aws/aws-bedrock/src/main/java/io/camunda/connector/aws/bedrock/model/BedrockResponse.java
new file mode 100644
index 0000000000..fdd984a83b
--- /dev/null
+++ b/connectors/aws/aws-bedrock/src/main/java/io/camunda/connector/aws/bedrock/model/BedrockResponse.java
@@ -0,0 +1,9 @@
+/*
+ * Copyright Camunda Services GmbH and/or licensed to Camunda Services GmbH
+ * under one or more contributor license agreements. Licensed under a proprietary license.
+ * See the License.txt file for more information. You may not use this file
+ * except in compliance with the proprietary license.
+ */
+package io.camunda.connector.aws.bedrock.model;
+
+public interface BedrockResponse {}
diff --git a/connectors/aws/aws-bedrock/src/main/java/io/camunda/connector/aws/bedrock/model/ConverseData.java b/connectors/aws/aws-bedrock/src/main/java/io/camunda/connector/aws/bedrock/model/ConverseData.java
new file mode 100644
index 0000000000..f7daeb50f4
--- /dev/null
+++ b/connectors/aws/aws-bedrock/src/main/java/io/camunda/connector/aws/bedrock/model/ConverseData.java
@@ -0,0 +1,176 @@
+/*
+ * Copyright Camunda Services GmbH and/or licensed to Camunda Services GmbH
+ * under one or more contributor license agreements. Licensed under a proprietary license.
+ * See the License.txt file for more information. You may not use this file
+ * except in compliance with the proprietary license.
+ */
+package io.camunda.connector.aws.bedrock.model;
+
+import com.fasterxml.jackson.annotation.JsonSetter;
+import com.fasterxml.jackson.annotation.Nulls;
+import com.fasterxml.jackson.databind.ObjectMapper;
+import io.camunda.connector.generator.dsl.Property;
+import io.camunda.connector.generator.java.annotation.TemplateProperty;
+import io.camunda.connector.generator.java.annotation.TemplateSubType;
+import jakarta.validation.Valid;
+import jakarta.validation.constraints.NotBlank;
+import jakarta.validation.constraints.NotNull;
+import java.util.ArrayList;
+import java.util.List;
+import java.util.Objects;
+import software.amazon.awssdk.services.bedrockruntime.BedrockRuntimeClient;
+import software.amazon.awssdk.services.bedrockruntime.model.ContentBlock;
+import software.amazon.awssdk.services.bedrockruntime.model.ConversationRole;
+import software.amazon.awssdk.services.bedrockruntime.model.ConverseResponse;
+import software.amazon.awssdk.services.bedrockruntime.model.Message;
+
+@TemplateSubType(id = "converse", label = "Converse")
+public final class ConverseData implements RequestData {
+
+ @TemplateProperty(
+ label = "Model id",
+ group = "converse",
+ id = "data.modelId1",
+ feel = Property.FeelMode.optional,
+ binding = @TemplateProperty.PropertyBinding(name = "data.modelId"))
+ @Valid
+ @NotNull
+ private String modelId;
+
+ @TemplateProperty(
+ label = "New Message",
+ group = "converse",
+ id = "data.nextMessage",
+ feel = Property.FeelMode.optional,
+ binding = @TemplateProperty.PropertyBinding(name = "data.nextMessage"))
+ @Valid
+ @NotBlank
+ private String nextMessage;
+
+ @TemplateProperty(
+ label = "Messages History",
+ group = "converse",
+ id = "data.messages",
+ feel = Property.FeelMode.optional,
+ optional = true,
+ binding = @TemplateProperty.PropertyBinding(name = "data.messages"))
+ @Valid
+ @JsonSetter(nulls = Nulls.SKIP)
+ private List messages = new ArrayList<>();
+
+ @TemplateProperty(
+ label = "Max token returned",
+ group = "converse",
+ id = "data.maxTokens",
+ feel = Property.FeelMode.optional,
+ optional = true,
+ binding = @TemplateProperty.PropertyBinding(name = "data.maxTokens"))
+ private Integer maxTokens = 512;
+
+ @TemplateProperty(
+ label = "Temperature",
+ group = "converse",
+ id = "data.temperature",
+ feel = Property.FeelMode.optional,
+ optional = true,
+ binding = @TemplateProperty.PropertyBinding(name = "data.temperature"))
+ private Float temperature = 0.5f;
+
+ @TemplateProperty(
+ label = "top P",
+ group = "converse",
+ id = "data.topP",
+ feel = Property.FeelMode.optional,
+ optional = true,
+ binding = @TemplateProperty.PropertyBinding(name = "data.topP"))
+ private Float topP = 0.9f;
+
+ @Override
+ public BedrockResponse execute(
+ BedrockRuntimeClient bedrockRuntimeClient, ObjectMapper mapperInstance) {
+ this.messages.add(new PreviousMessage(this.nextMessage, ConversationRole.USER.name()));
+ Message.Builder messageBuilder = Message.builder();
+ List messages =
+ this.messages.stream()
+ .map(
+ message ->
+ messageBuilder
+ .role(ConversationRole.valueOf(message.role()))
+ .content(ContentBlock.fromText(message.message()))
+ .build())
+ .toList();
+ ConverseResponse converseResponse =
+ bedrockRuntimeClient.converse(
+ builder ->
+ builder
+ .modelId(this.modelId)
+ .messages(messages)
+ .inferenceConfig(
+ config ->
+ config
+ .temperature(this.temperature)
+ .maxTokens(this.maxTokens)
+ .topP(this.topP)
+ .build()));
+ String newMessage = converseResponse.output().message().content().getFirst().text();
+ this.messages.add(new PreviousMessage(newMessage, ConversationRole.ASSISTANT.name()));
+ return new ConverseWrapperResponse(this.messages, newMessage);
+ }
+
+ @Override
+ public boolean equals(Object o) {
+ if (this == o) return true;
+ if (o == null || getClass() != o.getClass()) return false;
+ ConverseData that = (ConverseData) o;
+ return Objects.equals(modelId, that.modelId)
+ && Objects.equals(nextMessage, that.nextMessage)
+ && Objects.equals(messages, that.messages)
+ && Objects.equals(maxTokens, that.maxTokens)
+ && Objects.equals(temperature, that.temperature)
+ && Objects.equals(topP, that.topP);
+ }
+
+ @Override
+ public int hashCode() {
+ return Objects.hash(modelId, nextMessage, messages, maxTokens, temperature, topP);
+ }
+
+ public void setModelId(@Valid @NotNull String modelId) {
+ this.modelId = modelId;
+ }
+
+ public void setNextMessage(@Valid @NotBlank String nextMessage) {
+ this.nextMessage = nextMessage;
+ }
+
+ public void setMessages(@Valid List messages) {
+ this.messages = messages;
+ }
+
+ public void setMaxTokens(Integer maxTokens) {
+ this.maxTokens = maxTokens;
+ }
+
+ public void setTemperature(Float temperature) {
+ this.temperature = temperature;
+ }
+
+ public void setTopP(Float topP) {
+ this.topP = topP;
+ }
+
+ @Override
+ public String toString() {
+ return "ConverseData{"
+ + "modelId='"
+ + modelId
+ + '\''
+ + ", maxTokens="
+ + maxTokens
+ + ", temperature="
+ + temperature
+ + ", topP="
+ + topP
+ + '}';
+ }
+}
diff --git a/connectors/aws/aws-bedrock/src/main/java/io/camunda/connector/aws/bedrock/model/ConverseWrapperResponse.java b/connectors/aws/aws-bedrock/src/main/java/io/camunda/connector/aws/bedrock/model/ConverseWrapperResponse.java
new file mode 100644
index 0000000000..d3f0659d83
--- /dev/null
+++ b/connectors/aws/aws-bedrock/src/main/java/io/camunda/connector/aws/bedrock/model/ConverseWrapperResponse.java
@@ -0,0 +1,12 @@
+/*
+ * Copyright Camunda Services GmbH and/or licensed to Camunda Services GmbH
+ * under one or more contributor license agreements. Licensed under a proprietary license.
+ * See the License.txt file for more information. You may not use this file
+ * except in compliance with the proprietary license.
+ */
+package io.camunda.connector.aws.bedrock.model;
+
+import java.util.List;
+
+public record ConverseWrapperResponse(List messagesHistory, String newMessage)
+ implements BedrockResponse {}
diff --git a/connectors/aws/aws-bedrock/src/main/java/io/camunda/connector/aws/bedrock/model/InvokeModelData.java b/connectors/aws/aws-bedrock/src/main/java/io/camunda/connector/aws/bedrock/model/InvokeModelData.java
new file mode 100644
index 0000000000..2dcc25d620
--- /dev/null
+++ b/connectors/aws/aws-bedrock/src/main/java/io/camunda/connector/aws/bedrock/model/InvokeModelData.java
@@ -0,0 +1,85 @@
+/*
+ * Copyright Camunda Services GmbH and/or licensed to Camunda Services GmbH
+ * under one or more contributor license agreements. Licensed under a proprietary license.
+ * See the License.txt file for more information. You may not use this file
+ * except in compliance with the proprietary license.
+ */
+package io.camunda.connector.aws.bedrock.model;
+
+import com.fasterxml.jackson.core.JsonProcessingException;
+import com.fasterxml.jackson.databind.ObjectMapper;
+import io.camunda.connector.generator.dsl.Property;
+import io.camunda.connector.generator.java.annotation.TemplateProperty;
+import io.camunda.connector.generator.java.annotation.TemplateSubType;
+import jakarta.validation.Valid;
+import jakarta.validation.constraints.NotNull;
+import java.util.Objects;
+import software.amazon.awssdk.core.SdkBytes;
+import software.amazon.awssdk.services.bedrockruntime.BedrockRuntimeClient;
+
+@TemplateSubType(id = "invokeModel", label = "Invoke Model")
+public final class InvokeModelData implements RequestData {
+
+ @TemplateProperty(
+ label = "Model id",
+ group = "invokeModel",
+ id = "data.modelId0",
+ feel = Property.FeelMode.optional,
+ binding = @TemplateProperty.PropertyBinding(name = "data.modelId"))
+ @Valid
+ @NotNull
+ String modelId;
+
+ @TemplateProperty(
+ label = "Payload",
+ group = "invokeModel",
+ id = "data.payload",
+ feel = Property.FeelMode.required,
+ binding = @TemplateProperty.PropertyBinding(name = "data.payload"))
+ @Valid
+ @NotNull
+ Object payload;
+
+ @Override
+ public BedrockResponse execute(
+ BedrockRuntimeClient bedrockRuntimeClient, ObjectMapper mapperInstance) {
+ try {
+ String body = mapperInstance.writeValueAsString(this.payload);
+ String response =
+ bedrockRuntimeClient
+ .invokeModel(
+ builder -> builder.body(SdkBytes.fromUtf8String(body)).modelId(this.modelId))
+ .body()
+ .asUtf8String();
+ return new InvokeModelWrappedResponse(mapperInstance.readValue(response, Object.class));
+ } catch (JsonProcessingException e) {
+ throw new RuntimeException(e.getOriginalMessage(), e);
+ }
+ }
+
+ public void setModelId(@Valid @NotNull String modelId) {
+ this.modelId = modelId;
+ }
+
+ public void setPayload(@Valid @NotNull Object payload) {
+ this.payload = payload;
+ }
+
+ @Override
+ public boolean equals(Object o) {
+ if (this == o) return true;
+ if (o == null || getClass() != o.getClass()) return false;
+ InvokeModelData that = (InvokeModelData) o;
+ return Objects.equals(modelId, that.modelId) && Objects.equals(payload, that.payload);
+ }
+
+ @Override
+ public int hashCode() {
+ return Objects.hash(modelId, payload);
+ }
+
+ @Override
+ public String toString() {
+ return "InvokeModelData{" + "modelId='" + modelId + '\'' + '}';
+ }
+}
diff --git a/connectors/aws/aws-bedrock/src/main/java/io/camunda/connector/aws/bedrock/model/InvokeModelWrappedResponse.java b/connectors/aws/aws-bedrock/src/main/java/io/camunda/connector/aws/bedrock/model/InvokeModelWrappedResponse.java
new file mode 100644
index 0000000000..10635930c7
--- /dev/null
+++ b/connectors/aws/aws-bedrock/src/main/java/io/camunda/connector/aws/bedrock/model/InvokeModelWrappedResponse.java
@@ -0,0 +1,9 @@
+/*
+ * Copyright Camunda Services GmbH and/or licensed to Camunda Services GmbH
+ * under one or more contributor license agreements. Licensed under a proprietary license.
+ * See the License.txt file for more information. You may not use this file
+ * except in compliance with the proprietary license.
+ */
+package io.camunda.connector.aws.bedrock.model;
+
+public record InvokeModelWrappedResponse(Object body) implements BedrockResponse {}
diff --git a/connectors/aws/aws-bedrock/src/main/java/io/camunda/connector/aws/bedrock/model/PreviousMessage.java b/connectors/aws/aws-bedrock/src/main/java/io/camunda/connector/aws/bedrock/model/PreviousMessage.java
new file mode 100644
index 0000000000..1c6c0aacfb
--- /dev/null
+++ b/connectors/aws/aws-bedrock/src/main/java/io/camunda/connector/aws/bedrock/model/PreviousMessage.java
@@ -0,0 +1,9 @@
+/*
+ * Copyright Camunda Services GmbH and/or licensed to Camunda Services GmbH
+ * under one or more contributor license agreements. Licensed under a proprietary license.
+ * See the License.txt file for more information. You may not use this file
+ * except in compliance with the proprietary license.
+ */
+package io.camunda.connector.aws.bedrock.model;
+
+public record PreviousMessage(String message, String role) {}
diff --git a/connectors/aws/aws-bedrock/src/main/java/io/camunda/connector/aws/bedrock/model/RequestData.java b/connectors/aws/aws-bedrock/src/main/java/io/camunda/connector/aws/bedrock/model/RequestData.java
new file mode 100644
index 0000000000..5b2e1edb79
--- /dev/null
+++ b/connectors/aws/aws-bedrock/src/main/java/io/camunda/connector/aws/bedrock/model/RequestData.java
@@ -0,0 +1,22 @@
+/*
+ * Copyright Camunda Services GmbH and/or licensed to Camunda Services GmbH
+ * under one or more contributor license agreements. Licensed under a proprietary license.
+ * See the License.txt file for more information. You may not use this file
+ * except in compliance with the proprietary license.
+ */
+package io.camunda.connector.aws.bedrock.model;
+
+import com.fasterxml.jackson.databind.ObjectMapper;
+import io.camunda.connector.generator.java.annotation.TemplateDiscriminatorProperty;
+import io.camunda.connector.generator.java.annotation.TemplateSubType;
+import software.amazon.awssdk.services.bedrockruntime.BedrockRuntimeClient;
+
+@TemplateDiscriminatorProperty(
+ label = "Action",
+ group = "action",
+ name = "action",
+ defaultValue = "invokeModel")
+@TemplateSubType(id = "action", label = "Action")
+public sealed interface RequestData permits InvokeModelData, ConverseData {
+ BedrockResponse execute(BedrockRuntimeClient bedrockRuntimeClient, ObjectMapper mapperInstance);
+}
diff --git a/connectors/aws/aws-bedrock/src/main/resources/META-INF/services/io.camunda.connector.api.outbound.OutboundConnectorFunction b/connectors/aws/aws-bedrock/src/main/resources/META-INF/services/io.camunda.connector.api.outbound.OutboundConnectorFunction
new file mode 100644
index 0000000000..ae6fbf7303
--- /dev/null
+++ b/connectors/aws/aws-bedrock/src/main/resources/META-INF/services/io.camunda.connector.api.outbound.OutboundConnectorFunction
@@ -0,0 +1 @@
+io.camunda.connector.aws.bedrock.BedrockConnectorFunction
\ No newline at end of file
diff --git a/connectors/aws/aws-bedrock/src/main/resources/icon.svg b/connectors/aws/aws-bedrock/src/main/resources/icon.svg
new file mode 100644
index 0000000000..2e5077ba4e
--- /dev/null
+++ b/connectors/aws/aws-bedrock/src/main/resources/icon.svg
@@ -0,0 +1,12 @@
+
+
\ No newline at end of file
diff --git a/connectors/aws/aws-bedrock/src/test/java/io/camunda/connector/aws/bedrock/BaseTest.java b/connectors/aws/aws-bedrock/src/test/java/io/camunda/connector/aws/bedrock/BaseTest.java
new file mode 100644
index 0000000000..1a3138a855
--- /dev/null
+++ b/connectors/aws/aws-bedrock/src/test/java/io/camunda/connector/aws/bedrock/BaseTest.java
@@ -0,0 +1,57 @@
+/*
+ * Copyright Camunda Services GmbH and/or licensed to Camunda Services GmbH
+ * under one or more contributor license agreements. Licensed under a proprietary license.
+ * See the License.txt file for more information. You may not use this file
+ * except in compliance with the proprietary license.
+ */
+package io.camunda.connector.aws.bedrock;
+
+import static java.nio.charset.StandardCharsets.UTF_8;
+import static java.nio.file.Files.readString;
+
+import com.fasterxml.jackson.core.JsonProcessingException;
+import com.fasterxml.jackson.databind.ObjectMapper;
+import io.camunda.connector.api.json.ConnectorsObjectMapperSupplier;
+import java.io.File;
+import java.io.IOException;
+import java.util.ArrayList;
+import java.util.stream.Stream;
+import org.junit.jupiter.params.provider.Arguments;
+
+public class BaseTest {
+
+ public static Stream loadInvokeModelVariables() {
+ try {
+ return loadTestCasesFromResourceFile(
+ "src/test/resources/invokemodel/invokeModelExample.json");
+ } catch (IOException e) {
+ throw new RuntimeException(e);
+ }
+ }
+
+ public static Stream loadConverseVariables() {
+ try {
+ return loadTestCasesFromResourceFile("src/test/resources/converse/converseExample.json");
+ } catch (IOException e) {
+ throw new RuntimeException(e);
+ }
+ }
+
+ @SuppressWarnings("unchecked")
+ protected static Stream loadTestCasesFromResourceFile(final String fileWithTestCasesUri)
+ throws IOException {
+ final String cases = readString(new File(fileWithTestCasesUri).toPath(), UTF_8);
+ final ObjectMapper mapper = ConnectorsObjectMapperSupplier.getCopy();
+ var array = mapper.readValue(cases, ArrayList.class);
+ return array.stream()
+ .map(
+ value -> {
+ try {
+ return mapper.writeValueAsString(value);
+ } catch (JsonProcessingException e) {
+ throw new RuntimeException(e);
+ }
+ })
+ .map(Arguments::of);
+ }
+}
diff --git a/connectors/aws/aws-bedrock/src/test/java/io/camunda/connector/aws/bedrock/BedrockConnectorFunctionTest.java b/connectors/aws/aws-bedrock/src/test/java/io/camunda/connector/aws/bedrock/BedrockConnectorFunctionTest.java
new file mode 100644
index 0000000000..ba2f9ebc5b
--- /dev/null
+++ b/connectors/aws/aws-bedrock/src/test/java/io/camunda/connector/aws/bedrock/BedrockConnectorFunctionTest.java
@@ -0,0 +1,64 @@
+/*
+ * Copyright Camunda Services GmbH and/or licensed to Camunda Services GmbH
+ * under one or more contributor license agreements. Licensed under a proprietary license.
+ * See the License.txt file for more information. You may not use this file
+ * except in compliance with the proprietary license.
+ */
+package io.camunda.connector.aws.bedrock;
+
+import static org.mockito.ArgumentMatchers.any;
+import static org.mockito.Mockito.when;
+
+import io.camunda.connector.aws.bedrock.core.BedrockExecutor;
+import io.camunda.connector.aws.bedrock.model.InvokeModelWrappedResponse;
+import io.camunda.connector.test.outbound.OutboundConnectorContextBuilder;
+import org.junit.jupiter.api.Assertions;
+import org.junit.jupiter.params.ParameterizedTest;
+import org.junit.jupiter.params.provider.MethodSource;
+import org.mockito.MockedStatic;
+import org.mockito.Mockito;
+
+class BedrockConnectorFunctionTest extends BaseTest {
+
+ @ParameterizedTest
+ @MethodSource("loadInvokeModelVariables")
+ void executeInvokeModelReturnsCorrectResult(String variables) {
+
+ var bedrockConnectorFunction = new BedrockConnectorFunction();
+ var context = OutboundConnectorContextBuilder.create().variables(variables).build();
+
+ var bedrockExecutor = Mockito.mock(BedrockExecutor.class);
+
+ try (MockedStatic bedrockExecutorMockedStatic =
+ Mockito.mockStatic(BedrockExecutor.class)) {
+ bedrockExecutorMockedStatic
+ .when(() -> BedrockExecutor.create(any()))
+ .thenReturn(bedrockExecutor);
+ when(bedrockExecutor.execute()).thenReturn(new InvokeModelWrappedResponse("Hello"));
+ var response = bedrockConnectorFunction.execute(context);
+ Assertions.assertNotNull(response);
+ Assertions.assertInstanceOf(InvokeModelWrappedResponse.class, response);
+ }
+ }
+
+ @ParameterizedTest
+ @MethodSource("loadConverseVariables")
+ void executeConverseReturnsCorrectResult(String variables) {
+
+ var bedrockConnectorFunction = new BedrockConnectorFunction();
+ var context = OutboundConnectorContextBuilder.create().variables(variables).build();
+
+ var bedrockExecutor = Mockito.mock(BedrockExecutor.class);
+
+ try (MockedStatic bedrockExecutorMockedStatic =
+ Mockito.mockStatic(BedrockExecutor.class)) {
+ bedrockExecutorMockedStatic
+ .when(() -> BedrockExecutor.create(any()))
+ .thenReturn(bedrockExecutor);
+ when(bedrockExecutor.execute()).thenReturn(new InvokeModelWrappedResponse("Hello"));
+ var response = bedrockConnectorFunction.execute(context);
+ Assertions.assertNotNull(response);
+ Assertions.assertInstanceOf(InvokeModelWrappedResponse.class, response);
+ }
+ }
+}
diff --git a/connectors/aws/aws-bedrock/src/test/java/io/camunda/connector/aws/bedrock/model/ConverseDataTest.java b/connectors/aws/aws-bedrock/src/test/java/io/camunda/connector/aws/bedrock/model/ConverseDataTest.java
new file mode 100644
index 0000000000..e80ed0bbc7
--- /dev/null
+++ b/connectors/aws/aws-bedrock/src/test/java/io/camunda/connector/aws/bedrock/model/ConverseDataTest.java
@@ -0,0 +1,52 @@
+/*
+ * Copyright Camunda Services GmbH and/or licensed to Camunda Services GmbH
+ * under one or more contributor license agreements. Licensed under a proprietary license.
+ * See the License.txt file for more information. You may not use this file
+ * except in compliance with the proprietary license.
+ */
+package io.camunda.connector.aws.bedrock.model;
+
+import static org.mockito.ArgumentMatchers.any;
+import static org.mockito.Mockito.mock;
+import static org.mockito.Mockito.when;
+
+import com.fasterxml.jackson.databind.ObjectMapper;
+import java.util.ArrayList;
+import java.util.List;
+import java.util.function.Consumer;
+import org.junit.jupiter.api.Assertions;
+import org.junit.jupiter.api.Test;
+import org.mockito.Mockito;
+import software.amazon.awssdk.services.bedrockruntime.BedrockRuntimeClient;
+import software.amazon.awssdk.services.bedrockruntime.model.ConversationRole;
+import software.amazon.awssdk.services.bedrockruntime.model.ConverseResponse;
+
+class ConverseDataTest {
+
+ ObjectMapper mapper = new ObjectMapper();
+ BedrockRuntimeClient bedrockRuntimeClient = mock(BedrockRuntimeClient.class);
+ ConverseResponse converseResponse = mock(ConverseResponse.class, Mockito.RETURNS_DEEP_STUBS);
+
+ @Test
+ void execute_success() {
+
+ ConverseData converseData = new ConverseData();
+ converseData.setModelId("random-model-id");
+
+ List previousMessages = new ArrayList<>();
+ previousMessages.add(new PreviousMessage("Hey", ConversationRole.USER.name()));
+ previousMessages.add(new PreviousMessage("How are you?", ConversationRole.ASSISTANT.name()));
+ converseData.setMessages(previousMessages);
+ converseData.setNextMessage("I am good thanks, and you?");
+
+ when(bedrockRuntimeClient.converse(any(Consumer.class))).thenReturn(converseResponse);
+ when(converseResponse.output().message().content().getFirst().text())
+ .thenReturn("I am also good");
+ BedrockResponse bedrockResponse = converseData.execute(bedrockRuntimeClient, mapper);
+
+ Assertions.assertInstanceOf(ConverseWrapperResponse.class, bedrockResponse);
+ Assertions.assertEquals(
+ "I am also good",
+ ((ConverseWrapperResponse) bedrockResponse).messagesHistory().getLast().message());
+ }
+}
diff --git a/connectors/aws/aws-bedrock/src/test/java/io/camunda/connector/aws/bedrock/model/InvokeModelDataTest.java b/connectors/aws/aws-bedrock/src/test/java/io/camunda/connector/aws/bedrock/model/InvokeModelDataTest.java
new file mode 100644
index 0000000000..ee156d85c8
--- /dev/null
+++ b/connectors/aws/aws-bedrock/src/test/java/io/camunda/connector/aws/bedrock/model/InvokeModelDataTest.java
@@ -0,0 +1,65 @@
+/*
+ * Copyright Camunda Services GmbH and/or licensed to Camunda Services GmbH
+ * under one or more contributor license agreements. Licensed under a proprietary license.
+ * See the License.txt file for more information. You may not use this file
+ * except in compliance with the proprietary license.
+ */
+package io.camunda.connector.aws.bedrock.model;
+
+import static org.junit.jupiter.api.Assertions.*;
+import static org.mockito.ArgumentMatchers.any;
+import static org.mockito.Mockito.mock;
+import static org.mockito.Mockito.when;
+
+import com.fasterxml.jackson.core.JsonProcessingException;
+import com.fasterxml.jackson.databind.ObjectMapper;
+import io.camunda.connector.aws.bedrock.BaseTest;
+import java.util.function.Consumer;
+import org.junit.jupiter.api.Assertions;
+import org.junit.jupiter.api.Test;
+import software.amazon.awssdk.core.SdkBytes;
+import software.amazon.awssdk.services.bedrockruntime.BedrockRuntimeClient;
+import software.amazon.awssdk.services.bedrockruntime.model.InvokeModelResponse;
+
+class InvokeModelDataTest extends BaseTest {
+
+ ObjectMapper mapper = new ObjectMapper();
+ BedrockRuntimeClient bedrockRuntimeClient = mock(BedrockRuntimeClient.class);
+ InvokeModelResponse invokeModelResponse = mock(InvokeModelResponse.class);
+
+ @Test
+ void execute_success() throws JsonProcessingException {
+ String payload =
+ "{\"messages\":[{\"role\":\"user\", \"content\":\"Hello\"}], \"max_tokens\":256, \"top_p\":0.8, \"temperature\":0.7}";
+ InvokeModelData invokeModelData = new InvokeModelData();
+ invokeModelData.setModelId("random-model-id");
+ invokeModelData.setPayload(mapper.readTree(payload));
+
+ when(bedrockRuntimeClient.invokeModel(any(Consumer.class))).thenReturn(invokeModelResponse);
+ when(invokeModelResponse.body()).thenReturn(SdkBytes.fromUtf8String("{\"result\" : \"Hey\"}"));
+ BedrockResponse invokeModelWrappedResponse =
+ invokeModelData.execute(bedrockRuntimeClient, mapper);
+
+ Assertions.assertInstanceOf(InvokeModelWrappedResponse.class, invokeModelWrappedResponse);
+ }
+
+ @Test
+ void execute_wrongReturn() throws JsonProcessingException {
+ String payload =
+ "{\"messages\":[{\"role\":\"user\", \"content\":\"Hello\"}], \"max_tokens\":256, \"top_p\":0.8, \"temperature\":0.7}";
+ InvokeModelData invokeModelData = new InvokeModelData();
+ invokeModelData.setModelId("random-model-id");
+ invokeModelData.setPayload(mapper.readTree(payload));
+
+ when(bedrockRuntimeClient.invokeModel(any(Consumer.class))).thenReturn(invokeModelResponse);
+ when(invokeModelResponse.body()).thenReturn(SdkBytes.fromUtf8String("Hey"));
+
+ RuntimeException runtimeException =
+ Assertions.assertThrows(
+ RuntimeException.class, () -> invokeModelData.execute(bedrockRuntimeClient, mapper));
+
+ assertEquals(
+ "Unrecognized token 'Hey': was expecting (JSON String, Number, Array, Object or token 'null', 'true' or 'false')",
+ runtimeException.getMessage());
+ }
+}
diff --git a/connectors/aws/aws-bedrock/src/test/resources/converse/converseExample.json b/connectors/aws/aws-bedrock/src/test/resources/converse/converseExample.json
new file mode 100644
index 0000000000..b17a2a2967
--- /dev/null
+++ b/connectors/aws/aws-bedrock/src/test/resources/converse/converseExample.json
@@ -0,0 +1,20 @@
+[
+ {
+ "data":{
+ "modelId":"amazon.titan-text-premier-v1:0",
+ "nextMessage":"ok",
+ "messages":[
+ {"message":"Hey", "role":"USER"}, {"message":"Hello there! How can I assist you today?", "role":"ASSISTANT"},
+ {"message":"You feel good ?", "role":"USER"}, {
+ "message":"Thank you for asking! As an AI, I don't have personal feelings or experiences, but I'm always here to help with any questions or information you may need. How can I assist you today?",
+ "role":"ASSISTANT"
+ }
+ ]
+ },
+ "action":"converse",
+ "configuration":{"region":"us-east-1"},
+ "authentication":{
+ "type":"credentials", "accessKey":"access", "secretKey":"secret"
+ }
+ }
+]
\ No newline at end of file
diff --git a/connectors/aws/aws-bedrock/src/test/resources/invokemodel/invokeModelExample.json b/connectors/aws/aws-bedrock/src/test/resources/invokemodel/invokeModelExample.json
new file mode 100644
index 0000000000..a4883efe5f
--- /dev/null
+++ b/connectors/aws/aws-bedrock/src/test/resources/invokemodel/invokeModelExample.json
@@ -0,0 +1,10 @@
+[
+ {
+ "data":{
+ "modelId":"ai21.jamba-instruct-v1:0",
+ "payload":{"messages":[{"role":"user", "content":"Hello"}], "max_tokens":256, "top_p":0.8, "temperature":0.7}
+ }, "action":"invokeModel", "configuration":{"region":"us-east-1"}, "authentication":{
+ "type":"credentials", "accessKey":"access", "secretKey":"secret"
+ }
+ }
+]
\ No newline at end of file
diff --git a/connectors/aws/pom.xml b/connectors/aws/pom.xml
index caeca95e68..461acc64d6 100644
--- a/connectors/aws/pom.xml
+++ b/connectors/aws/pom.xml
@@ -24,6 +24,7 @@
aws-sns
aws-sqs
aws-sagemaker
+ aws-bedrock
diff --git a/parent/pom.xml b/parent/pom.xml
index 760077b5d5..49133a968e 100644
--- a/parent/pom.xml
+++ b/parent/pom.xml
@@ -91,6 +91,7 @@ limitations under the License.
1.5.6
1.12.745
+ 2.26.16
3.11.6
1.2.3