Skip to main content

Coming from MuleSoft

A guide for developers migrating integrations from MuleSoft Anypoint to WSO2 Integrator. This tutorial walks you through migrating a real MuleSoft project to WSO2 Integrator using the automated migration tool. No prior knowledge of Ballerina or WSO2 Integrator is required.

What you'll learn:

  • How to run the automated migration tool (UI wizard or CLI) to convert a MuleSoft project to Ballerina code
  • How to read the migration report and handle items the tool couldn't convert automatically
  • How MuleSoft concepts (flows, DataWeave, connectors) translate to their WSO2 Integrator equivalents
  • How to configure credentials, test, and run your migrated integration

Time: ~30–45 minutes

Prerequisites:

  • WSO2 Integrator installed (UI wizard path), or Ballerina installed with bal on your PATH (CLI path)
  • Git installed (to clone the sample project; not required if using your own project)

What You'll Build

You'll migrate the Mule Customer API Demo to WSO2 Integrator. It's a Mule 4.x project that exposes two REST endpoints backed by a MySQL database.

  • GET /customers: retrieves customer records from the database (with an optional ?mock=true mode for testing without a live database)
  • POST /customers: adds a new customer, applying a DataWeave transformation to normalise the input data

The project also includes a global error handler. This is a representative slice of what most real MuleSoft applications look like.

Sample project: Mule-API-Integration

The migration tool converts the Mule flows to Ballerina services, maps the MySQL connector, and generates a migration report. You'll review the report, fix up any remaining items, configure credentials in Config.toml, and run the result.

Using your own project? You can follow every step below using any Anypoint project(s) instead of the sample.

Step 1: Get the sample project

Clone the sample MuleSoft project:

git clone https://github.com/wso2/integration-samples.git
cd integration-samples/integrator-default-profile/third-party-integration-samples/Mule-API-Integration

Take a moment to look at the project structure. It follows the standard Mule 4 layout:

Mule-API-Integration/
├── pom.xml # Maven configuration
├── mule-artifact.json # Mule application descriptor
└── src/
└── main/
├── mule/
│ └── muledemo.xml # All Mule flows
└── resources/
├── config.properties # Database credentials
└── log4j2.xml # Logging configuration

The single flow file muledemo.xml contains the HTTP listener, the database operations, the DataWeave transformations, and the global error handler.

Using your own project? Skip this step and substitute your project's directory path wherever the sample path appears below.

Step 2: Run the migration tool

WSO2 Integrator provides an automated migration tool that converts MuleSoft Anypoint flows (XML configurations) to Ballerina code. The tool handles HTTP listeners, HTTP request connectors, DataWeave transformations, routers, error handling patterns, and more.

The migration wizard guides you through a 5-step process to convert your MuleSoft project(s) into a WSO2 Integrator project.

Prerequisite

  • Ensure WSO2 Integrator is installed and available on your system.

Step 1: Configure source

  1. Open WSO2 Integrator, click More Actions, and select Migrate Integrations from Other Vendors.

  2. Select MuleSoft as the source platform.

  3. Under Select a Project Folder or Directory, click Browse and select the Mule-API-Integration directory you cloned in Step 1 (or your own project directory).

  4. Under Source Layout, select Single Project (Mule-API-Integration is a single Mule project). If you have multiple projects in a parent directory, select Multiple Projects instead.

    Note: The Source Layout section appears only after you select a directory.

  5. Expand Configure MuleSoft Settings to set optional parameters:

    • Force Version: Select a specific Mule version, or use Auto Detect to let the tool determine it from your project.
  6. Click Generate Report.

    Configure source step

Step 2: Report generation

The wizard performs a dry run against your source project(s) to generate a coverage report before the actual migration begins.

