Konveyor AI (kai) is Konveyor's approach to easing modernization of application source code to a new target by leveraging LLMs with guidance from static code analysis augmented with data in Konveyor that helps to learn how an Organization solved a similar problem in the past.
Pronunciation of 'kai': https://www.howtopronounce.com/kai
Our approach is to use static code analysis to find the areas in source code that need to be transformed. 'kai' will iterate through analysis information and work with LLMs to generate code changes to resolve incidents identified from analysis.
This approach does not require fine-tuning of LLMs, we augment a LLMs knowledge via the prompt, similar to approaches with RAG by leveraging external data from inside of Konveyor and from Analysis Rules to aid the LLM in constructing better results.
For example, analyzer-lsp Rules such as these (Java EE to Quarkus rulesets) are leveraged to aid guiding a LLM to update a legacy Java EE application to Quarkus
Note: For purposes of this initial prototype we are using an example of Java EE to Quarkus. That is an arbitrary choice to show viability of this approach. The code and the approach will work on other targets that Konveyor has rules for.
-
Konveyor contains information related to an Organization's Application Portfolio, a view into all of the applications an Organization is managing. This view includes a history of analysis information over time, access to each applications source repositories, and metadata that tracks work in-progress/completed in regard to each application being migrated to a given technology.
-
When 'Konveyor AI' wants to fix a specific issue in a given application, it will mine data in Konveyor to extract 2 sources of information to inject into a given LLM prompt.
-
Static Code Analysis
-
We pinpoint where to begin work by leveraging static code analysis to guide us
-
The static code analysis is informed via a collection of crowd sourced knowledge contained in our rulesets plus augmented via custom-rules
-
We include in the prompt Analysis metadata information to give the LLM more context such as
remote-ejb-to-quarkus-00000: description: Remote EJBs are not supported in Quarkus incidents: - uri: file:///tmp/source-code/src/main/java/com/redhat/coolstore/service/ShippingService.java message: "Remote EJBs are not supported in Quarkus, and therefore its use must be removed and replaced with REST functionality. In order to do this:\n 1. Replace the `@Remote` annotation on the class with a `@jakarta.ws.rs.Path(\"<endpoint>\")` annotation. An endpoint must be added to the annotation in place of `<endpoint>` to specify the actual path to the REST service.\n 2. Remove `@Stateless` annotations if present. Given that REST services are stateless by nature, it makes it unnecessary.\n 3. For every public method on the EJB being converted, do the following:\n - Annotate the method with `@jakarta.ws.rs.GET`\n - Annotate the method with `@jakarta.ws.rs.Path(\"<endpoint>\")` and give it a proper endpoint path. As a rule of thumb... <snip for readability>" lineNumber: 12 variables: file: file:///tmp/source-code/src/main/java/com/redhat/coolstore/service/ShippingService.java kind: Class name: Stateless package: com.redhat.coolstore.service - url: https://jakarta.ee/specifications/restful-ws/ title: Jakarta RESTful Web Services
-
-
Solved Examples - these are source code diffs that show a LLM how a similar problem was seen in another application the Organization has and how that Organization decided to fix it.
- We mine data Konveyor has stored from the Application Hub to search for when other applications have fixed the same rule violations and learn how they fixed it and pass that info into the prompt to aid the LLM
- This ability to leverage how the issue was seen and fixed in the past helps to give the LLM extra context to give a higher quality result.
- This is an early prompt we created to help give a feel of this in action and the result we got back from a LLM
-
- We will walk through the migration of a sample application written for EAP with Java EE and bring it to Quarkus.
- Sample Application
- https://github.com/konveyor-ecosystem/coolstore
- We will use the
main
branch which has the Java EE version. - We have found these issues in the
main
branch which need to be addressed before we move to Quarkus:- This information was obtained by running Kantra (Konveyor's static code analyzer) with these custom-rules
- Full output from Kantra is checked into the git repo here: example/analysis/coolstore/markdown/
- This information was obtained by running Kantra (Konveyor's static code analyzer) with these custom-rules
- We will use the
- https://github.com/konveyor-ecosystem/coolstore
- We launch VSCode with our Kai VS Code extension which is a modified version of the MTA VSCode Plugin
- We open a git checkout of a sample application: coolstore
- We run Kantra inside of VSCode to do an analysis of the application to learn what issues are present that need to be addressed before migrating to Quarkus
- We view the analysis information in VSCode
- We look at the impacted files and choose what files/issues we want to fix
- We click 'Generate Fix' in VSCode on a given file/issue and wait ~45 seconds for the Kai backend to generate a fix
- We view the suggested fix as a 'Diff' in VSCode
- We accept the generated fix
- The file in question has now been updated
- We move onto the next file/issue and repeat
- Until we fix this open issue, you must have an IBM API Key to use kai
- Set the below environment variables in your shell:
GENAI_KEY=my-secret-api-key-value
- We plan to allow alternative ways of specifying coordinates in future, tracked via: Allow model credentials to be stored in an .env file #89
- Once the above issue is fixed we expect to have alternative ways to specify model coordinates
-
We are using:
IBM Big AI Model (BAM) laboratory is where IBM Research designs, builds, and iterates on what’s next in foundation models. Our goal is to help accelerate the transition from research to product. Come experiment with us.
-
Login: https://bam.res.ibm.com/
- In order to use this service an individual needs to obtain a w3id from IBM. The kai development team is unable to help obtaining this access.
-
Related client tooling:
-
Obtain your API key from IBM BAM:
- To access via an API you can look at ‘Documentation’ after logging into https://bam.res.ibm.com/
- You will see a field embedded in the 'Documentation' section where you can generate/obtain an API Key.
- To access via an API you can look at ‘Documentation’ after logging into https://bam.res.ibm.com/
-
We also support other models. To change which llm you are targeting, open config.toml
and change the [models]
section to one of the following:
IBM served granite
[models]
provider = "ChatIBMGenAI"
[models.args]
model_id = "ibm/granite-13b-chat-v2"
IBM served mistral
[models]
provider = "ChatIBMGenAI"
[models.args]
model_id = "mistralai/mixtral-8x7b-instruct-v01"
IBM served codellama
[models]
provider = "ChatIBMGenAI"
[models.args]
model_id = "meta-llama/llama-2-13b-chat"
IBM served llama3
# Note: llama3 complains if we use more than 2048 tokens
# See: https://github.com/konveyor-ecosystem/kai/issues/172
[models]
provider = "ChatIBMGenAI"
[models.args]
model_id = "meta-llama/llama-3-70b-instruct"
parameters.max_new_tokens = 2048
Ollama
[models]
provider = "ChatOllama"
[models.args]
model = "mistral"
OpenAI GPT 4
[models]
provider = "ChatOpenAI"
[models.args]
model = "gpt-4"
OpenAI GPT 3.5
[models]
provider = "ChatOpenAI"
[models.args]
model = "gpt-3.5-turbo"
- We want to run the 'kai' REST API Server, this will also require a running postgres database and we need to populate that postgres database with a collection of application analysis information from various sample applications. All of the needed data is contained in this repo, we just need to run a command to load it.
- Clone Repo and Ensure you have the virtual environment setup
git clone https://github.com/konveyor-ecosystem/kai.git
cd kai
python3 -m venv env
- We've tested this with Python 3.11 and 3.12
source env/bin/activate
pip install -r ./requirements.txt
- Run the DB via podman
- Open a new shell tab
source env/bin/activate
- Let this run in background:
make run-postgres
- Run the Kai server in background
- Open a new shell tab
source env/bin/activate
- Let this run in background:
DEMO_MODE=true make run-server
- Please double check that you have
GENAI_KEY=my-secret-api-key-value
defined in the environment variables prior to running the server. - The
DEMO_MODE
option will cache responses and play them back on subsequent runs.
- Please double check that you have
- Load data into the database
- Fetch sample apps:
pushd samples; ./fetch_apps.py; popd
- Can run this in current shell, command will run for a ~1 minute and complete
- Ensure you are running within our python virtual env:
source env/bin/activate
make load-data
- Fetch sample apps:
-
There are 2 means to use Kai
- IDE usage, this is the intended demo flow where we:
- Install our VSCode Plugin
- Above is a temporary google doc, we want to clean this up and bring into Markdown in near future.
git clone https://github.com/konveyor-ecosystem/coolstore
- Open coolstore in VSCode
- Run Konveyor Analysis via Kantra inside of the IDE
- TODO: Instructions forthcoming
- Install our VSCode Plugin
- IDE usage, this is the intended demo flow where we:
-
CLI usage (not intended for any demo, but useful for dev team to do test of backend)
-
Run the client: kai-service/mock-client.py
-
source env/bin/activate
-
cd kai-service
-
python ./mock-client.py
. (This needs the server to be running above)$ python ./mock-client.py 200 {"feeling": "OK!", "recv": {"test": "object"}} 200 ## Reasoning 1. In the Java EE code, we are using `@MessageDriven` annotation which is not supported in Quarkus. We need to replace it with a CDI scope annotation like `@ApplicationScoped`. 2. The `HelloWorldMDB` class is a Message Driven Bean (MDB) that listens to messages on a specified queue and processes them. 3. The MDB uses the `javax.jms.TextMessage` class to process the messages. 4. The MDB is activated using the `@ActivationConfigProperty` annotation which specifies the destination type, destination, and acknowledge mode. 5. To migrate this code to Quarkus, we need to replace the `@MessageDriven` annotation with `@ApplicationScoped` and use CDI for dependency injection. 6. We also need to update the `onMessage` method to use the `@Incoming` and `Log` annotations provided by Quarkus. ## Updated File ```java // Update the `HelloWorldMDB` class to use CDI and Quarkus annotations @ApplicationScoped public class HelloWorldMDB { @Incoming("CMTQueue") public void onMessage(String msg) { Log.info("Received Message: " + msg); } } ``` This updated file uses the `@ApplicationScoped` annotation to scope the MDB to the application and the `@Incoming` and `Log` annotations to process the messages and log them. input...
-
-
The kai server will always cache responses in the kai/data/vcr/<application_name>/<model>
directory. In non-demo mode, these responses will be overwritten whenever a new request is made.
When the server is run with DEMO_MODE=true
, these responses will be played back. The request will be matched on everything except for authorization headers, cookies, content-length and request body.
There are two ways to record new responses:
- Run the requests while the server is not in
DEMO_MODE
- Delete the specific existing cached response (under
kai/data/vcr/<application_name>/<model>/<source-file-path-with-slashes-replaced-with-dashes.java.yaml>
), then rerun. When a cached response does not exist, a new one will be recorded and played back on subsequent runs.
- Below information is only needed for those who are looking to contribute, run e2e tests, etc.
- If you are a developer working on Kai and you are updating requirements.txt, you will need to do some manual changes beyond just a
pip freeze &> ./requirements.txt
, we have a few directives that address differences in 'darwin' systems that need to be preserved. These need to be added manually after a 'freeze' as the freeze command is not aware of what exists in requirements.txt. Please consult the diff of changes you are making now from prior version and note the extra directions forpython_version
and orsys_platform
cd ./samples
./fetch_apps.py
- This will checkout the sample app source code to:
./samples/sample_repos
- This directory is in .gitignore
- This will checkout the sample app source code to:
Note: We have checked in analysis runs for all sample applications so you do NOT need to run analysis yourself. The instructions below are ONLY if you want to recreate, this is NOT required
- Install podman so you can run Kantra for static code analysis
cd samples
./fetch_apps.py
# this will git clone example source code appscd macos
./restart_podman_machine.sh
# setups the podman VM on MacOS so it will mount the host filesystem into the VM./get_latest_kantra_cli.sh
# fetches 'kantra' our analyzer tool and stores it in ../bincd ..
./analyze_apps.py
# Analyzes all sample apps we know about, in both the 'initial' and 'solved' states, expect this to run for ~2-3 hours.
Analysis data will be stored in: samples/analysis_reports/{APP_NAME}/<initial|solved>/output.yaml
- Install trunk via: https://docs.trunk.io/check#install-the-cli
- Run the linters:
trunk check
- Format code:
trunk fmt
- Install the prerequisites in Setup and activate the python virtual environment
- Ensure you've checked out the source code for sample applications: Run: ./samples/fetch_sample_apps.sh
- Run: ./run_tests.sh
This repository represents a prototype implementation as the team explores the solution space. The intent is for this work to remain in the konveyor-ecosystem as the team builds knowledge in the domain and experiments with solutions. As the approach matures we will integrate this properly into Konveyor and seek to promote to github.com/konveyor organization.
Refer to Konveyor's Code of Conduct here.