When the dry run completes, the wizard displays a summary of the migration coverage:

  • Migration Coverage: Percentage of code lines that were automatically migrated.

  • Total code lines: Total number of source code lines analyzed.

  • Migratable code lines: Lines successfully converted to Ballerina.

  • Non-migratable code lines: Lines that require manual attention.

    Report generation step

    Note: The exact coverage percentages shown in the screenshots may differ from what you see. They reflect the tool's capabilities and the reference project at the time the screenshots were taken, and both evolve over time.

Click View Full Report to open the full HTML report. The report includes:

  • Migration Coverage Overview: Overall coverage percentage with a breakdown of total, migratable, and non-migratable code lines.

  • Breakdown Components: Separate coverage for Mule Elements and DataWeave expressions.

  • Manual Work Estimation: Estimated effort (best, average, and worst case) for completing non-migratable items.

  • Currently Unsupported Elements: List of elements that could not be automatically migrated.

  • Element Blocks that Require Manual Conversion: Specific code blocks that need manual implementation.

    Full migration report

Click Save Report to download the report for future reference.

Click Configure Destination to proceed, or Done to exit the wizard.

Step 3: Configure destination

  1. Enter an Integration Name for your migrated project.

  2. Configure the project settings:

    • Project Name: Name of the project (defaults to Default).
    • Create within a project: Enable project mode to manage multiple integrations and libraries within a single repository.
    • Select Path: Choose where to create the migrated project.
  3. Click Start Migration.

    Configure destination step

Step 4: Rule-based migration

The wizard runs the automated rule-based migration and displays progress in the migration log.

After the migration completes successfully, the AI Enhancement (Recommended) section appears. You can select one of the following:

  • Enhance with AI (AI automatically resolves unmapped elements, fixes build errors, and improves migration quality)
  • Skip for Now – Enhance Later (Keep the project as-is. You can trigger AI enhancement later from WSO2 Integrator Copilot)

Click Start AI Enhancement to proceed to Step 5. If you chose to skip, click Open Project to open the migrated project or Done to exit.

Rule-based migration step

Step 5: AI enhancement

This step runs only if you selected Enhance with AI in Step 4.

The wizard first checks whether you are signed in. If not, a sign-in panel appears:

  1. Click Login using WSO2 Integration Platform to sign in using SSO, or use one of the alternative options:

    • Enter your Anthropic API key
    • Enter your AWS Bedrock credentials
    • Enter your Google Vertex AI credentials
  2. To skip AI enhancement and exit, click Skip and Done.

    Sign-in panel

After signing in, the AI agent runs automatically and streams its progress. The agent resolves unmapped elements, fixes build errors, and improves the overall quality of the migrated code.

While the agent is running:

  • Click Pause to pause the AI enhancement. Click Resume to continue.

  • Click Done to exit the wizard, or Open Project to open the project without waiting for the agent to finish.

    Enhancing with AI agent

When the AI enhancement completes, the status shows AI Enhancement completed. Click Open Project to open the migrated project or Done to exit.

Step 3: Fix up the generated code

The migration tool converts everything it can automatically. This step walks through reviewing the output and resolving any items that still need attention.

Handle items requiring manual attention

Navigate to the output directory created by the migration tool. You'll find:

  • migration_report.html: the rule-based migration report listing every Mule element, its conversion status, and any items that need manual attention.
  • ENHANCEMENT_SUMMARY.md: present only if you opted in to AI enhancement. It summarises the AI-assisted improvements applied on top of the rule-based migration.

Open migration_report.html and work through any non-migratable items. If you opted in to AI enhancement, check ENHANCEMENT_SUMMARY.md first. Many of these items may already have been resolved automatically. Only address what remains:

  1. Unsupported Mule elements: Implement the equivalent Ballerina logic manually. Refer to the concept and component mapping table below.
  2. DataWeave transformations: Simple field mappings can be redone with the Visual Data Mapper. Complex transformations should be rewritten as Ballerina query expressions.
  3. Custom connectors: Check Connectors for a Ballerina equivalent, or call the service's REST API directly using http:Client.

Configure credentials

The migration tool automatically converts your Mule properties file (config.properties) to a Ballerina Config.toml in the output directory, so you don't need to create it manually. However, the file will contain placeholder values, you need to replace them with the correct values for your environment before running the migrated project:

# Config.toml — already created by the tool; update values before running
db_user = "root"
db_password = "abc123"

Step 4: Test it

If your Mule project had tests, the migration tool will have converted them to Ballerina. Run them with:

bal test

You can also use the Try-It tool built into WSO2 Integrator to send requests to your HTTP service interactively without leaving the IDE. Open the service file, click Try it, and test each endpoint.

Compare the responses with those returned by your original Mule application.

Step 5: Deploy

Once tests pass, run the integration locally:

bal run

To build a deployable artifact:

bal build

See Deploy for Docker, Kubernetes, and cloud deployment options.

Concept and component mapping

MuleSoftWSO2 IntegratorNotes
Mule FlowService / AutomationHTTP-triggered flows become services; scheduler-triggered flows become automations
Sub-flow / Flow ReferenceFunctionReusable logic extracted as a Ballerina function; called with a direct function invocation
Mule Event (payload + attributes + vars)Function parameters + recordsTyped parameters replace the mutable implicit event object. There is no global payload variable
HTTP Listener / API Kit Routerhttp:Listener + service resourcesOne service declaration combines transport and routing; each resource function maps to an endpoint
HTTP Requesthttp:ClientOutbound HTTP calls to external services
Database Connectorballerinax/mysql, ballerinax/postgresql, etc.SQL database access via type-safe queries
ConnectorConnector (Ballerina Central)Pre-built modules for Salesforce, Kafka, FTP, email, and more
Object Storeballerina/cache or a database connectorUse cache for transient in-memory state; a DB connector for durable persistence
Schedulertask:ListenerInterval or cron-based trigger for automations
Choice Routerif/else or matchConditional branching using standard Ballerina control flow
Scatter-GatherWorkersParallel execution; results are collected when all workers complete
For Each / Batchforeach / query expressionsIteration over collections; batch processing via chunked foreach or streaming
Asyncstart / WorkersFire-and-forget execution using the start keyword
Try Scope / Error Handlerdo/on failTyped, structural error handling; catch and handle specific error types
DataWeave (Transform Message)Visual Data Mapper + query expressionsDrag-and-drop for field mappings; Ballerina query expressions for complex transforms
DataWeave expressionBallerina expressionInline transformations; statically typed unlike DataWeave's dynamic typing
RAML / OASOpenAPI / Ballerina service contractImport an OAS spec to generate a service, or auto-export OAS from your Ballerina service
Connection Configuration / properties fileConfig.tomlAll credentials and environment-specific config in one file; override per environment with BAL_CONFIG_FILES
Maven (pom.xml)Ballerina.tomlProject metadata, dependencies, and build configuration
MUnitbal testBuilt-in test framework with mocking and test isolation support
CloudHub / Runtime FabricWSO2 Devant / KubernetesManaged cloud or self-hosted Kubernetes deployment
Runtime ManagerIntegration Control Plane (ICP)Centralized monitoring, observability, and management

Key differences

DataWeave vs. Ballerina

MuleSoft's DataWeave is a domain-specific language for data transformation. In WSO2 Integrator, the equivalent is Ballerina itself. Ballerina is a general-purpose language with first-class support for data transformation.

DataWeave:

%dw 2.0
output application/json
---
payload.orders map (order) -> {
id: order.orderId,
total: order.items reduce ((item, acc = 0) -> acc + item.price * item.qty),
status: if (order.shipped) "shipped" else "pending"
}

Ballerina equivalent:

type OrderOutput record {|
string id;
decimal total;
string status;
|};

function transformOrders(Order[] orders) returns OrderOutput[] {
return from Order order in orders
select {
id: order.orderId,
total: order.items.reduce(
isolated function(decimal acc, OrderItem item) returns decimal =>
acc + item.price * <decimal>item.qty, 0d
),
status: order.shipped ? "shipped" : "pending"
};
}

Key differences:

  • DataWeave is dynamically typed. Ballerina is statically typed. Schema errors are caught at compile time.
  • DataWeave uses map, filter, reduce with a functional style. Ballerina offers both functional style and query expressions (from ... where ... select).
  • DataWeave handles format conversion (JSON to XML, etc.) implicitly. In Ballerina, use explicit conversion functions or the Visual Data Mapper.

Mule flows vs. Ballerina services

In MuleSoft, a flow is a pipeline of message processors triggered by an inbound endpoint. In WSO2 Integrator, a service is a collection of HTTP resources (similar to APIkit Router). Each resource is a function.

MuleSoft flow (conceptual XML):

<flow name="getOrderFlow">
<http:listener path="/orders/{orderId}" method="GET"/>
<db:select config-ref="Database_Config">
<db:sql>SELECT * FROM orders WHERE id = :orderId</db:sql>
</db:select>
<ee:transform>
<!-- DataWeave transformation -->
</ee:transform>
</flow>

WSO2 Integrator equivalent:

import ballerina/http;
import ballerinax/postgresql;

final postgresql:Client db = check new (/* config */);

service /api on new http:Listener(8090) {
resource function get orders/[string orderId]() returns Order|http:NotFound|error {
Order? order = check db->queryRow(
`SELECT * FROM orders WHERE id = ${orderId}`
);
if order is () {
return http:NOT_FOUND;
}
return order;
}
}

Differences:

  • MuleSoft uses XML configuration. WSO2 Integrator uses either the visual designer or Ballerina code (bidirectionally synced).
  • MuleSoft passes data through a mutable MuleEvent (payload, attributes, variables). Ballerina uses typed function parameters and return values. There is no implicit state.
  • In MuleSoft, the flow pipeline processes one message at a time. In Ballerina, each resource function handles one request and returns a typed response.

Error handling

MuleSoft:

<error-handler>
<on-error-continue type="DB:CONNECTIVITY">
<set-payload value="Database unavailable"/>
</on-error-continue>
<on-error-propagate type="ANY">
<set-payload value="Unexpected error"/>
</on-error-propagate>
</error-handler>

WSO2 Integrator:

do {
Order order = check db->queryRow(`SELECT * FROM orders WHERE id = ${orderId}`);
return order;
} on fail postgresql:Error dbErr {
// Equivalent to on-error-continue for DB errors
log:printError("Database error", dbErr);
return <http:ServiceUnavailable>{body: {message: "Database unavailable"}};
} on fail error err {
// Equivalent to on-error-propagate for all other errors
log:printError("Unexpected error", err);
return <http:InternalServerError>{body: {message: "Unexpected error"}};
}

The key advantage is that Ballerina errors are typed. You handle postgresql:Error specifically rather than matching on a string error type like DB:CONNECTIVITY.

Understanding the migration in depth

The migration tool handles the conversion automatically. This section explains what the tool does under the hood. This is useful if you want to understand the decisions it makes, manually convert a component it couldn't handle, or build a deeper understanding of Ballerina alongside the migration.

1. Inventory your Mule applications

Categorize each Mule flow:

  • API flows (HTTP Listener + APIkit) --> WSO2 Integrator services
  • Scheduler flows --> automations
  • JMS/Kafka listener flows --> event handlers
  • Batch jobs --> automations with streaming/query expressions

2. Convert DataWeave to Ballerina

For each DataWeave transformation:

  1. Define the input and output record types in Ballerina (equivalent to DataWeave type definitions).
  2. For simple field mappings, use the Visual Data Mapper (drag and drop).
  3. For complex transformations, write Ballerina query expressions (similar to DataWeave map/filter).
  4. For format conversions (JSON to XML, CSV, etc.), use ballerina/data.xmldata, ballerina/data.csv, etc.

3. Map connectors

For each MuleSoft connector:

  • Database (db:select, db:insert): Use ballerinax/postgresql, ballerinax/mysql, etc.
  • HTTP Request: Use the ballerina/http client
  • Salesforce: Use ballerinax/salesforce
  • Kafka: Use ballerinax/kafka
  • File/FTP: Use ballerina/ftp, ballerina/io
  • JMS: Use ballerinax/java.jms or migrate to Kafka
  • Email: Use ballerina/email
  • Check the Connectors page for the full list.

4. Flow control constructs

The tool maps Mule's structural flow constructs to their Ballerina equivalents:

  • APIkit Router --> service resource functions with path parameters
  • Choice Router --> if/else or match
  • Scatter-Gather --> Ballerina workers
  • For Each --> foreach loop or query expression
  • Try Scope / Error Handler --> do/on fail
  • Set Variable --> local variable
  • Set Payload --> function return value
  • flow-ref --> direct function call

Common gotchas

  • No mutable event object: MuleSoft's MuleEvent carries payload, attributes, and variables through the flow. In Ballerina, pass data explicitly as function parameters. There is no global mutable state.
  • DataWeave payload keyword: There is no equivalent. Data arrives as typed function parameters or return values from connector calls.
  • Flow references: MuleSoft's flow-ref calls a sub-flow. In Ballerina, extract a function and call it directly.
  • Object Store persistence: MuleSoft Object Store provides key-value persistence. Use a database (ballerinax/postgresql) or ballerina/cache for in-memory caching.
  • Batch processing: MuleSoft has a dedicated batch module. In Ballerina, use streaming with query expressions or chunked processing in a foreach loop.
  • MEL expressions: Mule Expression Language is replaced by Ballerina expressions. All expressions are statically typed.

Before/After examples

These are additional reference examples showing how common MuleSoft patterns map to WSO2 Integrator. They are not from the sample project used in this tutorial.

REST API with database

MuleSoft (simplified):

  • HTTP Listener on /api/customers/{id}
  • db:select with query SELECT * FROM customers WHERE id = :id
  • DataWeave to transform { "customerId": payload.id, "fullName": payload.first_name ++ " " ++ payload.last_name }
  • Return JSON response

WSO2 Integrator:

import ballerina/http;
import ballerinax/postgresql;

final postgresql:Client db = check new ("localhost", "user", "pass", "mydb", 5432);

type DbCustomer record {|
int id;
string first_name;
string last_name;
string email;
|};

type CustomerResponse record {|
int customerId;
string fullName;
string email;
|};

service /api on new http:Listener(8090) {
resource function get customers/[int id]() returns CustomerResponse|http:NotFound|error {
DbCustomer? row = check db->queryRow(
`SELECT id, first_name, last_name, email FROM customers WHERE id = ${id}`
);
if row is () {
return http:NOT_FOUND;
}
return {
customerId: row.id,
fullName: string `${row.first_name} ${row.last_name}`,
email: row.email
};
}
}

Kafka consumer with transformation

MuleSoft: Kafka Listener --> DataWeave transform --> HTTP POST to downstream service

WSO2 Integrator:

import ballerinax/kafka;
import ballerina/http;
import ballerina/log;

listener kafka:Listener orderListener = check new ({
bootstrapServers: "localhost:9092",
groupId: "order-processor",
topics: ["orders"]
});

final http:Client downstream = check new ("http://fulfillment-service:8080");

type KafkaOrder record {|
string orderId;
string customerId;
decimal total;
|};

type FulfillmentRequest record {|
string orderRef;
string customer;
decimal amount;
|};

service on orderListener {
remote function onConsumerRecord(kafka:Caller caller, kafka:ConsumerRecord[] records) returns error? {
foreach kafka:ConsumerRecord rec in records {
json payload = check (check string:fromBytes(rec.value)).fromJsonString();
KafkaOrder order = check payload.cloneWithType();

// Transform (equivalent to DataWeave)
FulfillmentRequest req = {
orderRef: order.orderId,
customer: order.customerId,
amount: order.total
};

_ = check downstream->post("/fulfill", req);
}
check caller->commit();
}
}