diff --git a/.changes/3.42.json b/.changes/3.42.json
new file mode 100644
index 0000000000..dacf014390
--- /dev/null
+++ b/.changes/3.42.json
@@ -0,0 +1,23 @@
+{
+ "date" : "2024-11-27",
+ "version" : "3.42",
+ "entries" : [ {
+ "type" : "feature",
+ "description" : "Amazon Q /dev: support `Dockerfile` files"
+ }, {
+ "type" : "feature",
+ "description" : "Feature(Amazon Q Code Transformation): allow users to view results in 5 smaller diffs"
+ }, {
+ "type" : "feature",
+ "description" : "Introduce @workspace command to enhance chat context fetching for Chat"
+ }, {
+ "type" : "bugfix",
+ "description" : "Correct search text for Amazon Q inline suggestion keybindings"
+ }, {
+ "type" : "bugfix",
+ "description" : "Fix(Amazon Q Code Transformation): always show user latest/correct transformation results"
+ }, {
+ "type" : "bugfix",
+ "description" : "Amazon Q /dev: Fix error when accepting changes if leading slash is present."
+ } ]
+}
\ No newline at end of file
diff --git a/.changes/3.43.json b/.changes/3.43.json
new file mode 100644
index 0000000000..3ff7f67e04
--- /dev/null
+++ b/.changes/3.43.json
@@ -0,0 +1,17 @@
+{
+ "date" : "2024-12-03",
+ "version" : "3.43",
+ "entries" : [ {
+ "type" : "feature",
+ "description" : "`/review` in Q chat to scan your code for vulnerabilities and quality issues, and generate fixes"
+ }, {
+ "type" : "feature",
+ "description" : "`/test` in Q chat to generate unit tests for java and python"
+ }, {
+ "type" : "feature",
+ "description" : "`/doc` in Q chat to generate and update documentation for your project"
+ }, {
+ "type" : "feature",
+ "description" : "Added system notifications to inform users about critical plugin updates and potential issues with available workarounds"
+ } ]
+}
\ No newline at end of file
diff --git a/.changes/3.44.json b/.changes/3.44.json
new file mode 100644
index 0000000000..22545117d6
--- /dev/null
+++ b/.changes/3.44.json
@@ -0,0 +1,20 @@
+{
+ "date" : "2024-12-04",
+ "version" : "3.44",
+ "entries" : [ {
+ "type" : "feature",
+ "description" : "Amazon Q: UI improvements to chat: New splash loader animation, initial streaming card animation, improved button colours"
+ }, {
+ "type" : "feature",
+ "description" : "Amazon Q: Navigate through prompt history by using the up/down arrows"
+ }, {
+ "type" : "bugfix",
+ "description" : "Fix issue where Amazon Q Code Transform is unable to start"
+ }, {
+ "type" : "bugfix",
+ "description" : "Fix DynamoDB viewer throwing 'ActionGroup should be registered using tag' on IDE start (#5012) (#5120)"
+ }, {
+ "type" : "bugfix",
+ "description" : "Amazon Q: Fix chat syntax highlighting when using several different themes"
+ } ]
+}
\ No newline at end of file
diff --git a/.changes/3.45.json b/.changes/3.45.json
new file mode 100644
index 0000000000..001c781e7b
--- /dev/null
+++ b/.changes/3.45.json
@@ -0,0 +1,23 @@
+{
+ "date" : "2024-12-10",
+ "version" : "3.45",
+ "entries" : [ {
+ "type" : "feature",
+ "description" : "Add acknowledgement button for Amazon Q Chat disclaimer"
+ }, {
+ "type" : "bugfix",
+ "description" : " Chosing cancel on sign out confirmation now cancels the sign out and does not delete profiles from ~/.aws/config (#5167)"
+ }, {
+ "type" : "bugfix",
+ "description" : "Fix `@workspace` missing from the Amazon Q Chat welcome tab"
+ }, {
+ "type" : "bugfix",
+ "description" : "Fix for /review LLM based code issues for file review on windows"
+ }, {
+ "type" : "bugfix",
+ "description" : "Fix for File Review payload and Regex error for payload generation"
+ }, {
+ "type" : "bugfix",
+ "description" : "Amazon Q Code Transformation: show build logs when server-side build fails"
+ } ]
+}
\ No newline at end of file
diff --git a/.changes/3.46.json b/.changes/3.46.json
new file mode 100644
index 0000000000..4d998c9ac6
--- /dev/null
+++ b/.changes/3.46.json
@@ -0,0 +1,26 @@
+{
+ "date" : "2024-12-17",
+ "version" : "3.46",
+ "entries" : [ {
+ "type" : "feature",
+ "description" : "/review: Code fix automatically scrolls into view after generation."
+ }, {
+ "type" : "feature",
+ "description" : "Chat: improve font size and line-height in footer (below prompt input field)"
+ }, {
+ "type" : "feature",
+ "description" : "Adds capability to send new context commands to AB groups"
+ }, {
+ "type" : "bugfix",
+ "description" : "Chat: When writing a prompt without sending it, navigating via up/down arrows sometimes deletes the unsent prompt."
+ }, {
+ "type" : "bugfix",
+ "description" : "Fix chat not retaining history when interaction is through onboarding tab type (#5189)"
+ }, {
+ "type" : "bugfix",
+ "description" : "Chat: When navigating to previous prompts, code attachments are sometimes displayed incorrectly"
+ }, {
+ "type" : "bugfix",
+ "description" : "Reduce frequency of system information query"
+ } ]
+}
\ No newline at end of file
diff --git a/.changes/3.47.json b/.changes/3.47.json
new file mode 100644
index 0000000000..c7e3dbe830
--- /dev/null
+++ b/.changes/3.47.json
@@ -0,0 +1,35 @@
+{
+ "date" : "2025-01-09",
+ "version" : "3.47",
+ "entries" : [ {
+ "type" : "bugfix",
+ "description" : "Fix issue where users are unable to login to Amazon Q if they have previously authenticated (#5214)"
+ }, {
+ "type" : "bugfix",
+ "description" : "Fix incorrect text shown while updating documentation in /doc"
+ }, {
+ "type" : "bugfix",
+ "description" : "Amazon Q Code Transformation: retry initial project upload on failure"
+ }, {
+ "type" : "bugfix",
+ "description" : "/transform: use correct doc link in SQL conversion help message"
+ }, {
+ "type" : "bugfix",
+ "description" : "Amazon Q /dev: Fix issue when files are deleted while preparing context"
+ }, {
+ "type" : "bugfix",
+ "description" : "Amazon Q /test: Test generation fails for files outside the project"
+ }, {
+ "type" : "bugfix",
+ "description" : "Amazon Q Code Transformation: allow PostgreSQL as target DB for SQL conversions"
+ }, {
+ "type" : "bugfix",
+ "description" : "Fix incorrect accept and reject buttons shows up while hovering over the generated file"
+ }, {
+ "type" : "bugfix",
+ "description" : "Prevent customization override if user has manually selected a customization"
+ }, {
+ "type" : "bugfix",
+ "description" : "Align UX text of document generation flow with vs code version"
+ } ]
+}
\ No newline at end of file
diff --git a/.changes/3.48.json b/.changes/3.48.json
new file mode 100644
index 0000000000..c3ed0456a7
--- /dev/null
+++ b/.changes/3.48.json
@@ -0,0 +1,32 @@
+{
+ "date" : "2025-01-16",
+ "version" : "3.48",
+ "entries" : [ {
+ "type" : "feature",
+ "description" : "Enhance Q inline completion context fetching for better suggestion quality"
+ }, {
+ "type" : "feature",
+ "description" : "/doc: Add error message if updated README is too large"
+ }, {
+ "type" : "bugfix",
+ "description" : "/transform: always include button to start a new transformation at the end of a job"
+ }, {
+ "type" : "bugfix",
+ "description" : "Amazon Q can update mvn and gradle build files"
+ }, {
+ "type" : "bugfix",
+ "description" : "Fix doc generation for modules that are a part of the project"
+ }, {
+ "type" : "bugfix",
+ "description" : "Amazon Q /dev: Remove hard-coded limits and instead rely server-side data to communicate number of code generations remaining"
+ }, {
+ "type" : "bugfix",
+ "description" : "/transform: automatically open pre-build error logs when available"
+ }, {
+ "type" : "bugfix",
+ "description" : "/doc: Fix code generation error when cancelling a documentation task"
+ }, {
+ "type" : "bugfix",
+ "description" : "Amazon Q - update messaging for /doc agent"
+ } ]
+}
\ No newline at end of file
diff --git a/.changes/next-release/.changes/next-release/bugfix-21701bb3-5189-474e-868b-9ec46ecde6ee.json b/.changes/next-release/.changes/next-release/bugfix-21701bb3-5189-474e-868b-9ec46ecde6ee.json
deleted file mode 100644
index 8d2e0694ab..0000000000
--- a/.changes/next-release/.changes/next-release/bugfix-21701bb3-5189-474e-868b-9ec46ecde6ee.json
+++ /dev/null
@@ -1,4 +0,0 @@
-{
- "type" : "bugfix",
- "description" : "Feature Development: Fix file rejections for files outside of src/"
-}
diff --git a/.changes/next-release/feature-07ba60de-7043-4965-8fb5-7038c8ea6c5e.json b/.changes/next-release/feature-07ba60de-7043-4965-8fb5-7038c8ea6c5e.json
deleted file mode 100644
index 93b0197b9f..0000000000
--- a/.changes/next-release/feature-07ba60de-7043-4965-8fb5-7038c8ea6c5e.json
+++ /dev/null
@@ -1,4 +0,0 @@
-{
- "type" : "feature",
- "description" : "Feature(Amazon Q Code Transformation): allow users to view results in 5 smaller diffs"
-}
\ No newline at end of file
diff --git a/.changes/next-release/feature-b05e3cd8-abd5-47f9-bd5c-02c7b7eb378e.json b/.changes/next-release/feature-b05e3cd8-abd5-47f9-bd5c-02c7b7eb378e.json
deleted file mode 100644
index f28cb4e973..0000000000
--- a/.changes/next-release/feature-b05e3cd8-abd5-47f9-bd5c-02c7b7eb378e.json
+++ /dev/null
@@ -1,4 +0,0 @@
-{
- "type" : "feature",
- "description" : "Introduce @workspace command to enhance chat context fetching for Chat"
-}
\ No newline at end of file
diff --git a/.editorconfig b/.editorconfig
index 7b39413db6..55acdaad72 100644
--- a/.editorconfig
+++ b/.editorconfig
@@ -42,7 +42,7 @@ ij_kotlin_allow_trailing_comma = true
ij_kotlin_code_style_defaults = KOTLIN_OFFICIAL
ij_kotlin_name_count_to_use_star_import = 2147483647
ij_kotlin_name_count_to_use_star_import_for_members = 2147483647
-ij_kotlin_packages_to_use_import_on_demand = unset
+ij_kotlin_packages_to_use_import_on_demand = ""
[{*.markdown,*.md}]
ij_markdown_force_one_space_after_blockquote_symbol = true
diff --git a/.github/CODEOWNERS b/.github/CODEOWNERS
index d9f66ea3a2..bddc4d68df 100644
--- a/.github/CODEOWNERS
+++ b/.github/CODEOWNERS
@@ -5,9 +5,9 @@ codemodernizer/ @aws/elastic-gumby
codetransform/ @aws/elastic-gumby
codewhisperer/ @aws/codewhisperer-team
-mynah-ui/ @aws/dexp
-mynah-ui/package.json @aws/dexp @aws/elastic-gumby @aws/earlybird @aws/codewhisperer-team
-mynah-ui/package-lock.json @aws/dexp @aws/elastic-gumby @aws/earlybird @aws/codewhisperer-team
+mynah-ui/ @aws/flare
+mynah-ui/package.json @aws/flare @aws/elastic-gumby @aws/earlybird @aws/codewhisperer-team
+mynah-ui/package-lock.json @aws/flare @aws/elastic-gumby @aws/earlybird @aws/codewhisperer-team
# nested within chat/, so needs to be after to take precedence
amazonqFeatureDev/ @aws/earlybird
diff --git a/.github/workflows/prerelease.yml b/.github/workflows/prerelease.yml
index 9ebe77f813..4e5d0a67b2 100644
--- a/.github/workflows/prerelease.yml
+++ b/.github/workflows/prerelease.yml
@@ -9,6 +9,9 @@ on:
push:
branches: [ main, feature/* ]
+concurrency:
+ group: ${{ github.workflow }}${{ github.event.pull_request.number || github.ref }}
+ cancel-in-progress: true
jobs:
generate_artifact_toolkit_standalone:
diff --git a/.github/workflows/q-mega-prerelease.yml b/.github/workflows/q-mega-prerelease.yml
deleted file mode 100644
index e919987d23..0000000000
--- a/.github/workflows/q-mega-prerelease.yml
+++ /dev/null
@@ -1,164 +0,0 @@
-name: Prerelease Q-only
-on:
- workflow_dispatch:
- inputs:
- tag_name:
- description: 'Tag name for release'
- required: false
- default: prerelease
- push:
- branches: [ feature/q-mega ]
-
-jobs:
- time:
- outputs:
- time: ${{ steps.time.outputs.time }}
- runs-on: ubuntu-latest
- steps:
- - id: time
- run: |
- echo "time=$(date +%s)" >> "$GITHUB_OUTPUT"
-
- generate_artifact_q:
- needs: [ time ]
- strategy:
- matrix:
- supported_versions: [ '2024.1', '2024.2', '2024.3' ]
- runs-on: ubuntu-latest
- steps:
- - uses: actions/checkout@v4
- with:
- fetch-depth: 0
- - name: remove unwanted dependencies
- run: |
- sudo rm -rf /usr/share/dotnet
- sudo rm -rf /opt/ghc
- sudo rm -rf "/usr/local/share/boost"
- sudo rm -rf "$AGENT_TOOLSDIRECTORY"
- - uses: actions/setup-java@v4
- with:
- distribution: 'corretto'
- java-version: '21'
- - name: Generate artifact
- run: |
- sed -i.bak "s/\(toolkitVersion\s*=\s*\)\(.*\)/\1\2-${{ needs.time.outputs.time }}/" gradle.properties
- ./gradlew -PideProfileName=${{ matrix.supported_versions }} :plugin-amazonq:buildPlugin
- - name: Upload artifact
- uses: actions/upload-artifact@v4
- with:
- name: plugin-amazonq-${{ matrix.supported_versions }}
- path: ./plugins/amazonq/build/distributions/*.zip
- retention-days: 1
-
- generate_artifact_core:
- needs: [ time ]
- strategy:
- matrix:
- supported_versions: [ '2024.1', '2024.2', '2024.3' ]
- runs-on: ubuntu-latest
- steps:
- - uses: actions/checkout@v4
- with:
- fetch-depth: 0
- - name: remove unwanted dependencies
- run: |
- sudo rm -rf /usr/share/dotnet
- sudo rm -rf /opt/ghc
- sudo rm -rf "/usr/local/share/boost"
- sudo rm -rf "$AGENT_TOOLSDIRECTORY"
- - uses: actions/setup-java@v4
- with:
- distribution: 'corretto'
- java-version: '21'
-
- - name: Generate artifact
- run: |
- sed -i.bak "s/\(toolkitVersion\s*=\s*\)\(.*\)/\1\2-${{ needs.time.outputs.time }}/" gradle.properties
- ./gradlew -PideProfileName=${{ matrix.supported_versions }} :plugin-core:buildPlugin
- - name: Upload artifact
- uses: actions/upload-artifact@v4
- with:
- name: plugin-core-${{ matrix.supported_versions }}
- path: ./plugins/core/build/distributions/*.zip
- retention-days: 1
-
- generate_changelog:
- needs: [ time ]
- runs-on: ubuntu-latest
- outputs:
- feature: ${{ steps.assign_output.outputs.feature }}
- tagname: ${{ steps.assign_output.outputs.tagname }}
- version: ${{ steps.assign_output.outputs.version }}
- changes: ${{ steps.assign_output.outputs.changes }}
- steps:
- - uses: actions/checkout@v4
-
- - uses: actions/setup-java@v4
- with:
- distribution: 'corretto'
- java-version: '21'
-
- - if: github.event_name == 'workflow_dispatch'
- run: |
- echo "TAG_NAME=${{ github.event.inputs.tag_name }}" >> $GITHUB_ENV
- - if: github.ref_name != 'main'
- run: |
- TAG_NAME=${{ github.ref_name }}
- FEAT_NAME=$(echo $TAG_NAME | sed 's/feature\///')
- echo "FEAT_NAME=$FEAT_NAME" >> $GITHUB_ENV
- echo "TAG_NAME=pre-$FEAT_NAME" >> $GITHUB_ENV
- - if: github.ref_name == 'main'
- run: |
- echo "FEAT_NAME=" >> $GITHUB_ENV
- echo "TAG_NAME=prerelease" >> $GITHUB_ENV
- - name: Generate changelog
- id: changelog
- run: |
- sed -i.bak "s/\(toolkitVersion\s*=\s*\)\(.*\)/\1\2-${{ needs.time.outputs.time }}/" gradle.properties
- ./gradlew :createRelease :generateChangelog
- - name: Provide the metadata to output
- id: assign_output
- run: |
- echo "feature=$FEAT_NAME" >> $GITHUB_OUTPUT
- echo "tagname=$TAG_NAME" >> $GITHUB_OUTPUT
- echo "version=$(cat gradle.properties | grep toolkitVersion | cut -d'=' -f2)" >> $GITHUB_OUTPUT
- echo 'changes<> $GITHUB_OUTPUT
- cat CHANGELOG.md | perl -ne 'BEGIN{$/="\n\n"} print; exit if $. == 1' >> $GITHUB_OUTPUT
- echo 'EOF' >> $GITHUB_OUTPUT
-
- publish:
- needs: [ generate_artifact_core, generate_artifact_q, generate_changelog ]
- env:
- GH_TOKEN: ${{ secrets.GITHUB_TOKEN }}
- GH_REPO: ${{ github.repository }}
- FEAT_NAME: ${{ needs.generate_changelog.outputs.feature }}
- TAG_NAME: ${{ needs.generate_changelog.outputs.tagname }}
- AWS_TOOLKIT_VERSION: ${{ needs.generate_changelog.outputs.version }}
- BRANCH: ${{ github.ref_name }}
- AWS_TOOLKIT_CHANGES: ${{ needs.generate_changelog.outputs.changes }}
- runs-on: ubuntu-latest
- permissions:
- contents: write
- steps:
- - uses: actions/checkout@v4
- - uses: actions/download-artifact@v4
- - name: Delete existing prerelease
- # "prerelease" (main branch) or "pre-"
- if: "env.TAG_NAME == 'prerelease' || startsWith(env.TAG_NAME, 'pre-')"
- run: |
- echo "SUBJECT=AWS Toolkit ${AWS_TOOLKIT_VERSION}: ${FEAT_NAME:-${TAG_NAME}}" >> $GITHUB_ENV
- gh release delete "$TAG_NAME" --cleanup-tag --yes || true
- - name: Publish to GitHub Releases
- run: |
- envsubst < "$GITHUB_WORKSPACE/.github/workflows/prerelease_notes.md" > "$RUNNER_TEMP/prerelease_notes.md"
- gh release create "$TAG_NAME" --prerelease --notes-file "$RUNNER_TEMP/prerelease_notes.md" --title "$SUBJECT" --target $GITHUB_SHA
- - name: Publish core
- run: |
- gh release upload "$TAG_NAME" plugin-core-*/*.zip
- - name: Publish Q
- run: |
- gh release upload "$TAG_NAME" plugin-amazonq-*/*.zip
- - name: Publish XML manifest
- run: |
- gh release view "$TAG_NAME" --repo "$GITHUB_REPOSITORY" --json assets | python3 "$GITHUB_WORKSPACE/.github/workflows/generateUpdatesXml.py" - > updatePlugins.xml
- gh release upload "$TAG_NAME" updatePlugins.xml
diff --git a/.github/workflows/ssm-integ.yml b/.github/workflows/ssm-integ.yml
index 3d50adbb86..d18f63d2d0 100644
--- a/.github/workflows/ssm-integ.yml
+++ b/.github/workflows/ssm-integ.yml
@@ -9,6 +9,7 @@ on:
branches: [ main, feature/* ]
# PRs only need to run this if the SSM plugin logic has changed
paths:
+ - '.github/workflows/ssm-integ.yml'
- 'jetbrains-core/src/software/aws/toolkits/jetbrains/services/ssm/SsmPlugin.kt'
- 'jetbrains-core/it/software/aws/toolkits/jetbrains/services/ssm/SsmPluginTest.kt'
@@ -28,10 +29,10 @@ jobs:
run: git config --system core.longpaths true
if: ${{ matrix.os == 'windows-latest' }}
- uses: actions/checkout@v2
- - name: Set up JDK 17
- uses: actions/setup-java@v1
+ - uses: actions/setup-java@v4
with:
- java-version: 17
+ distribution: 'corretto'
+ java-version: '21'
- name: Grant execute permission for gradlew
run: chmod +x gradlew
- name: Build with Gradle
diff --git a/.gitignore b/.gitignore
index 41dcb232c8..5b3693c327 100644
--- a/.gitignore
+++ b/.gitignore
@@ -1,5 +1,6 @@
.gradle
.intellijPlatform
+.kotlin
out/
target/
.idea
diff --git a/.run/Run Amazon Q - Ultimate [2024.1].run.xml b/.run/Run Amazon Q - Ultimate [2024.1].run.xml
index 4296908019..bae51e0465 100644
--- a/.run/Run Amazon Q - Ultimate [2024.1].run.xml
+++ b/.run/Run Amazon Q - Ultimate [2024.1].run.xml
@@ -22,4 +22,4 @@
false
-
\ No newline at end of file
+
diff --git a/CHANGELOG.md b/CHANGELOG.md
index e7a6630993..30f7bd5bd0 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -1,3 +1,64 @@
+# _3.48_ (2025-01-16)
+- **(Feature)** Enhance Q inline completion context fetching for better suggestion quality
+- **(Feature)** /doc: Add error message if updated README is too large
+- **(Bug Fix)** /transform: always include button to start a new transformation at the end of a job
+- **(Bug Fix)** Amazon Q can update mvn and gradle build files
+- **(Bug Fix)** Fix doc generation for modules that are a part of the project
+- **(Bug Fix)** Amazon Q /dev: Remove hard-coded limits and instead rely server-side data to communicate number of code generations remaining
+- **(Bug Fix)** /transform: automatically open pre-build error logs when available
+- **(Bug Fix)** /doc: Fix code generation error when cancelling a documentation task
+- **(Bug Fix)** Amazon Q - update messaging for /doc agent
+
+# _3.47_ (2025-01-09)
+- **(Bug Fix)** Fix issue where users are unable to login to Amazon Q if they have previously authenticated ([#5214](https://github.com/aws/aws-toolkit-jetbrains/issues/5214))
+- **(Bug Fix)** Fix incorrect text shown while updating documentation in /doc
+- **(Bug Fix)** Amazon Q Code Transformation: retry initial project upload on failure
+- **(Bug Fix)** /transform: use correct doc link in SQL conversion help message
+- **(Bug Fix)** Amazon Q /dev: Fix issue when files are deleted while preparing context
+- **(Bug Fix)** Amazon Q /test: Test generation fails for files outside the project
+- **(Bug Fix)** Amazon Q Code Transformation: allow PostgreSQL as target DB for SQL conversions
+- **(Bug Fix)** Fix incorrect accept and reject buttons shows up while hovering over the generated file
+- **(Bug Fix)** Prevent customization override if user has manually selected a customization
+- **(Bug Fix)** Align UX text of document generation flow with vs code version
+
+# _3.46_ (2024-12-17)
+- **(Feature)** /review: Code fix automatically scrolls into view after generation.
+- **(Feature)** Chat: improve font size and line-height in footer (below prompt input field)
+- **(Feature)** Adds capability to send new context commands to AB groups
+- **(Bug Fix)** Chat: When writing a prompt without sending it, navigating via up/down arrows sometimes deletes the unsent prompt.
+- **(Bug Fix)** Fix chat not retaining history when interaction is through onboarding tab type ([#5189](https://github.com/aws/aws-toolkit-jetbrains/issues/5189))
+- **(Bug Fix)** Chat: When navigating to previous prompts, code attachments are sometimes displayed incorrectly
+- **(Bug Fix)** Reduce frequency of system information query
+
+# _3.45_ (2024-12-10)
+- **(Feature)** Add acknowledgement button for Amazon Q Chat disclaimer
+- **(Bug Fix)** Chosing cancel on sign out confirmation now cancels the sign out and does not delete profiles from ~/.aws/config ([#5167](https://github.com/aws/aws-toolkit-jetbrains/issues/5167))
+- **(Bug Fix)** Fix `@workspace` missing from the Amazon Q Chat welcome tab
+- **(Bug Fix)** Fix for /review LLM based code issues for file review on windows
+- **(Bug Fix)** Fix for File Review payload and Regex error for payload generation
+- **(Bug Fix)** Amazon Q Code Transformation: show build logs when server-side build fails
+
+# _3.44_ (2024-12-04)
+- **(Feature)** Amazon Q: UI improvements to chat: New splash loader animation, initial streaming card animation, improved button colours
+- **(Feature)** Amazon Q: Navigate through prompt history by using the up/down arrows
+- **(Bug Fix)** Fix issue where Amazon Q Code Transform is unable to start
+- **(Bug Fix)** Fix DynamoDB viewer throwing 'ActionGroup should be registered using tag' on IDE start ([#5012](https://github.com/aws/aws-toolkit-jetbrains/issues/5012)) ([#5120](https://github.com/aws/aws-toolkit-jetbrains/issues/5120))
+- **(Bug Fix)** Amazon Q: Fix chat syntax highlighting when using several different themes
+
+# _3.43_ (2024-12-03)
+- **(Feature)** `/review` in Q chat to scan your code for vulnerabilities and quality issues, and generate fixes
+- **(Feature)** `/test` in Q chat to generate unit tests for java and python
+- **(Feature)** `/doc` in Q chat to generate and update documentation for your project
+- **(Feature)** Added system notifications to inform users about critical plugin updates and potential issues with available workarounds
+
+# _3.42_ (2024-11-27)
+- **(Feature)** Amazon Q /dev: support `Dockerfile` files
+- **(Feature)** Feature(Amazon Q Code Transformation): allow users to view results in 5 smaller diffs
+- **(Feature)** Introduce @workspace command to enhance chat context fetching for Chat
+- **(Bug Fix)** Correct search text for Amazon Q inline suggestion keybindings
+- **(Bug Fix)** Fix(Amazon Q Code Transformation): always show user latest/correct transformation results
+- **(Bug Fix)** Amazon Q /dev: Fix error when accepting changes if leading slash is present.
+
# _3.41_ (2024-11-22)
- **(Feature)** Amazon Q /dev: support `.gradle` files
- **(Feature)** Inline Auto trigger will now happen more consistently and will not conflict with JetBrains code completion.
diff --git a/buildSrc/build.gradle.kts b/buildSrc/build.gradle.kts
index a8ffb0067f..45238ea7b2 100644
--- a/buildSrc/build.gradle.kts
+++ b/buildSrc/build.gradle.kts
@@ -34,8 +34,6 @@ dependencies {
testImplementation(libs.junit4)
testImplementation(libs.bundles.mockito)
testImplementation(gradleTestKit())
-
- testRuntimeOnly(libs.junit5.jupiterVintage)
}
tasks.test {
diff --git a/buildSrc/src/main/kotlin/toolkit-generate-sdks.gradle.kts b/buildSrc/src/main/kotlin/toolkit-generate-sdks.gradle.kts
index b051f7e913..314ff99426 100644
--- a/buildSrc/src/main/kotlin/toolkit-generate-sdks.gradle.kts
+++ b/buildSrc/src/main/kotlin/toolkit-generate-sdks.gradle.kts
@@ -38,6 +38,8 @@ java {
tasks.withType().configureEach {
options.encoding = "UTF-8"
+ sourceCompatibility = "17"
+ targetCompatibility = "17"
}
val generateTask = tasks.register("generateSdks")
diff --git a/buildSrc/src/main/kotlin/toolkit-publishing-conventions.gradle.kts b/buildSrc/src/main/kotlin/toolkit-publishing-conventions.gradle.kts
index 9eeb990c92..7b5994ab3e 100644
--- a/buildSrc/src/main/kotlin/toolkit-publishing-conventions.gradle.kts
+++ b/buildSrc/src/main/kotlin/toolkit-publishing-conventions.gradle.kts
@@ -66,3 +66,8 @@ tasks.verifyPlugin {
// give each instance its own home dir
systemProperty("plugin.verifier.home.dir", temporaryDir)
}
+
+val pluginZip by configurations.creating
+artifacts {
+ add("pluginZip", tasks.buildPlugin)
+}
diff --git a/buildSrc/src/main/kotlin/toolkit-testing.gradle.kts b/buildSrc/src/main/kotlin/toolkit-testing.gradle.kts
index 9c83f195e5..ca37a88d9c 100644
--- a/buildSrc/src/main/kotlin/toolkit-testing.gradle.kts
+++ b/buildSrc/src/main/kotlin/toolkit-testing.gradle.kts
@@ -20,10 +20,9 @@ dependencies {
// Everything uses junit4/5 except rider, which uses TestNG
testFixturesApi(platform(versionCatalog.findLibrary("junit5-bom").get()))
- testFixturesApi(versionCatalog.findLibrary("junit5-jupiterApi").get())
- testFixturesApi(versionCatalog.findLibrary("junit5-jupiterParams").get())
+ testFixturesApi(versionCatalog.findLibrary("junit5-jupiter").get())
- testRuntimeOnly(versionCatalog.findLibrary("junit5-jupiterEngine").get())
+ testRuntimeOnly(versionCatalog.findLibrary("junit-platform-launcher").get())
testRuntimeOnly(versionCatalog.findLibrary("junit5-jupiterVintage").get())
}
diff --git a/buildspec/linuxUiTests.yml b/buildspec/linuxUiTests.yml
index c200f631cc..607736f9f4 100644
--- a/buildspec/linuxUiTests.yml
+++ b/buildspec/linuxUiTests.yml
@@ -19,7 +19,7 @@ env:
phases:
install:
commands:
- - dnf install -y marco mate-media
+ - dnf install -y marco mate-media e2fsprogs
- startDesktop.sh
# login to DockerHub so we don't get throttled
@@ -45,10 +45,9 @@ phases:
credential_source=EcsContainer"
- chmod +x gradlew
- - ./gradlew -PideProfileName=$ALTERNATIVE_IDE_PROFILE_NAME :sandbox-all:prepareTestIdeUiSandbox --console plain --info
- ffmpeg -loglevel quiet -nostdin -f x11grab -video_size ${SCREEN_WIDTH}x${SCREEN_HEIGHT} -i ${DISPLAY} -codec:v libx264 -pix_fmt yuv420p -vf drawtext="fontsize=48:box=1:boxcolor=black@0.75:boxborderw=5:fontcolor=white:x=0:y=h-text_h:text='%{gmtime\:%H\\\\\:%M\\\\\:%S}'" -framerate 12 -g 12 /tmp/screen_recording.mp4 &
- - ./gradlew -PideProfileName=$ALTERNATIVE_IDE_PROFILE_NAME uiTestCore coverageReport --console plain --info
+ - ./gradlew -PideProfileName=$ALTERNATIVE_IDE_PROFILE_NAME :ui-tests-starter:test coverageReport --console plain --info
post_build:
commands:
diff --git a/detekt-rules/build.gradle.kts b/detekt-rules/build.gradle.kts
index aa42eb403d..045281b95a 100644
--- a/detekt-rules/build.gradle.kts
+++ b/detekt-rules/build.gradle.kts
@@ -15,7 +15,6 @@ dependencies {
// only used to make test work
testRuntimeOnly(libs.slf4j.api)
- testRuntimeOnly(libs.junit5.jupiterVintage)
}
tasks.test {
diff --git a/detekt-rules/detekt.yml b/detekt-rules/detekt.yml
index 3176ddae30..09fdafb5a9 100644
--- a/detekt-rules/detekt.yml
+++ b/detekt-rules/detekt.yml
@@ -72,7 +72,7 @@ formatting:
maxLineLength: 160
ignoreBackTickedIdentifier: true
NoWildcardImports:
- # no `packagesToUseImportOnDemandProperty` because we don't want to allow any star imports
+ packagesToUseImportOnDemandProperty: ""
active: true
ParameterListWrapping:
active: true
diff --git a/gradle.properties b/gradle.properties
index 3daceb7fcf..575fd640fb 100644
--- a/gradle.properties
+++ b/gradle.properties
@@ -2,13 +2,13 @@
# SPDX-License-Identifier: Apache-2.0
# Toolkit Version
-toolkitVersion=3.42-SNAPSHOT
+toolkitVersion=3.49-SNAPSHOT
# Publish Settings
publishToken=
publishChannel=
-ideProfileName=2024.1
+ideProfileName=2024.2
remoteRobotPort=8080
diff --git a/gradle/libs.versions.toml b/gradle/libs.versions.toml
index de7d410a66..7b55c86cec 100644
--- a/gradle/libs.versions.toml
+++ b/gradle/libs.versions.toml
@@ -1,6 +1,6 @@
[versions]
apache-commons-collections = "4.4"
-apache-commons-io = "2.16.0"
+apache-commons-io = "2.16.1"
apache-commons-text = "1.12.0"
assertJ = "3.26.3"
# match with /settings.gradle.kts
@@ -10,7 +10,7 @@ detekt = "1.23.7"
diff-util = "4.12"
intellijExt = "1.1.8"
# match with /settings.gradle.kts
-intellijGradle = "2.2.0-SNAPSHOT"
+intellijGradle = "2.2.1"
intellijRemoteRobot = "0.11.22"
jackson = "2.17.2"
jacoco = "0.8.12"
@@ -27,7 +27,7 @@ mockitoKotlin = "5.4.0"
mockk = "1.13.10"
nimbus-jose-jwt = "9.40"
node-gradle = "7.0.2"
-telemetryGenerator = "1.0.278"
+telemetryGenerator = "1.0.293"
testLogger = "4.0.0"
testRetry = "1.5.10"
# test-only; platform provides slf4j transitively at runtime
@@ -91,11 +91,13 @@ jackson-xml = { module = "com.fasterxml.jackson.dataformat:jackson-dataformat-xm
jackson-yaml = { module = "com.fasterxml.jackson.dataformat:jackson-dataformat-yaml", version.ref = "jackson" }
jacoco = { module = "org.jacoco:org.jacoco.core", version.ref = "jacoco" }
jgit = { module = "org.eclipse.jgit:org.eclipse.jgit", version.ref = "jgit" }
+
+# platfom launcher version selected by BOM
+junit-platform-launcher = { module = "org.junit.platform:junit-platform-launcher" }
+
junit4 = { module = "junit:junit", version.ref = "junit4" }
junit5-bom = { module = "org.junit:junit-bom", version.ref = "junit5" }
-junit5-jupiterApi = { module = "org.junit.jupiter:junit-jupiter-api", version.ref = "junit5" }
-junit5-jupiterEngine = { module = "org.junit.jupiter:junit-jupiter-engine", version.ref = "junit5" }
-junit5-jupiterParams = { module = "org.junit.jupiter:junit-jupiter-params", version.ref = "junit5" }
+junit5-jupiter = { module = "org.junit.jupiter:junit-jupiter", version.ref = "junit5" }
junit5-jupiterVintage = { module = "org.junit.vintage:junit-vintage-engine", version.ref = "junit5" }
kotlin-coroutines = { module = "org.jetbrains.kotlinx:kotlinx-coroutines-core", version.ref = "kotlinCoroutines" }
kotlin-coroutinesDebug = { module = "org.jetbrains.kotlinx:kotlinx-coroutines-debug", version.ref = "kotlinCoroutines" }
diff --git a/noop/build.gradle.kts b/noop/build.gradle.kts
new file mode 100644
index 0000000000..54de3e72a4
--- /dev/null
+++ b/noop/build.gradle.kts
@@ -0,0 +1,5 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+// project that does nothing
+tasks.register("test")
diff --git a/plugins/amazonq/chat/jetbrains-community/resources/META-INF/plugin-chat.xml b/plugins/amazonq/chat/jetbrains-community/resources/META-INF/plugin-chat.xml
index d08356231f..6949f4ba39 100644
--- a/plugins/amazonq/chat/jetbrains-community/resources/META-INF/plugin-chat.xml
+++ b/plugins/amazonq/chat/jetbrains-community/resources/META-INF/plugin-chat.xml
@@ -26,6 +26,9 @@
+
+
+
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/common/clients/AmazonQCodeGenerateClient.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/common/clients/AmazonQCodeGenerateClient.kt
new file mode 100644
index 0000000000..7741758d49
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/common/clients/AmazonQCodeGenerateClient.kt
@@ -0,0 +1,153 @@
+// Copyright 2023 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.common.clients
+
+import com.intellij.openapi.components.Service
+import com.intellij.openapi.components.service
+import com.intellij.openapi.project.Project
+import com.intellij.openapi.util.SystemInfo
+import software.amazon.awssdk.services.codewhispererruntime.CodeWhispererRuntimeClient
+import software.amazon.awssdk.services.codewhispererruntime.model.ArtifactType
+import software.amazon.awssdk.services.codewhispererruntime.model.ContentChecksumType
+import software.amazon.awssdk.services.codewhispererruntime.model.CreateTaskAssistConversationRequest
+import software.amazon.awssdk.services.codewhispererruntime.model.CreateTaskAssistConversationResponse
+import software.amazon.awssdk.services.codewhispererruntime.model.CreateUploadUrlResponse
+import software.amazon.awssdk.services.codewhispererruntime.model.DocGenerationEvent
+import software.amazon.awssdk.services.codewhispererruntime.model.GetTaskAssistCodeGenerationResponse
+import software.amazon.awssdk.services.codewhispererruntime.model.IdeCategory
+import software.amazon.awssdk.services.codewhispererruntime.model.OperatingSystem
+import software.amazon.awssdk.services.codewhispererruntime.model.OptOutPreference
+import software.amazon.awssdk.services.codewhispererruntime.model.SendTelemetryEventResponse
+import software.amazon.awssdk.services.codewhispererruntime.model.StartTaskAssistCodeGenerationResponse
+import software.amazon.awssdk.services.codewhispererruntime.model.TaskAssistPlanningUploadContext
+import software.amazon.awssdk.services.codewhispererruntime.model.UploadContext
+import software.amazon.awssdk.services.codewhispererruntime.model.UploadIntent
+import software.amazon.awssdk.services.codewhispererruntime.model.UserContext
+import software.amazon.awssdk.services.codewhispererstreaming.model.ExportIntent
+import software.aws.toolkits.core.utils.error
+import software.aws.toolkits.core.utils.getLogger
+import software.aws.toolkits.core.utils.info
+import software.aws.toolkits.jetbrains.common.session.Intent
+import software.aws.toolkits.jetbrains.core.awsClient
+import software.aws.toolkits.jetbrains.core.credentials.ToolkitConnectionManager
+import software.aws.toolkits.jetbrains.core.credentials.pinning.QConnection
+import software.aws.toolkits.jetbrains.services.amazonq.clients.AmazonQStreamingClient
+import software.aws.toolkits.jetbrains.services.amazonqDoc.FEATURE_EVALUATION_PRODUCT_NAME
+import software.aws.toolkits.jetbrains.services.codemodernizer.utils.calculateTotalLatency
+import software.aws.toolkits.jetbrains.services.telemetry.ClientMetadata
+import software.aws.toolkits.jetbrains.settings.AwsSettings
+import java.time.Instant
+import software.amazon.awssdk.services.codewhispererruntime.model.ChatTriggerType as SyncChatTriggerType
+
+@Service(Service.Level.PROJECT)
+class AmazonQCodeGenerateClient(private val project: Project) {
+ private fun getTelemetryOptOutPreference() =
+ if (AwsSettings.getInstance().isTelemetryEnabled) {
+ OptOutPreference.OPTIN
+ } else {
+ OptOutPreference.OPTOUT
+ }
+
+ private val docGenerationUserContext = ClientMetadata.getDefault().let {
+ val osForFeatureDev: OperatingSystem =
+ when {
+ SystemInfo.isWindows -> OperatingSystem.WINDOWS
+ SystemInfo.isMac -> OperatingSystem.MAC
+ // For now, categorize everything else as "Linux" (Linux/FreeBSD/Solaris/etc.)
+ else -> OperatingSystem.LINUX
+ }
+
+ UserContext.builder()
+ .ideCategory(IdeCategory.JETBRAINS)
+ .operatingSystem(osForFeatureDev)
+ .product(FEATURE_EVALUATION_PRODUCT_NAME)
+ .clientId(it.clientId)
+ .ideVersion(it.awsVersion)
+ .build()
+ }
+
+ fun connection() = ToolkitConnectionManager.getInstance(project).activeConnectionForFeature(QConnection.getInstance())
+ ?: error("Attempted to use connection while one does not exist")
+
+ fun bearerClient() = connection().getConnectionSettings().awsClient()
+
+ private val amazonQStreamingClient
+ get() = AmazonQStreamingClient.getInstance(project)
+
+ fun sendDocGenerationTelemetryEvent(
+ docGenerationEvent: DocGenerationEvent,
+ ): SendTelemetryEventResponse =
+ bearerClient().sendTelemetryEvent { requestBuilder ->
+ requestBuilder.telemetryEvent { telemetryEventBuilder ->
+ telemetryEventBuilder.docGenerationEvent(docGenerationEvent)
+ }
+ requestBuilder.optOutPreference(getTelemetryOptOutPreference())
+ requestBuilder.userContext(docGenerationUserContext)
+ }
+
+ fun createTaskAssistConversation(): CreateTaskAssistConversationResponse = bearerClient().createTaskAssistConversation(
+ CreateTaskAssistConversationRequest.builder().build()
+ )
+
+ fun createTaskAssistUploadUrl(conversationId: String, contentChecksumSha256: String, contentLength: Long): CreateUploadUrlResponse =
+ bearerClient().createUploadUrl {
+ it.contentChecksumType(ContentChecksumType.SHA_256)
+ .contentChecksum(contentChecksumSha256)
+ .contentLength(contentLength)
+ .artifactType(ArtifactType.SOURCE_CODE)
+ .uploadIntent(UploadIntent.TASK_ASSIST_PLANNING)
+ .uploadContext(
+ UploadContext.builder()
+ .taskAssistPlanningUploadContext(
+ TaskAssistPlanningUploadContext.builder()
+ .conversationId(conversationId)
+ .build()
+ )
+ .build()
+ )
+ }
+
+ fun startTaskAssistCodeGeneration(conversationId: String, uploadId: String, userMessage: String, intent: Intent): StartTaskAssistCodeGenerationResponse =
+ bearerClient()
+ .startTaskAssistCodeGeneration { request ->
+ request
+ .conversationState {
+ it
+ .conversationId(conversationId)
+ .chatTriggerType(SyncChatTriggerType.MANUAL)
+ .currentMessage { cm -> cm.userInputMessage { um -> um.content(userMessage) } }
+ }
+ .workspaceState {
+ it
+ .programmingLanguage { pl -> pl.languageName("javascript") }
+ .uploadId(uploadId)
+ }
+ .intent(intent.name)
+ }
+
+ fun getTaskAssistCodeGeneration(conversationId: String, codeGenerationId: String): GetTaskAssistCodeGenerationResponse = bearerClient()
+ .getTaskAssistCodeGeneration {
+ it
+ .conversationId(conversationId)
+ .codeGenerationId(codeGenerationId)
+ }
+
+ suspend fun exportTaskAssistResultArchive(conversationId: String): MutableList = amazonQStreamingClient.exportResultArchive(
+ conversationId,
+ ExportIntent.TASK_ASSIST,
+ null,
+ { e ->
+ LOG.error(e) { "TaskAssist - ExportResultArchive stream exportId=$conversationId exportIntent=${ExportIntent.TASK_ASSIST} Failed: ${e.message} " }
+ },
+ { startTime ->
+ LOG.info { "TaskAssist - ExportResultArchive latency: ${calculateTotalLatency(startTime, Instant.now())}" }
+ }
+ )
+
+ companion object {
+ private val LOG = getLogger()
+
+ fun getInstance(project: Project) = project.service()
+ }
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/common/session/ConversationNotStartedState.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/common/session/ConversationNotStartedState.kt
new file mode 100644
index 0000000000..ad300449df
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/common/session/ConversationNotStartedState.kt
@@ -0,0 +1,21 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.common.session
+
+import software.aws.toolkits.jetbrains.services.amazonqDoc.session.SessionStateInteraction
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.SessionStateAction
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.SessionStatePhase
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.util.CancellationTokenSource
+
+class ConversationNotStartedState(
+ override var approach: String,
+ override val tabID: String,
+ override var token: CancellationTokenSource?,
+) : SessionState {
+ override val phase = SessionStatePhase.INIT
+
+ override suspend fun interact(action: SessionStateAction): SessionStateInteraction {
+ error("Illegal transition between states, restart the conversation")
+ }
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/common/session/SessionState.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/common/session/SessionState.kt
new file mode 100644
index 0000000000..1585096382
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/common/session/SessionState.kt
@@ -0,0 +1,17 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.common.session
+
+import software.aws.toolkits.jetbrains.services.amazonqDoc.session.SessionStateInteraction
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.SessionStateAction
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.SessionStatePhase
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.util.CancellationTokenSource
+
+interface SessionState {
+ val tabID: String
+ val phase: SessionStatePhase?
+ var token: CancellationTokenSource?
+ var approach: String
+ suspend fun interact(action: SessionStateAction): SessionStateInteraction
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/common/session/SessionStateTypes.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/common/session/SessionStateTypes.kt
new file mode 100644
index 0000000000..3b50f10ca9
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/common/session/SessionStateTypes.kt
@@ -0,0 +1,24 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.common.session
+
+import software.aws.toolkits.jetbrains.common.util.AmazonQCodeGenService
+import software.aws.toolkits.jetbrains.services.amazonq.FeatureDevSessionContext
+
+open class SessionStateConfig(
+ open val conversationId: String,
+ open val repoContext: FeatureDevSessionContext,
+ open val amazonQCodeGenService: AmazonQCodeGenService,
+)
+
+data class SessionStateConfigData(
+ override val conversationId: String,
+ override val repoContext: FeatureDevSessionContext,
+ override val amazonQCodeGenService: AmazonQCodeGenService,
+) : SessionStateConfig(conversationId, repoContext, amazonQCodeGenService)
+
+enum class Intent {
+ DEV,
+ DOC,
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/common/util/AmazonQCodeGenService.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/common/util/AmazonQCodeGenService.kt
new file mode 100644
index 0000000000..afe9da01b2
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/common/util/AmazonQCodeGenService.kt
@@ -0,0 +1,214 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.common.util
+
+import com.fasterxml.jackson.module.kotlin.jacksonObjectMapper
+import com.fasterxml.jackson.module.kotlin.readValue
+import com.intellij.openapi.project.Project
+import software.amazon.awssdk.services.codewhispererruntime.model.CodeWhispererRuntimeException
+import software.amazon.awssdk.services.codewhispererruntime.model.CreateUploadUrlResponse
+import software.amazon.awssdk.services.codewhispererruntime.model.GetTaskAssistCodeGenerationResponse
+import software.amazon.awssdk.services.codewhispererruntime.model.ServiceQuotaExceededException
+import software.amazon.awssdk.services.codewhispererruntime.model.StartTaskAssistCodeGenerationResponse
+import software.amazon.awssdk.services.codewhispererruntime.model.ThrottlingException
+import software.amazon.awssdk.services.codewhispererruntime.model.ValidationException
+import software.amazon.awssdk.services.codewhispererstreaming.model.CodeWhispererStreamingException
+import software.aws.toolkits.core.utils.debug
+import software.aws.toolkits.core.utils.error
+import software.aws.toolkits.core.utils.getLogger
+import software.aws.toolkits.core.utils.warn
+import software.aws.toolkits.jetbrains.common.clients.AmazonQCodeGenerateClient
+import software.aws.toolkits.jetbrains.common.session.Intent
+import software.aws.toolkits.jetbrains.services.amazonqDoc.session.DocGenerationStreamResult
+import software.aws.toolkits.jetbrains.services.amazonqDoc.session.ExportDocTaskAssistResultArchiveStreamResult
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.CodeIterationLimitException
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.ContentLengthException
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.ExportParseException
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.FEATURE_NAME
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.FeatureDevException
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.FeatureDevOperation
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.MonthlyConversationLimitError
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.ZipFileCorruptedException
+import software.aws.toolkits.jetbrains.services.cwc.controller.chat.telemetry.getStartUrl
+import software.aws.toolkits.telemetry.AmazonqTelemetry
+import software.aws.toolkits.telemetry.MetricResult
+
+class AmazonQCodeGenService(val proxyClient: AmazonQCodeGenerateClient, val project: Project) {
+ fun createConversation(): String {
+ val startTime = System.currentTimeMillis()
+ var failureReason: String? = null
+ var failureReasonDesc: String? = null
+ var result: MetricResult = MetricResult.Succeeded
+ var conversationId: String? = null
+ try {
+ logger.debug { "Executing createTaskAssistConversation" }
+ val taskAssistConversationResult = proxyClient.createTaskAssistConversation()
+ conversationId = taskAssistConversationResult.conversationId()
+ logger.debug {
+ "$FEATURE_NAME: Created conversation: {conversationId: $conversationId, requestId: ${
+ taskAssistConversationResult.responseMetadata().requestId()
+ }"
+ }
+
+ return conversationId
+ } catch (e: Exception) {
+ logger.warn(e) { "$FEATURE_NAME: Failed to start conversation: ${e.message}" }
+ result = MetricResult.Failed
+ failureReason = e.javaClass.simpleName
+ if (e is FeatureDevException) {
+ failureReason = e.reason()
+ failureReasonDesc = e.reasonDesc()
+ }
+ var errMssg = e.message
+ if (e is CodeWhispererRuntimeException) {
+ errMssg = e.awsErrorDetails().errorMessage()
+ logger.warn(e) { "Start conversation failed for request: ${e.requestId()}" }
+
+ if (e is ServiceQuotaExceededException) {
+ throw MonthlyConversationLimitError(errMssg, operation = FeatureDevOperation.CreateConversation.toString(), desc = null, cause = e.cause)
+ }
+ }
+ throw FeatureDevException(errMssg, operation = FeatureDevOperation.CreateConversation.toString(), desc = null, e.cause)
+ } finally {
+ AmazonqTelemetry.startConversationInvoke(
+ amazonqConversationId = conversationId,
+ result = result,
+ reason = failureReason,
+ reasonDesc = failureReasonDesc,
+ duration = (System.currentTimeMillis() - startTime).toDouble(),
+ credentialStartUrl = getStartUrl(project = this.project),
+ )
+ }
+ }
+
+ fun createUploadUrl(conversationId: String, contentChecksumSha256: String, contentLength: Long, uploadId: String):
+ CreateUploadUrlResponse {
+ try {
+ logger.debug { "Executing createUploadUrl with conversationId $conversationId" }
+ val uploadUrlResponse = proxyClient.createTaskAssistUploadUrl(
+ conversationId,
+ contentChecksumSha256,
+ contentLength,
+ )
+ logger.debug {
+ "$FEATURE_NAME: Created upload url: {uploadId: $uploadId, requestId: ${uploadUrlResponse.responseMetadata().requestId()}}"
+ }
+ return uploadUrlResponse
+ } catch (e: Exception) {
+ logger.warn(e) { "$FEATURE_NAME: Failed to generate presigned url: ${e.message}" }
+
+ var errMssg = e.message
+ if (e is CodeWhispererRuntimeException) {
+ errMssg = e.awsErrorDetails().errorMessage()
+ logger.warn(e) { "Create UploadUrl failed for request: ${e.requestId()}" }
+
+ if (e is ValidationException && e.message?.contains("Invalid contentLength") == true) {
+ throw ContentLengthException(operation = FeatureDevOperation.CreateUploadUrl.toString(), desc = null, cause = e.cause)
+ }
+ }
+ throw FeatureDevException(errMssg, operation = FeatureDevOperation.CreateUploadUrl.toString(), desc = null, e.cause)
+ }
+ }
+
+ open fun startTaskAssistCodeGeneration(conversationId: String, uploadId: String, message: String, intent: Intent):
+ StartTaskAssistCodeGenerationResponse {
+ try {
+ logger.debug { "Executing startTaskAssistCodeGeneration with conversationId: $conversationId , uploadId: $uploadId" }
+ val startCodeGenerationResponse = proxyClient.startTaskAssistCodeGeneration(
+ conversationId,
+ uploadId,
+ message,
+ intent
+ )
+
+ logger.debug { "$FEATURE_NAME: Started code generation with requestId: ${startCodeGenerationResponse.responseMetadata().requestId()}" }
+ return startCodeGenerationResponse
+ } catch (e: Exception) {
+ logger.warn(e) { "$FEATURE_NAME: Failed to execute startTaskAssistCodeGeneration ${e.message}" }
+
+ var errMssg = e.message
+ if (e is CodeWhispererRuntimeException) {
+ errMssg = e.awsErrorDetails().errorMessage()
+ logger.warn(e) { "StartTaskAssistCodeGeneration failed for request: ${e.requestId()}" }
+
+ // API Front-end will throw Throttling if conversation limit is reached. API Front-end monitors StartCodeGeneration for throttling
+ if (e is software.amazon.awssdk.services.codewhispererruntime.model.ThrottlingException &&
+ e.message?.contains("StartTaskAssistCodeGeneration reached for this month.") == true
+ ) {
+ throw MonthlyConversationLimitError(errMssg, operation = FeatureDevOperation.StartTaskAssistCodeGeneration.toString(), desc = null, e.cause)
+ }
+
+ if (e is ServiceQuotaExceededException || (
+ e is ThrottlingException && e.message?.contains(
+ "limit for number of iterations on a code generation"
+ ) == true
+ )
+ ) {
+ throw CodeIterationLimitException(operation = FeatureDevOperation.StartTaskAssistCodeGeneration.toString(), desc = null, e.cause)
+ } else if (e is ValidationException && e.message?.contains("repo size is exceeding the limits") == true) {
+ throw ContentLengthException(operation = FeatureDevOperation.StartTaskAssistCodeGeneration.toString(), desc = null, cause = e.cause)
+ } else if (e is ValidationException && e.message?.contains("zipped file is corrupted") == true) {
+ throw ZipFileCorruptedException(operation = FeatureDevOperation.StartTaskAssistCodeGeneration.toString(), desc = null, e.cause)
+ }
+ }
+ throw FeatureDevException(errMssg, operation = FeatureDevOperation.StartTaskAssistCodeGeneration.toString(), desc = null, e.cause)
+ }
+ }
+
+ fun getTaskAssistCodeGeneration(conversationId: String, codeGenerationId: String): GetTaskAssistCodeGenerationResponse {
+ try {
+ logger.debug { "Executing GetTaskAssistCodeGeneration with conversationId: $conversationId , codeGenerationId: $codeGenerationId" }
+ val getCodeGenerationResponse = proxyClient.getTaskAssistCodeGeneration(conversationId, codeGenerationId)
+
+ logger.debug {
+ "$FEATURE_NAME: Received code generation status $getCodeGenerationResponse with requestId ${
+ getCodeGenerationResponse.responseMetadata()
+ .requestId()
+ }"
+ }
+ return getCodeGenerationResponse
+ } catch (e: Exception) {
+ logger.warn(e) { "$FEATURE_NAME: Failed to execute GetTaskAssistCodeGeneration ${e.message}" }
+
+ var errMssg = e.message
+ if (e is CodeWhispererRuntimeException) {
+ errMssg = e.awsErrorDetails().errorMessage()
+ logger.warn(e) { "GetTaskAssistCodeGeneration failed for request: ${e.requestId()}" }
+ }
+ throw FeatureDevException(errMssg, operation = FeatureDevOperation.GetTaskAssistCodeGeneration.toString(), desc = null, e.cause)
+ }
+ }
+
+ suspend fun exportTaskAssistArchiveResult(conversationId: String): DocGenerationStreamResult {
+ val exportResponse: MutableList
+ try {
+ exportResponse = proxyClient.exportTaskAssistResultArchive(conversationId)
+ logger.debug { "$FEATURE_NAME: Received export task assist result archive response" }
+ } catch (e: Exception) {
+ logger.warn(e) { "$FEATURE_NAME: Failed to export archive result: ${e.message}" }
+
+ var errMssg = e.message
+ if (e is CodeWhispererStreamingException) {
+ errMssg = e.awsErrorDetails().errorMessage()
+ logger.warn(e) { "ExportTaskAssistArchiveResult failed for request: ${e.requestId()}" }
+ }
+ throw FeatureDevException(errMssg, operation = FeatureDevOperation.ExportTaskAssistArchiveResult.toString(), desc = null, e.cause)
+ }
+
+ val parsedResult: ExportDocTaskAssistResultArchiveStreamResult
+ try {
+ val result = exportResponse.reduce { acc, next -> acc + next } // To map the result it is needed to combine the full byte array
+ parsedResult = jacksonObjectMapper().readValue(result)
+ } catch (e: Exception) {
+ logger.error(e) { "Failed to parse downloaded code results" }
+ throw ExportParseException(operation = FeatureDevOperation.ExportTaskAssistArchiveResult.toString(), desc = null, e.cause)
+ }
+
+ return parsedResult.codeGenerationResult
+ }
+
+ companion object {
+ private val logger = getLogger()
+ }
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/common/util/FileUtils.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/common/util/FileUtils.kt
new file mode 100644
index 0000000000..6cf8cc2476
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/common/util/FileUtils.kt
@@ -0,0 +1,82 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.common.util
+
+import com.github.difflib.DiffUtils
+import com.github.difflib.patch.DeltaType
+import com.intellij.openapi.fileChooser.FileChooser
+import com.intellij.openapi.fileChooser.FileChooserDescriptorFactory
+import com.intellij.openapi.project.Project
+import com.intellij.openapi.vfs.CharsetToolkit
+import com.intellij.openapi.vfs.VirtualFile
+import java.io.File
+import java.nio.charset.Charset
+import java.nio.file.Path
+import kotlin.io.path.createDirectories
+import kotlin.io.path.deleteIfExists
+import kotlin.io.path.writeBytes
+
+fun resolveAndCreateOrUpdateFile(projectRootPath: Path, relativeFilePath: String, fileContent: String) {
+ val filePath = projectRootPath.resolve(relativeFilePath)
+ filePath.parent.createDirectories() // Create directories if needed
+ filePath.writeBytes(fileContent.toByteArray(Charsets.UTF_8))
+}
+
+fun resolveAndDeleteFile(projectRootPath: Path, relativePath: String) {
+ val filePath = projectRootPath.resolve(relativePath)
+ filePath.deleteIfExists()
+}
+
+fun selectFolder(project: Project, openOn: VirtualFile): VirtualFile? {
+ val fileChooserDescriptor = FileChooserDescriptorFactory.createSingleFolderDescriptor()
+ return FileChooser.chooseFile(fileChooserDescriptor, project, openOn)
+}
+
+fun readFileToString(file: File): String {
+ val charsetToolkit = CharsetToolkit(file.readBytes(), Charset.forName("UTF-8"), false)
+ val charset = charsetToolkit.guessEncoding(4096)
+ return file.readText(charset)
+}
+
+/**
+ * Calculates the number of added characters and lines between existing content and LLM response
+ *
+ * @param existingContent The original text content before changes
+ * @param llmResponse The new text content from the LLM
+ * @return A Map containing:
+ * - "addedChars": Total number of new characters added
+ * - "addedLines": Total number of new lines added
+ */
+data class DiffResult(val addedChars: Int, val addedLines: Int)
+
+fun getDiffCharsAndLines(
+ existingContent: String,
+ llmResponse: String,
+): DiffResult {
+ var addedChars = 0
+ var addedLines = 0
+
+ val existingLines = existingContent.lines()
+ val llmLines = llmResponse.lines()
+
+ val patch = DiffUtils.diff(existingLines, llmLines)
+
+ for (delta in patch.deltas) {
+ when (delta.type) {
+ DeltaType.INSERT -> {
+ addedChars += delta.target.lines.sumOf { it.length }
+ addedLines += delta.target.lines.size
+ }
+
+ DeltaType.CHANGE -> {
+ addedChars += delta.target.lines.sumOf { it.length }
+ addedLines += delta.target.lines.size
+ }
+
+ else -> {} // Do nothing for DELETE
+ }
+ }
+
+ return DiffResult(addedChars, addedLines)
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/QOpenPanelAction.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/QOpenPanelAction.kt
index 95fbabd3a4..de01e99bad 100644
--- a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/QOpenPanelAction.kt
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/QOpenPanelAction.kt
@@ -9,6 +9,8 @@ import com.intellij.openapi.actionSystem.CommonDataKeys
import com.intellij.openapi.wm.ToolWindowManager
import icons.AwsIcons
import software.aws.toolkits.jetbrains.services.amazonq.toolwindow.AMAZON_Q_WINDOW_ID
+import software.aws.toolkits.jetbrains.services.amazonq.toolwindow.AmazonQToolWindow
+import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererConstants.runScanKey
import software.aws.toolkits.jetbrains.utils.isRunningOnRemoteBackend
import software.aws.toolkits.resources.message
import software.aws.toolkits.telemetry.UiTelemetry
@@ -19,5 +21,8 @@ class QOpenPanelAction : AnAction(message("action.q.openchat.text"), null, AwsIc
val project = e.getRequiredData(CommonDataKeys.PROJECT)
UiTelemetry.click(project, "q_openChat")
ToolWindowManager.getInstance(project).getToolWindow(AMAZON_Q_WINDOW_ID)?.activate(null, true)
+ if (e.getData(runScanKey) == true) {
+ AmazonQToolWindow.openScanTab(project)
+ }
}
}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/toolwindow/AmazonQToolWindow.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/toolwindow/AmazonQToolWindow.kt
index e5673cc6da..15a533d119 100644
--- a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/toolwindow/AmazonQToolWindow.kt
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/toolwindow/AmazonQToolWindow.kt
@@ -24,9 +24,14 @@ import software.aws.toolkits.jetbrains.services.amazonq.messages.AmazonQMessage
import software.aws.toolkits.jetbrains.services.amazonq.messages.MessageConnector
import software.aws.toolkits.jetbrains.services.amazonq.onboarding.OnboardingPageInteraction
import software.aws.toolkits.jetbrains.services.amazonq.onboarding.OnboardingPageInteractionType
+import software.aws.toolkits.jetbrains.services.amazonq.util.highlightCommand
import software.aws.toolkits.jetbrains.services.amazonq.webview.BrowserConnector
import software.aws.toolkits.jetbrains.services.amazonq.webview.FqnWebviewAdapter
import software.aws.toolkits.jetbrains.services.amazonq.webview.theme.EditorThemeAdapter
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.auth.isCodeScanAvailable
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.runCodeScanMessage
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.auth.isCodeTestAvailable
+import software.aws.toolkits.jetbrains.services.amazonqDoc.auth.isDocAvailable
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.auth.isFeatureDevAvailable
import software.aws.toolkits.jetbrains.services.codemodernizer.utils.isCodeTransformAvailable
import javax.swing.JComponent
@@ -72,6 +77,14 @@ class AmazonQToolWindow private constructor(
}
}
+ private fun sendMessageAppToUi(message: AmazonQMessage, tabType: String) {
+ appConnections.filter { it.app.tabTypes.contains(tabType) }.forEach {
+ scope.launch {
+ it.messagesFromAppToUi.publish(message)
+ }
+ }
+ }
+
private fun initConnections() {
val apps = appSource.getApps(project)
apps.forEach { app ->
@@ -110,7 +123,11 @@ class AmazonQToolWindow private constructor(
chatBrowser.init(
isCodeTransformAvailable = isCodeTransformAvailable(project),
- isFeatureDevAvailable = isFeatureDevAvailable(project)
+ isFeatureDevAvailable = isFeatureDevAvailable(project),
+ isCodeScanAvailable = isCodeScanAvailable(project),
+ isCodeTestAvailable = isCodeTestAvailable(project),
+ isDocAvailable = isDocAvailable(project),
+ highlightCommand = highlightCommand()
)
scope.launch {
@@ -134,17 +151,25 @@ class AmazonQToolWindow private constructor(
companion object {
fun getInstance(project: Project): AmazonQToolWindow = project.service()
+ private fun showChatWindow(project: Project) = runInEdt {
+ val toolWindow = ToolWindowManager.getInstance(project).getToolWindow(AmazonQToolWindowFactory.WINDOW_ID)
+ toolWindow?.show()
+ }
+
fun getStarted(project: Project) {
// Make sure the window is shown
- runInEdt {
- val toolWindow = ToolWindowManager.getInstance(project).getToolWindow(AmazonQToolWindowFactory.WINDOW_ID)
- toolWindow?.show()
- }
+ showChatWindow(project)
// Send the interaction message
val window = getInstance(project)
window.sendMessage(OnboardingPageInteraction(OnboardingPageInteractionType.CwcButtonClick), "cwc")
}
+
+ fun openScanTab(project: Project) {
+ showChatWindow(project)
+ val window = getInstance(project)
+ window.sendMessageAppToUi(runCodeScanMessage, tabType = "codescan")
+ }
}
override fun dispose() {
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/toolwindow/AmazonQToolWindowFactory.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/toolwindow/AmazonQToolWindowFactory.kt
index e31a74e57d..f540bfeca8 100644
--- a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/toolwindow/AmazonQToolWindowFactory.kt
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/toolwindow/AmazonQToolWindowFactory.kt
@@ -10,8 +10,8 @@ import com.intellij.openapi.project.Project
import com.intellij.openapi.wm.ToolWindow
import com.intellij.openapi.wm.ToolWindowFactory
import com.intellij.openapi.wm.ex.ToolWindowEx
-import com.intellij.ui.content.Content
-import com.intellij.ui.content.ContentManager
+import com.intellij.ui.components.panels.Wrapper
+import com.intellij.util.ui.components.BorderLayoutPanel
import software.aws.toolkits.core.utils.debug
import software.aws.toolkits.core.utils.getLogger
import software.aws.toolkits.jetbrains.core.credentials.AwsBearerTokenConnection
@@ -22,6 +22,8 @@ import software.aws.toolkits.jetbrains.core.credentials.pinning.QConnection
import software.aws.toolkits.jetbrains.core.credentials.sono.Q_SCOPES
import software.aws.toolkits.jetbrains.core.credentials.sso.bearer.BearerTokenAuthState
import software.aws.toolkits.jetbrains.core.credentials.sso.bearer.BearerTokenProviderListener
+import software.aws.toolkits.jetbrains.core.notifications.NotificationPanel
+import software.aws.toolkits.jetbrains.core.notifications.ProcessNotificationsBase
import software.aws.toolkits.jetbrains.core.webview.BrowserState
import software.aws.toolkits.jetbrains.services.amazonq.QWebviewPanel
import software.aws.toolkits.jetbrains.services.amazonq.RefreshQChatPanelButtonPressedListener
@@ -35,7 +37,21 @@ import java.awt.event.ComponentAdapter
import java.awt.event.ComponentEvent
class AmazonQToolWindowFactory : ToolWindowFactory, DumbAware {
+
override fun createToolWindowContent(project: Project, toolWindow: ToolWindow) {
+ val mainPanel = BorderLayoutPanel()
+ val qPanel = Wrapper()
+ val notificationPanel = NotificationPanel()
+
+ mainPanel.addToTop(notificationPanel)
+ mainPanel.add(qPanel)
+ val notifListener = ProcessNotificationsBase.getInstance(project)
+ notifListener.addListenerForNotification { bannerContent ->
+ runInEdt {
+ notificationPanel.updateNotificationPanel(bannerContent)
+ }
+ }
+
if (toolWindow is ToolWindowEx) {
val actionManager = ActionManager.getInstance()
toolWindow.setTitleActions(listOf(actionManager.getAction("aws.q.toolwindow.titleBar")))
@@ -46,7 +62,7 @@ class AmazonQToolWindowFactory : ToolWindowFactory, DumbAware {
ToolkitConnectionManagerListener.TOPIC,
object : ToolkitConnectionManagerListener {
override fun activeConnectionChanged(newConnection: ToolkitConnection?) {
- onConnectionChanged(project, newConnection, toolWindow)
+ onConnectionChanged(project, newConnection, qPanel)
}
}
)
@@ -56,8 +72,7 @@ class AmazonQToolWindowFactory : ToolWindowFactory, DumbAware {
object : RefreshQChatPanelButtonPressedListener {
override fun onRefresh() {
runInEdt {
- contentManager.removeAllContents(true)
- prepareChatContent(project, contentManager)
+ prepareChatContent(project, qPanel)
}
}
}
@@ -68,43 +83,37 @@ class AmazonQToolWindowFactory : ToolWindowFactory, DumbAware {
object : BearerTokenProviderListener {
override fun onChange(providerId: String, newScopes: List?) {
if (ToolkitConnectionManager.getInstance(project).connectionStateForFeature(QConnection.getInstance()) == BearerTokenAuthState.AUTHORIZED) {
- val content = contentManager.factory.createContent(AmazonQToolWindow.getInstance(project).component, null, false).also {
- it.isCloseable = true
- it.isPinnable = true
- }
+ val qComponent = AmazonQToolWindow.getInstance(project).component
runInEdt {
- contentManager.removeAllContents(true)
- contentManager.addContent(content)
+ qPanel.setContent(qComponent)
}
}
}
}
)
- val content = prepareChatContent(project, contentManager)
+ prepareChatContent(project, qPanel)
+ val content = contentManager.factory.createContent(mainPanel, null, false).also {
+ it.isCloseable = true
+ it.isPinnable = true
+ }
toolWindow.activate(null)
- contentManager.setSelectedContent(content)
+ contentManager.addContent(content)
}
private fun prepareChatContent(
project: Project,
- contentManager: ContentManager,
- ): Content {
+ qPanel: Wrapper,
+ ) {
val component = if (isQConnected(project) && !isQExpired(project)) {
AmazonQToolWindow.getInstance(project).component
} else {
QWebviewPanel.getInstance(project).browser?.prepareBrowser(BrowserState(FeatureId.AmazonQ))
QWebviewPanel.getInstance(project).component
}
-
- val content = contentManager.factory.createContent(component, null, false).also {
- it.isCloseable = true
- it.isPinnable = true
- }
- contentManager.addContent(content)
- return content
+ qPanel.setContent(component)
}
override fun init(toolWindow: ToolWindow) {
@@ -125,8 +134,7 @@ class AmazonQToolWindowFactory : ToolWindowFactory, DumbAware {
override fun shouldBeAvailable(project: Project): Boolean = isQWebviewsAvailable()
- private fun onConnectionChanged(project: Project, newConnection: ToolkitConnection?, toolWindow: ToolWindow) {
- val contentManager = toolWindow.contentManager
+ private fun onConnectionChanged(project: Project, newConnection: ToolkitConnection?, qPanel: Wrapper) {
val isNewConnectionForQ = newConnection?.let {
(it as? AwsBearerTokenConnection)?.let { conn ->
val scopeShouldHave = Q_SCOPES
@@ -151,15 +159,8 @@ class AmazonQToolWindowFactory : ToolWindowFactory, DumbAware {
LOG.debug { "returning login window; no Q connection found" }
QWebviewPanel.getInstance(project).component
}
-
- val content = contentManager.factory.createContent(component, null, false).also {
- it.isCloseable = true
- it.isPinnable = true
- }
-
runInEdt {
- contentManager.removeAllContents(true)
- contentManager.addContent(content)
+ qPanel.setContent(component)
}
}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/toolwindow/OuterAmazonQPanel.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/toolwindow/OuterAmazonQPanel.kt
new file mode 100644
index 0000000000..72524e49b2
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/toolwindow/OuterAmazonQPanel.kt
@@ -0,0 +1,39 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonq.toolwindow
+
+import com.intellij.openapi.project.Project
+import com.intellij.ui.components.panels.Wrapper
+import com.intellij.util.ui.components.BorderLayoutPanel
+import software.aws.toolkits.core.utils.error
+import software.aws.toolkits.core.utils.getLogger
+import software.aws.toolkits.jetbrains.core.webview.BrowserState
+import software.aws.toolkits.jetbrains.services.amazonq.QWebviewPanel
+import software.aws.toolkits.jetbrains.utils.isQConnected
+import software.aws.toolkits.jetbrains.utils.isQExpired
+import software.aws.toolkits.telemetry.FeatureId
+import javax.swing.JComponent
+
+class OuterAmazonQPanel(val project: Project) : BorderLayoutPanel() {
+ private val wrapper = Wrapper()
+ init {
+ isOpaque = false
+ addToCenter(wrapper)
+ val component = if (isQConnected(project) && !isQExpired(project)) {
+ AmazonQToolWindow.getInstance(project).component
+ } else {
+ QWebviewPanel.getInstance(project).browser?.prepareBrowser(BrowserState(FeatureId.AmazonQ))
+ QWebviewPanel.getInstance(project).component
+ }
+ updateQPanel(component)
+ }
+
+ fun updateQPanel(content: JComponent) {
+ try {
+ wrapper.setContent(content)
+ } catch (e: Exception) {
+ getLogger().error { "Error while creating window" }
+ }
+ }
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/util/HighlightCommand.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/util/HighlightCommand.kt
new file mode 100644
index 0000000000..5019d051c1
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/util/HighlightCommand.kt
@@ -0,0 +1,16 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonq.util
+
+import software.aws.toolkits.jetbrains.services.amazonq.CodeWhispererFeatureConfigService
+
+data class HighlightCommand(val command: String, val description: String)
+
+fun highlightCommand(): HighlightCommand? {
+ val feature = CodeWhispererFeatureConfigService.getInstance().getHighlightCommandFeature()
+
+ if (feature == null || feature.value.stringValue().isEmpty()) return null
+
+ return HighlightCommand(feature.value.stringValue(), feature.variation)
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/webview/Browser.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/webview/Browser.kt
index 9a4f30222d..cc39985dd3 100644
--- a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/webview/Browser.kt
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/webview/Browser.kt
@@ -3,11 +3,14 @@
package software.aws.toolkits.jetbrains.services.amazonq.webview
+import com.fasterxml.jackson.module.kotlin.jacksonObjectMapper
import com.intellij.openapi.Disposable
import com.intellij.openapi.util.Disposer
import com.intellij.ui.jcef.JBCefJSQuery
import org.cef.CefApp
+import software.aws.toolkits.jetbrains.services.amazonq.util.HighlightCommand
import software.aws.toolkits.jetbrains.services.amazonq.util.createBrowser
+import software.aws.toolkits.jetbrains.settings.MeetQSettings
/*
Displays the web view for the Amazon Q tool window
@@ -18,7 +21,14 @@ class Browser(parent: Disposable) : Disposable {
val receiveMessageQuery = JBCefJSQuery.create(jcefBrowser)
- fun init(isCodeTransformAvailable: Boolean, isFeatureDevAvailable: Boolean) {
+ fun init(
+ isCodeTransformAvailable: Boolean,
+ isFeatureDevAvailable: Boolean,
+ isDocAvailable: Boolean,
+ isCodeScanAvailable: Boolean,
+ isCodeTestAvailable: Boolean,
+ highlightCommand: HighlightCommand?,
+ ) {
// register the scheme handler to route http://mynah/ URIs to the resources/assets directory on classpath
CefApp.getInstance()
.registerSchemeHandlerFactory(
@@ -27,7 +37,7 @@ class Browser(parent: Disposable) : Disposable {
AssetResourceHandler.AssetResourceHandlerFactory(),
)
- loadWebView(isCodeTransformAvailable, isFeatureDevAvailable)
+ loadWebView(isCodeTransformAvailable, isFeatureDevAvailable, isDocAvailable, isCodeScanAvailable, isCodeTestAvailable, highlightCommand)
}
override fun dispose() {
@@ -42,19 +52,35 @@ class Browser(parent: Disposable) : Disposable {
.executeJavaScript("window.postMessage(JSON.stringify($message))", jcefBrowser.cefBrowser.url, 0)
// Load the chat web app into the jcefBrowser
- private fun loadWebView(isCodeTransformAvailable: Boolean, isFeatureDevAvailable: Boolean) {
+ private fun loadWebView(
+ isCodeTransformAvailable: Boolean,
+ isFeatureDevAvailable: Boolean,
+ isDocAvailable: Boolean,
+ isCodeScanAvailable: Boolean,
+ isCodeTestAvailable: Boolean,
+ highlightCommand: HighlightCommand?,
+ ) {
// setup empty state. The message request handlers use this for storing state
// that's persistent between page loads.
jcefBrowser.setProperty("state", "")
// load the web app
- jcefBrowser.loadHTML(getWebviewHTML(isCodeTransformAvailable, isFeatureDevAvailable))
+ jcefBrowser.loadHTML(
+ getWebviewHTML(isCodeTransformAvailable, isFeatureDevAvailable, isDocAvailable, isCodeScanAvailable, isCodeTestAvailable, highlightCommand)
+ )
}
/**
* Generate index.html for the web view
* @return HTML source
*/
- private fun getWebviewHTML(isCodeTransformAvailable: Boolean, isFeatureDevAvailable: Boolean): String {
+ private fun getWebviewHTML(
+ isCodeTransformAvailable: Boolean,
+ isFeatureDevAvailable: Boolean,
+ isDocAvailable: Boolean,
+ isCodeScanAvailable: Boolean,
+ isCodeTestAvailable: Boolean,
+ highlightCommand: HighlightCommand?,
+ ): String {
val postMessageToJavaJsCode = receiveMessageQuery.inject("JSON.stringify(message)")
val jsScripts = """
@@ -67,9 +93,15 @@ class Browser(parent: Disposable) : Disposable {
$postMessageToJavaJsCode
}
},
+ ${MeetQSettings.getInstance().reinvent2024OnboardingCount < MAX_ONBOARDING_PAGE_COUNT},
+ ${MeetQSettings.getInstance().disclaimerAcknowledged},
$isFeatureDevAvailable, // whether /dev is available
$isCodeTransformAvailable, // whether /transform is available
- );
+ $isDocAvailable, // whether /doc is available
+ $isCodeScanAvailable, // whether /scan is available
+ $isCodeTestAvailable, // whether /test is available
+ ${OBJECT_MAPPER.writeValueAsString(highlightCommand)}
+ );
}
""".trimIndent()
@@ -89,5 +121,7 @@ class Browser(parent: Disposable) : Disposable {
companion object {
private const val WEB_SCRIPT_URI = "http://mynah/js/mynah-ui.js"
+ private const val MAX_ONBOARDING_PAGE_COUNT = 3
+ private val OBJECT_MAPPER = jacksonObjectMapper()
}
}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/webview/BrowserConnector.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/webview/BrowserConnector.kt
index b2571123b7..af9c4b5fce 100644
--- a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/webview/BrowserConnector.kt
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/webview/BrowserConnector.kt
@@ -3,6 +3,8 @@
package software.aws.toolkits.jetbrains.services.amazonq.webview
+import com.intellij.ide.BrowserUtil
+import com.intellij.ide.util.RunOnceUtil
import com.intellij.ui.jcef.JBCefJSQuery.Response
import kotlinx.coroutines.CompletableDeferred
import kotlinx.coroutines.channels.awaitClose
@@ -21,6 +23,8 @@ import software.aws.toolkits.jetbrains.services.amazonq.util.command
import software.aws.toolkits.jetbrains.services.amazonq.util.tabType
import software.aws.toolkits.jetbrains.services.amazonq.webview.theme.AmazonQTheme
import software.aws.toolkits.jetbrains.services.amazonq.webview.theme.ThemeBrowserAdapter
+import software.aws.toolkits.jetbrains.settings.MeetQSettings
+import software.aws.toolkits.telemetry.Telemetry
import java.util.function.Function
class BrowserConnector(
@@ -37,9 +41,40 @@ class BrowserConnector(
addMessageHook(browser)
.onEach { json ->
val node = serializer.toNode(json)
- if (node.command == "ui-is-ready") {
- uiReady.complete(true)
+ when (node.command) {
+ "ui-is-ready" -> {
+ uiReady.complete(true)
+ RunOnceUtil.runOnceForApp("AmazonQ-UI-Ready") {
+ MeetQSettings.getInstance().reinvent2024OnboardingCount += 1
+ }
+ }
+
+ "disclaimer-acknowledged" -> {
+ MeetQSettings.getInstance().disclaimerAcknowledged = true
+ }
+
+ // some weird issue preventing deserialization from working
+ "open-user-guide" -> {
+ BrowserUtil.browse(node.get("userGuideLink").asText())
+ }
+ "send-telemetry" -> {
+ val source = node.get("source")
+ val module = node.get("module")
+ val trigger = node.get("trigger")
+
+ if (source != null) {
+ Telemetry.ui.click.use {
+ it.elementId(source.asText())
+ }
+ } else if (module != null && trigger != null) {
+ Telemetry.toolkit.openModule.use {
+ it.module(module.asText())
+ it.source(trigger.asText())
+ }
+ }
+ }
}
+
val tabType = node.tabType ?: return@onEach
connections.filter { connection -> connection.app.tabTypes.contains(tabType) }.forEach { connection ->
launch {
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/CodeScanChatApp.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/CodeScanChatApp.kt
new file mode 100644
index 0000000000..166223d623
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/CodeScanChatApp.kt
@@ -0,0 +1,175 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqCodeScan
+
+import com.intellij.openapi.application.ApplicationManager
+import com.intellij.openapi.components.service
+import com.intellij.openapi.project.Project
+import kotlinx.coroutines.CoroutineScope
+import kotlinx.coroutines.flow.merge
+import kotlinx.coroutines.launch
+import software.aws.toolkits.jetbrains.core.credentials.AwsBearerTokenConnection
+import software.aws.toolkits.jetbrains.core.credentials.ToolkitConnection
+import software.aws.toolkits.jetbrains.core.credentials.ToolkitConnectionManager
+import software.aws.toolkits.jetbrains.core.credentials.ToolkitConnectionManagerListener
+import software.aws.toolkits.jetbrains.core.credentials.pinning.QConnection
+import software.aws.toolkits.jetbrains.core.credentials.sso.bearer.BearerTokenAuthState
+import software.aws.toolkits.jetbrains.core.credentials.sso.bearer.BearerTokenProvider
+import software.aws.toolkits.jetbrains.core.credentials.sso.bearer.BearerTokenProviderListener
+import software.aws.toolkits.jetbrains.services.amazonq.apps.AmazonQApp
+import software.aws.toolkits.jetbrains.services.amazonq.apps.AmazonQAppInitContext
+import software.aws.toolkits.jetbrains.services.amazonq.auth.AuthController
+import software.aws.toolkits.jetbrains.services.amazonq.messages.AmazonQMessage
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.auth.isCodeScanAvailable
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.commands.CodeScanActionMessage
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.commands.CodeScanMessageListener
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.controller.CodeScanChatController
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.messages.AuthenticationNeededExceptionMessage
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.messages.AuthenticationUpdateMessage
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.messages.CODE_SCAN_TAB_NAME
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.messages.IncomingCodeScanMessage
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.storage.ChatSessionStorage
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.auth.isCodeTestAvailable
+import software.aws.toolkits.jetbrains.services.amazonqDoc.auth.isDocAvailable
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.auth.isFeatureDevAvailable
+import software.aws.toolkits.jetbrains.services.codemodernizer.utils.isCodeTransformAvailable
+import java.util.concurrent.atomic.AtomicBoolean
+
+private enum class CodeScanMessageTypes(val type: String) {
+ ClearChat("clear"),
+ Help("help"),
+ TabCreated("new-tab-was-created"),
+ TabRemoved("tab-was-removed"),
+ Scan("scan"),
+ StartProjectScan("codescan_start_project_scan"),
+ StartFileScan("codescan_start_file_scan"),
+ StopFileScan("codescan_stop_file_scan"),
+ StopProjectScan("codescan_stop_project_scan"),
+ OpenIssuesPanel("codescan_open_issues"),
+ ResponseBodyLinkClicked("response-body-link-click"),
+}
+
+class CodeScanChatApp(private val scope: CoroutineScope) : AmazonQApp {
+ private val isProcessingAuthChanged = AtomicBoolean(false)
+ override val tabTypes = listOf(CODE_SCAN_TAB_NAME)
+ override fun init(context: AmazonQAppInitContext) {
+ val chatSessionStorage = ChatSessionStorage()
+ val inboundAppMessagesHandler: InboundAppMessagesHandler = CodeScanChatController(context, chatSessionStorage)
+
+ context.messageTypeRegistry.register(
+ CodeScanMessageTypes.ClearChat.type to IncomingCodeScanMessage.ClearChat::class,
+ CodeScanMessageTypes.Help.type to IncomingCodeScanMessage.Help::class,
+ CodeScanMessageTypes.TabCreated.type to IncomingCodeScanMessage.TabCreated::class,
+ CodeScanMessageTypes.TabRemoved.type to IncomingCodeScanMessage.TabRemoved::class,
+ CodeScanMessageTypes.Scan.type to IncomingCodeScanMessage.Scan::class,
+ CodeScanMessageTypes.StartProjectScan.type to IncomingCodeScanMessage.StartProjectScan::class,
+ CodeScanMessageTypes.StartFileScan.type to IncomingCodeScanMessage.StartFileScan::class,
+ CodeScanMessageTypes.StopProjectScan.type to IncomingCodeScanMessage.StopProjectScan::class,
+ CodeScanMessageTypes.StopFileScan.type to IncomingCodeScanMessage.StopFileScan::class,
+ CodeScanMessageTypes.ResponseBodyLinkClicked.type to IncomingCodeScanMessage.ResponseBodyLinkClicked::class,
+ CodeScanMessageTypes.OpenIssuesPanel.type to IncomingCodeScanMessage.OpenIssuesPanel::class
+ )
+
+ scope.launch {
+ merge(service().flow, context.messagesFromUiToApp.flow).collect { message ->
+ // Launch a new coroutine to handle each message
+ scope.launch { handleMessage(message, inboundAppMessagesHandler) }
+ }
+ }
+
+ fun authChanged() {
+ val isAnotherThreadProcessing = !isProcessingAuthChanged.compareAndSet(false, true)
+ if (isAnotherThreadProcessing) return
+ scope.launch {
+ val authController = AuthController()
+ val credentialState = authController.getAuthNeededStates(context.project).amazonQ
+ if (credentialState == null) {
+ // Notify tabs about restoring authentication
+ context.messagesFromAppToUi.publish(
+ AuthenticationUpdateMessage(
+ featureDevEnabled = isFeatureDevAvailable(context.project),
+ codeTransformEnabled = isCodeTransformAvailable(context.project),
+ codeScanEnabled = isCodeScanAvailable(context.project),
+ codeTestEnabled = isCodeTestAvailable(context.project),
+ docEnabled = isDocAvailable(context.project),
+ authenticatingTabIDs = chatSessionStorage.getAuthenticatingSessions().map { it.tabId }
+ )
+ )
+
+ chatSessionStorage.changeAuthenticationNeeded(false)
+ chatSessionStorage.changeAuthenticationNeededNotified(false)
+ } else {
+ chatSessionStorage.changeAuthenticationNeeded(true)
+
+ // Ask for reauth
+ chatSessionStorage.getAuthenticatingSessions().filter { !it.authNeededNotified }.forEach {
+ context.messagesFromAppToUi.publish(
+ AuthenticationNeededExceptionMessage(
+ tabId = it.tabId,
+ authType = credentialState.authType,
+ message = credentialState.message
+ )
+ )
+ }
+
+ // Prevent multiple calls to activeConnectionChanged
+ chatSessionStorage.changeAuthenticationNeededNotified(true)
+ }
+ isProcessingAuthChanged.set(false)
+ }
+ }
+
+ ApplicationManager.getApplication().messageBus.connect(this).subscribe(
+ BearerTokenProviderListener.TOPIC,
+ object : BearerTokenProviderListener {
+ override fun onChange(providerId: String, newScopes: List?) {
+ val qProvider = getQTokenProvider(context.project)
+ val isQ = qProvider?.id == providerId
+ val isAuthorized = qProvider?.state() == BearerTokenAuthState.AUTHORIZED
+ if (!isQ || !isAuthorized) return
+ authChanged()
+ }
+ }
+ )
+
+ ApplicationManager.getApplication().messageBus.connect(this).subscribe(
+ ToolkitConnectionManagerListener.TOPIC,
+ object : ToolkitConnectionManagerListener {
+ override fun activeConnectionChanged(newConnection: ToolkitConnection?) {
+ authChanged()
+ }
+ }
+ )
+ }
+
+ private fun getQTokenProvider(project: Project) = (
+ ToolkitConnectionManager
+ .getInstance(project)
+ .activeConnectionForFeature(QConnection.getInstance()) as? AwsBearerTokenConnection
+ )
+ ?.getConnectionSettings()
+ ?.tokenProvider
+ ?.delegate as? BearerTokenProvider
+
+ private suspend fun handleMessage(message: AmazonQMessage, inboundAppMessagesHandler: InboundAppMessagesHandler) {
+ when (message) {
+ is IncomingCodeScanMessage.ClearChat -> inboundAppMessagesHandler.processClearQuickAction(message)
+ is IncomingCodeScanMessage.Help -> inboundAppMessagesHandler.processHelpQuickAction(message)
+ is IncomingCodeScanMessage.TabCreated -> inboundAppMessagesHandler.processTabCreated(message)
+ is IncomingCodeScanMessage.TabRemoved -> inboundAppMessagesHandler.processTabRemoved(message)
+ is IncomingCodeScanMessage.Scan -> inboundAppMessagesHandler.processScanQuickAction(message)
+ is IncomingCodeScanMessage.StartProjectScan -> inboundAppMessagesHandler.processStartProjectScan(message)
+ is IncomingCodeScanMessage.StartFileScan -> inboundAppMessagesHandler.processStartFileScan(message)
+ is CodeScanActionMessage -> inboundAppMessagesHandler.processCodeScanCommand(message)
+ is IncomingCodeScanMessage.StopProjectScan -> inboundAppMessagesHandler.processStopProjectScan(message)
+ is IncomingCodeScanMessage.StopFileScan -> inboundAppMessagesHandler.processStopFileScan(message)
+ is IncomingCodeScanMessage.ResponseBodyLinkClicked -> inboundAppMessagesHandler.processResponseBodyLinkClicked(message)
+ is IncomingCodeScanMessage.OpenIssuesPanel -> inboundAppMessagesHandler.processOpenIssuesPanel(message)
+ }
+ }
+
+ override fun dispose() {
+ // nothing to do
+ }
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/CodeScanChatAppFactory.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/CodeScanChatAppFactory.kt
new file mode 100644
index 0000000000..2ac70b38b0
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/CodeScanChatAppFactory.kt
@@ -0,0 +1,12 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqCodeScan
+
+import com.intellij.openapi.project.Project
+import kotlinx.coroutines.CoroutineScope
+import software.aws.toolkits.jetbrains.services.amazonq.apps.AmazonQAppFactory
+
+class CodeScanChatAppFactory(private val cs: CoroutineScope) : AmazonQAppFactory {
+ override fun createApp(project: Project) = CodeScanChatApp(cs)
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/CodeScanChatItems.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/CodeScanChatItems.kt
new file mode 100644
index 0000000000..ee2dc09e3c
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/CodeScanChatItems.kt
@@ -0,0 +1,165 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqCodeScan
+
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.messages.Button
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.messages.CodeScanButtonId
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.messages.CodeScanChatMessage
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.messages.CodeScanChatMessageContent
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.messages.ProgressField
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.messages.PromptProgressMessage
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.CodeWhispererCodeScanIssue
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils.IssueSeverity
+import software.aws.toolkits.jetbrains.services.cwc.controller.chat.StaticPrompt
+import software.aws.toolkits.jetbrains.services.cwc.controller.chat.StaticTextResponse
+import software.aws.toolkits.jetbrains.services.cwc.messages.ChatMessageType
+import software.aws.toolkits.resources.message
+import java.util.UUID
+
+private val projectScanButton = Button(
+ id = CodeScanButtonId.StartProjectScan.id,
+ text = message("codescan.chat.message.button.projectScan")
+)
+
+private val fileScanButton = Button(
+ id = CodeScanButtonId.StartFileScan.id,
+ text = message("codescan.chat.message.button.fileScan")
+)
+
+private val openIssuesPanelButton = Button(
+ id = CodeScanButtonId.OpenIssuesPanel.id,
+ text = message("codescan.chat.message.button.openIssues"),
+ keepCardAfterClick = true
+)
+
+fun buildStartNewScanChatContent() = CodeScanChatMessageContent(
+ type = ChatMessageType.Answer,
+ message = message("codescan.chat.new_scan.input.message"),
+ buttons = listOf(
+ fileScanButton,
+ projectScanButton
+ ),
+ canBeVoted = false
+)
+
+// TODO: Replace StaticPrompt and StaticTextResponse message according to Fnf
+fun buildHelpChatPromptContent() = CodeScanChatMessageContent(
+ type = ChatMessageType.Prompt,
+ message = StaticPrompt.Help.message,
+ canBeVoted = false
+)
+
+fun buildHelpChatAnswerContent() = CodeScanChatMessageContent(
+ type = ChatMessageType.Answer,
+ message = StaticTextResponse.Help.message,
+ canBeVoted = false
+)
+
+fun buildUserSelectionProjectScanChatContent() = CodeScanChatMessageContent(
+ type = ChatMessageType.Prompt,
+ message = message("codescan.chat.message.button.projectScan"),
+ canBeVoted = false
+)
+
+fun buildUserSelectionFileScanChatContent() = CodeScanChatMessageContent(
+ type = ChatMessageType.Prompt,
+ message = message("codescan.chat.message.button.fileScan"),
+ canBeVoted = false
+)
+
+fun buildNotInGitRepoChatContent() = CodeScanChatMessageContent(
+ type = ChatMessageType.Answer,
+ message = message("codescan.chat.message.not_git_repo"),
+ canBeVoted = false
+)
+
+fun buildScanInProgressChatContent(currentStep: Int, isProject: Boolean) = CodeScanChatMessageContent(
+ type = ChatMessageType.AnswerPart,
+ message = buildString {
+ appendLine(if (isProject) message("codescan.chat.message.scan_begin_project") else message("codescan.chat.message.scan_begin_file"))
+ appendLine("")
+ appendLine(message("codescan.chat.message.scan_begin_wait_time"))
+ appendLine("")
+ appendLine("${getIconForStep(0, currentStep)} " + message("codescan.chat.message.scan_step_1"))
+ appendLine("${getIconForStep(1, currentStep)} " + message("codescan.chat.message.scan_step_2"))
+ appendLine("${getIconForStep(2, currentStep)} " + message("codescan.chat.message.scan_step_3"))
+ }
+)
+
+val cancellingProgressField = ProgressField(
+ status = "warning",
+ text = message("general.canceling"),
+ value = -1,
+ actions = emptyList()
+)
+
+fun buildScanCompleteChatContent(issues: List, isProject: Boolean = false): CodeScanChatMessageContent {
+ val issueCountMap = IssueSeverity.entries.associate { it.displayName to 0 }.toMutableMap()
+ val aggregatedIssues = issues.groupBy { it.severity }
+ aggregatedIssues.forEach { (key, list) -> if (list.isNotEmpty()) issueCountMap[key] = list.size }
+
+ val message = buildString {
+ appendLine(if (isProject) message("codewhisperer.codescan.scan_complete_project") else message("codewhisperer.codescan.scan_complete_file"))
+ issueCountMap.entries.forEach { (severity, count) ->
+ appendLine(message("codewhisperer.codescan.scan_complete_count", severity, count))
+ }
+ }
+
+ return CodeScanChatMessageContent(
+ type = ChatMessageType.Answer,
+ message = message,
+ buttons = listOf(
+ openIssuesPanelButton
+ ),
+ )
+}
+
+fun buildPromptProgressMessage(tabId: String, isProject: Boolean = false, isCanceling: Boolean = false) = PromptProgressMessage(
+ progressField = when {
+ isCanceling -> cancellingProgressField
+ isProject -> projectScanProgressField
+ else -> fileScanProgressField
+ },
+ tabId = tabId
+)
+
+fun buildClearPromptProgressMessage(tabId: String) = PromptProgressMessage(
+ tabId = tabId
+)
+
+val runCodeScanMessage
+ get() = CodeScanChatMessage(messageType = ChatMessageType.Prompt, command = "review", tabId = UUID.randomUUID().toString())
+
+val cancelFileScanButton = Button(
+ id = CodeScanButtonId.StopFileScan.id,
+ text = message("general.cancel"),
+ icon = "cancel"
+)
+
+val cancelProjectScanButton = cancelFileScanButton.copy(
+ id = CodeScanButtonId.StopProjectScan.id
+)
+
+val fileScanProgressField = ProgressField(
+ status = "default",
+ text = message("codescan.chat.message.scan_file_in_progress"),
+ value = -1,
+ actions = listOf(cancelFileScanButton)
+)
+
+val projectScanProgressField = fileScanProgressField.copy(
+ text = message("codescan.chat.message.scan_project_in_progress"),
+ actions = listOf(cancelProjectScanButton)
+)
+
+fun buildProjectScanFailedChatContent(errorMessage: String?) = CodeScanChatMessageContent(
+ type = ChatMessageType.Answer,
+ message = errorMessage ?: message("codescan.chat.message.project_scan_failed")
+)
+
+fun getIconForStep(targetStep: Int, currentStep: Int) = when {
+ currentStep == targetStep -> "☐"
+ currentStep > targetStep -> "☑"
+ else -> "☐"
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/CodeScanConstants.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/CodeScanConstants.kt
new file mode 100644
index 0000000000..c27b7ef416
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/CodeScanConstants.kt
@@ -0,0 +1,6 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqCodeScan
+
+const val FEATURE_NAME = "Amazon Q Code Scan"
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/InboundAppMessagesHandler.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/InboundAppMessagesHandler.kt
new file mode 100644
index 0000000000..31ca09503d
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/InboundAppMessagesHandler.kt
@@ -0,0 +1,33 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqCodeScan
+
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.commands.CodeScanActionMessage
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.messages.IncomingCodeScanMessage
+
+interface InboundAppMessagesHandler {
+ suspend fun processScanQuickAction(message: IncomingCodeScanMessage.Scan)
+
+ suspend fun processStartProjectScan(message: IncomingCodeScanMessage.StartProjectScan)
+
+ suspend fun processStartFileScan(message: IncomingCodeScanMessage.StartFileScan)
+
+ suspend fun processStopProjectScan(message: IncomingCodeScanMessage.StopProjectScan)
+
+ suspend fun processStopFileScan(message: IncomingCodeScanMessage.StopFileScan)
+
+ suspend fun processTabCreated(message: IncomingCodeScanMessage.TabCreated)
+
+ suspend fun processClearQuickAction(message: IncomingCodeScanMessage.ClearChat)
+
+ suspend fun processHelpQuickAction(message: IncomingCodeScanMessage.Help)
+
+ suspend fun processTabRemoved(message: IncomingCodeScanMessage.TabRemoved)
+
+ suspend fun processCodeScanCommand(message: CodeScanActionMessage)
+
+ suspend fun processResponseBodyLinkClicked(message: IncomingCodeScanMessage.ResponseBodyLinkClicked)
+
+ suspend fun processOpenIssuesPanel(message: IncomingCodeScanMessage.OpenIssuesPanel)
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/auth/CodeScanAuthUtils.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/auth/CodeScanAuthUtils.kt
new file mode 100644
index 0000000000..2e5744b176
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/auth/CodeScanAuthUtils.kt
@@ -0,0 +1,11 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqCodeScan.auth
+
+import com.intellij.openapi.project.Project
+import software.aws.toolkits.jetbrains.core.gettingstarted.editor.ActiveConnection
+import software.aws.toolkits.jetbrains.core.gettingstarted.editor.BearerTokenFeatureSet
+import software.aws.toolkits.jetbrains.core.gettingstarted.editor.checkBearerConnectionValidity
+
+fun isCodeScanAvailable(project: Project): Boolean = checkBearerConnectionValidity(project, BearerTokenFeatureSet.Q) is ActiveConnection.ValidBearer
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/commands/CodeScanActionMessage.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/commands/CodeScanActionMessage.kt
new file mode 100644
index 0000000000..01532c154f
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/commands/CodeScanActionMessage.kt
@@ -0,0 +1,16 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqCodeScan.commands
+
+import com.intellij.openapi.project.Project
+import software.aws.toolkits.jetbrains.services.amazonq.messages.AmazonQMessage
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.CodeScanResponse
+import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererConstants
+
+data class CodeScanActionMessage(
+ val command: CodeScanCommand,
+ val project: Project,
+ val scanResult: CodeScanResponse? = null,
+ val scope: CodeWhispererConstants.CodeAnalysisScope,
+) : AmazonQMessage
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/commands/CodeScanCommand.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/commands/CodeScanCommand.kt
new file mode 100644
index 0000000000..f7b16705b1
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/commands/CodeScanCommand.kt
@@ -0,0 +1,8 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqCodeScan.commands
+
+enum class CodeScanCommand {
+ ScanComplete,
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/commands/CodeScanMessageListener.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/commands/CodeScanMessageListener.kt
new file mode 100644
index 0000000000..832741a1fc
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/commands/CodeScanMessageListener.kt
@@ -0,0 +1,21 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqCodeScan.commands
+
+import com.intellij.openapi.components.Service
+import com.intellij.openapi.project.Project
+import kotlinx.coroutines.flow.MutableSharedFlow
+import kotlinx.coroutines.flow.asSharedFlow
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.CodeScanResponse
+import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererConstants
+
+@Service
+class CodeScanMessageListener {
+ private val _messages by lazy { MutableSharedFlow(extraBufferCapacity = 10) }
+ val flow = _messages.asSharedFlow()
+
+ fun onScanResult(result: CodeScanResponse?, scope: CodeWhispererConstants.CodeAnalysisScope, project: Project) {
+ _messages.tryEmit(CodeScanActionMessage(CodeScanCommand.ScanComplete, scanResult = result, scope = scope, project = project))
+ }
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/controller/CodeScanChatController.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/controller/CodeScanChatController.kt
new file mode 100644
index 0000000000..59a69c7ce4
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/controller/CodeScanChatController.kt
@@ -0,0 +1,197 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqCodeScan.controller
+
+import com.intellij.ide.BrowserUtil
+import software.aws.toolkits.core.utils.debug
+import software.aws.toolkits.core.utils.getLogger
+import software.aws.toolkits.jetbrains.services.amazonq.apps.AmazonQAppInitContext
+import software.aws.toolkits.jetbrains.services.amazonq.auth.AuthController
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.FEATURE_NAME
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.InboundAppMessagesHandler
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.buildHelpChatAnswerContent
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.buildHelpChatPromptContent
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.buildNotInGitRepoChatContent
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.buildProjectScanFailedChatContent
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.buildScanCompleteChatContent
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.buildScanInProgressChatContent
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.buildStartNewScanChatContent
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.buildUserSelectionFileScanChatContent
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.buildUserSelectionProjectScanChatContent
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.commands.CodeScanActionMessage
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.commands.CodeScanCommand
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.messages.AuthenticationNeededExceptionMessage
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.messages.CodeScanChatMessage
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.messages.IncomingCodeScanMessage
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.storage.ChatSessionStorage
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.CodeScanResponse
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.CodeWhispererCodeScanManager
+import software.aws.toolkits.jetbrains.services.codewhisperer.telemetry.CodeWhispererTelemetryService
+import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererConstants
+import software.aws.toolkits.jetbrains.services.codewhisperer.util.getAuthType
+import software.aws.toolkits.jetbrains.services.cwc.messages.ChatMessageType
+import software.aws.toolkits.resources.message
+
+class CodeScanChatController(
+ private val context: AmazonQAppInitContext,
+ private val chatSessionStorage: ChatSessionStorage,
+ private val authController: AuthController = AuthController(),
+) : InboundAppMessagesHandler {
+
+ private val messenger = context.messagesFromAppToUi
+ private val codeScanManager = CodeWhispererCodeScanManager.getInstance(context.project)
+ private val codeScanChatHelper = CodeScanChatHelper(context.messagesFromAppToUi, chatSessionStorage)
+ private val scanInProgressMessageId = "scanProgressMessage"
+
+ override suspend fun processScanQuickAction(message: IncomingCodeScanMessage.Scan) {
+ // TODO: telemetry
+
+ if (!checkForAuth(message.tabId)) {
+ return
+ }
+ if (message.tabId != codeScanChatHelper.getActiveCodeScanTabId()) return
+
+ codeScanChatHelper.setActiveCodeScanTabId(message.tabId)
+ codeScanChatHelper.addNewMessage(buildStartNewScanChatContent())
+ codeScanChatHelper.sendChatInputEnabledMessage(false)
+ }
+
+ override suspend fun processStartProjectScan(message: IncomingCodeScanMessage.StartProjectScan) {
+ if (message.tabId != codeScanChatHelper.getActiveCodeScanTabId()) return
+ codeScanChatHelper.addNewMessage(buildUserSelectionProjectScanChatContent())
+ if (!codeScanManager.isInsideWorkTree()) {
+ codeScanChatHelper.addNewMessage(buildNotInGitRepoChatContent())
+ }
+ codeScanChatHelper.addNewMessage(buildScanInProgressChatContent(currentStep = 1, isProject = true), messageIdOverride = scanInProgressMessageId)
+ codeScanManager.runCodeScan(CodeWhispererConstants.CodeAnalysisScope.PROJECT, initiatedByChat = true)
+ codeScanChatHelper.updateProgress(isProject = true, isCanceling = false)
+ }
+
+ override suspend fun processStopProjectScan(message: IncomingCodeScanMessage.StopProjectScan) {
+ if (message.tabId != codeScanChatHelper.getActiveCodeScanTabId()) return
+ codeScanChatHelper.updateProgress(isProject = true, isCanceling = true)
+ codeScanManager.stopCodeScan(CodeWhispererConstants.CodeAnalysisScope.PROJECT)
+ }
+
+ override suspend fun processStopFileScan(message: IncomingCodeScanMessage.StopFileScan) {
+ if (message.tabId != codeScanChatHelper.getActiveCodeScanTabId()) return
+ codeScanChatHelper.updateProgress(isProject = false, isCanceling = true)
+ codeScanManager.stopCodeScan(CodeWhispererConstants.CodeAnalysisScope.FILE)
+ }
+
+ override suspend fun processStartFileScan(message: IncomingCodeScanMessage.StartFileScan) {
+ if (message.tabId != codeScanChatHelper.getActiveCodeScanTabId()) return
+ codeScanChatHelper.addNewMessage(buildUserSelectionFileScanChatContent())
+ codeScanChatHelper.addNewMessage(buildScanInProgressChatContent(currentStep = 1, isProject = false), messageIdOverride = scanInProgressMessageId)
+ codeScanManager.runCodeScan(CodeWhispererConstants.CodeAnalysisScope.FILE, initiatedByChat = true)
+ codeScanChatHelper.updateProgress(isProject = false, isCanceling = false)
+ }
+
+ override suspend fun processTabCreated(message: IncomingCodeScanMessage.TabCreated) {
+ logger.debug { "$FEATURE_NAME: New tab created: $message" }
+ codeScanChatHelper.setActiveCodeScanTabId(message.tabId)
+ CodeWhispererTelemetryService.getInstance().sendCodeScanNewTabEvent(getAuthType(context.project))
+ }
+
+ override suspend fun processClearQuickAction(message: IncomingCodeScanMessage.ClearChat) {
+ chatSessionStorage.deleteSession(message.tabId)
+ }
+
+ override suspend fun processHelpQuickAction(message: IncomingCodeScanMessage.Help) {
+ if (message.tabId != codeScanChatHelper.getActiveCodeScanTabId()) return
+ codeScanChatHelper.addNewMessage(buildHelpChatPromptContent())
+ codeScanChatHelper.addNewMessage(buildHelpChatAnswerContent())
+ }
+
+ override suspend fun processTabRemoved(message: IncomingCodeScanMessage.TabRemoved) {
+ chatSessionStorage.deleteSession(message.tabId)
+ }
+
+ override suspend fun processResponseBodyLinkClicked(message: IncomingCodeScanMessage.ResponseBodyLinkClicked) {
+ BrowserUtil.browse(message.link)
+ }
+
+ override suspend fun processCodeScanCommand(message: CodeScanActionMessage) {
+ if (message.project != context.project) return
+ val isProject = message.scope == CodeWhispererConstants.CodeAnalysisScope.PROJECT
+ when (message.command) {
+ CodeScanCommand.ScanComplete -> {
+ codeScanChatHelper.addNewMessage(
+ buildScanInProgressChatContent(currentStep = 2, isProject = isProject),
+ messageIdOverride = scanInProgressMessageId
+ )
+ val result = message.scanResult
+ if (result != null) {
+ handleCodeScanResult(result, message.scope)
+ } else {
+ codeScanChatHelper.addNewMessage(buildProjectScanFailedChatContent("Cancelled"))
+ codeScanChatHelper.clearProgress()
+ }
+ }
+ }
+ }
+
+ private suspend fun handleCodeScanResult(result: CodeScanResponse, scope: CodeWhispererConstants.CodeAnalysisScope) {
+ val isProject = scope == CodeWhispererConstants.CodeAnalysisScope.PROJECT
+ when (result) {
+ is CodeScanResponse.Success -> {
+ codeScanChatHelper.addNewMessage(
+ buildScanInProgressChatContent(currentStep = 3, isProject = isProject),
+ messageIdOverride = scanInProgressMessageId
+ )
+ codeScanChatHelper.addNewMessage(buildScanCompleteChatContent(result.issues, isProject = isProject))
+ codeScanChatHelper.clearProgress()
+ }
+ is CodeScanResponse.Failure -> {
+ codeScanChatHelper.addNewMessage(buildScanInProgressChatContent(3, isProject = isProject), messageIdOverride = scanInProgressMessageId)
+ codeScanChatHelper.addNewMessage(buildProjectScanFailedChatContent(result.failureReason.message))
+ codeScanChatHelper.clearProgress()
+ }
+ }
+ }
+
+ /**
+ * Return true if authenticated, else show authentication message and return false
+ * // TODO: Refactor this to avoid code duplication with other controllers
+ */
+ private suspend fun checkForAuth(tabId: String): Boolean {
+ try {
+ val session = chatSessionStorage.getSession(tabId)
+ logger.debug { "$FEATURE_NAME: Session created with id: ${session.tabId}" }
+
+ val credentialState = authController.getAuthNeededStates(context.project).amazonQ
+ if (credentialState != null) {
+ messenger.publish(
+ AuthenticationNeededExceptionMessage(
+ tabId = session.tabId,
+ authType = credentialState.authType,
+ message = credentialState.message
+ )
+ )
+ session.isAuthenticating = true
+ return false
+ }
+ } catch (err: Exception) {
+ messenger.publish(
+ CodeScanChatMessage(
+ tabId = tabId,
+ messageType = ChatMessageType.Answer,
+ message = message("codescan.chat.message.error_request")
+ )
+ )
+ return false
+ }
+
+ return true
+ }
+
+ override suspend fun processOpenIssuesPanel(message: IncomingCodeScanMessage.OpenIssuesPanel) {
+ if (message.tabId != codeScanChatHelper.getActiveCodeScanTabId()) return
+ codeScanManager.showCodeScanUI()
+ }
+
+ companion object {
+ private val logger = getLogger()
+ }
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/controller/CodeScanChatHelper.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/controller/CodeScanChatHelper.kt
new file mode 100644
index 0000000000..6a9acbbfc0
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/controller/CodeScanChatHelper.kt
@@ -0,0 +1,81 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqCodeScan.controller
+
+import software.aws.toolkits.jetbrains.services.amazonq.messages.MessagePublisher
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.buildClearPromptProgressMessage
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.buildPromptProgressMessage
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.messages.ChatInputEnabledMessage
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.messages.CodeScanChatMessage
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.messages.CodeScanChatMessageContent
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.messages.UpdatePlaceholderMessage
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.storage.ChatSessionStorage
+import software.aws.toolkits.jetbrains.services.cwc.messages.ChatMessageType
+import java.util.UUID
+
+class CodeScanChatHelper(
+ private val messagePublisher: MessagePublisher,
+ private val chatSessionStorage: ChatSessionStorage,
+) {
+ private var activeCodeScanTabId: String? = null
+
+ fun setActiveCodeScanTabId(tabId: String) {
+ activeCodeScanTabId = tabId
+ }
+
+ fun getActiveCodeScanTabId(): String? = activeCodeScanTabId
+
+ private fun isInValidSession() = activeCodeScanTabId == null || chatSessionStorage.getSession(activeCodeScanTabId as String).isAuthenticating
+
+ suspend fun addNewMessage(
+ content: CodeScanChatMessageContent,
+ messageIdOverride: String? = null,
+ clearPreviousItemButtons: Boolean? = false,
+ ) {
+ if (isInValidSession()) return
+
+ messagePublisher.publish(
+ CodeScanChatMessage(
+ tabId = activeCodeScanTabId as String,
+ messageId = messageIdOverride ?: UUID.randomUUID().toString(),
+ messageType = content.type,
+ message = content.message,
+ buttons = content.buttons,
+ formItems = content.formItems,
+ followUps = content.followUps,
+ canBeVoted = content.canBeVoted,
+ isLoading = content.type == ChatMessageType.AnswerPart,
+ clearPreviousItemButtons = clearPreviousItemButtons as Boolean
+ )
+ )
+ }
+
+ suspend fun updateProgress(isProject: Boolean = false, isCanceling: Boolean = false) {
+ if (isInValidSession()) return
+ messagePublisher.publish(buildPromptProgressMessage(activeCodeScanTabId as String, isProject, isCanceling))
+ sendChatInputEnabledMessage(false)
+ }
+
+ suspend fun clearProgress() {
+ if (isInValidSession()) return
+ messagePublisher.publish(buildClearPromptProgressMessage(activeCodeScanTabId as String))
+ sendChatInputEnabledMessage(true)
+ }
+
+ suspend fun sendChatInputEnabledMessage(isEnabled: Boolean) {
+ if (isInValidSession()) return
+ messagePublisher.publish(ChatInputEnabledMessage(activeCodeScanTabId as String, enabled = isEnabled))
+ }
+
+ suspend fun updatePlaceholder(newPlaceholder: String) {
+ if (isInValidSession()) return
+
+ messagePublisher.publish(
+ UpdatePlaceholderMessage(
+ tabId = activeCodeScanTabId as String,
+ newPlaceholder = newPlaceholder
+ )
+ )
+ }
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/messages/CodeScanMessage.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/messages/CodeScanMessage.kt
new file mode 100644
index 0000000000..062e974336
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/messages/CodeScanMessage.kt
@@ -0,0 +1,185 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqCodeScan.messages
+
+import com.fasterxml.jackson.annotation.JsonProperty
+import software.aws.toolkits.jetbrains.services.amazonq.auth.AuthFollowUpType
+import software.aws.toolkits.jetbrains.services.amazonq.messages.AmazonQMessage
+import software.aws.toolkits.jetbrains.services.cwc.messages.ChatMessageType
+import software.aws.toolkits.jetbrains.services.cwc.messages.FollowUp
+import java.time.Instant
+import java.util.UUID
+
+const val CODE_SCAN_TAB_NAME = "codescan"
+
+enum class CodeScanButtonId(val id: String) {
+ StartProjectScan("codescan_start_project_scan"),
+ StartFileScan("codescan_start_file_scan"),
+ StopProjectScan("codescan_stop_project_scan"),
+ StopFileScan("codescan_stop_file_scan"),
+ OpenIssuesPanel("codescan_open_issues"),
+}
+
+data class Button(
+ val id: String,
+ val text: String,
+ val description: String? = null,
+ val icon: String? = null,
+ val keepCardAfterClick: Boolean? = false,
+ val disabled: Boolean? = false,
+ val waitMandatoryFormItems: Boolean? = false,
+)
+data class ProgressField(
+ val title: String? = null,
+ val value: Int? = null,
+ val status: String? = null,
+ val actions: List? = null,
+ val text: String? = null,
+)
+
+data class FormItemOption(
+ val label: String,
+ val value: String,
+)
+
+data class FormItem(
+ val id: String,
+ val type: String = "select",
+ val title: String,
+ val mandatory: Boolean = true,
+ val options: List = emptyList(),
+)
+
+sealed interface CodeScanBaseMessage : AmazonQMessage
+
+// === UI -> App Messages ===
+sealed interface IncomingCodeScanMessage : CodeScanBaseMessage {
+ data class Scan(
+ @JsonProperty("tabID") val tabId: String,
+ ) : IncomingCodeScanMessage
+
+ data class StartProjectScan(
+ @JsonProperty("tabID") val tabId: String,
+ ) : IncomingCodeScanMessage
+
+ data class StartFileScan(
+ @JsonProperty("tabID") val tabId: String,
+ ) : IncomingCodeScanMessage
+
+ data class StopProjectScan(
+ @JsonProperty("tabID") val tabId: String,
+ ) : IncomingCodeScanMessage
+
+ data class StopFileScan(
+ @JsonProperty("tabID") val tabId: String,
+ ) : IncomingCodeScanMessage
+
+ data class OpenIssuesPanel(
+ @JsonProperty("tabID") val tabId: String,
+ ) : IncomingCodeScanMessage
+
+ data class TabCreated(
+ @JsonProperty("tabID") val tabId: String,
+ ) : IncomingCodeScanMessage
+
+ data class Help(
+ @JsonProperty("tabID") val tabId: String,
+ ) : IncomingCodeScanMessage
+
+ data class ClearChat(
+ @JsonProperty("tabID") val tabId: String,
+ ) : IncomingCodeScanMessage
+
+ data class TabRemoved(
+ @JsonProperty("tabID") val tabId: String,
+ ) : IncomingCodeScanMessage
+
+ data class ResponseBodyLinkClicked(
+ @JsonProperty("tabID") val tabId: String,
+ val link: String,
+ ) : IncomingCodeScanMessage
+}
+
+// === App -> UI messages ===
+sealed class CodeScanUiMessage(
+ open val tabId: String?,
+ open val type: String,
+ open val messageId: String? = UUID.randomUUID().toString(),
+) : CodeScanBaseMessage {
+ val time = Instant.now().epochSecond
+ val sender = CODE_SCAN_TAB_NAME
+}
+
+data class PromptProgressMessage(
+ @JsonProperty("tabID") override val tabId: String,
+ val progressField: ProgressField? = null,
+) : CodeScanUiMessage(
+ tabId = tabId,
+ type = "updatePromptProgress",
+)
+
+data class ChatInputEnabledMessage(
+ @JsonProperty("tabID") override val tabId: String,
+ val enabled: Boolean,
+) : CodeScanUiMessage(
+ tabId = tabId,
+ type = "chatInputEnabledMessage"
+)
+
+data class AuthenticationUpdateMessage(
+ val authenticatingTabIDs: List,
+ val featureDevEnabled: Boolean,
+ val codeTransformEnabled: Boolean,
+ val codeScanEnabled: Boolean,
+ val codeTestEnabled: Boolean,
+ val docEnabled: Boolean,
+ val message: String? = null,
+) : CodeScanUiMessage(
+ null,
+ type = "authenticationUpdateMessage"
+)
+
+data class AuthenticationNeededExceptionMessage(
+ @JsonProperty("tabID") override val tabId: String,
+ val authType: AuthFollowUpType,
+ val message: String? = null,
+) : CodeScanUiMessage(
+ tabId = tabId,
+ type = "authNeededException"
+)
+
+data class CodeScanChatMessage(
+ @JsonProperty("tabID") override val tabId: String,
+ override val messageId: String? = UUID.randomUUID().toString(),
+ val messageType: ChatMessageType,
+ val message: String? = null,
+ val buttons: List? = null,
+ val formItems: List? = null,
+ val followUps: List? = null,
+ val isLoading: Boolean = false,
+ val canBeVoted: Boolean = true,
+ val clearPreviousItemButtons: Boolean = true,
+ val command: String? = null,
+) : CodeScanUiMessage(
+ messageId = messageId,
+ tabId = tabId,
+ type = "chatMessage",
+)
+
+data class UpdatePlaceholderMessage(
+ @JsonProperty("tabID") override val tabId: String,
+ val newPlaceholder: String,
+) : CodeScanUiMessage(
+ tabId = tabId,
+ type = "updatePlaceholderMessage"
+)
+
+data class CodeScanChatMessageContent(
+ val message: String? = null,
+ val buttons: List? = null,
+ val formItems: List? = null,
+ val followUps: List? = null,
+ val type: ChatMessageType,
+ val canBeVoted: Boolean = true,
+)
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/session/Session.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/session/Session.kt
new file mode 100644
index 0000000000..749e39fb24
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/session/Session.kt
@@ -0,0 +1,9 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqCodeScan.session
+
+data class Session(val tabId: String) {
+ var isAuthenticating: Boolean = false
+ var authNeededNotified: Boolean = false
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/storage/ChatSessionStorage.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/storage/ChatSessionStorage.kt
new file mode 100644
index 0000000000..fb05a7beda
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeScan/storage/ChatSessionStorage.kt
@@ -0,0 +1,33 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqCodeScan.storage
+
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.session.Session
+
+class ChatSessionStorage {
+ private val sessions = mutableMapOf()
+
+ private fun createSession(tabId: String): Session {
+ val session = Session(tabId)
+ sessions[tabId] = session
+ return session
+ }
+
+ fun getSession(tabId: String): Session = sessions[tabId] ?: createSession(tabId)
+
+ fun deleteSession(tabId: String) {
+ sessions.remove(tabId)
+ }
+
+ // Find all sessions that are currently waiting to be authenticated
+ fun getAuthenticatingSessions(): List = this.sessions.values.filter { it.isAuthenticating }
+
+ fun changeAuthenticationNeeded(isAuthenticating: Boolean) {
+ sessions.keys.forEach { sessions[it]?.isAuthenticating = isAuthenticating }
+ }
+
+ fun changeAuthenticationNeededNotified(authNeededNotified: Boolean) {
+ sessions.keys.forEach { sessions[it]?.authNeededNotified = authNeededNotified }
+ }
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/CodeTestChatApp.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/CodeTestChatApp.kt
new file mode 100644
index 0000000000..036a048273
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/CodeTestChatApp.kt
@@ -0,0 +1,87 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqCodeTest
+
+import com.intellij.openapi.application.ApplicationManager
+import kotlinx.coroutines.CoroutineScope
+import kotlinx.coroutines.launch
+import software.aws.toolkits.jetbrains.core.credentials.ToolkitConnection
+import software.aws.toolkits.jetbrains.core.credentials.ToolkitConnectionManagerListener
+import software.aws.toolkits.jetbrains.services.amazonq.apps.AmazonQApp
+import software.aws.toolkits.jetbrains.services.amazonq.apps.AmazonQAppInitContext
+import software.aws.toolkits.jetbrains.services.amazonq.messages.AmazonQMessage
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.auth.isCodeScanAvailable
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.auth.isCodeTestAvailable
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.controller.CodeTestChatController
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.messages.AuthenticationUpdateMessage
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.messages.IncomingCodeTestMessage
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.storage.ChatSessionStorage
+import software.aws.toolkits.jetbrains.services.amazonqDoc.auth.isDocAvailable
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.auth.isFeatureDevAvailable
+import software.aws.toolkits.jetbrains.services.codemodernizer.utils.isCodeTransformAvailable
+
+class CodeTestChatApp(private val scope: CoroutineScope) : AmazonQApp {
+
+ override val tabTypes = listOf("codetest")
+
+ override fun init(context: AmazonQAppInitContext) {
+ val chatSessionStorage = ChatSessionStorage()
+ val inboundAppMessagesHandler =
+ CodeTestChatController(context, chatSessionStorage, cs = scope)
+
+ context.messageTypeRegistry.register(
+ "clear" to IncomingCodeTestMessage.ClearChat::class,
+ "help" to IncomingCodeTestMessage.Help::class,
+ "chat-prompt" to IncomingCodeTestMessage.ChatPrompt::class,
+ "new-tab-was-created" to IncomingCodeTestMessage.NewTabCreated::class,
+ "tab-was-removed" to IncomingCodeTestMessage.TabRemoved::class,
+ "start-test-gen" to IncomingCodeTestMessage.StartTestGen::class,
+ "response-body-link-click" to IncomingCodeTestMessage.ClickedLink::class,
+ "button-click" to IncomingCodeTestMessage.ButtonClicked::class
+ )
+
+ scope.launch {
+ context.messagesFromUiToApp.flow.collect { message ->
+ // Launch a new coroutine to handle each message
+ scope.launch { handleMessage(message, inboundAppMessagesHandler) }
+ }
+ }
+
+ ApplicationManager.getApplication().messageBus.connect(this).subscribe(
+ ToolkitConnectionManagerListener.TOPIC,
+ object : ToolkitConnectionManagerListener {
+ override fun activeConnectionChanged(newConnection: ToolkitConnection?) {
+ scope.launch {
+ context.messagesFromAppToUi.publish(
+ AuthenticationUpdateMessage(
+ featureDevEnabled = isFeatureDevAvailable(context.project),
+ codeTransformEnabled = isCodeTransformAvailable(context.project),
+ codeScanEnabled = isCodeScanAvailable(context.project),
+ codeTestEnabled = isCodeTestAvailable(context.project),
+ docEnabled = isDocAvailable(context.project),
+ authenticatingTabIDs = chatSessionStorage.getAuthenticatingSessions().map { it.tabId }
+ )
+ )
+ }
+ }
+ }
+ )
+ }
+
+ private suspend fun handleMessage(message: AmazonQMessage, inboundAppMessagesHandler: InboundAppMessagesHandler) {
+ when (message) {
+ is IncomingCodeTestMessage.ClearChat -> inboundAppMessagesHandler.processClearQuickAction(message)
+ is IncomingCodeTestMessage.Help -> inboundAppMessagesHandler.processHelpQuickAction(message)
+ is IncomingCodeTestMessage.ChatPrompt -> inboundAppMessagesHandler.processPromptChatMessage(message)
+ is IncomingCodeTestMessage.NewTabCreated -> inboundAppMessagesHandler.processNewTabCreatedMessage(message)
+ is IncomingCodeTestMessage.TabRemoved -> inboundAppMessagesHandler.processTabRemovedMessage(message)
+ is IncomingCodeTestMessage.StartTestGen -> inboundAppMessagesHandler.processStartTestGen(message)
+ is IncomingCodeTestMessage.ClickedLink -> inboundAppMessagesHandler.processLinkClick(message)
+ is IncomingCodeTestMessage.ButtonClicked -> inboundAppMessagesHandler.processButtonClickedMessage(message)
+ }
+ }
+
+ override fun dispose() {
+ }
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/CodeTestChatAppFactory.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/CodeTestChatAppFactory.kt
new file mode 100644
index 0000000000..46ccd461d9
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/CodeTestChatAppFactory.kt
@@ -0,0 +1,11 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqCodeTest
+import com.intellij.openapi.project.Project
+import kotlinx.coroutines.CoroutineScope
+import software.aws.toolkits.jetbrains.services.amazonq.apps.AmazonQAppFactory
+
+class CodeTestChatAppFactory(private val cs: CoroutineScope) : AmazonQAppFactory {
+ override fun createApp(project: Project) = CodeTestChatApp(cs)
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/CodeTestChatItems.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/CodeTestChatItems.kt
new file mode 100644
index 0000000000..4ed35e3506
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/CodeTestChatItems.kt
@@ -0,0 +1,38 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqCodeTest
+
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.messages.Button
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.messages.CodeTestButtonId
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.messages.ProgressField
+import software.aws.toolkits.resources.message
+
+val cancellingProgressField = ProgressField(
+ status = "warning",
+ text = message("general.canceling"),
+ value = -1,
+ actions = emptyList()
+)
+
+// TODO: Need to change the string after the F2F
+val testGenCompletedField = ProgressField(
+ status = "success",
+ text = message("general.success"),
+ value = 100,
+ actions = emptyList()
+)
+
+val cancelTestGenButton = Button(
+ id = CodeTestButtonId.StopTestGeneration.id,
+ text = message("general.cancel"),
+ icon = "cancel"
+)
+
+fun testGenProgressField(value: Int) = ProgressField(
+ status = "default",
+ text = message("testgen.progressbar.generate_unit_tests"),
+ value = value,
+ valueText = "$value%",
+ actions = listOf(cancelTestGenButton)
+)
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/CodeTestConstants.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/CodeTestConstants.kt
new file mode 100644
index 0000000000..e5e25ed7cb
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/CodeTestConstants.kt
@@ -0,0 +1,19 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqCodeTest
+
+const val FEATURE_NAME = "Amazon Q Unit Test Generation"
+
+enum class ConversationState {
+ IDLE,
+ WAITING_FOR_BUILD_COMMAND_INPUT,
+ WAITING_FOR_REGENERATE_INPUT,
+ IN_PROGRESS,
+}
+
+fun generateSummaryMessage(fileName: String): String = """
+ Sure. This may take a few minutes. I'll share updates here as I work on this.
+ **Generating unit tests for the following methods in $fileName**:
+
+""".trimIndent()
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/CodeWhispererCodeTestSession.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/CodeWhispererCodeTestSession.kt
new file mode 100644
index 0000000000..127106f9cd
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/CodeWhispererCodeTestSession.kt
@@ -0,0 +1,146 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqCodeTest
+import com.intellij.openapi.project.Project
+import kotlinx.coroutines.ensureActive
+import kotlinx.coroutines.time.withTimeout
+import software.amazon.awssdk.services.codewhispererruntime.model.CreateUploadUrlResponse
+import software.aws.toolkits.core.utils.debug
+import software.aws.toolkits.core.utils.getLogger
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.controller.CodeTestChatHelper
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.messages.CodeTestChatMessageContent
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.model.PreviousUTGIterationContext
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.sessionconfig.PayloadContext
+import software.aws.toolkits.jetbrains.services.codewhisperer.codetest.sessionconfig.CodeTestSessionConfig
+import software.aws.toolkits.jetbrains.services.codewhisperer.model.CreateUploadUrlServiceInvocationContext
+import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererConstants
+import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererConstants.TOTAL_BYTES_IN_KB
+import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererConstants.TOTAL_MILLIS_IN_SECOND
+import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererZipUploadManager
+import software.aws.toolkits.jetbrains.services.cwc.messages.ChatMessageType
+import software.aws.toolkits.jetbrains.utils.assertIsNonDispatchThread
+import java.nio.file.Path
+import java.time.Duration
+import java.time.Instant
+import java.util.UUID
+import kotlin.coroutines.coroutineContext
+
+// TODO: Refactor with CodeWhispererCodeScanSession code since both are about zip CreateUploadUrl logic
+class CodeWhispererCodeTestSession(val sessionContext: CodeTestSessionContext) {
+ private fun now() = Instant.now().toEpochMilli()
+
+ /**
+ * Run UTG sessions are follow steps:
+ * 1. Zipping project
+ * 2. Creating Upload url & Upload to S3 bucket
+ */
+ suspend fun run(codeTestChatHelper: CodeTestChatHelper, previousIterationContext: PreviousUTGIterationContext?): CodeTestResponseContext {
+ try {
+ assertIsNonDispatchThread()
+ coroutineContext.ensureActive()
+
+ val path = sessionContext.sessionConfig.getRelativePath()
+ ?: throw RuntimeException("Can not determine current file path for adding unit tests")
+
+ // Add card answer to show UTG in progress
+ val testSummaryMessageId =
+ if (previousIterationContext == null) {
+ codeTestChatHelper.addAnswer(
+ CodeTestChatMessageContent(
+ message = generateSummaryMessage(path.fileName.toString()),
+ type = ChatMessageType.AnswerStream
+ )
+ ).also {
+ // For streaming effect
+ codeTestChatHelper.updateAnswer(
+ CodeTestChatMessageContent(type = ChatMessageType.AnswerPart)
+ )
+ codeTestChatHelper.updateUI(
+ loadingChat = true,
+ promptInputDisabledState = true
+ )
+ if (it == null) {
+ throw RuntimeException("Can not add test summary card")
+ }
+ }
+ } else {
+ // non-first iteration doesn't have a test summary card
+ null
+ }
+
+ val (payloadContext, sourceZip) = withTimeout(Duration.ofSeconds(sessionContext.sessionConfig.createPayloadTimeoutInSeconds())) {
+ sessionContext.sessionConfig.createPayload()
+ }
+
+ LOG.debug {
+ "Total size of source payload in KB: ${payloadContext.srcPayloadSize * 1.0 / TOTAL_BYTES_IN_KB} \n" +
+ "Total size of build payload in KB: ${(payloadContext.buildPayloadSize ?: 0) * 1.0 / TOTAL_BYTES_IN_KB} \n" +
+ "Total size of source zip file in KB: ${payloadContext.srcZipFileSize * 1.0 / TOTAL_BYTES_IN_KB} \n" +
+ "Total number of lines included: ${payloadContext.totalLines} \n" +
+ "Total number of files included in payload: ${payloadContext.totalFiles} \n" +
+ "Total time taken for creating payload: ${payloadContext.totalTimeInMilliseconds * 1.0 / TOTAL_MILLIS_IN_SECOND} seconds\n" +
+ "Payload context language: ${payloadContext.language}"
+ }
+
+ // 2 & 3. CreateUploadURL and upload the context.
+ val artifactsUploadStartTime = now()
+ val taskName = UUID.randomUUID().toString()
+ val sourceZipUploadResponse =
+ CodeWhispererZipUploadManager.getInstance(sessionContext.project).createUploadUrlAndUpload(
+ sourceZip,
+ "SourceCode",
+ CodeWhispererConstants.UploadTaskType.UTG,
+ taskName,
+ CodeWhispererConstants.FeatureName.TEST_GENERATION
+ )
+
+ sourceZipUploadResponse.uploadId()
+
+ LOG.debug {
+ "Successfully uploaded source zip to s3: " +
+ "Upload id: ${sourceZipUploadResponse.uploadId()} " +
+ "Request id: ${sourceZipUploadResponse.responseMetadata().requestId()}"
+ }
+ val artifactsUploadDuration = now() - artifactsUploadStartTime
+
+ val codeTestResponseContext = CodeTestResponseContext(
+ payloadContext,
+ CreateUploadUrlServiceInvocationContext(artifactsUploadDuration = artifactsUploadDuration),
+ path,
+ sourceZipUploadResponse,
+ testSummaryMessageId
+ )
+
+ return codeTestResponseContext
+ } catch (e: Exception) {
+ LOG.debug(e) { "Error while creating zip and uploading to S3" }
+ throw e
+ }
+ }
+
+ companion object {
+ private val LOG = getLogger()
+ }
+}
+
+sealed class CodeTestResponse {
+ abstract val responseContext: CodeTestResponseContext
+ data class Success(val message: String, override val responseContext: CodeTestResponseContext) : CodeTestResponse()
+
+ data class Error(val errorMessage: String, override val responseContext: CodeTestResponseContext) : CodeTestResponse()
+}
+
+data class CodeTestSessionContext(
+ val project: Project,
+ val sessionConfig: CodeTestSessionConfig,
+)
+
+data class CodeTestResponseContext(
+ val payloadContext: PayloadContext,
+ val serviceInvocationContext: CreateUploadUrlServiceInvocationContext,
+ val currentFileRelativePath: Path,
+ val createUploadUrlResponse: CreateUploadUrlResponse,
+ val testSummaryMessageId: String?,
+ val reason: String? = null,
+)
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/CodeWhispererUTGChatManager.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/CodeWhispererUTGChatManager.kt
new file mode 100644
index 0000000000..e1419c4c10
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/CodeWhispererUTGChatManager.kt
@@ -0,0 +1,601 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqCodeTest
+
+import com.fasterxml.jackson.core.JsonParseException
+import com.fasterxml.jackson.module.kotlin.jacksonObjectMapper
+import com.intellij.openapi.application.ApplicationManager
+import com.intellij.openapi.components.Service
+import com.intellij.openapi.components.service
+import com.intellij.openapi.fileEditor.FileDocumentManager
+import com.intellij.openapi.fileEditor.FileEditorManager
+import com.intellij.openapi.fileEditor.impl.FileDocumentManagerImpl
+import com.intellij.openapi.project.Project
+import kotlinx.coroutines.CoroutineScope
+import kotlinx.coroutines.Job
+import kotlinx.coroutines.delay
+import kotlinx.coroutines.launch
+import software.amazon.awssdk.core.exception.SdkServiceException
+import software.amazon.awssdk.services.codewhispererruntime.model.GetTestGenerationResponse
+import software.amazon.awssdk.services.codewhispererruntime.model.Range
+import software.amazon.awssdk.services.codewhispererruntime.model.StartTestGenerationResponse
+import software.amazon.awssdk.services.codewhispererruntime.model.TargetCode
+import software.amazon.awssdk.services.codewhispererruntime.model.TestGenerationJobStatus
+import software.amazon.awssdk.services.codewhispererstreaming.model.ExportContext
+import software.amazon.awssdk.services.codewhispererstreaming.model.ExportIntent
+import software.aws.toolkits.core.utils.debug
+import software.aws.toolkits.core.utils.error
+import software.aws.toolkits.core.utils.getLogger
+import software.aws.toolkits.core.utils.info
+import software.aws.toolkits.jetbrains.services.amazonq.clients.AmazonQStreamingClient
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.controller.CodeTestChatHelper
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.messages.Button
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.messages.CodeTestChatMessageContent
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.model.PreviousUTGIterationContext
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.model.ShortAnswer
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.session.BuildAndExecuteProgressStatus
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.session.Session
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.utils.combineBuildAndExecuteLogFiles
+import software.aws.toolkits.jetbrains.services.codemodernizer.utils.calculateTotalLatency
+import software.aws.toolkits.jetbrains.services.codewhisperer.codetest.CodeTestException
+import software.aws.toolkits.jetbrains.services.codewhisperer.codetest.sessionconfig.CodeTestSessionConfig
+import software.aws.toolkits.jetbrains.services.codewhisperer.codetest.testGenStoppedError
+import software.aws.toolkits.jetbrains.services.codewhisperer.credentials.CodeWhispererClientAdaptor
+import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererConstants
+import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererUtil.promptReAuth
+import software.aws.toolkits.jetbrains.services.codewhisperer.util.getTelemetryErrorMessage
+import software.aws.toolkits.jetbrains.services.cwc.controller.chat.telemetry.getStartUrl
+import software.aws.toolkits.jetbrains.services.cwc.messages.ChatMessageType
+import software.aws.toolkits.jetbrains.services.cwc.messages.CodeReference
+import software.aws.toolkits.jetbrains.settings.CodeWhispererSettings
+import software.aws.toolkits.jetbrains.utils.isQConnected
+import software.aws.toolkits.resources.message
+import software.aws.toolkits.telemetry.AmazonqTelemetry
+import software.aws.toolkits.telemetry.MetricResult
+import java.io.ByteArrayInputStream
+import java.io.ByteArrayOutputStream
+import java.io.File
+import java.io.IOException
+import java.nio.file.Paths
+import java.time.Instant
+import java.util.concurrent.atomic.AtomicBoolean
+import java.util.zip.ZipInputStream
+
+@Service
+class CodeWhispererUTGChatManager(val project: Project, private val cs: CoroutineScope) {
+ // TODO: consider combining this with session.isGeneratingTests
+ private val isUTGInProgress = AtomicBoolean(false)
+ private val mapper = jacksonObjectMapper()
+ private val generatedTestDiffs = mutableMapOf()
+
+ private fun throwIfCancelled(session: Session) {
+ if (!session.isGeneratingTests) {
+ testGenStoppedError()
+ }
+ }
+
+ private suspend fun launchTestGenFlow(
+ prompt: String,
+ codeTestChatHelper: CodeTestChatHelper,
+ previousIterationContext: PreviousUTGIterationContext?,
+ selectionRange: Range?,
+ ) {
+ // 1st API call: Zip project and call CreateUploadUrl
+ val session = codeTestChatHelper.getActiveSession()
+ session.isGeneratingTests = true
+ session.iteration++
+
+ // Set the Progress bar to "Generating unit tests..."
+ codeTestChatHelper.updateUI(
+ promptInputDisabledState = true,
+ promptInputProgress = testGenProgressField(0),
+ )
+
+ val codeTestResponseContext = createUploadUrl(codeTestChatHelper, previousIterationContext)
+ session.srcPayloadSize = codeTestResponseContext.payloadContext.srcPayloadSize
+ session.srcZipFileSize = codeTestResponseContext.payloadContext.srcZipFileSize
+ session.artifactUploadDuration = codeTestResponseContext.serviceInvocationContext.artifactsUploadDuration
+ val path = codeTestResponseContext.currentFileRelativePath
+
+ val createUploadUrlResponse = codeTestResponseContext.createUploadUrlResponse ?: return
+ throwIfCancelled(session)
+
+ LOG.debug {
+ "Q TestGen StartTestGenerationRequest: TabId= ${codeTestChatHelper.getActiveCodeTestTabId()}: " +
+ "uploadId: ${createUploadUrlResponse.uploadId()}, relativeTargetPath: ${codeTestResponseContext.currentFileRelativePath}, " +
+ "selectionRange: $selectionRange, "
+ }
+
+ // 2nd API call: StartTestGeneration
+ val startTestGenerationResponse = try {
+ startTestGeneration(
+ uploadId = createUploadUrlResponse.uploadId(),
+ targetCode = listOf(
+ TargetCode.builder()
+ .relativeTargetPath(codeTestResponseContext.currentFileRelativePath.toString())
+ .targetLineRangeList(
+ if (selectionRange != null) {
+ listOf(
+ selectionRange
+ )
+ } else {
+ emptyList()
+ }
+ )
+ .build()
+ ),
+ userInput = prompt
+ )
+ } catch (e: Exception) {
+ val statusCode = when {
+ e is SdkServiceException -> e.statusCode()
+ else -> 400
+ }
+ LOG.error(e) { "Unexpected error while creating test generation job" }
+ val errorMessage = getTelemetryErrorMessage(e, CodeWhispererConstants.FeatureName.TEST_GENERATION)
+ throw CodeTestException(
+ "CreateTestJobError: $errorMessage",
+ "CreateTestJobError",
+ message("testgen.error.generic_technical_error_message")
+ )
+ }
+
+ val job = startTestGenerationResponse.testGenerationJob()
+ session.startTestGenerationRequestId = startTestGenerationResponse.responseMetadata().requestId()
+ session.testGenerationJobGroupName = job.testGenerationJobGroupName()
+ session.testGenerationJob = job.testGenerationJobId()
+ throwIfCancelled(session)
+
+ // 3rd API call: Step 3: Polling mechanism on test job status with getTestGenStatus getTestGeneration
+ var finished = false
+ var testGenerationResponse: GetTestGenerationResponse? = null
+
+ var shortAnswer = ShortAnswer()
+ LOG.debug {
+ "Q TestGen session: ${codeTestChatHelper.getActiveCodeTestTabId()}: " +
+ "polling result for id: ${job.testGenerationJobId()}, group name: ${job.testGenerationJobGroupName()}, " +
+ "request id: ${startTestGenerationResponse.responseMetadata().requestId()}"
+ }
+
+ while (!finished) {
+ throwIfCancelled(session)
+ testGenerationResponse = getTestGenerationStatus(job.testGenerationJobId(), job.testGenerationJobGroupName())
+
+ val status = testGenerationResponse.testGenerationJob().status()
+ if (status == TestGenerationJobStatus.COMPLETED) {
+ LOG.debug {
+ "Q TestGen session: ${codeTestChatHelper.getActiveCodeTestTabId()}: " +
+ "Test generation completed, short answer string: ${testGenerationResponse.testGenerationJob().shortAnswer()}"
+ }
+ finished = true
+ if (testGenerationResponse.testGenerationJob().shortAnswer() != null) {
+ shortAnswer = parseShortAnswerString(testGenerationResponse.testGenerationJob().shortAnswer())
+
+ val testFileName = shortAnswer.testFilePath?.let { File(it).name }.orEmpty()
+ session.testFileName = testFileName
+ // Setting default value to 0 if the value is null or invalid
+ session.numberOfUnitTestCasesGenerated = shortAnswer.numberOfTestMethods
+ session.testFileRelativePathToProjectRoot = getTestFilePathRelativeToRoot(shortAnswer)
+
+ // update test summary card in success case
+ if (previousIterationContext == null) {
+ codeTestChatHelper.updateAnswer(
+ CodeTestChatMessageContent(
+ message = generateSummaryMessage(path.fileName.toString()) + shortAnswer.planSummary,
+ type = ChatMessageType.Answer,
+ footer = listOf(testFileName)
+ ),
+ messageIdOverride = codeTestResponseContext.testSummaryMessageId
+ )
+ }
+ // update test summary card
+ } else {
+ // If job status is Completed and has no ShortAnswer then there might be some issue in the backend.
+ throw CodeTestException(
+ "TestGenFailedError: " + message("testgen.message.failed"),
+ "TestGenFailedError",
+ message("testgen.error.generic_technical_error_message")
+ )
+ }
+ } else if (status == TestGenerationJobStatus.FAILED) {
+ LOG.debug {
+ "Q TestGen session: ${codeTestChatHelper.getActiveCodeTestTabId()}: " +
+ "Test generation failed, short answer string: ${testGenerationResponse.testGenerationJob().shortAnswer()}"
+ }
+ if (testGenerationResponse.testGenerationJob().shortAnswer() != null) {
+ shortAnswer = parseShortAnswerString(testGenerationResponse.testGenerationJob().shortAnswer())
+ if (shortAnswer.stopIteration == "true") {
+ throw CodeTestException("TestGenFailedError: ${shortAnswer.planSummary}", "TestGenFailedError", shortAnswer.planSummary)
+ }
+ }
+
+ // If job status is Failed and has no ShortAnswer then there might be some issue in the backend.
+ throw CodeTestException(
+ "TestGenFailedError: " + message("testgen.message.failed"),
+ "TestGenFailedError",
+ message("testgen.error.generic_technical_error_message")
+ )
+ } else {
+ // In progress
+ LOG.debug {
+ "Q TestGen session: ${codeTestChatHelper.getActiveCodeTestTabId()}: " +
+ "Test generation in progress, progress rate ${testGenerationResponse.testGenerationJob().progressRate()}}"
+ }
+ val progressRate = testGenerationResponse.testGenerationJob().progressRate() ?: 0
+
+ if (previousIterationContext == null && testGenerationResponse.testGenerationJob().shortAnswer() != null) {
+ shortAnswer = parseShortAnswerString(testGenerationResponse.testGenerationJob().shortAnswer())
+ if (shortAnswer.stopIteration == "true") {
+ throw CodeTestException("TestGenFailedError: ${shortAnswer.planSummary}", "TestGenFailedError", shortAnswer.planSummary)
+ }
+ codeTestChatHelper.updateAnswer(
+ CodeTestChatMessageContent(
+ message = generateSummaryMessage(path.fileName.toString()) + shortAnswer.planSummary,
+ type = ChatMessageType.Answer
+ ),
+ messageIdOverride = codeTestResponseContext.testSummaryMessageId
+ )
+ }
+ codeTestChatHelper.updateUI(
+ promptInputDisabledState = true,
+ promptInputProgress = testGenProgressField(progressRate),
+ )
+ }
+
+ // polling every 2 seconds to reduce # of API calls
+ delay(2000)
+ }
+
+ throwIfCancelled(session)
+
+ // 4th API call: Step 4: ExportResultsArchive
+ val byteArray = AmazonQStreamingClient.getInstance(project).exportResultArchive(
+ createUploadUrlResponse.uploadId(),
+ ExportIntent.UNIT_TESTS,
+ ExportContext.fromUnitTestGenerationExportContext {
+ it.testGenerationJobId(job.testGenerationJobId())
+ it.testGenerationJobGroupName(job.testGenerationJobGroupName())
+ },
+ { e ->
+ LOG.error(e) { "ExportResultArchive failed: ${e.message}" }
+ throw CodeTestException(
+ "ExportResultsArchiveError: ${e.message}",
+ "ExportResultsArchiveError",
+ message("testgen.error.generic_technical_error_message")
+ )
+ },
+ { startTime ->
+ LOG.info { "ExportResultArchive latency: ${calculateTotalLatency(startTime, Instant.now())}" }
+ }
+ )
+ val result = byteArray.reduce { acc, next -> acc + next } // To map the result it is needed to combine the full byte array
+ storeGeneratedTestDiffs(result, session)
+ if (!session.isGeneratingTests) {
+ // TODO: Modify text according to FnF
+ codeTestChatHelper.addAnswer(
+ CodeTestChatMessageContent(
+ message = message("testgen.message.failed"),
+ type = ChatMessageType.Answer,
+ canBeVoted = true
+ )
+ )
+ return
+ }
+
+ val codeReference = shortAnswer.codeReferences?.map { ref ->
+ CodeReference(
+ licenseName = ref.licenseName,
+ url = ref.url,
+ information = "${ref.licenseName} - ${ref.repository} "
+ )
+ }
+ shortAnswer.codeReferences?.let { session.codeReferences = it }
+ val isReferenceAllowed = CodeWhispererSettings.getInstance().isIncludeCodeWithReference()
+ if (!isReferenceAllowed && codeReference?.isNotEmpty() == true) {
+ codeTestChatHelper.addAnswer(
+ CodeTestChatMessageContent(
+ message = """
+ Your settings do not allow code generation with references.
+ """.trimIndent(),
+ type = ChatMessageType.Answer,
+ )
+ )
+ } else {
+ if (previousIterationContext == null) {
+ // show another card as the answer
+ val viewDiffMessageId = codeTestChatHelper.addAnswer(
+ CodeTestChatMessageContent(
+ message = """
+ Please see the unit tests generated below. Click "View Diff" to review the changes in the code editor.
+ """.trimIndent(),
+ type = ChatMessageType.Answer,
+ buttons = listOf(Button("utg_view_diff", "View Diff", keepCardAfterClick = true, position = "outside", status = "info")),
+ fileList = listOf(getTestFilePathRelativeToRoot(shortAnswer)),
+ projectRootName = project.name,
+ canBeVoted = true,
+ codeReference = codeReference
+ )
+ )
+ session.viewDiffMessageId = viewDiffMessageId
+ codeTestChatHelper.updateUI(
+ promptInputDisabledState = false,
+ promptInputPlaceholder = "Specify a function(s) in the current file(optional)",
+ promptInputProgress = testGenCompletedField,
+ )
+ } else {
+ codeTestChatHelper.updateAnswer(
+ CodeTestChatMessageContent(
+ type = ChatMessageType.Answer,
+ buttons = listOf(Button("utg_view_diff", "View Diff", keepCardAfterClick = true, position = "outside", status = "info")),
+ fileList = listOf(getTestFilePathRelativeToRoot(shortAnswer)),
+ projectRootName = project.name,
+ codeReference = codeReference
+ ),
+ messageIdOverride = previousIterationContext.buildAndExecuteMessageId
+ )
+ session.viewDiffMessageId = previousIterationContext.buildAndExecuteMessageId
+ codeTestChatHelper.updateUI(
+ loadingChat = false,
+ )
+ }
+ codeTestChatHelper.updateUI(
+ promptInputDisabledState = true,
+ promptInputPlaceholder = message("testgen.placeholder.view_diff"),
+ promptInputProgress = testGenCompletedField,
+ )
+ delay(1000)
+ }
+
+ codeTestChatHelper.sendUpdatePromptProgress(codeTestChatHelper.getActiveSession().tabId, null)
+ }
+
+ // Input: test file path relative to project root's parent .
+ // Output: test file path relative to project root.
+ // shortAnswer.testFilePath has a format of /.
+ // test file path in generatedTestDiffs map has a format of resultArtifacts/.
+ // both needs to be handled the same way which is remove the first sub-directory
+ private fun getTestFilePathRelativeToRoot(shortAnswer: ShortAnswer): String {
+ val pathString = shortAnswer.testFilePath ?: generatedTestDiffs.keys.firstOrNull() ?: throw RuntimeException("No test file path found")
+ val path = Paths.get(pathString)
+ val updatedPath = path.subpath(1, path.nameCount).toString()
+ return updatedPath
+ }
+
+ private fun parseShortAnswerString(shortAnswerString: String): ShortAnswer {
+ // Step 1: Replace single quotes with double quotes
+ var jsonString = shortAnswerString.replace("'", "\"").replace("```", "")
+
+ // Step 2: Replace Python's None with JSON's null
+ jsonString = jsonString.replace(": None", ": null")
+
+ // Step 3: remove extra quotes in the head and tail
+ if (jsonString.startsWith("\"") && jsonString.endsWith("\"")) {
+ jsonString = jsonString.substring(1, jsonString.length - 1) // Remove the first and last quote
+ }
+
+ // Step 4: unescape it
+ jsonString = jsonString.replace("\\\"", "\"")
+ .replace("\\\\", "\\")
+ // Deserialize JSON to Kotlin data class
+ try {
+ val shortAnswer: ShortAnswer = mapper.readValue(jsonString, ShortAnswer::class.java)
+ return shortAnswer
+ } catch (e: JsonParseException) {
+ LOG.debug(e) { "Test Generation JSON parsing error: ${e.message}" }
+ throw e
+ } catch (e: Exception) {
+ LOG.debug(e) { "Error parsing JSON" }
+ throw e
+ }
+ }
+
+ private fun storeGeneratedTestDiffs(byteArray: ByteArray, session: Session) {
+ try {
+ val byteArrayInputStream = ByteArrayInputStream(byteArray)
+ ZipInputStream(byteArrayInputStream).use { zipInputStream ->
+ var zipEntry = zipInputStream.nextEntry
+
+ while (zipEntry != null) {
+ if (zipEntry.isDirectory) {
+ zipInputStream.closeEntry()
+ zipEntry = zipInputStream.nextEntry
+ // We are only interested in test file diff in zip entries
+ continue
+ }
+
+ val baos = ByteArrayOutputStream()
+ val buffer = ByteArray(1024)
+ var len: Int
+
+ while (zipInputStream.read(buffer).also { len = it } > 0) {
+ baos.write(buffer, 0, len)
+ }
+
+ val fileContent = baos.toByteArray()
+ if (fileContent.toString(Charsets.UTF_8).isEmpty()) {
+ session.isGeneratingTests = false
+ return
+ }
+ val zipEntryPath = Paths.get(zipEntry.name)
+
+ // relative path to project root
+ val updatedZipEntryPath = zipEntryPath.subpath(1, zipEntryPath.nameCount).toString()
+ session.generatedTestDiffs[updatedZipEntryPath] = fileContent.toString(Charsets.UTF_8)
+
+ zipInputStream.closeEntry()
+ zipEntry = zipInputStream.nextEntry
+ }
+ }
+ } catch (e: IOException) {
+ LOG.debug(e) { "Error reading ZIP entries" }
+ throw e
+ }
+ }
+
+ private suspend fun createUploadUrl(
+ codeTestChatHelper: CodeTestChatHelper,
+ previousIterationContext: PreviousUTGIterationContext?,
+ ): CodeTestResponseContext {
+ throwIfCancelled(codeTestChatHelper.getActiveSession())
+ val file =
+ if (previousIterationContext == null) {
+ FileEditorManager.getInstance(project).selectedEditor?.file.also {
+ codeTestChatHelper.getActiveSession().selectedFile = it
+ }
+ } else {
+ previousIterationContext.selectedFile
+ }
+
+ val combinedBuildAndExecuteLogFile = combineBuildAndExecuteLogFiles(
+ previousIterationContext?.buildLogFile,
+ previousIterationContext?.testLogFile
+ )
+ val codeTestSessionConfig = CodeTestSessionConfig(file, project, combinedBuildAndExecuteLogFile)
+ codeTestChatHelper.getActiveSession().projectRoot = codeTestSessionConfig.projectRoot.path
+
+ val codeTestSessionContext = CodeTestSessionContext(project, codeTestSessionConfig)
+ val codeWhispererCodeTestSession = CodeWhispererCodeTestSession(codeTestSessionContext)
+ return codeWhispererCodeTestSession.run(codeTestChatHelper, previousIterationContext)
+ }
+
+ private fun startTestGeneration(uploadId: String, targetCode: List, userInput: String): StartTestGenerationResponse =
+ CodeWhispererClientAdaptor.getInstance(project).startTestGeneration(uploadId, targetCode, userInput)
+
+ private fun getTestGenerationStatus(jobId: String, jobGroupName: String): GetTestGenerationResponse =
+ CodeWhispererClientAdaptor.getInstance(project).getTestGeneration(jobId, jobGroupName)
+
+ /**
+ * Returns true if the UTG is in progress.
+ * This function will return true for a cancelled UTG job which is in cancellation state.
+ */
+ fun isUTGInProgress(): Boolean = isUTGInProgress.get()
+
+ private fun beforeTestGenFlow(session: Session) {
+ resetTestGenFlowSession(session)
+ session.isGeneratingTests = true
+ isUTGInProgress.set(true)
+ // Show in progress indicator
+
+ ApplicationManager.getApplication().invokeLater {
+ (FileDocumentManager.getInstance() as FileDocumentManagerImpl).saveAllDocuments(false)
+ }
+ }
+
+ private fun resetTestGenFlowSession(session: Session) {
+ // session.selectedFile doesn't need to be reset since it will remain unchanged
+ session.conversationState = ConversationState.IN_PROGRESS
+ session.shortAnswer = ShortAnswer()
+ session.openedDiffFile = null
+ session.testFileRelativePathToProjectRoot = ""
+ session.testFileName = ""
+ session.openedDiffFile = null
+ session.generatedTestDiffs.clear()
+ session.buildAndExecuteTaskContext.apply {
+ buildExitCode = -1
+ testExitCode = -1
+ progressStatus = BuildAndExecuteProgressStatus.START_STEP
+ }
+ }
+
+ private fun afterTestGenFlow() {
+ isUTGInProgress.set(false)
+ }
+
+ /**
+ * Triggers a unit test generation flow based on current open file.
+ */
+ fun generateTests(
+ prompt: String,
+ codeTestChatHelper: CodeTestChatHelper,
+ previousIterationContext: PreviousUTGIterationContext?,
+ selectionRange: Range?,
+ ): Job? {
+ val shouldStart = performTestGenPreChecks()
+ val session = codeTestChatHelper.getActiveSession()
+ if (!shouldStart) {
+ session.conversationState = ConversationState.IDLE
+ return null
+ }
+
+ beforeTestGenFlow(session)
+
+ return cs.launch {
+ try {
+ launchTestGenFlow(prompt, codeTestChatHelper, previousIterationContext, selectionRange)
+ } catch (e: Exception) {
+ // reset number of unitTestGenerated to null
+ session.numberOfUnitTestCasesGenerated = null
+ // Add an answer for displaying error message
+ val errorMessage = when {
+ e is CodeTestException &&
+ e.message?.startsWith("CreateTestJobError: Maximum") == true ->
+ message("testgen.error.maximum_generations_reach")
+
+ e is CodeTestException -> e.uiMessage
+ e is JsonParseException -> message("testgen.error.generic_technical_error_message")
+ else -> message("testgen.error.generic_error_message")
+ }
+
+ codeTestChatHelper.addAnswer(
+ CodeTestChatMessageContent(
+ message = errorMessage,
+ type = ChatMessageType.Answer,
+ canBeVoted = false
+ )
+ )
+
+ AmazonqTelemetry.utgGenerateTests(
+ cwsprChatProgrammingLanguage = session.programmingLanguage.languageId,
+ hasUserPromptSupplied = session.hasUserPromptSupplied,
+ isFileInWorkspace = true,
+ isSupportedLanguage = true,
+ credentialStartUrl = getStartUrl(project),
+ jobGroup = session.testGenerationJobGroupName,
+ jobId = session.testGenerationJob,
+ result = if (e.message == message("testgen.message.cancelled")) MetricResult.Cancelled else MetricResult.Failed,
+ reason = (e as CodeTestException).code ?: "DefaultError",
+ reasonDesc = if (e.message == message("testgen.message.cancelled")) "${e.code}: ${e.message}" else e.message,
+ perfClientLatency = (Instant.now().toEpochMilli() - session.startTimeOfTestGeneration),
+ isCodeBlockSelected = session.isCodeBlockSelected,
+ artifactsUploadDuration = session.artifactUploadDuration,
+ buildPayloadBytes = session.srcPayloadSize,
+ buildZipFileBytes = session.srcZipFileSize,
+ requestId = session.startTestGenerationRequestId
+ )
+ session.isGeneratingTests = false
+ } finally {
+ // Reset the flow if there is any error
+ if (!session.isGeneratingTests) {
+ codeTestChatHelper.updateUI(
+ promptInputProgress = cancellingProgressField
+ )
+ delay(1000)
+ codeTestChatHelper.sendUpdatePromptProgress(session.tabId, null)
+ codeTestChatHelper.deleteSession(session.tabId)
+ codeTestChatHelper.updateUI(
+ promptInputDisabledState = false,
+ promptInputPlaceholder = message("testgen.placeholder.newtab"),
+ )
+ }
+ session.isGeneratingTests = false
+ session.conversationState = ConversationState.IDLE
+ afterTestGenFlow()
+ // send message displaying card
+ }
+ }
+ }
+
+ private fun performTestGenPreChecks(): Boolean {
+ if (!isQConnected(project)) return false
+ if (isUTGInProgress()) return false
+ val connectionExpired = promptReAuth(project)
+ if (connectionExpired) return false
+ return true
+ }
+
+ companion object {
+ fun getInstance(project: Project) = project.service()
+ private val LOG = getLogger()
+ }
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/InboundAppMessagesHandler.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/InboundAppMessagesHandler.kt
new file mode 100644
index 0000000000..1c41befeb9
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/InboundAppMessagesHandler.kt
@@ -0,0 +1,24 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqCodeTest
+
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.messages.IncomingCodeTestMessage
+
+interface InboundAppMessagesHandler {
+ suspend fun processPromptChatMessage(message: IncomingCodeTestMessage.ChatPrompt)
+
+ suspend fun processStartTestGen(message: IncomingCodeTestMessage.StartTestGen)
+
+ suspend fun processLinkClick(message: IncomingCodeTestMessage.ClickedLink)
+
+ suspend fun processNewTabCreatedMessage(message: IncomingCodeTestMessage.NewTabCreated)
+
+ suspend fun processClearQuickAction(message: IncomingCodeTestMessage.ClearChat)
+
+ suspend fun processHelpQuickAction(message: IncomingCodeTestMessage.Help)
+
+ suspend fun processTabRemovedMessage(message: IncomingCodeTestMessage.TabRemoved)
+
+ suspend fun processButtonClickedMessage(message: IncomingCodeTestMessage.ButtonClicked)
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/auth/CodeTestAuthUtils.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/auth/CodeTestAuthUtils.kt
new file mode 100644
index 0000000000..bd8161d53a
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/auth/CodeTestAuthUtils.kt
@@ -0,0 +1,11 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqCodeTest.auth
+
+import com.intellij.openapi.project.Project
+import software.aws.toolkits.jetbrains.core.gettingstarted.editor.ActiveConnection
+import software.aws.toolkits.jetbrains.core.gettingstarted.editor.BearerTokenFeatureSet
+import software.aws.toolkits.jetbrains.core.gettingstarted.editor.checkBearerConnectionValidity
+
+fun isCodeTestAvailable(project: Project): Boolean = checkBearerConnectionValidity(project, BearerTokenFeatureSet.Q) is ActiveConnection.ValidBearer
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/controller/CodeTestChatController.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/controller/CodeTestChatController.kt
new file mode 100644
index 0000000000..9abaf0e5d2
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/controller/CodeTestChatController.kt
@@ -0,0 +1,1285 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqCodeTest.controller
+
+import com.intellij.diff.DiffContentFactory
+import com.intellij.diff.DiffManager
+import com.intellij.diff.DiffManagerEx
+import com.intellij.diff.requests.SimpleDiffRequest
+import com.intellij.ide.BrowserUtil
+import com.intellij.openapi.application.ApplicationManager
+import com.intellij.openapi.fileEditor.FileEditorManager
+import com.intellij.openapi.project.Project
+import com.intellij.openapi.project.guessProjectDir
+import com.intellij.openapi.vfs.LocalFileSystem
+import com.intellij.openapi.vfs.VirtualFile
+import com.intellij.openapi.vfs.VirtualFileManager
+import kotlinx.coroutines.CoroutineScope
+import kotlinx.coroutines.currentCoroutineContext
+import kotlinx.coroutines.delay
+import kotlinx.coroutines.future.await
+import kotlinx.coroutines.launch
+import kotlinx.coroutines.withContext
+import migration.software.aws.toolkits.jetbrains.services.codewhisperer.customization.CodeWhispererModelConfigurator
+import software.amazon.awssdk.services.codewhispererruntime.model.IdeCategory
+import software.amazon.awssdk.services.codewhispererruntime.model.Position
+import software.amazon.awssdk.services.codewhispererruntime.model.Range
+import software.amazon.awssdk.services.codewhispererruntime.model.Reference
+import software.amazon.awssdk.services.codewhispererruntime.model.Span
+import software.amazon.awssdk.services.codewhispererstreaming.CodeWhispererStreamingAsyncClient
+import software.amazon.awssdk.services.codewhispererstreaming.model.AssistantResponseEvent
+import software.amazon.awssdk.services.codewhispererstreaming.model.ChatMessage
+import software.amazon.awssdk.services.codewhispererstreaming.model.ChatResponseStream
+import software.amazon.awssdk.services.codewhispererstreaming.model.ChatTriggerType
+import software.amazon.awssdk.services.codewhispererstreaming.model.CursorState
+import software.amazon.awssdk.services.codewhispererstreaming.model.DocumentSymbol
+import software.amazon.awssdk.services.codewhispererstreaming.model.EditorState
+import software.amazon.awssdk.services.codewhispererstreaming.model.GenerateAssistantResponseRequest
+import software.amazon.awssdk.services.codewhispererstreaming.model.GenerateAssistantResponseResponseHandler
+import software.amazon.awssdk.services.codewhispererstreaming.model.ProgrammingLanguage
+import software.amazon.awssdk.services.codewhispererstreaming.model.RelevantTextDocument
+import software.amazon.awssdk.services.codewhispererstreaming.model.SymbolType
+import software.amazon.awssdk.services.codewhispererstreaming.model.TextDocument
+import software.amazon.awssdk.services.codewhispererstreaming.model.UserInputMessage
+import software.amazon.awssdk.services.codewhispererstreaming.model.UserInputMessageContext
+import software.amazon.awssdk.services.codewhispererstreaming.model.UserIntent
+import software.aws.toolkits.core.utils.debug
+import software.aws.toolkits.core.utils.getLogger
+import software.aws.toolkits.jetbrains.core.AwsClientManager
+import software.aws.toolkits.jetbrains.core.coroutines.EDT
+import software.aws.toolkits.jetbrains.core.credentials.ToolkitConnectionManager
+import software.aws.toolkits.jetbrains.core.credentials.pinning.QConnection
+import software.aws.toolkits.jetbrains.services.amazonq.apps.AmazonQAppInitContext
+import software.aws.toolkits.jetbrains.services.amazonq.auth.AuthController
+import software.aws.toolkits.jetbrains.services.amazonq.project.RelevantDocument
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.CodeWhispererUTGChatManager
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.ConversationState
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.FEATURE_NAME
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.InboundAppMessagesHandler
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.messages.Button
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.messages.CodeTestChatMessage
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.messages.CodeTestChatMessageContent
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.messages.IncomingCodeTestMessage
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.model.PreviousUTGIterationContext
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.session.BuildAndExecuteProgressStatus
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.session.Session
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.storage.ChatSessionStorage
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.utils.constructBuildAndExecutionSummaryText
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.utils.runBuildOrTestCommand
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.messages.sendAuthNeededException
+import software.aws.toolkits.jetbrains.services.codewhisperer.credentials.CodeWhispererClientAdaptor
+import software.aws.toolkits.jetbrains.services.codewhisperer.language.CodeWhispererProgrammingLanguage
+import software.aws.toolkits.jetbrains.services.codewhisperer.language.programmingLanguage
+import software.aws.toolkits.jetbrains.services.codewhisperer.toolwindow.CodeWhispererCodeReferenceManager
+import software.aws.toolkits.jetbrains.services.cwc.ChatConstants
+import software.aws.toolkits.jetbrains.services.cwc.clients.chat.model.ChatRequestData
+import software.aws.toolkits.jetbrains.services.cwc.clients.chat.model.TriggerType
+import software.aws.toolkits.jetbrains.services.cwc.clients.chat.v1.ChatSessionV1.Companion.validLanguages
+import software.aws.toolkits.jetbrains.services.cwc.controller.chat.StaticPrompt
+import software.aws.toolkits.jetbrains.services.cwc.controller.chat.StaticTextResponse
+import software.aws.toolkits.jetbrains.services.cwc.controller.chat.telemetry.getStartUrl
+import software.aws.toolkits.jetbrains.services.cwc.editor.context.ActiveFileContext
+import software.aws.toolkits.jetbrains.services.cwc.editor.context.ActiveFileContextExtractor
+import software.aws.toolkits.jetbrains.services.cwc.editor.context.ExtractionTriggerType
+import software.aws.toolkits.jetbrains.services.cwc.editor.context.file.FileContext
+import software.aws.toolkits.jetbrains.services.cwc.messages.ChatMessageType
+import software.aws.toolkits.resources.message
+import software.aws.toolkits.telemetry.AmazonqTelemetry
+import software.aws.toolkits.telemetry.MetricResult
+import software.aws.toolkits.telemetry.UiTelemetry
+import java.io.File
+import java.nio.file.Files
+import java.nio.file.Path
+import java.nio.file.Paths
+import java.time.Instant
+import java.util.UUID
+import software.amazon.awssdk.services.codewhispererstreaming.model.Position as StreamingPosition
+import software.amazon.awssdk.services.codewhispererstreaming.model.Range as StreamingRange
+
+class CodeTestChatController(
+ private val context: AmazonQAppInitContext,
+ private val chatSessionStorage: ChatSessionStorage,
+ private val authController: AuthController = AuthController(),
+ private val cs: CoroutineScope,
+) : InboundAppMessagesHandler {
+ val messenger = context.messagesFromAppToUi
+ private val codeTestChatHelper = CodeTestChatHelper(context.messagesFromAppToUi, chatSessionStorage)
+ private val supportedLanguage = setOf("python", "java")
+ val client = CodeWhispererClientAdaptor.getInstance(context.project)
+ override suspend fun processPromptChatMessage(message: IncomingCodeTestMessage.ChatPrompt) {
+ handleChat(tabId = message.tabId, message = message.chatMessage)
+ }
+
+ private fun isLanguageSupported(languageId: String): Boolean =
+ supportedLanguage.contains(languageId.lowercase())
+
+ private fun getEditorSelectionRange(project: Project): Range? {
+ var selectionRange: Range? = null
+
+ ApplicationManager.getApplication().invokeAndWait {
+ selectionRange = ApplicationManager.getApplication().runReadAction {
+ val editor = FileEditorManager.getInstance(project).selectedTextEditor
+ editor?.let {
+ val selectionModel = it.selectionModel
+ val startOffset = selectionModel.selectionStart
+ val endOffset = selectionModel.selectionEnd
+
+ val startLogicalPosition = editor.offsetToLogicalPosition(startOffset)
+ val endLogicalPosition = editor.offsetToLogicalPosition(endOffset)
+
+ if (startOffset != endOffset) {
+ val start = Position.builder()
+ .line(startLogicalPosition.line)
+ .character(startLogicalPosition.column)
+ .build()
+
+ val end = Position.builder()
+ .line(endLogicalPosition.line)
+ .character(endLogicalPosition.column)
+ .build()
+
+ Range.builder()
+ .start(start)
+ .end(end)
+ .build()
+ } else {
+ null
+ }
+ }
+ }
+ }
+
+ return selectionRange
+ }
+
+ override suspend fun processStartTestGen(message: IncomingCodeTestMessage.StartTestGen) {
+ codeTestChatHelper.setActiveCodeTestTabId(message.tabId)
+ val session = codeTestChatHelper.getActiveSession()
+ // check if IDE has active file open, yes return (fileName and filePath) else return null
+ val project = context.project
+ val fileInfo = checkActiveFileInIDE(project, message) ?: return
+ session.programmingLanguage = fileInfo.fileLanguage
+ if (session.isGeneratingTests === true) {
+ return
+ }
+ session.startTimeOfTestGeneration = Instant.now().toEpochMilli().toDouble()
+ session.isGeneratingTests = true
+
+ var requestId: String = ""
+ var statusCode: Int = 0
+ var conversationId: String? = null
+ var testResponseMessageId: String? = null
+ var testResponseText: String = ""
+
+ val userMessage = when {
+ message.prompt != "" -> {
+ "/test ${message.prompt}"
+ }
+ else -> "/test Generate unit tests for `${fileInfo.fileName}`"
+ }
+ session.hasUserPromptSupplied = message.prompt.isNotEmpty()
+
+ // Send user prompt to chat
+ codeTestChatHelper.addNewMessage(
+ CodeTestChatMessageContent(message = userMessage, type = ChatMessageType.Prompt, canBeVoted = false),
+ message.tabId,
+ false
+ )
+ if (fileInfo.fileInWorkspace && isLanguageSupported(fileInfo.fileLanguage.languageId)) {
+ // Send Capability card to chat
+ codeTestChatHelper.addNewMessage(
+ CodeTestChatMessageContent(informationCard = true, message = null, type = ChatMessageType.Answer, canBeVoted = false),
+ message.tabId,
+ false
+ )
+
+ var selectionRange = getEditorSelectionRange(project)
+
+ session.isCodeBlockSelected = selectionRange !== null
+
+ // This check is added to remove /test if user accidentally added while doing Regenerate unit tests.
+ val userPrompt = if (message.prompt.startsWith("/test")) {
+ message.prompt.substringAfter("/test ").trim()
+ } else {
+ message.prompt
+ }
+ CodeWhispererUTGChatManager.getInstance(project).generateTests(userPrompt, codeTestChatHelper, null, selectionRange)
+ } else {
+ // Not adding a progress bar to unsupported language cases
+ val responseHandler = GenerateAssistantResponseResponseHandler.builder()
+ .onResponse {
+ requestId = it.responseMetadata().requestId()
+ statusCode = it.sdkHttpResponse().statusCode()
+ conversationId = it.conversationId()
+ }
+ .subscriber { stream: ChatResponseStream ->
+ stream.accept(object : GenerateAssistantResponseResponseHandler.Visitor {
+
+ override fun visitAssistantResponseEvent(event: AssistantResponseEvent) {
+ testResponseText += event.content()
+ cs.launch {
+ codeTestChatHelper.updateAnswer(
+ CodeTestChatMessageContent(
+ message = testResponseText,
+ type = ChatMessageType.AnswerPart
+ ),
+ messageIdOverride = testResponseMessageId
+ )
+ }
+ }
+ })
+ }
+ .build()
+
+ val messageContent = if (fileInfo.fileInWorkspace) {
+ "⚠ ${fileInfo.fileLanguage.languageId} is not a " +
+ "language I support specialized unit test generation for at the moment. The languages " +
+ "I support now are Python and Java. I can still provide examples, instructions and code suggestions."
+ } else {
+ "⚠ I can't generate tests for ${fileInfo.fileName}" +
+ " because it's outside the project directory. " +
+ "I can still provide examples, instructions and code suggestions."
+ }
+
+ codeTestChatHelper.addNewMessage(
+ CodeTestChatMessageContent(
+ message = messageContent,
+ type = ChatMessageType.Answer,
+ canBeVoted = false
+ ),
+ message.tabId,
+ false
+ )
+ testResponseMessageId = codeTestChatHelper.addAnswer(
+ CodeTestChatMessageContent(
+ message = "",
+ type = ChatMessageType.AnswerStream
+ )
+ )
+ codeTestChatHelper.updateUI(
+ loadingChat = true,
+ promptInputDisabledState = true,
+ )
+ // Send Request to Sync UTG API
+ val connection = ToolkitConnectionManager.getInstance(project).activeConnectionForFeature(QConnection.getInstance())
+ // this should never happen because it should have been handled upstream by [AuthController]
+ ?: error("connection was found to be null")
+ val contextExtractor = ActiveFileContextExtractor.create(fqnWebviewAdapter = null, project = project)
+ val activeFileContext = ActiveFileContext(
+ fileContext = FileContext(
+ fileLanguage = fileInfo.fileLanguage.languageId,
+ filePath = fileInfo.filePath,
+ matchPolicy = null
+ ),
+ focusAreaContext = contextExtractor.extractContextForTrigger(ExtractionTriggerType.ChatMessage).focusAreaContext,
+ )
+
+ val requestData = ChatRequestData(
+ tabId = session.tabId,
+ message = "Generate unit tests for the following part of my code: ${message.prompt}",
+ activeFileContext = activeFileContext,
+ userIntent = UserIntent.GENERATE_UNIT_TESTS,
+ triggerType = TriggerType.ContextMenu,
+ customization = CodeWhispererModelConfigurator.getInstance().activeCustomization(context.project),
+ relevantTextDocuments = emptyList(),
+ useRelevantDocuments = false,
+ )
+
+ val client = AwsClientManager.getInstance().getClient(connection.getConnectionSettings())
+ val request = requestData.toChatRequest()
+ client.generateAssistantResponse(request, responseHandler).await()
+ // TODO: Need to send isCodeBlockSelected field
+ requestId.let { id ->
+ LOG.debug { "$FEATURE_NAME: Unit test generation requestId: $id" }
+ AmazonqTelemetry.utgGenerateTests(
+ cwsprChatProgrammingLanguage = session.programmingLanguage.languageId,
+ hasUserPromptSupplied = session.hasUserPromptSupplied,
+ isFileInWorkspace = fileInfo.fileInWorkspace,
+ isSupportedLanguage = isLanguageSupported(fileInfo.fileLanguage.languageId),
+ credentialStartUrl = getStartUrl(project),
+ result = MetricResult.Succeeded,
+ perfClientLatency = (Instant.now().toEpochMilli() - session.startTimeOfTestGeneration),
+ requestId = id
+ )
+ }
+ session.isGeneratingTests = false
+ codeTestChatHelper.updateUI(
+ loadingChat = false,
+ promptInputDisabledState = false
+ )
+ }
+ }
+ private fun ActiveFileContext.toEditorState(relevantDocuments: List, useRelevantDocuments: Boolean): EditorState {
+ val editorStateBuilder = EditorState.builder()
+ if (fileContext != null) {
+ val cursorStateBuilder = CursorState.builder()
+ // Cursor State
+ val start = focusAreaContext?.codeSelectionRange?.start
+ val end = focusAreaContext?.codeSelectionRange?.end
+
+ if (start != null && end != null) {
+ cursorStateBuilder.range(
+ StreamingRange.builder()
+ .start(
+ StreamingPosition.builder()
+ .line(start.row)
+ .character(start.column)
+ .build(),
+ )
+ .end(
+ StreamingPosition.builder()
+ .line(end.row)
+ .character(end.column)
+ .build(),
+ ).build(),
+ )
+ }
+ editorStateBuilder.cursorState(cursorStateBuilder.build())
+
+ // Code Names -> DocumentSymbols
+ val documentBuilder = TextDocument.builder()
+ val codeNames = focusAreaContext?.codeNames
+
+ val documentSymbolList = codeNames?.fullyQualifiedNames?.used?.map {
+ DocumentSymbol.builder()
+ .name(it.symbol?.joinToString(separator = "."))
+ .type(SymbolType.USAGE)
+ .source(it.source?.joinToString(separator = "."))
+ .build()
+ }?.filter { it.name().length in ChatConstants.FQN_SIZE_MIN until ChatConstants.FQN_SIZE_LIMIT }.orEmpty()
+ documentBuilder.documentSymbols(documentSymbolList)
+ // TODO: Do conditional check for focusAreaContext?.codeSelectionRange if undefined then get entire file
+ // File Text
+ val fileContent = Files.readString(Paths.get(fileContext.filePath))
+ documentBuilder.text(fileContent)
+
+ // Programming Language
+ val programmingLanguage = fileContext.fileLanguage
+ if (programmingLanguage != null && validLanguages.contains(programmingLanguage)) {
+ documentBuilder.programmingLanguage(
+ ProgrammingLanguage.builder()
+ .languageName(programmingLanguage).build(),
+ )
+ }
+
+ // Relative File Path
+ val filePath = fileContext.filePath
+ if (filePath != null) {
+ documentBuilder.relativeFilePath(filePath.take(ChatConstants.FILE_PATH_SIZE_LIMIT))
+ }
+ editorStateBuilder.document(documentBuilder.build())
+ }
+
+ // Relevant Documents
+ val documents: List = relevantDocuments.map { doc ->
+ RelevantTextDocument.builder().text(doc.text).relativeFilePath(doc.relativeFilePath.take(ChatConstants.FILE_PATH_SIZE_LIMIT)).build()
+ }
+
+ editorStateBuilder.relevantDocuments(documents)
+ editorStateBuilder.useRelevantDocuments(useRelevantDocuments)
+ return editorStateBuilder.build()
+ }
+
+ private fun ChatRequestData.toChatRequest(): GenerateAssistantResponseRequest {
+ val userInputMessageContextBuilder = UserInputMessageContext.builder()
+ userInputMessageContextBuilder.editorState(activeFileContext.toEditorState(relevantTextDocuments, useRelevantDocuments))
+ val userInputMessageContext = userInputMessageContextBuilder.build() //
+ val userInput = UserInputMessage.builder()
+ .content(message.take(ChatConstants.CUSTOMER_MESSAGE_SIZE_LIMIT))
+ .userInputMessageContext(userInputMessageContext)
+ .userIntent(userIntent)
+ .build()
+ println("UserInput Message: $userInput")
+ val conversationState = software.amazon.awssdk.services.codewhispererstreaming.model.ConversationState.builder()
+ .currentMessage(ChatMessage.fromUserInputMessage(userInput))
+ .chatTriggerType(if (triggerType == TriggerType.Inline) ChatTriggerType.INLINE_CHAT else ChatTriggerType.MANUAL)
+ .customizationArn(customization?.arn)
+ .build()
+ return GenerateAssistantResponseRequest.builder()
+ .conversationState(conversationState)
+ .build()
+ }
+
+ override suspend fun processNewTabCreatedMessage(message: IncomingCodeTestMessage.NewTabCreated) {
+ newTabOpened(message.tabId)
+ LOG.debug { "$FEATURE_NAME: New tab created: $message" }
+ codeTestChatHelper.setActiveCodeTestTabId(message.tabId)
+ }
+
+ override suspend fun processTabRemovedMessage(message: IncomingCodeTestMessage.TabRemoved) {
+ chatSessionStorage.deleteSession(message.tabId)
+ }
+
+ override suspend fun processClearQuickAction(message: IncomingCodeTestMessage.ClearChat) {
+ chatSessionStorage.deleteSession(message.tabId)
+ }
+
+ override suspend fun processHelpQuickAction(message: IncomingCodeTestMessage.Help) {
+ // TODO: Replace StaticPrompt and StaticTextResponse message according to Fnf
+ codeTestChatHelper.addNewMessage(
+ CodeTestChatMessageContent(
+ message = StaticPrompt.Help.message,
+ type = ChatMessageType.Prompt,
+ canBeVoted = false
+ ),
+ message.tabId,
+ false
+ )
+ codeTestChatHelper.addNewMessage(
+ CodeTestChatMessageContent(
+ message = StaticTextResponse.Help.message,
+ type = ChatMessageType.Answer,
+ canBeVoted = false
+ ),
+ message.tabId,
+ false
+ )
+ }
+
+ override suspend fun processLinkClick(message: IncomingCodeTestMessage.ClickedLink) {
+ BrowserUtil.browse(message.link)
+ }
+
+ override suspend fun processButtonClickedMessage(message: IncomingCodeTestMessage.ButtonClicked) {
+ val session = codeTestChatHelper.getActiveSession()
+ var numberOfLinesGenerated = 0
+ var numberOfLinesSelected = 0
+ var lineDifference = 0
+ var numberOfCharsGenerated = 0
+ var numberOfCharsSelected = 0
+ var charDifference = 0
+ var generatedFileContent = ""
+ var selectedFileContent = ""
+
+ when (message.actionID) {
+ "utg_view_diff" -> {
+ withContext(EDT) {
+ (DiffManager.getInstance() as DiffManagerEx).showDiffBuiltin(
+ context.project,
+ SimpleDiffRequest(
+ session.testFileName,
+ DiffContentFactory.getInstance().create(
+ getFileContentAtTestFilePath(
+ session.projectRoot,
+ session.testFileRelativePathToProjectRoot
+ )
+ ),
+ DiffContentFactory.getInstance().create(session.generatedTestDiffs.values.first()),
+ "Before",
+ "After"
+ )
+ )
+ session.openedDiffFile = FileEditorManager.getInstance(context.project).selectedEditor?.file
+ ApplicationManager.getApplication().runReadAction {
+ generatedFileContent = getGeneratedFileContent(session)
+ }
+
+ selectedFileContent = getFileContentAtTestFilePath(
+ session.projectRoot,
+ session.testFileRelativePathToProjectRoot,
+ )
+
+ // Line difference calculation: linesOfCodeGenerated = number of lines in generated test file - number of lines in original test file
+ numberOfLinesGenerated = generatedFileContent.lines().size
+ numberOfLinesSelected = selectedFileContent.lines().size
+ lineDifference = numberOfLinesGenerated - numberOfLinesSelected
+
+ // Character difference calculation: charsOfCodeGenerated = number of characters in generated test file - number of characters in original test file
+ numberOfCharsGenerated = generatedFileContent.length
+ numberOfCharsSelected = selectedFileContent.length
+ charDifference = numberOfCharsGenerated - numberOfCharsSelected
+
+ session.linesOfCodeGenerated = lineDifference.coerceAtLeast(0)
+ session.charsOfCodeGenerated = charDifference.coerceAtLeast(0)
+ session.latencyOfTestGeneration = (Instant.now().toEpochMilli() - session.startTimeOfTestGeneration)
+ UiTelemetry.click(null as Project?, "unitTestGeneration_viewDiff")
+
+ val buttonList = mutableListOf()
+ buttonList.add(
+ Button(
+ "utg_reject",
+ "Reject",
+ keepCardAfterClick = true,
+ position = "outside",
+ status = "error",
+ ),
+ )
+ /*
+ // TODO: for unit test regeneration loop
+ if (session.iteration < 2) {
+ buttonList.add(
+ Button(
+ "utg_regenerate",
+ "Regenerate",
+ keepCardAfterClick = true,
+ position = "outside",
+ status = "info",
+ ),
+ )
+ }
+ */
+
+ buttonList.add(
+ Button(
+ "utg_accept",
+ "Accept",
+ keepCardAfterClick = true,
+ position = "outside",
+ status = "success",
+ ),
+ )
+
+ codeTestChatHelper.updateUI(
+ promptInputDisabledState = true,
+ promptInputPlaceholder = message("testgen.placeholder.select_an_option"),
+ )
+
+ codeTestChatHelper.updateAnswer(
+ CodeTestChatMessageContent(
+ type = ChatMessageType.AnswerPart,
+ buttons = buttonList,
+ ),
+ messageIdOverride = session.viewDiffMessageId
+ )
+ }
+ }
+ "utg_accept" -> {
+ // open the file at test path relative to the project root
+ val testFileAbsolutePath = Paths.get(session.projectRoot, session.testFileRelativePathToProjectRoot)
+ openOrCreateTestFileAndApplyDiff(context.project, testFileAbsolutePath, session.generatedTestDiffs.values.first(), session.openedDiffFile)
+ session.codeReferences?.let { references ->
+ LOG.debug { "Accepted unit tests with references: $references" }
+ val manager = CodeWhispererCodeReferenceManager.getInstance(context.project)
+ references.forEach { ref ->
+ var referenceContentSpan: Span? = null
+ ref.recommendationContentSpan?.let {
+ referenceContentSpan = Span.builder().start(ref.recommendationContentSpan.start).end(ref.recommendationContentSpan.end).build()
+ }
+ val reference = Reference.builder().url(
+ ref.url
+ ).licenseName(ref.licenseName).repository(ref.repository).recommendationContentSpan(referenceContentSpan).build()
+ var originalContent: String? = null
+ ref.recommendationContentSpan?.let {
+ originalContent = session.generatedTestDiffs.values.first().substring(
+ ref.recommendationContentSpan.start,
+ ref.recommendationContentSpan.end
+ )
+ }
+ LOG.debug { "Original code content from reference span: $originalContent" }
+ withContext(EDT) {
+ manager.addReferenceLogPanelEntry(reference = reference, null, null, originalContent?.split("\n"))
+ manager.toolWindow?.show()
+ }
+ }
+ }
+ val testGenerationEventResponse = client.sendTestGenerationEvent(
+ session.testGenerationJob,
+ session.testGenerationJobGroupName,
+ session.programmingLanguage,
+ IdeCategory.JETBRAINS,
+ session.numberOfUnitTestCasesGenerated,
+ session.numberOfUnitTestCasesGenerated,
+ session.linesOfCodeGenerated,
+ session.linesOfCodeGenerated,
+ session.charsOfCodeGenerated,
+ session.charsOfCodeGenerated
+ )
+ LOG.debug {
+ "Successfully sent test generation telemetry. RequestId: ${
+ testGenerationEventResponse.responseMetadata().requestId()}"
+ }
+
+ UiTelemetry.click(null as Project?, "unitTestGeneration_acceptDiff")
+
+ AmazonqTelemetry.utgGenerateTests(
+ cwsprChatProgrammingLanguage = session.programmingLanguage.languageId,
+ hasUserPromptSupplied = session.hasUserPromptSupplied,
+ isFileInWorkspace = true,
+ isSupportedLanguage = true,
+ credentialStartUrl = getStartUrl(project = context.project),
+ jobGroup = session.testGenerationJobGroupName,
+ jobId = session.testGenerationJob,
+ acceptedCount = session.numberOfUnitTestCasesGenerated?.toLong(),
+ generatedCount = session.numberOfUnitTestCasesGenerated?.toLong(),
+ acceptedLinesCount = session.linesOfCodeGenerated?.toLong(),
+ generatedLinesCount = session.linesOfCodeGenerated?.toLong(),
+ acceptedCharactersCount = session.charsOfCodeGenerated?.toLong(),
+ generatedCharactersCount = session.charsOfCodeGenerated?.toLong(),
+ result = MetricResult.Succeeded,
+ perfClientLatency = session.latencyOfTestGeneration,
+ isCodeBlockSelected = session.isCodeBlockSelected,
+ artifactsUploadDuration = session.artifactUploadDuration,
+ buildPayloadBytes = session.srcPayloadSize,
+ buildZipFileBytes = session.srcZipFileSize,
+ requestId = session.startTestGenerationRequestId
+ )
+ codeTestChatHelper.addAnswer(
+ CodeTestChatMessageContent(
+ message = message("testgen.message.success"),
+ type = ChatMessageType.Answer,
+ canBeVoted = false
+ )
+ )
+ sessionCleanUp(session.tabId)
+ codeTestChatHelper.updateUI(
+ promptInputDisabledState = false,
+ promptInputPlaceholder = message("testgen.placeholder.waiting_on_your_inputs"),
+ )
+ /*
+ val taskContext = session.buildAndExecuteTaskContext
+ if (session.iteration < 2) {
+ taskContext.buildCommand = getBuildCommand(message.tabId)
+ taskContext.executionCommand = getExecutionCommand(message.tabId)
+ codeTestChatHelper.addAnswer(
+ CodeTestChatMessageContent(
+ message = """
+ Would you like me to help build and execute the test? I'll run following commands
+
+ ```sh
+ ${taskContext.buildCommand}
+ ${taskContext.executionCommand}
+ ```
+ """.trimIndent(),
+ type = ChatMessageType.Answer,
+ canBeVoted = true,
+ buttons = listOf(
+ Button(
+ "utg_skip_and_finish",
+ "Skip and finish",
+ keepCardAfterClick = true,
+ position = "outside",
+ status = "info",
+ ),
+ Button(
+ "utg_modify_command",
+ "Modify commands",
+ keepCardAfterClick = true,
+ position = "outside",
+ status = "info",
+ ),
+ Button(
+ "utg_build_and_execute",
+ "Build and execute",
+ keepCardAfterClick = true,
+ position = "outside",
+ status = "info",
+ ),
+ )
+ )
+ )
+ codeTestChatHelper.updateUI(
+ promptInputDisabledState = true,
+ )
+ } else if (session.iteration < 4) {
+ // Already built and executed once, display # of iterations left message
+ val remainingIterationsCount = UTG_CHAT_MAX_ITERATION - session.iteration
+ val iterationCountString = "$remainingIterationsCount ${if (remainingIterationsCount > 1) "iterations" else "iteration"}"
+ codeTestChatHelper.addAnswer(
+ CodeTestChatMessageContent(
+ message = """
+ Would you like Amazon Q to build and execute again, and fix errors?
+
+ You have $iterationCountString left.
+
+ """.trimIndent(),
+ type = ChatMessageType.AIPrompt,
+ buttons = listOf(
+ Button(
+ "utg_skip_and_finish",
+ "Skip and finish",
+ keepCardAfterClick = true,
+ position = "outside",
+ status = "info",
+ ),
+ Button(
+ "utg_proceed",
+ "Proceed",
+ keepCardAfterClick = true,
+ position = "outside",
+ status = "info",
+ ),
+ ),
+ )
+ )
+ codeTestChatHelper.updateUI(
+ promptInputDisabledState = true,
+ )
+ } else {
+ // TODO: change this hardcoded string
+ val monthlyLimitString = "25 out of 30"
+ codeTestChatHelper.addAnswer(
+ CodeTestChatMessageContent(
+ message = """
+ You have gone through all three iterations and this unit test generation workflow is complete. You have $monthlyLimitString Amazon Q Developer Agent invocations left this month.
+ """.trimIndent(),
+ type = ChatMessageType.Answer,
+ )
+ )
+ codeTestChatHelper.updateUI(
+ promptInputPlaceholder = message("testgen.placeholder.newtab")
+ )
+ }
+ */
+ }
+ /*
+ //TODO: this is for unit test regeneration build iteration loop
+ "utg_regenerate" -> {
+ // close the existing open diff in the editor.
+ ApplicationManager.getApplication().invokeLater {
+ session.openedDiffFile?.let { FileEditorManager.getInstance(context.project).closeFile(it) }
+ }
+ codeTestChatHelper.addAnswer(
+ CodeTestChatMessageContent(
+ message = message("testgen.message.regenerate_input"),
+ type = ChatMessageType.Answer,
+ canBeVoted = false
+ )
+ )
+ val testGenerationEventResponse = client.sendTestGenerationEvent(
+ session.testGenerationJob,
+ session.testGenerationJobGroupName,
+ session.programmingLanguage,
+ session.numberOfUnitTestCasesGenerated,
+ 0,
+ session.linesOfCodeGenerated,
+ 0,
+ session.charsOfCodeGenerated,
+ 0
+ )
+ LOG.debug {
+ "Successfully sent test generation telemetry. RequestId: ${
+ testGenerationEventResponse.responseMetadata().requestId()}"
+ }
+ sessionCleanUp(session.tabId)
+ codeTestChatHelper.updateUI(
+ promptInputDisabledState = false,
+ promptInputPlaceholder = message("testgen.placeholder.waiting_on_your_inputs"),
+ )
+ }
+ */
+
+ "utg_reject" -> {
+ ApplicationManager.getApplication().invokeLater {
+ session.openedDiffFile?.let { FileEditorManager.getInstance(context.project).closeFile(it) }
+ }
+ codeTestChatHelper.addAnswer(
+ CodeTestChatMessageContent(
+ message = message("testgen.message.success"),
+ type = ChatMessageType.Answer,
+ canBeVoted = false
+ )
+ )
+ val testGenerationEventResponse = client.sendTestGenerationEvent(
+ session.testGenerationJob,
+ session.testGenerationJobGroupName,
+ session.programmingLanguage,
+ IdeCategory.JETBRAINS,
+ session.numberOfUnitTestCasesGenerated,
+ 0,
+ session.linesOfCodeGenerated,
+ 0,
+ session.charsOfCodeGenerated,
+ 0
+ )
+ LOG.debug {
+ "Successfully sent test generation telemetry. RequestId: ${
+ testGenerationEventResponse.responseMetadata().requestId()}"
+ }
+
+ UiTelemetry.click(null as Project?, "unitTestGeneration_rejectDiff")
+ AmazonqTelemetry.utgGenerateTests(
+ cwsprChatProgrammingLanguage = session.programmingLanguage.languageId,
+ hasUserPromptSupplied = session.hasUserPromptSupplied,
+ isFileInWorkspace = true,
+ isSupportedLanguage = true,
+ credentialStartUrl = getStartUrl(project = context.project),
+ jobGroup = session.testGenerationJobGroupName,
+ jobId = session.testGenerationJob,
+ acceptedCount = 0,
+ generatedCount = session.numberOfUnitTestCasesGenerated?.toLong(),
+ acceptedLinesCount = 0,
+ generatedLinesCount = session.linesOfCodeGenerated?.toLong(),
+ acceptedCharactersCount = 0,
+ generatedCharactersCount = session.charsOfCodeGenerated?.toLong(),
+ result = MetricResult.Succeeded,
+ perfClientLatency = session.latencyOfTestGeneration,
+ isCodeBlockSelected = session.isCodeBlockSelected,
+ artifactsUploadDuration = session.artifactUploadDuration,
+ buildPayloadBytes = session.srcPayloadSize,
+ buildZipFileBytes = session.srcZipFileSize,
+ requestId = session.startTestGenerationRequestId
+ )
+ sessionCleanUp(message.tabId)
+ }
+ "utg_skip_and_finish" -> {
+ codeTestChatHelper.addAnswer(
+ CodeTestChatMessageContent(
+ message = message("testgen.message.success"),
+ type = ChatMessageType.Answer,
+ canBeVoted = false
+ )
+ )
+ sessionCleanUp(message.tabId)
+ }
+ "utg_proceed", "utg_build_and_execute" -> {
+ // handle both "Proceed" and "Build and execute" button clicks since their actions are similar
+ // TODO: show install dependencies card if needed
+ session.conversationState = ConversationState.IN_PROGRESS
+
+ // display build in progress card
+ val taskContext = session.buildAndExecuteTaskContext
+
+ taskContext.progressStatus = BuildAndExecuteProgressStatus.RUN_BUILD
+ val messageId = updateBuildAndExecuteProgressCard(taskContext.progressStatus, null, session.iteration)
+ // TODO: build and execute case
+ val buildLogsFile = VirtualFileManager.getInstance().findFileByNioPath(
+ withContext(currentCoroutineContext()) {
+ Files.createTempFile(null, null)
+ }
+ )
+ if (buildLogsFile == null) {
+ // TODO: handle no log file case
+ return
+ }
+ LOG.debug {
+ "Q TestGen session: ${codeTestChatHelper.getActiveCodeTestTabId()}: " +
+ "tmpFile for build logs:\n ${buildLogsFile.path}"
+ }
+
+ runBuildOrTestCommand(taskContext.buildCommand, buildLogsFile, context.project, isBuildCommand = true, taskContext)
+ while (taskContext.buildExitCode < 0) {
+ // wait until build command finished
+ delay(1000)
+ }
+
+ // TODO: only go to future iterations when buildExitCode or testExitCode > 0, right now iterate regardless
+ if (taskContext.buildExitCode > 0) {
+ // TODO: handle build failure case
+ // ...
+// return
+ }
+ taskContext.progressStatus = BuildAndExecuteProgressStatus.RUN_EXECUTION_TESTS
+ updateBuildAndExecuteProgressCard(taskContext.progressStatus, messageId, session.iteration)
+
+ val testLogsFile = VirtualFileManager.getInstance().findFileByNioPath(
+ withContext(currentCoroutineContext()) {
+ Files.createTempFile(null, null)
+ }
+ )
+ if (testLogsFile == null) {
+ // TODO: handle no log file case
+ return
+ }
+ LOG.debug {
+ "Q TestGen session: ${codeTestChatHelper.getActiveCodeTestTabId()}: " +
+ "tmpFile for test logs:\n ${buildLogsFile.path}"
+ }
+ delay(1000)
+ runBuildOrTestCommand(taskContext.executionCommand, testLogsFile, context.project, isBuildCommand = false, taskContext)
+ while (taskContext.testExitCode < 0) {
+ // wait until test command finished
+ delay(1000)
+ }
+
+ if (taskContext.testExitCode == 0) {
+ taskContext.progressStatus = BuildAndExecuteProgressStatus.TESTS_EXECUTED
+ updateBuildAndExecuteProgressCard(taskContext.progressStatus, messageId, session.iteration)
+ codeTestChatHelper.addAnswer(
+ CodeTestChatMessageContent(
+ message = message("testgen.message.success"),
+ type = ChatMessageType.Answer,
+ canBeVoted = false
+ )
+ )
+ sessionCleanUp(message.tabId)
+ return
+ }
+
+ // has test failure, we will zip the latest project and invoke backend again
+ taskContext.progressStatus = BuildAndExecuteProgressStatus.FIXING_TEST_CASES
+ val buildAndExecuteMessageId = updateBuildAndExecuteProgressCard(taskContext.progressStatus, messageId, session.iteration)
+
+ val previousUTGIterationContext = PreviousUTGIterationContext(
+ buildLogFile = buildLogsFile,
+ testLogFile = testLogsFile,
+ selectedFile = session.selectedFile,
+ buildAndExecuteMessageId = buildAndExecuteMessageId
+ )
+
+ val job = CodeWhispererUTGChatManager.getInstance(context.project).generateTests("", codeTestChatHelper, previousUTGIterationContext, null)
+ job?.join()
+
+ taskContext.progressStatus = BuildAndExecuteProgressStatus.PROCESS_TEST_RESULTS
+ // session.iteration already updated in generateTests
+ updateBuildAndExecuteProgressCard(taskContext.progressStatus, messageId, session.iteration - 1)
+ }
+ "utg_modify_command" -> {
+ // TODO allow user input to modify the command
+ codeTestChatHelper.addAnswer(
+ CodeTestChatMessageContent(
+ message = """
+ Sure. Let me know which command you'd like to modify or you could also provide all command lines you'd like me to run.
+
+ """.trimIndent(),
+ type = ChatMessageType.Answer,
+ canBeVoted = false
+ )
+ )
+ session.conversationState = ConversationState.WAITING_FOR_BUILD_COMMAND_INPUT
+ }
+ "utg_install_and_continue" -> {
+ // TODO: install dependencies and build
+ }
+ "stop_test_generation" -> {
+ UiTelemetry.click(null as Project?, "unitTestGeneration_cancelTestGenerationProgress")
+ session.isGeneratingTests = false
+ sessionCleanUp(message.tabId)
+ return
+ }
+ else -> {
+ // Handle other cases or do nothing
+ }
+ }
+ }
+
+ private suspend fun updateBuildAndExecuteProgressCard(
+ currentStatus: BuildAndExecuteProgressStatus,
+ messageId: String?,
+ iterationNum: Int,
+ ): String? {
+ val updatedText = constructBuildAndExecutionSummaryText(currentStatus, iterationNum)
+
+ if (currentStatus == BuildAndExecuteProgressStatus.RUN_BUILD) {
+ val buildAndExecuteMessageId = codeTestChatHelper.addAnswer(
+ CodeTestChatMessageContent(
+ message = updatedText,
+ type = ChatMessageType.AnswerStream,
+ canBeVoted = true,
+ )
+ )
+ // For streaming effect
+ codeTestChatHelper.updateAnswer(
+ CodeTestChatMessageContent(type = ChatMessageType.AnswerPart),
+ messageIdOverride = buildAndExecuteMessageId
+ )
+ codeTestChatHelper.updateUI(
+ loadingChat = true,
+ promptInputDisabledState = true,
+ )
+ return buildAndExecuteMessageId
+ } else {
+ val isLastStage = currentStatus == BuildAndExecuteProgressStatus.PROCESS_TEST_RESULTS
+ codeTestChatHelper.updateAnswer(
+ CodeTestChatMessageContent(
+ message = updatedText,
+ type = if (isLastStage) ChatMessageType.Answer else ChatMessageType.AnswerPart,
+ canBeVoted = true
+ ),
+ messageId
+ )
+ codeTestChatHelper.updateUI(
+ loadingChat = !isLastStage,
+ promptInputDisabledState = true,
+ )
+ return messageId
+ }
+ }
+
+ /**
+ * Perform Session CleanUp in below cases
+ * 1. UTG success workflow or UTG build success.
+ * 2. If user click Reject or SkipAndFinish button
+ * 3. Error while generating unit tests
+ * 4. After finishing 3 build loop iterations
+ * 5. Closing a Q-Test tab
+ * 6. Progress bar cancel
+ */
+ private suspend fun sessionCleanUp(tabId: String) {
+ // TODO: May be need to clear all the session data like jobId, jobGroupName and etc along with temp build log files
+ chatSessionStorage.deleteSession(tabId)
+ codeTestChatHelper.updateUI(
+ promptInputDisabledState = false
+ )
+ codeTestChatHelper.sendUpdatePlaceholder(tabId, message("testgen.placeholder.newtab"))
+ }
+
+ private fun openOrCreateTestFileAndApplyDiff(
+ project: Project,
+ testFileAbsolutePath: Path,
+ afterContent: String,
+ openedDiffFile: VirtualFile?,
+ ) {
+ val virtualFile: VirtualFile?
+
+ // Check if the file exists
+ if (Files.exists(testFileAbsolutePath)) {
+ // File exists, get the VirtualFile
+ virtualFile = LocalFileSystem.getInstance().findFileByPath(testFileAbsolutePath.toString())
+ if (virtualFile == null) return
+ val beforeContent = String(virtualFile.contentsToByteArray()) // Read the existing content
+
+ ApplicationManager.getApplication().invokeLater {
+ ApplicationManager.getApplication().runWriteAction {
+ applyDiffAndWriteContent(virtualFile, beforeContent, afterContent)
+ }
+ }
+ } else {
+ // File does not exist, create it
+ virtualFile = createFile(testFileAbsolutePath, afterContent)
+ }
+ if (virtualFile == null) return
+ ApplicationManager.getApplication().invokeLater {
+ openedDiffFile?.let { FileEditorManager.getInstance(project).closeFile(it) }
+ FileEditorManager.getInstance(project).openFile(virtualFile, true) // Open the file in editor
+ }
+ }
+
+ // Function to create the file and write content
+ private fun createFile(path: Path, content: String): VirtualFile? {
+ val parentPath = path.parent
+ if (!Files.exists(parentPath)) {
+ Files.createDirectories(parentPath) // Ensure parent directories exist
+ }
+
+ val file = Files.createFile(path) // Create the file
+ Files.writeString(file, content) // Write the afterContent to the file
+ return LocalFileSystem.getInstance().refreshAndFindFileByPath(path.toString())
+ }
+
+ // Function to apply the diff and write the new content
+ private fun applyDiffAndWriteContent(
+ virtualFile: VirtualFile,
+ beforeContent: String,
+ afterContent: String,
+ ) {
+ if (beforeContent == afterContent) return
+ virtualFile.setBinaryContent(afterContent.toByteArray()) // Update the file content
+ }
+
+ // Return test file content if it exists, return an empty string otherwise.
+ private fun getFileContentAtTestFilePath(projectRoot: String, testFileRelativePathToProjectRoot: String): String {
+ val testFileAbsolutePath = Paths.get(projectRoot, testFileRelativePathToProjectRoot)
+ return if (Files.exists(testFileAbsolutePath)) {
+ Files.readString(testFileAbsolutePath) // Read and return the file content
+ } else {
+ "" // Return an empty string if the file does not exist
+ }
+ }
+
+ // Return generated test file content
+ private fun getGeneratedFileContent(session: Session): String {
+ val generateFileContent = session.generatedTestDiffs[session.testFileRelativePathToProjectRoot].toString()
+ return generateFileContent
+ }
+
+ /*
+ If shortAnswer has buildCommand, use it, if it doesn't hardcode it according to the user type(internal or not)
+ private fun getBuildCommand(tabId: String): String {
+ val buildCommand = codeTestChatHelper.getSession(tabId).shortAnswer.buildCommand
+ if (buildCommand != null) return buildCommand
+
+ // TODO: remove hardcode
+ return "pip install -e ."
+ }
+
+ private fun getExecutionCommand(tabId: String): String {
+ val executionCommand = codeTestChatHelper.getSession(tabId).shortAnswer.executionCommand
+ if (executionCommand != null) return executionCommand
+
+ // TODO: remove hardcode
+ return "pytest"
+ }
+ */
+
+ private suspend fun newTabOpened(tabId: String) {
+ // TODO: the logic of checking auth is needed (for calling APIs) but need refactor with FeatureDev
+ val session: Session?
+ try {
+ session = codeTestChatHelper.getSession(tabId)
+ LOG.debug {
+ "$FEATURE_NAME:" +
+ " Session created with id: ${session.tabId}"
+ }
+ val credentialState = authController.getAuthNeededStates(context.project).amazonQ
+ if (credentialState != null) {
+ messenger.sendAuthNeededException(
+ tabId = tabId,
+ triggerId = UUID.randomUUID().toString(),
+ credentialState = credentialState,
+ )
+ session.isAuthenticating = true
+ return
+ }
+ } catch (err: Exception) {
+ messenger.publish(
+ CodeTestChatMessage(
+ tabId = tabId,
+ messageType = ChatMessageType.Answer,
+ message = message("codescan.chat.message.error_request")
+ )
+ )
+ return
+ }
+ }
+
+ data class ActiveFileInfo(
+ val filePath: String,
+ val fileName: String,
+ val fileLanguage: CodeWhispererProgrammingLanguage,
+ val fileInWorkspace: Boolean = true,
+ )
+
+ private suspend fun updateUIState() {
+ codeTestChatHelper.updateUI(
+ promptInputDisabledState = false,
+ promptInputPlaceholder = message("testgen.placeholder.newtab")
+ )
+ }
+
+ private suspend fun handleInvalidFileState(tabId: String) {
+ codeTestChatHelper.addNewMessage(
+ CodeTestChatMessageContent(
+ message = message("testgen.no_file_found"),
+ type = ChatMessageType.Answer,
+ canBeVoted = false
+ ),
+ tabId,
+ false
+ )
+ sessionCleanUp(codeTestChatHelper.getActiveSession().tabId)
+ updateUIState()
+ }
+
+ private suspend fun checkActiveFileInIDE(
+ project: Project,
+ message: IncomingCodeTestMessage.StartTestGen,
+ ): ActiveFileInfo? {
+ try {
+ val fileEditorManager = FileEditorManager.getInstance(project)
+ val activeEditor = fileEditorManager.selectedEditor
+ val activeFile = fileEditorManager.selectedFiles.firstOrNull()
+ val projectRoot = project.basePath?.let { Path.of(it) }?.toFile()?.toVirtualFile() ?: run {
+ project.guessProjectDir() ?: error("Cannot guess base directory for project ${project.name}")
+ }
+
+ if (activeEditor == null || activeFile == null) {
+ handleInvalidFileState(message.tabId)
+ return null
+ }
+ val programmingLanguage: CodeWhispererProgrammingLanguage = activeFile.programmingLanguage()
+ if (programmingLanguage.languageId.equals("unknown", ignoreCase = true)) {
+ handleInvalidFileState(message.tabId)
+ return null
+ }
+ return ActiveFileInfo(
+ filePath = activeFile.path,
+ fileName = activeFile.name,
+ fileLanguage = programmingLanguage,
+ fileInWorkspace = activeFile.path.startsWith(projectRoot.path)
+ )
+ } catch (e: Exception) {
+ LOG.debug { "Error checking active file: $e" }
+ updateUIState()
+ codeTestChatHelper.addNewMessage(
+ CodeTestChatMessageContent(message = e.message, type = ChatMessageType.Answer, canBeVoted = false),
+ message.tabId,
+ false
+ )
+ return null
+ }
+ }
+
+ private fun File.toVirtualFile() = LocalFileSystem.getInstance().findFileByIoFile(this)
+
+ /* UTG Tab Chat input use cases:
+ * 1. If User exits the flow and want to start a new generate unit test cycle.
+ * 2. If User clicks on Modify build command option and can enter the build command from chat input
+ * 3. If User trys to regenerate the unit tests case using Regenerate button
+ * */
+ private suspend fun handleChat(tabId: String, message: String) {
+ val session = codeTestChatHelper.getActiveSession()
+ LOG.debug {
+ "$FEATURE_NAME: " +
+ "Processing message: $message " +
+ "tabId: $tabId"
+ }
+ when (session.conversationState) {
+ ConversationState.WAITING_FOR_BUILD_COMMAND_INPUT -> handleBuildCommandInput(session, message)
+ ConversationState.WAITING_FOR_REGENERATE_INPUT -> handleRegenerateInput(session, message)
+ else -> this.processStartTestGen(
+ message = IncomingCodeTestMessage.StartTestGen(
+ tabId = session.tabId,
+ prompt = message,
+ )
+ )
+ }
+ }
+
+ private suspend fun handleRegenerateInput(session: Session, message: String) {
+ codeTestChatHelper.addAnswer(
+ CodeTestChatMessageContent(
+ message,
+ type = ChatMessageType.Prompt,
+ canBeVoted = false
+ )
+ )
+ session.conversationState = ConversationState.IDLE
+ // Start the UTG workflow with new user prompt
+ CodeWhispererUTGChatManager.getInstance(
+ context.project
+ ).generateTests(message, codeTestChatHelper, null, getEditorSelectionRange(context.project))
+ }
+
+ private suspend fun handleBuildCommandInput(session: Session, message: String) {
+ // TODO: Logic to store modified build command
+ session.conversationState = ConversationState.IDLE
+ // for now treat user's input as a single build command.
+ session.buildAndExecuteTaskContext.buildCommand = message
+ session.buildAndExecuteTaskContext.executionCommand = ""
+ codeTestChatHelper.addAnswer(
+ CodeTestChatMessageContent(
+ message = """
+ Would you like me to help build and execute the test? I'll run following commands
+
+ ```sh
+ $message
+ ```
+ """.trimIndent(),
+ type = ChatMessageType.Answer,
+ canBeVoted = true,
+ buttons = listOf(
+ Button(
+ "utg_skip_and_finish",
+ "Skip and finish task",
+ keepCardAfterClick = true,
+ position = "outside",
+ status = "info",
+ ),
+ Button(
+ "utg_modify_command",
+ "Modify commands",
+ keepCardAfterClick = true,
+ position = "outside",
+ status = "info",
+ ),
+ Button(
+ "utg_build_and_execute",
+ "Build and execute",
+ keepCardAfterClick = true,
+ position = "outside",
+ status = "info",
+ ),
+ )
+ )
+ )
+ codeTestChatHelper.updateUI(
+ promptInputDisabledState = true,
+ )
+ println(message)
+ }
+
+ companion object {
+ private val LOG = getLogger()
+ }
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/controller/CodeTestChatHelper.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/controller/CodeTestChatHelper.kt
new file mode 100644
index 0000000000..d4cbe27e7d
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/controller/CodeTestChatHelper.kt
@@ -0,0 +1,156 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqCodeTest.controller
+
+import software.aws.toolkits.jetbrains.services.amazonq.messages.MessagePublisher
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.messages.CodeTestAddAnswerMessage
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.messages.CodeTestChatMessage
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.messages.CodeTestChatMessageContent
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.messages.CodeTestUpdateAnswerMessage
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.messages.CodeTestUpdateUIMessage
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.messages.ProgressField
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.messages.PromptProgressMessage
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.messages.UpdatePlaceholderMessage
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.session.Session
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.storage.ChatSessionStorage
+import software.aws.toolkits.jetbrains.services.cwc.messages.ChatMessageType
+import java.util.UUID
+
+class CodeTestChatHelper(
+ private val messagePublisher: MessagePublisher,
+ private val chatSessionStorage: ChatSessionStorage,
+) {
+ private var activeCodeTestTabId: String = ""
+
+ fun setActiveCodeTestTabId(tabId: String) {
+ activeCodeTestTabId = tabId
+ }
+
+ fun getActiveCodeTestTabId(): String = activeCodeTestTabId
+
+ fun getSession(tabId: String): Session = chatSessionStorage.getSession(tabId)
+
+ fun deleteSession(tabId: String) = chatSessionStorage.deleteSession(tabId)
+
+ fun getActiveSession(): Session = chatSessionStorage.getSession(activeCodeTestTabId)
+
+ private fun isInvalidSession() = chatSessionStorage.getSession(activeCodeTestTabId).isAuthenticating
+
+ // helper for adding a brand new answer(card) to chat UI.
+ suspend fun addAnswer(
+ content: CodeTestChatMessageContent,
+ messageIdOverride: String? = null,
+ ): String? {
+ if (isInvalidSession()) return null
+
+ val messageId = messageIdOverride ?: UUID.randomUUID().toString()
+ messagePublisher.publish(
+ CodeTestAddAnswerMessage(
+ tabId = activeCodeTestTabId,
+ messageId = messageId,
+ messageType = content.type,
+ message = content.message,
+ buttons = content.buttons,
+ formItems = content.formItems,
+ followUps = content.followUps,
+ canBeVoted = content.canBeVoted,
+ isAddingNewItem = true,
+ isLoading = content.type == ChatMessageType.AnswerStream,
+ clearPreviousItemButtons = false,
+ fileList = content.fileList,
+ footer = content.footer,
+ projectRootName = content.projectRootName,
+ codeReference = content.codeReference
+ )
+ )
+ return messageId
+ }
+
+ // helper for updating a specific chat card to chat UI. If messageId is not specified, update the last card.
+ suspend fun updateAnswer(
+ content: CodeTestChatMessageContent,
+ messageIdOverride: String? = null,
+ ) {
+ if (isInvalidSession()) return
+
+ messagePublisher.publish(
+ CodeTestUpdateAnswerMessage(
+ tabId = activeCodeTestTabId,
+ messageId = messageIdOverride,
+ messageType = content.type,
+ message = content.message,
+ buttons = content.buttons,
+ formItems = content.formItems,
+ followUps = content.followUps,
+ isAddingNewItem = true,
+ isLoading = content.type == ChatMessageType.AnswerPart,
+ clearPreviousItemButtons = false,
+ fileList = content.fileList,
+ footer = content.footer,
+ projectRootName = content.projectRootName,
+ codeReference = content.codeReference
+ )
+ )
+ }
+
+ suspend fun sendUpdatePlaceholder(tabId: String, newPlaceholder: String) {
+ messagePublisher.publish(
+ UpdatePlaceholderMessage(
+ tabId = tabId,
+ newPlaceholder = newPlaceholder,
+ )
+ )
+ }
+
+ // Everything to be nullable so that only those that are assigned are changed
+ suspend fun updateUI(
+ loadingChat: Boolean? = null,
+ cancelButtonWhenLoading: Boolean? = null,
+ promptInputPlaceholder: String? = null,
+ promptInputDisabledState: Boolean? = null,
+ promptInputProgress: ProgressField? = null,
+ ) {
+ messagePublisher.publish(
+ CodeTestUpdateUIMessage(
+ activeCodeTestTabId,
+ loadingChat,
+ cancelButtonWhenLoading,
+ promptInputPlaceholder,
+ promptInputDisabledState,
+ promptInputProgress
+ )
+ )
+ }
+
+ // currently only used for removing progress bar
+ suspend fun sendUpdatePromptProgress(tabId: String, progressField: ProgressField?) {
+ if (isInvalidSession()) return
+ messagePublisher.publish(PromptProgressMessage(tabId, progressField))
+ }
+
+ suspend fun addNewMessage(
+ content: CodeTestChatMessageContent,
+ messageIdOverride: String? = null,
+ clearPreviousItemButtons: Boolean? = false,
+ ) {
+ if (isInvalidSession()) return
+
+ messagePublisher.publish(
+ CodeTestChatMessage(
+ tabId = activeCodeTestTabId,
+ messageId = messageIdOverride ?: UUID.randomUUID().toString(),
+ messageType = content.type,
+ message = content.message,
+ buttons = content.buttons,
+ formItems = content.formItems,
+ followUps = content.followUps,
+ canBeVoted = content.canBeVoted,
+ informationCard = content.informationCard,
+ isAddingNewItem = true,
+ isLoading = content.type == ChatMessageType.AnswerPart,
+ clearPreviousItemButtons = clearPreviousItemButtons as Boolean
+ )
+ )
+ }
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/messages/CodeTestMessage.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/messages/CodeTestMessage.kt
new file mode 100644
index 0000000000..9ece7fc48a
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/messages/CodeTestMessage.kt
@@ -0,0 +1,241 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqCodeTest.messages
+
+import com.fasterxml.jackson.annotation.JsonProperty
+import software.aws.toolkits.jetbrains.services.amazonq.auth.AuthFollowUpType
+import software.aws.toolkits.jetbrains.services.amazonq.messages.AmazonQMessage
+import software.aws.toolkits.jetbrains.services.cwc.messages.ChatMessageType
+import software.aws.toolkits.jetbrains.services.cwc.messages.CodeReference
+import software.aws.toolkits.jetbrains.services.cwc.messages.FollowUp
+import java.time.Instant
+import java.util.UUID
+
+const val CODE_TEST_TAB_NAME = "codetest"
+
+enum class CodeTestButtonId(val id: String) {
+ StopTestGeneration("stop_test_generation"),
+}
+
+data class Button(
+ val id: String,
+ val text: String,
+ val description: String? = null,
+ val icon: String? = null,
+ val keepCardAfterClick: Boolean? = false,
+ val disabled: Boolean? = false,
+ val waitMandatoryFormItems: Boolean? = false,
+ val position: String = "inside",
+ val status: String = "primary",
+)
+
+data class ProgressField(
+ val title: String? = null,
+ val value: Int? = null,
+ val valueText: String? = null,
+ val status: String? = null,
+ val actions: List? = null,
+ val text: String? = null,
+)
+
+data class FormItemOption(
+ val label: String,
+ val value: String,
+)
+
+data class FormItem(
+ val id: String,
+ val type: String = "select",
+ val title: String,
+ val mandatory: Boolean = true,
+ val options: List = emptyList(),
+)
+
+sealed interface CodeTestBaseMessage : AmazonQMessage
+
+// === UI -> App Messages ===
+sealed interface IncomingCodeTestMessage : CodeTestBaseMessage {
+ data class ChatPrompt(
+ val chatMessage: String,
+ val command: String,
+ @JsonProperty("tabID") val tabId: String,
+ ) : IncomingCodeTestMessage
+
+ data class StartTestGen(
+ @JsonProperty("tabID") val tabId: String,
+ val prompt: String,
+ ) : IncomingCodeTestMessage
+
+ data class ClickedLink(
+ @JsonProperty("tabID") val tabId: String,
+ val command: String,
+ val messageId: String?,
+ val link: String,
+ ) : IncomingCodeTestMessage
+
+ data class ClearChat(
+ @JsonProperty("tabID") val tabId: String,
+ ) : IncomingCodeTestMessage
+
+ data class Help(
+ @JsonProperty("tabID") val tabId: String,
+ ) : IncomingCodeTestMessage
+
+ data class NewTabCreated(
+ @JsonProperty("tabID") val tabId: String,
+ ) : IncomingCodeTestMessage
+
+ data class TabRemoved(
+ @JsonProperty("tabID") val tabId: String,
+ ) : IncomingCodeTestMessage
+
+ data class ButtonClicked(
+ @JsonProperty("tabID") val tabId: String,
+ @JsonProperty("actionID") val actionID: String,
+ ) : IncomingCodeTestMessage
+}
+
+data class UpdatePlaceholderMessage(
+ @JsonProperty("tabID") override val tabId: String,
+ val newPlaceholder: String,
+) : UiMessage(
+ tabId = tabId,
+ type = "updatePlaceholderMessage"
+)
+
+data class ChatInputEnabledMessage(
+ @JsonProperty("tabID") override val tabId: String,
+ val enabled: Boolean,
+) : UiMessage(
+ tabId = tabId,
+ type = "chatInputEnabledMessage"
+)
+
+data class CodeTestUpdateUIMessage(
+ @JsonProperty("tabID") override val tabId: String,
+ val loadingChat: Boolean?,
+ val cancelButtonWhenLoading: Boolean?,
+ val promptInputPlaceholder: String?,
+ val promptInputDisabledState: Boolean?,
+ val promptInputProgress: ProgressField?,
+) : UiMessage(
+ tabId = tabId,
+ type = "updateUI"
+)
+
+// === App -> UI messages ===
+sealed class UiMessage(
+ open val tabId: String?,
+ open val type: String,
+ open val messageId: String? = UUID.randomUUID().toString(),
+) : CodeTestBaseMessage {
+ val time = Instant.now().epochSecond
+ val sender = CODE_TEST_TAB_NAME
+}
+
+data class AuthenticationUpdateMessage(
+ val authenticatingTabIDs: List,
+ val featureDevEnabled: Boolean,
+ val codeTransformEnabled: Boolean,
+ val codeScanEnabled: Boolean,
+ val codeTestEnabled: Boolean,
+ val docEnabled: Boolean,
+ val message: String? = null,
+) : UiMessage(
+ null,
+ type = "authenticationUpdateMessage"
+)
+
+data class AuthenticationNeededExceptionMessage(
+ @JsonProperty("tabID") override val tabId: String,
+ val authType: AuthFollowUpType,
+ val message: String? = null,
+) : UiMessage(
+ tabId = tabId,
+ type = "authNeededException"
+)
+
+data class PromptProgressMessage(
+ @JsonProperty("tabID") override val tabId: String,
+ val progressField: ProgressField? = null,
+) : UiMessage(
+ tabId = tabId,
+ type = "updatePromptProgress",
+)
+
+data class CodeTestChatMessage(
+ @JsonProperty("tabID") override val tabId: String,
+ override val messageId: String? = UUID.randomUUID().toString(),
+ val messageType: ChatMessageType,
+ val message: String? = null,
+ val buttons: List? = null,
+ val canBeVoted: Boolean? = false,
+ val formItems: List? = null,
+ val followUps: List? = null,
+ val informationCard: Boolean? = false,
+ val isAddingNewItem: Boolean = true,
+ val isLoading: Boolean = false,
+ val clearPreviousItemButtons: Boolean = true,
+) : UiMessage(
+ messageId = messageId,
+ tabId = tabId,
+ type = "chatMessage",
+)
+
+data class CodeTestUpdateAnswerMessage(
+ @JsonProperty("tabID") override val tabId: String,
+ override val messageId: String? = UUID.randomUUID().toString(),
+ val messageType: ChatMessageType,
+ val message: String? = null,
+ val buttons: List? = null,
+ val formItems: List? = null,
+ val followUps: List? = null,
+ val isAddingNewItem: Boolean = true,
+ val isLoading: Boolean = false,
+ val clearPreviousItemButtons: Boolean = true,
+ val fileList: List? = null,
+ val footer: List? = null,
+ val projectRootName: String? = null,
+ val codeReference: List? = null,
+) : UiMessage(
+ messageId = messageId,
+ tabId = tabId,
+ type = "updateAnswer",
+)
+
+data class CodeTestAddAnswerMessage(
+ @JsonProperty("tabID") override val tabId: String,
+ override val messageId: String? = UUID.randomUUID().toString(),
+ val messageType: ChatMessageType,
+ val message: String? = null,
+ val buttons: List? = null,
+ val formItems: List? = null,
+ val followUps: List? = null,
+ val isAddingNewItem: Boolean = true,
+ val isLoading: Boolean = false,
+ val clearPreviousItemButtons: Boolean = true,
+ val fileList: List? = null,
+ val footer: List? = null,
+ val projectRootName: String? = null,
+ val canBeVoted: Boolean = false,
+ val codeReference: List? = null,
+) : UiMessage(
+ messageId = messageId,
+ tabId = tabId,
+ type = "addAnswer",
+)
+
+data class CodeTestChatMessageContent(
+ val message: String? = null,
+ val buttons: List? = null,
+ val formItems: List? = null,
+ val followUps: List? = null,
+ val type: ChatMessageType,
+ val canBeVoted: Boolean = false,
+ val informationCard: Boolean? = false,
+ val fileList: List? = null,
+ val footer: List? = null,
+ val projectRootName: String? = null,
+ val codeReference: List? = null,
+)
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/model/BuildAndExecuteStatusIcon.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/model/BuildAndExecuteStatusIcon.kt
new file mode 100644
index 0000000000..0847cec649
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/model/BuildAndExecuteStatusIcon.kt
@@ -0,0 +1,39 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqCodeTest.model
+
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.session.BuildAndExecuteProgressStatus
+
+enum class BuildAndExecuteStatusIcon(val icon: String) {
+ WAIT("☐ "),
+ CURRENT("☐ "),
+ DONE("✔ "),
+}
+
+fun getBuildIcon(progressStatus: BuildAndExecuteProgressStatus) =
+ if (progressStatus < BuildAndExecuteProgressStatus.RUN_BUILD) {
+ BuildAndExecuteStatusIcon.WAIT.icon
+ } else if (progressStatus == BuildAndExecuteProgressStatus.RUN_BUILD) {
+ BuildAndExecuteStatusIcon.CURRENT.icon
+ } else {
+ BuildAndExecuteStatusIcon.DONE.icon
+ }
+
+fun getExecutionIcon(progressStatus: BuildAndExecuteProgressStatus) =
+ if (progressStatus < BuildAndExecuteProgressStatus.RUN_EXECUTION_TESTS) {
+ BuildAndExecuteStatusIcon.WAIT.icon
+ } else if (progressStatus == BuildAndExecuteProgressStatus.RUN_EXECUTION_TESTS) {
+ BuildAndExecuteStatusIcon.CURRENT.icon
+ } else {
+ BuildAndExecuteStatusIcon.DONE.icon
+ }
+
+fun getFixingTestCasesIcon(progressStatus: BuildAndExecuteProgressStatus) =
+ if (progressStatus < BuildAndExecuteProgressStatus.FIXING_TEST_CASES) {
+ BuildAndExecuteStatusIcon.WAIT.icon
+ } else if (progressStatus == BuildAndExecuteProgressStatus.FIXING_TEST_CASES) {
+ BuildAndExecuteStatusIcon.CURRENT.icon
+ } else {
+ BuildAndExecuteStatusIcon.DONE.icon
+ }
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/model/PreviousUTGIterationContext.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/model/PreviousUTGIterationContext.kt
new file mode 100644
index 0000000000..4576a0ea0c
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/model/PreviousUTGIterationContext.kt
@@ -0,0 +1,13 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqCodeTest.model
+
+import com.intellij.openapi.vfs.VirtualFile
+
+data class PreviousUTGIterationContext(
+ val buildLogFile: VirtualFile,
+ val testLogFile: VirtualFile,
+ val selectedFile: VirtualFile?,
+ val buildAndExecuteMessageId: String?,
+)
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/model/ShortAnswer.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/model/ShortAnswer.kt
new file mode 100644
index 0000000000..fb8caef7aa
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/model/ShortAnswer.kt
@@ -0,0 +1,41 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqCodeTest.model
+
+import com.fasterxml.jackson.annotation.JsonIgnoreProperties
+
+data class ShortAnswerReference(
+ val licenseName: String? = null,
+ val repository: String? = null,
+ val url: String? = null,
+ val recommendationContentSpan: RecommendationContentSpan? = null,
+) {
+ data class RecommendationContentSpan(
+ val start: Int,
+ val end: Int,
+ )
+}
+
+@JsonIgnoreProperties(ignoreUnknown = true)
+data class ShortAnswer(
+ val sourceFilePath: String? = null,
+
+ val testFramework: String? = null,
+
+ val testFilePath: String? = null,
+
+ val buildCommand: String? = null,
+
+ val executionCommand: String? = null,
+
+ val testCoverage: String? = null,
+
+ val stopIteration: String? = null,
+
+ val planSummary: String? = null,
+
+ val codeReferences: List? = null,
+
+ val numberOfTestMethods: Int? = 0,
+)
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/session/Session.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/session/Session.kt
new file mode 100644
index 0000000000..f7c067b5b8
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/session/Session.kt
@@ -0,0 +1,71 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqCodeTest.session
+
+import com.intellij.openapi.vfs.VirtualFile
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.ConversationState
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.model.ShortAnswer
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.model.ShortAnswerReference
+import software.aws.toolkits.jetbrains.services.codewhisperer.language.CodeWhispererProgrammingLanguage
+import software.aws.toolkits.jetbrains.services.codewhisperer.language.languages.CodeWhispererUnknownLanguage
+
+data class Session(val tabId: String) {
+ var isAuthenticating: Boolean = false
+ var authNeededNotified: Boolean = false
+ var conversationState: ConversationState = ConversationState.IDLE
+
+ // Generating unit tests
+ var isGeneratingTests: Boolean = false
+ var programmingLanguage: CodeWhispererProgrammingLanguage = CodeWhispererUnknownLanguage.INSTANCE
+ var testGenerationJob: String = ""
+ var testGenerationJobGroupName: String = ""
+ var startTestGenerationRequestId: String = ""
+
+ // Telemetry
+ var hasUserPromptSupplied: Boolean = false
+ var numberOfUnitTestCasesGenerated: Int? = null
+ var linesOfCodeGenerated: Int? = null
+ var charsOfCodeGenerated: Int? = null
+ var startTimeOfTestGeneration: Double = 0.0
+ var latencyOfTestGeneration: Double = 0.0
+ var isCodeBlockSelected: Boolean = false
+ var srcPayloadSize: Long = 0
+ var srcZipFileSize: Long = 0
+ var artifactUploadDuration: Long = 0
+
+ // First iteration will have a value of 1
+ var iteration: Int = 0
+ var projectRoot: String = "/"
+ var shortAnswer: ShortAnswer = ShortAnswer()
+ var selectedFile: VirtualFile? = null
+ var testFileRelativePathToProjectRoot: String = ""
+ var testFileName: String = ""
+ var viewDiffMessageId: String? = null
+ var openedDiffFile: VirtualFile? = null
+ val generatedTestDiffs = mutableMapOf()
+ var codeReferences: List? = null
+
+ // Build loop execution
+ val buildAndExecuteTaskContext = BuildAndExecuteTaskContext()
+}
+
+data class BuildAndExecuteTaskContext(
+ var buildCommand: String = "",
+ var executionCommand: String = "",
+ var buildExitCode: Int = -1,
+ var testExitCode: Int = -1,
+ var progressStatus: BuildAndExecuteProgressStatus = BuildAndExecuteProgressStatus.START_STEP,
+)
+
+enum class BuildAndExecuteProgressStatus {
+ START_STEP,
+ INSTALL_DEPENDENCIES,
+ RUN_BUILD,
+ RUN_EXECUTION_TESTS,
+ TESTS_EXECUTED,
+ FIXING_TEST_CASES,
+ PROCESS_TEST_RESULTS,
+}
+
+const val UTG_CHAT_MAX_ITERATION = 4
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/storage/ChatSessionStorage.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/storage/ChatSessionStorage.kt
new file mode 100644
index 0000000000..0e38f06c6d
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/storage/ChatSessionStorage.kt
@@ -0,0 +1,20 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqCodeTest.storage
+
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.session.Session
+
+class ChatSessionStorage {
+ private val sessions = mutableMapOf()
+
+ @Synchronized
+ fun getSession(tabId: String): Session = sessions.getOrPut(tabId) { Session(tabId) }
+
+ fun deleteSession(tabId: String) {
+ sessions.remove(tabId)
+ }
+
+ // Find all sessions that are currently waiting to be authenticated
+ fun getAuthenticatingSessions(): List = this.sessions.values.filter { it.isAuthenticating }
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/utils/UTGChatUtil.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/utils/UTGChatUtil.kt
new file mode 100644
index 0000000000..9b1615c461
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqCodeTest/utils/UTGChatUtil.kt
@@ -0,0 +1,200 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqCodeTest.utils
+
+import com.intellij.build.BuildContentManager
+import com.intellij.execution.impl.ConsoleViewImpl
+import com.intellij.execution.process.OSProcessHandler
+import com.intellij.execution.process.ProcessAdapter
+import com.intellij.execution.process.ProcessEvent
+import com.intellij.execution.process.ProcessHandler
+import com.intellij.execution.ui.ConsoleView
+import com.intellij.execution.ui.ConsoleViewContentType
+import com.intellij.openapi.application.ApplicationManager
+import com.intellij.openapi.project.Project
+import com.intellij.openapi.util.Key
+import com.intellij.openapi.vfs.VirtualFile
+import com.intellij.openapi.vfs.VirtualFileManager
+import com.intellij.ui.content.impl.ContentImpl
+import kotlinx.coroutines.currentCoroutineContext
+import kotlinx.coroutines.withContext
+import software.aws.toolkits.jetbrains.core.coroutines.EDT
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.model.getBuildIcon
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.model.getExecutionIcon
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.model.getFixingTestCasesIcon
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.session.BuildAndExecuteProgressStatus
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.session.BuildAndExecuteTaskContext
+import java.io.File
+import java.nio.charset.StandardCharsets
+import java.nio.file.Files
+
+fun constructBuildAndExecutionSummaryText(currentStatus: BuildAndExecuteProgressStatus, iterationNum: Int): String {
+ val progressMessages = mutableListOf()
+
+ if (currentStatus >= BuildAndExecuteProgressStatus.RUN_BUILD) {
+ val verb = if (currentStatus == BuildAndExecuteProgressStatus.RUN_BUILD) "in progress" else "complete"
+ progressMessages.add("${getBuildIcon(currentStatus)}: Build $verb")
+ }
+
+ if (currentStatus >= BuildAndExecuteProgressStatus.RUN_EXECUTION_TESTS) {
+ val verb = if (currentStatus == BuildAndExecuteProgressStatus.RUN_EXECUTION_TESTS) "Executing" else "Executed"
+ progressMessages.add("${getExecutionIcon(currentStatus)}: $verb passed tests")
+ }
+
+ if (currentStatus >= BuildAndExecuteProgressStatus.FIXING_TEST_CASES) {
+ val verb = if (currentStatus == BuildAndExecuteProgressStatus.FIXING_TEST_CASES) "Fixing" else "Fixed"
+ progressMessages.add("${getFixingTestCasesIcon(currentStatus)}: $verb errors in tests")
+ }
+
+ if (currentStatus >= BuildAndExecuteProgressStatus.PROCESS_TEST_RESULTS) {
+ progressMessages.add("\n")
+ progressMessages.add("**Test case summary**")
+ progressMessages.add("\n")
+ progressMessages.add("Unit test coverage X%")
+ progressMessages.add("Build fails Y")
+ progressMessages.add("Assertion fails Z")
+ }
+
+ val prefix =
+ if (iterationNum < 2) {
+ "Sure"
+ } else {
+ val timeString = when (iterationNum) {
+ 2 -> "second"
+ 3 -> "third"
+ 4 -> "fourth"
+ // shouldn't reach
+ else -> "fifth"
+ }
+ "Working on the $timeString iteration now"
+ }
+
+ // Join all progress messages into a single string
+ return """
+ $prefix. This may take a few minutes and I'll update the progress here.
+
+ **Progress summary**
+
+ """.trimIndent() + progressMessages.joinToString("\n")
+}
+
+fun runBuildOrTestCommand(
+ localCommand: String,
+ tmpFile: VirtualFile,
+ project: Project,
+ isBuildCommand: Boolean,
+ buildAndExecuteTaskContext: BuildAndExecuteTaskContext,
+) {
+ if (localCommand.isEmpty()) {
+ buildAndExecuteTaskContext.testExitCode = 0
+ return
+ }
+ val repositoryPath = project.basePath ?: return
+ val commandParts = localCommand.split(" ")
+ val command = commandParts.first()
+ val args = commandParts.drop(1)
+ val file = File(tmpFile.path)
+
+ // Create Console View for Build Output
+ val console: ConsoleView = ConsoleViewImpl(project, true)
+
+ // Attach Console View to Build Tool Window
+ ApplicationManager.getApplication().invokeLater {
+ val tabName = if (isBuildCommand) "Q TestGen Build Output" else "Q Test Gen Test Execution Output"
+ val content = ContentImpl(console.component, tabName, true)
+ BuildContentManager.getInstance(project).addContent(content)
+ // TODO: remove these tabs when they are not needed
+ BuildContentManager.getInstance(project).setSelectedContent(content, false, false, true, null)
+ }
+
+ val processBuilder = ProcessBuilder()
+ .command(listOf(command) + args)
+ .directory(File(repositoryPath))
+ .redirectErrorStream(true)
+
+ try {
+ val process = processBuilder.start()
+ val processHandler: ProcessHandler = OSProcessHandler(process, localCommand, null)
+
+ // Attach Process Listener for Output Handling
+ processHandler.addProcessListener(object : ProcessAdapter() {
+ override fun onTextAvailable(event: ProcessEvent, outputType: Key<*>) {
+ val cleanedText = cleanText(event.text)
+ ApplicationManager.getApplication().invokeLater {
+ ApplicationManager.getApplication().runWriteAction {
+ file.appendText(cleanedText)
+ }
+ }
+ }
+
+ override fun processTerminated(event: ProcessEvent) {
+ val exitCode = event.exitCode
+ if (exitCode == 0) {
+ // green color
+ console.print("\nBUILD SUCCESSFUL\n", ConsoleViewContentType.USER_INPUT)
+ } else {
+ // red color
+ console.print("\nBUILD FAILED with exit code $exitCode\n", ConsoleViewContentType.ERROR_OUTPUT)
+ }
+ if (isBuildCommand) {
+ buildAndExecuteTaskContext.buildExitCode = exitCode
+ } else {
+ buildAndExecuteTaskContext.testExitCode = exitCode
+ }
+ }
+ })
+
+ // Start Process and Notify
+ console.attachToProcess(processHandler)
+ processHandler.startNotify()
+ console.print("\n", ConsoleViewContentType.NORMAL_OUTPUT)
+ } catch (e: Exception) {
+ console.print("Error executing command: $localCommand\n", ConsoleViewContentType.ERROR_OUTPUT)
+ console.print("$e", ConsoleViewContentType.ERROR_OUTPUT)
+ if (isBuildCommand) {
+ buildAndExecuteTaskContext.buildExitCode = 1
+ } else {
+ buildAndExecuteTaskContext.testExitCode = 1
+ }
+ return
+ }
+}
+
+private fun cleanText(input: String): String {
+ val cleaned = StringBuilder()
+ for (char in input) {
+ if (char == '\b' && cleaned.isNotEmpty()) {
+ // Remove the last character when encountering a backspace
+ cleaned.deleteCharAt(cleaned.length - 1)
+ } else if (char != '\b') {
+ cleaned.append(char)
+ }
+ }
+ return cleaned.toString()
+}
+
+suspend fun combineBuildAndExecuteLogFiles(
+ buildLogFile: VirtualFile?,
+ testLogFile: VirtualFile?,
+): VirtualFile? {
+ if (buildLogFile == null || testLogFile == null) return null
+ val buildLogFileContent = String(buildLogFile.contentsToByteArray(), StandardCharsets.UTF_8)
+ val testLogFileContent = String(testLogFile.contentsToByteArray(), StandardCharsets.UTF_8)
+
+ val combinedContent = "Build Output:\n$buildLogFileContent\nTest Execution Output:\n$testLogFileContent"
+
+ // Create a new virtual file and write combined content
+ val newFile = VirtualFileManager.getInstance().findFileByNioPath(
+ withContext(currentCoroutineContext()) {
+ Files.createTempFile(null, null)
+ }
+ )
+ withContext(EDT) {
+ ApplicationManager.getApplication().runWriteAction {
+ newFile?.setBinaryContent(combinedContent.toByteArray(StandardCharsets.UTF_8))
+ }
+ }
+
+ return newFile
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/DocApp.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/DocApp.kt
new file mode 100644
index 0000000000..0e55a65557
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/DocApp.kt
@@ -0,0 +1,96 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqDoc
+
+import com.intellij.openapi.application.ApplicationManager
+import kotlinx.coroutines.launch
+import software.aws.toolkits.jetbrains.core.coroutines.disposableCoroutineScope
+import software.aws.toolkits.jetbrains.core.credentials.ToolkitConnection
+import software.aws.toolkits.jetbrains.core.credentials.ToolkitConnectionManagerListener
+import software.aws.toolkits.jetbrains.services.amazonq.apps.AmazonQApp
+import software.aws.toolkits.jetbrains.services.amazonq.apps.AmazonQAppInitContext
+import software.aws.toolkits.jetbrains.services.amazonq.messages.AmazonQMessage
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.auth.isCodeScanAvailable
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.auth.isCodeTestAvailable
+import software.aws.toolkits.jetbrains.services.amazonqDoc.auth.isDocAvailable
+import software.aws.toolkits.jetbrains.services.amazonqDoc.controller.DocController
+import software.aws.toolkits.jetbrains.services.amazonqDoc.messages.AuthenticationUpdateMessage
+import software.aws.toolkits.jetbrains.services.amazonqDoc.messages.IncomingDocMessage
+import software.aws.toolkits.jetbrains.services.amazonqDoc.storage.ChatSessionStorage
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.auth.isFeatureDevAvailable
+import software.aws.toolkits.jetbrains.services.codemodernizer.utils.isCodeTransformAvailable
+
+class DocApp : AmazonQApp {
+
+ private val scope = disposableCoroutineScope(this)
+
+ override val tabTypes = listOf("doc")
+
+ override fun init(context: AmazonQAppInitContext) {
+ val chatSessionStorage = ChatSessionStorage()
+ // Create Doc controller
+ val inboundAppMessagesHandler =
+ DocController(context, chatSessionStorage)
+
+ context.messageTypeRegistry.register(
+ "chat-prompt" to IncomingDocMessage.ChatPrompt::class,
+ "new-tab-was-created" to IncomingDocMessage.NewTabCreated::class,
+ "tab-was-removed" to IncomingDocMessage.TabRemoved::class,
+ "auth-follow-up-was-clicked" to IncomingDocMessage.AuthFollowUpWasClicked::class,
+ "follow-up-was-clicked" to IncomingDocMessage.FollowupClicked::class,
+ "chat-item-voted" to IncomingDocMessage.ChatItemVotedMessage::class,
+ "chat-item-feedback" to IncomingDocMessage.ChatItemFeedbackMessage::class,
+ "response-body-link-click" to IncomingDocMessage.ClickedLink::class,
+ "insert_code_at_cursor_position" to IncomingDocMessage.InsertCodeAtCursorPosition::class,
+ "open-diff" to IncomingDocMessage.OpenDiff::class,
+ "file-click" to IncomingDocMessage.FileClicked::class,
+ "doc_stop_generate" to IncomingDocMessage.StopDocGeneration::class
+ )
+
+ scope.launch {
+ context.messagesFromUiToApp.flow.collect { message ->
+ // Launch a new coroutine to handle each message
+ scope.launch { handleMessage(message, inboundAppMessagesHandler) }
+ }
+ }
+
+ ApplicationManager.getApplication().messageBus.connect(this).subscribe(
+ ToolkitConnectionManagerListener.TOPIC,
+ object : ToolkitConnectionManagerListener {
+ override fun activeConnectionChanged(newConnection: ToolkitConnection?) {
+ scope.launch {
+ context.messagesFromAppToUi.publish(
+ AuthenticationUpdateMessage(
+ featureDevEnabled = isFeatureDevAvailable(context.project),
+ codeTransformEnabled = isCodeTransformAvailable(context.project),
+ codeScanEnabled = isCodeScanAvailable(context.project),
+ codeTestEnabled = isCodeTestAvailable(context.project),
+ docEnabled = isDocAvailable(context.project),
+ authenticatingTabIDs = chatSessionStorage.getAuthenticatingSessions().map { it.tabID }
+ )
+ )
+ }
+ }
+ }
+ )
+ }
+
+ private suspend fun handleMessage(message: AmazonQMessage, inboundAppMessagesHandler: InboundAppMessagesHandler) {
+ when (message) {
+ is IncomingDocMessage.ChatPrompt -> inboundAppMessagesHandler.processPromptChatMessage(message)
+ is IncomingDocMessage.NewTabCreated -> inboundAppMessagesHandler.processNewTabCreatedMessage(message)
+ is IncomingDocMessage.TabRemoved -> inboundAppMessagesHandler.processTabRemovedMessage(message)
+ is IncomingDocMessage.AuthFollowUpWasClicked -> inboundAppMessagesHandler.processAuthFollowUpClick(message)
+ is IncomingDocMessage.FollowupClicked -> inboundAppMessagesHandler.processFollowupClickedMessage(message)
+ is IncomingDocMessage.ClickedLink -> inboundAppMessagesHandler.processLinkClick(message)
+ is IncomingDocMessage.OpenDiff -> inboundAppMessagesHandler.processOpenDiff(message)
+ is IncomingDocMessage.FileClicked -> inboundAppMessagesHandler.processFileClicked(message)
+ is IncomingDocMessage.StopDocGeneration -> inboundAppMessagesHandler.processStopDocGeneration(message)
+ }
+ }
+
+ override fun dispose() {
+ // nothing to do
+ }
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/DocAppFactory.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/DocAppFactory.kt
new file mode 100644
index 0000000000..de2526a665
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/DocAppFactory.kt
@@ -0,0 +1,11 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqDoc
+
+import com.intellij.openapi.project.Project
+import software.aws.toolkits.jetbrains.services.amazonq.apps.AmazonQAppFactory
+
+class DocAppFactory : AmazonQAppFactory {
+ override fun createApp(project: Project) = DocApp()
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/DocChatItems.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/DocChatItems.kt
new file mode 100644
index 0000000000..b4abb0540c
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/DocChatItems.kt
@@ -0,0 +1,45 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqDoc
+
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.messages.Button
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.messages.ProgressField
+import software.aws.toolkits.resources.message
+
+val cancellingProgressField = ProgressField(
+ status = "warning",
+ text = message("general.canceling"),
+ value = -1,
+ actions = emptyList()
+)
+
+// TODO: Need to change the string after the F2F
+val docGenCompletedField = ProgressField(
+ status = "success",
+ text = message("general.success"),
+ value = 100,
+ actions = emptyList()
+)
+
+val cancelTestGenButton = Button(
+ id = "doc_stop_generate",
+ text = message("general.cancel"),
+ icon = "cancel"
+)
+
+fun inProgress(progress: Int, message: String? = null): ProgressField? {
+ // Constants to improve readability and maintainability
+ val completionProgress = 100
+ val completionValue = -1
+
+ // Pre-calculate the conditions to avoid repeated evaluations
+ val isComplete = progress >= completionProgress
+
+ return ProgressField(
+ status = "default",
+ text = message ?: message("amazonqDoc.inprogress_message.generating"),
+ value = if (isComplete) completionValue else progress,
+ actions = listOf(cancelTestGenButton)
+ )
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/DocConstants.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/DocConstants.kt
new file mode 100644
index 0000000000..0d3c7c4f2c
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/DocConstants.kt
@@ -0,0 +1,27 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqDoc
+
+const val FEATURE_EVALUATION_PRODUCT_NAME = "DocGeneration"
+
+const val FEATURE_NAME = "Amazon Q Documentation Generation"
+
+// Max number of times a user can attempt to retry a code generation request if it fails
+const val CODE_GENERATION_RETRY_LIMIT = 3
+
+// The default retry limit used when the session could not be found
+const val DEFAULT_RETRY_LIMIT = 0
+
+// Max allowed size for a repository in bytes
+const val MAX_PROJECT_SIZE_BYTES: Long = 200 * 1024 * 1024
+
+enum class ModifySourceFolderErrorReason(
+ private val reasonText: String,
+) {
+ ClosedBeforeSelection("ClosedBeforeSelection"),
+ NotInWorkspaceFolder("NotInWorkspaceFolder"),
+ ;
+
+ override fun toString(): String = reasonText
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/DocExceptions.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/DocExceptions.kt
new file mode 100644
index 0000000000..b27592b465
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/DocExceptions.kt
@@ -0,0 +1,25 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqDoc
+
+import software.aws.toolkits.resources.message
+
+open class DocException(override val message: String?, override val cause: Throwable? = null) : RuntimeException()
+
+class ZipFileError(override val message: String, override val cause: Throwable?) : RuntimeException()
+
+class CodeIterationLimitError(override val message: String, override val cause: Throwable?) : RuntimeException()
+
+internal fun docServiceError(message: String?): Nothing =
+ throw DocException(message)
+
+internal fun codeGenerationFailedError(): Nothing =
+ throw DocException(message("amazonqFeatureDev.code_generation.failed_generation"))
+
+internal fun conversationIdNotFound(): Nothing =
+ throw DocException(message("amazonqFeatureDev.exception.conversation_not_found"))
+
+val denyListedErrors = arrayOf("Deserialization error", "Inaccessible host", "UnknownHost")
+fun createUserFacingErrorMessage(message: String?): String? =
+ if (message != null && denyListedErrors.any { message.contains(it) }) "$FEATURE_NAME API request failed" else message
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/InboundAppMessagesHandler.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/InboundAppMessagesHandler.kt
new file mode 100644
index 0000000000..3227335196
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/InboundAppMessagesHandler.kt
@@ -0,0 +1,18 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqDoc
+
+import software.aws.toolkits.jetbrains.services.amazonqDoc.messages.IncomingDocMessage
+
+interface InboundAppMessagesHandler {
+ suspend fun processPromptChatMessage(message: IncomingDocMessage.ChatPrompt)
+ suspend fun processNewTabCreatedMessage(message: IncomingDocMessage.NewTabCreated)
+ suspend fun processTabRemovedMessage(message: IncomingDocMessage.TabRemoved)
+ suspend fun processAuthFollowUpClick(message: IncomingDocMessage.AuthFollowUpWasClicked)
+ suspend fun processFollowupClickedMessage(message: IncomingDocMessage.FollowupClicked)
+ suspend fun processLinkClick(message: IncomingDocMessage.ClickedLink)
+ suspend fun processOpenDiff(message: IncomingDocMessage.OpenDiff)
+ suspend fun processFileClicked(message: IncomingDocMessage.FileClicked)
+ suspend fun processStopDocGeneration(message: IncomingDocMessage.StopDocGeneration)
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/auth/DocAuthUtils.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/auth/DocAuthUtils.kt
new file mode 100644
index 0000000000..3390a8c5bb
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/auth/DocAuthUtils.kt
@@ -0,0 +1,16 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqDoc.auth
+
+import com.intellij.openapi.project.Project
+import software.aws.toolkits.jetbrains.core.gettingstarted.editor.ActiveConnection
+import software.aws.toolkits.jetbrains.core.gettingstarted.editor.ActiveConnectionType
+import software.aws.toolkits.jetbrains.core.gettingstarted.editor.BearerTokenFeatureSet
+import software.aws.toolkits.jetbrains.core.gettingstarted.editor.checkBearerConnectionValidity
+
+fun isDocAvailable(project: Project): Boolean {
+ val connection = checkBearerConnectionValidity(project, BearerTokenFeatureSet.Q)
+ return (connection.connectionType == ActiveConnectionType.IAM_IDC || connection.connectionType == ActiveConnectionType.BUILDER_ID) &&
+ connection is ActiveConnection.ValidBearer
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/controller/DocController.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/controller/DocController.kt
new file mode 100644
index 0000000000..2d3729532a
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/controller/DocController.kt
@@ -0,0 +1,995 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqDoc.controller
+
+import com.intellij.diff.DiffContentFactory
+import com.intellij.diff.DiffManager
+import com.intellij.diff.contents.EmptyContent
+import com.intellij.diff.requests.SimpleDiffRequest
+import com.intellij.diff.util.DiffUserDataKeys
+import com.intellij.ide.BrowserUtil
+import com.intellij.openapi.application.runInEdt
+import com.intellij.openapi.project.Project
+import com.intellij.openapi.roots.ProjectRootManager
+import com.intellij.openapi.vfs.LocalFileSystem
+import com.intellij.openapi.vfs.VfsUtil
+import com.intellij.openapi.wm.ToolWindowManager
+import kotlinx.coroutines.withContext
+import software.amazon.awssdk.services.codewhispererruntime.model.DocGenerationFolderLevel
+import software.amazon.awssdk.services.codewhispererruntime.model.DocGenerationInteractionType
+import software.amazon.awssdk.services.codewhispererruntime.model.DocGenerationUserDecision
+import software.aws.toolkits.core.utils.debug
+import software.aws.toolkits.core.utils.error
+import software.aws.toolkits.core.utils.getLogger
+import software.aws.toolkits.core.utils.info
+import software.aws.toolkits.core.utils.warn
+import software.aws.toolkits.jetbrains.common.util.selectFolder
+import software.aws.toolkits.jetbrains.core.coroutines.EDT
+import software.aws.toolkits.jetbrains.services.amazonq.RepoSizeError
+import software.aws.toolkits.jetbrains.services.amazonq.apps.AmazonQAppInitContext
+import software.aws.toolkits.jetbrains.services.amazonq.auth.AuthController
+import software.aws.toolkits.jetbrains.services.amazonq.toolwindow.AmazonQToolWindowFactory
+import software.aws.toolkits.jetbrains.services.amazonqDoc.DEFAULT_RETRY_LIMIT
+import software.aws.toolkits.jetbrains.services.amazonqDoc.DocException
+import software.aws.toolkits.jetbrains.services.amazonqDoc.FEATURE_NAME
+import software.aws.toolkits.jetbrains.services.amazonqDoc.InboundAppMessagesHandler
+import software.aws.toolkits.jetbrains.services.amazonqDoc.ZipFileError
+import software.aws.toolkits.jetbrains.services.amazonqDoc.cancellingProgressField
+import software.aws.toolkits.jetbrains.services.amazonqDoc.createUserFacingErrorMessage
+import software.aws.toolkits.jetbrains.services.amazonqDoc.denyListedErrors
+import software.aws.toolkits.jetbrains.services.amazonqDoc.inProgress
+import software.aws.toolkits.jetbrains.services.amazonqDoc.messages.DocMessageType
+import software.aws.toolkits.jetbrains.services.amazonqDoc.messages.FollowUp
+import software.aws.toolkits.jetbrains.services.amazonqDoc.messages.FollowUpIcons
+import software.aws.toolkits.jetbrains.services.amazonqDoc.messages.FollowUpStatusType
+import software.aws.toolkits.jetbrains.services.amazonqDoc.messages.FollowUpTypes
+import software.aws.toolkits.jetbrains.services.amazonqDoc.messages.IncomingDocMessage
+import software.aws.toolkits.jetbrains.services.amazonqDoc.messages.initialExamples
+import software.aws.toolkits.jetbrains.services.amazonqDoc.messages.sendAnswer
+import software.aws.toolkits.jetbrains.services.amazonqDoc.messages.sendAsyncEventProgress
+import software.aws.toolkits.jetbrains.services.amazonqDoc.messages.sendAuthNeededException
+import software.aws.toolkits.jetbrains.services.amazonqDoc.messages.sendAuthenticationInProgressMessage
+import software.aws.toolkits.jetbrains.services.amazonqDoc.messages.sendChatInputEnabledMessage
+import software.aws.toolkits.jetbrains.services.amazonqDoc.messages.sendCodeResult
+import software.aws.toolkits.jetbrains.services.amazonqDoc.messages.sendError
+import software.aws.toolkits.jetbrains.services.amazonqDoc.messages.sendFolderConfirmationMessage
+import software.aws.toolkits.jetbrains.services.amazonqDoc.messages.sendMonthlyLimitError
+import software.aws.toolkits.jetbrains.services.amazonqDoc.messages.sendSystemPrompt
+import software.aws.toolkits.jetbrains.services.amazonqDoc.messages.sendUpdatePlaceholder
+import software.aws.toolkits.jetbrains.services.amazonqDoc.messages.sendUpdatePromptProgress
+import software.aws.toolkits.jetbrains.services.amazonqDoc.messages.updateFileComponent
+import software.aws.toolkits.jetbrains.services.amazonqDoc.session.DocSession
+import software.aws.toolkits.jetbrains.services.amazonqDoc.session.PrepareDocGenerationState
+import software.aws.toolkits.jetbrains.services.amazonqDoc.storage.ChatSessionStorage
+import software.aws.toolkits.jetbrains.services.amazonqDoc.util.getFollowUpOptions
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.CodeIterationLimitException
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.MonthlyConversationLimitError
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.CodeReferenceGenerated
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.DeletedFileInfo
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.NewFileZipInfo
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.SessionStatePhase
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.util.CancellationTokenSource
+import software.aws.toolkits.resources.message
+import java.nio.file.Paths
+import java.util.UUID
+
+enum class DocGenerationStep {
+ UPLOAD_TO_S3,
+ CREATE_KNOWLEDGE_GRAPH,
+ SUMMARIZING_FILES,
+ GENERATING_ARTIFACTS,
+ COMPLETE,
+}
+
+enum class Mode(val value: String) {
+ NONE("None"),
+ CREATE("Create"),
+ SYNC("Sync"),
+ EDIT("Edit"),
+}
+
+val checkIcons = mapOf(
+ "wait" to "☐",
+ "current" to "☐",
+ "done" to "☑"
+)
+
+fun getIconForStep(targetStep: DocGenerationStep, currentStep: DocGenerationStep): String? = when {
+ currentStep == targetStep -> checkIcons["current"]
+ currentStep > targetStep -> checkIcons["done"]
+ else -> checkIcons["wait"]
+}
+
+fun docGenerationProgressMessage(currentStep: DocGenerationStep, mode: Mode?): String {
+ val isCreationMode = mode == Mode.CREATE
+ val baseLine = if (isCreationMode) message("amazonqDoc.progress_message.creating") else message("amazonqDoc.progress_message.updating")
+
+ return """
+ $baseLine ${message("amazonqDoc.progress_message.baseline")}
+
+ ${getIconForStep(DocGenerationStep.UPLOAD_TO_S3, currentStep)} ${message("amazonqDoc.progress_message.scanning")}
+
+ ${getIconForStep(DocGenerationStep.SUMMARIZING_FILES, currentStep)} ${message("amazonqDoc.progress_message.summarizing")}
+
+ ${getIconForStep(DocGenerationStep.GENERATING_ARTIFACTS, currentStep)} ${message("amazonqDoc.progress_message.generating")}
+ """.trimIndent()
+}
+
+class DocController(
+ private val context: AmazonQAppInitContext,
+ private val chatSessionStorage: ChatSessionStorage,
+ private val authController: AuthController = AuthController(),
+) : InboundAppMessagesHandler {
+ val messenger = context.messagesFromAppToUi
+ var mode: Mode = Mode.CREATE
+ val toolWindow = ToolWindowManager.getInstance(context.project).getToolWindow(AmazonQToolWindowFactory.WINDOW_ID)
+ var docGenerationTask = DocGenerationTask()
+
+ override suspend fun processPromptChatMessage(message: IncomingDocMessage.ChatPrompt) {
+ handleChat(
+ tabId = message.tabId,
+ message = message.chatMessage
+ )
+ }
+
+ override suspend fun processNewTabCreatedMessage(message: IncomingDocMessage.NewTabCreated) {
+ newTabOpened(message.tabId)
+ }
+
+ override suspend fun processTabRemovedMessage(message: IncomingDocMessage.TabRemoved) {
+ docGenerationTask.reset()
+ chatSessionStorage.deleteSession(message.tabId)
+ }
+
+ override suspend fun processAuthFollowUpClick(message: IncomingDocMessage.AuthFollowUpWasClicked) {
+ authController.handleAuth(context.project, message.authType)
+ messenger.sendAuthenticationInProgressMessage(message.tabId) // show user that authentication is in progress
+ messenger.sendChatInputEnabledMessage(message.tabId, enabled = false) // disable the input field while authentication is in progress
+ }
+
+ override suspend fun processFollowupClickedMessage(message: IncomingDocMessage.FollowupClicked) {
+ val session = getSessionInfo(message.tabId)
+
+ session.preloader(message.followUp.pillText, messenger) // also stores message in session history
+
+ when (message.followUp.type) {
+ FollowUpTypes.RETRY -> retryRequests(message.tabId)
+ FollowUpTypes.MODIFY_DEFAULT_SOURCE_FOLDER -> modifyDefaultSourceFolder(message.tabId)
+ FollowUpTypes.DEV_EXAMPLES -> messenger.initialExamples(message.tabId)
+ FollowUpTypes.INSERT_CODE -> insertCode(message.tabId)
+ FollowUpTypes.PROVIDE_FEEDBACK_AND_REGENERATE_CODE -> provideFeedbackAndRegenerateCode(message.tabId)
+ FollowUpTypes.NEW_TASK -> newTask(message.tabId)
+ FollowUpTypes.CLOSE_SESSION -> closeSession(message.tabId)
+ FollowUpTypes.CREATE_DOCUMENTATION -> {
+ docGenerationTask.interactionType = DocGenerationInteractionType.GENERATE_README
+ mode = Mode.CREATE
+ promptForDocTarget(message.tabId)
+ }
+
+ FollowUpTypes.UPDATE_DOCUMENTATION -> {
+ docGenerationTask.interactionType = DocGenerationInteractionType.UPDATE_README
+ updateDocumentation(message.tabId)
+ }
+
+ FollowUpTypes.CANCEL_FOLDER_SELECTION -> {
+ docGenerationTask.reset()
+ newTask(message.tabId)
+ }
+
+ FollowUpTypes.PROCEED_FOLDER_SELECTION -> if (mode == Mode.EDIT) makeChanges(message.tabId) else onDocsGeneration(message)
+ FollowUpTypes.ACCEPT_CHANGES -> {
+ docGenerationTask.userDecision = DocGenerationUserDecision.ACCEPT
+ sendDocGenerationTelemetry(message.tabId)
+ acceptChanges(message)
+ }
+
+ FollowUpTypes.MAKE_CHANGES -> {
+ mode = Mode.EDIT
+ makeChanges(message.tabId)
+ }
+
+ FollowUpTypes.REJECT_CHANGES -> {
+ docGenerationTask.userDecision = DocGenerationUserDecision.REJECT
+ sendDocGenerationTelemetry(message.tabId)
+ rejectChanges(message)
+ }
+
+ FollowUpTypes.SYNCHRONIZE_DOCUMENTATION -> {
+ mode = Mode.SYNC
+ promptForDocTarget(message.tabId)
+ }
+
+ FollowUpTypes.EDIT_DOCUMENTATION -> {
+ mode = Mode.EDIT
+ docGenerationTask.interactionType = DocGenerationInteractionType.EDIT_README
+ promptForDocTarget(message.tabId)
+ }
+ }
+ }
+
+ override suspend fun processStopDocGeneration(message: IncomingDocMessage.StopDocGeneration) {
+ messenger.sendUpdatePromptProgress(
+ tabId = message.tabId,
+ progressField = cancellingProgressField
+ )
+
+ messenger.sendAnswer(
+ tabId = message.tabId,
+ message("amazonqFeatureDev.code_generation.stopping_code_generation"),
+ messageType = DocMessageType.Answer,
+ canBeVoted = false
+ )
+ messenger.sendUpdatePlaceholder(
+ tabId = message.tabId,
+ newPlaceholder = message("amazonqFeatureDev.code_generation.stopping_code_generation")
+ )
+ messenger.sendChatInputEnabledMessage(tabId = message.tabId, enabled = false)
+ val session = getSessionInfo(message.tabId)
+
+ if (session.sessionState.token?.token !== null) {
+ session.sessionState.token?.cancel()
+ }
+
+ docGenerationTask.reset()
+ newTask(message.tabId)
+ }
+
+ private suspend fun updateDocumentation(tabId: String) {
+ messenger.sendAnswer(
+ tabId,
+ messageType = DocMessageType.Answer,
+ followUp = listOf(
+ FollowUp(
+ type = FollowUpTypes.SYNCHRONIZE_DOCUMENTATION,
+ pillText = message("amazonqDoc.prompt.update.follow_up.sync"),
+ prompt = message("amazonqDoc.prompt.update.follow_up.sync"),
+ ),
+ FollowUp(
+ type = FollowUpTypes.EDIT_DOCUMENTATION,
+ pillText = message("amazonqDoc.prompt.update.follow_up.edit"),
+ prompt = message("amazonqDoc.prompt.update.follow_up.edit"),
+ )
+ )
+ )
+
+ messenger.sendChatInputEnabledMessage(tabId, enabled = false)
+ }
+
+ private suspend fun makeChanges(tabId: String) {
+ messenger.sendAnswer(
+ tabId = tabId,
+ message = message("amazonqDoc.edit.message"),
+ messageType = DocMessageType.Answer
+ )
+
+ messenger.sendUpdatePlaceholder(tabId, message("amazonqDoc.edit.placeholder"))
+ messenger.sendChatInputEnabledMessage(tabId, true)
+ }
+
+ private suspend fun rejectChanges(message: IncomingDocMessage.FollowupClicked) {
+ messenger.sendAnswer(
+ tabId = message.tabId,
+ message = message("amazonqDoc.prompt.reject.message"),
+ followUp = listOf(
+ FollowUp(
+ pillText = message("amazonqFeatureDev.follow_up.new_task"),
+ prompt = message("amazonqFeatureDev.follow_up.new_task"),
+ status = FollowUpStatusType.Info,
+ type = FollowUpTypes.NEW_TASK
+ ),
+ FollowUp(
+ pillText = message("amazonqFeatureDev.follow_up.close_session"),
+ prompt = message("amazonqFeatureDev.follow_up.close_session"),
+ status = FollowUpStatusType.Info,
+ type = FollowUpTypes.CLOSE_SESSION
+ )
+ ),
+ messageType = DocMessageType.Answer
+ )
+
+ messenger.sendChatInputEnabledMessage(message.tabId, false)
+ }
+
+ private suspend fun acceptChanges(message: IncomingDocMessage.FollowupClicked) {
+ insertCode(message.tabId)
+ }
+
+ private suspend fun promptForDocTarget(tabId: String) {
+ val session = getSessionInfo(tabId)
+
+ val currentSourceFolder = session.context.selectedSourceFolder
+
+ try {
+ messenger.sendFolderConfirmationMessage(
+ tabId = tabId,
+ message = if (mode == Mode.CREATE) message("amazonqDoc.prompt.create.confirmation") else message("amazonqDoc.prompt.update"),
+ folderPath = currentSourceFolder.name,
+ followUps = listOf(
+ FollowUp(
+ icon = FollowUpIcons.Ok,
+ pillText = message("amazonqDoc.prompt.folder.proceed"),
+ prompt = message("amazonqDoc.prompt.folder.proceed"),
+ status = FollowUpStatusType.Success,
+ type = FollowUpTypes.PROCEED_FOLDER_SELECTION
+ ),
+ FollowUp(
+ icon = FollowUpIcons.Refresh,
+ pillText = message("amazonqDoc.prompt.folder.change"),
+ prompt = message("amazonqDoc.prompt.folder.change"),
+ status = FollowUpStatusType.Info,
+ type = FollowUpTypes.MODIFY_DEFAULT_SOURCE_FOLDER
+ ),
+ FollowUp(
+ icon = FollowUpIcons.Cancel,
+ pillText = message("general.cancel"),
+ prompt = message("general.cancel"),
+ status = FollowUpStatusType.Error,
+ type = FollowUpTypes.CANCEL_FOLDER_SELECTION
+ ),
+ )
+ )
+
+ messenger.sendChatInputEnabledMessage(tabId, false)
+ } catch (e: Exception) {
+ logger.error { "Error sending answer: ${e.message}" }
+ // Consider logging the error or handling it appropriately
+ }
+ }
+
+ override suspend fun processLinkClick(message: IncomingDocMessage.ClickedLink) {
+ BrowserUtil.browse(message.link)
+ }
+
+ override suspend fun processOpenDiff(message: IncomingDocMessage.OpenDiff) {
+ val session = getSessionInfo(message.tabId)
+
+ val project = context.project
+ val sessionState = session.sessionState
+
+ when (sessionState) {
+ is PrepareDocGenerationState -> {
+ runInEdt {
+ val existingFile = VfsUtil.findRelativeFile(message.filePath, session.context.selectedSourceFolder)
+
+ val leftDiffContent = if (existingFile == null) {
+ EmptyContent()
+ } else {
+ DiffContentFactory.getInstance().create(project, existingFile)
+ }
+
+ val newFileContent = sessionState.filePaths.find { it.zipFilePath == message.filePath }?.fileContent
+
+ val rightDiffContent = if (message.deleted || newFileContent == null) {
+ EmptyContent()
+ } else {
+ DiffContentFactory.getInstance().create(newFileContent)
+ }
+
+ val request = SimpleDiffRequest(message.filePath, leftDiffContent, rightDiffContent, null, null)
+ request.putUserData(DiffUserDataKeys.FORCE_READ_ONLY, true)
+
+ DiffManager.getInstance().showDiff(project, request)
+ }
+ }
+
+ else -> {
+ logger.error { "$FEATURE_NAME: OpenDiff event is received for a conversation that has ${session.sessionState.phase} phase" }
+ messenger.sendError(
+ tabId = message.tabId,
+ errMessage = message("amazonqFeatureDev.exception.open_diff_failed"),
+ retries = 0,
+ conversationId = session.conversationIdUnsafe
+ )
+ }
+ }
+ }
+
+ override suspend fun processFileClicked(message: IncomingDocMessage.FileClicked) {
+ val fileToUpdate = message.filePath
+ val session = getSessionInfo(message.tabId)
+ val messageId = message.messageId
+
+ var filePaths: List = emptyList()
+ var deletedFiles: List = emptyList()
+ when (val state = session.sessionState) {
+ is PrepareDocGenerationState -> {
+ filePaths = state.filePaths
+ deletedFiles = state.deletedFiles
+ }
+ }
+
+ // Mark the file as rejected or not depending on the previous state
+ filePaths.find { it.zipFilePath == fileToUpdate }?.let { it.rejected = !it.rejected }
+ deletedFiles.find { it.zipFilePath == fileToUpdate }?.let { it.rejected = !it.rejected }
+
+ messenger.updateFileComponent(message.tabId, filePaths, deletedFiles, messageId)
+ }
+
+ private suspend fun newTabOpened(tabId: String) {
+ var session: DocSession? = null
+ try {
+ session = getSessionInfo(tabId)
+ logger.debug { "$FEATURE_NAME: Session created with id: ${session.tabID}" }
+
+ val credentialState = authController.getAuthNeededStates(context.project).amazonQ
+ if (credentialState != null) {
+ messenger.sendAuthNeededException(
+ tabId = tabId,
+ triggerId = UUID.randomUUID().toString(),
+ credentialState = credentialState,
+ )
+ session.isAuthenticating = true
+ return
+ }
+ docGenerationTask.userIdentity = session.getUserIdentity()
+ docGenerationTask.numberOfNavigation += 1
+ messenger.sendUpdatePlaceholder(tabId, message("amazonqDoc.prompt.placeholder"))
+ } catch (err: Exception) {
+ val message = createUserFacingErrorMessage(err.message)
+ messenger.sendError(
+ tabId = tabId,
+ errMessage = message ?: message("amazonqFeatureDev.exception.request_failed"),
+ retries = retriesRemaining(session),
+ conversationId = session?.conversationIdUnsafe
+ )
+ }
+ }
+
+ private suspend fun insertCode(tabId: String) {
+ var session: DocSession? = null
+ try {
+ session = getSessionInfo(tabId)
+
+ var filePaths: List = emptyList()
+ var deletedFiles: List = emptyList()
+
+ when (val state = session.sessionState) {
+ is PrepareDocGenerationState -> {
+ filePaths = state.filePaths
+ deletedFiles = state.deletedFiles
+ }
+ }
+
+ session.insertChanges(
+ filePaths = filePaths.filterNot { it.rejected },
+ deletedFiles = deletedFiles.filterNot { it.rejected }
+ )
+
+ messenger.sendSystemPrompt(
+ tabId = tabId,
+ followUp = listOf(
+ FollowUp(
+ pillText = message("amazonqDoc.prompt.reject.new_task"),
+ type = FollowUpTypes.NEW_TASK,
+ status = FollowUpStatusType.Info
+ ),
+ FollowUp(
+ pillText = message("amazonqDoc.prompt.reject.close_session"),
+ type = FollowUpTypes.CLOSE_SESSION,
+ status = FollowUpStatusType.Info
+ )
+ )
+ )
+
+ messenger.sendUpdatePlaceholder(
+ tabId = tabId,
+ newPlaceholder = message("amazonqFeatureDev.placeholder.additional_improvements")
+ )
+ } catch (err: Exception) {
+ val message = createUserFacingErrorMessage("Failed to insert code changes: ${err.message}")
+ messenger.sendError(
+ tabId = tabId,
+ errMessage = message ?: message("amazonqFeatureDev.exception.insert_code_failed"),
+ retries = retriesRemaining(session),
+ conversationId = session?.conversationIdUnsafe
+ )
+ }
+ }
+
+ private suspend fun newTask(tabId: String) {
+ docGenerationTask = DocGenerationTask()
+ chatSessionStorage.deleteSession(tabId)
+
+ messenger.sendAnswer(
+ tabId = tabId,
+ messageType = DocMessageType.Answer,
+ message = message("amazonqFeatureDev.chat_message.ask_for_new_task")
+ )
+
+ messenger.sendUpdatePromptProgress(tabId, null)
+
+ messenger.sendUpdatePlaceholder(
+ tabId = tabId,
+ newPlaceholder = message("amazonqFeatureDev.placeholder.after_code_generation")
+ )
+
+ newTabOpened(tabId)
+
+ messenger.sendSystemPrompt(
+ tabId = tabId,
+ followUp = listOf(
+ FollowUp(
+ pillText = message("amazonqDoc.prompt.create"),
+ prompt = message("amazonqDoc.prompt.create"),
+ type = FollowUpTypes.CREATE_DOCUMENTATION,
+ ),
+ FollowUp(
+ pillText = message("amazonqDoc.prompt.update"),
+ prompt = message("amazonqDoc.prompt.update"),
+ type = FollowUpTypes.UPDATE_DOCUMENTATION,
+ )
+ )
+ )
+ }
+
+ private suspend fun closeSession(tabId: String) {
+ messenger.sendAnswer(
+ tabId = tabId,
+ messageType = DocMessageType.Answer,
+ message = message("amazonqFeatureDev.chat_message.closed_session"),
+ canBeVoted = true
+ )
+
+ messenger.sendUpdatePlaceholder(
+ tabId = tabId,
+ newPlaceholder = message("amazonqFeatureDev.placeholder.closed_session")
+ )
+
+ messenger.sendChatInputEnabledMessage(tabId = tabId, enabled = false)
+ docGenerationTask.reset()
+ }
+
+ private suspend fun provideFeedbackAndRegenerateCode(tabId: String) {
+ // Unblock the message button
+ messenger.sendAsyncEventProgress(tabId = tabId, inProgress = false)
+
+ messenger.sendAnswer(
+ tabId = tabId,
+ message = message("amazonqFeatureDev.code_generation.provide_code_feedback"),
+ messageType = DocMessageType.Answer,
+ canBeVoted = true
+ )
+ messenger.sendUpdatePlaceholder(tabId, message("amazonqFeatureDev.placeholder.provide_code_feedback"))
+ }
+
+ private suspend fun processErrorChatMessage(err: Exception, session: DocSession?, tabId: String) {
+ logger.warn(err) { "Encountered ${err.message} for tabId: $tabId" }
+ messenger.sendUpdatePromptProgress(tabId, null)
+
+ when (err) {
+ is RepoSizeError -> {
+ messenger.sendError(
+ tabId = tabId,
+ errMessage = err.message,
+ retries = retriesRemaining(session),
+ conversationId = session?.conversationIdUnsafe
+ )
+ messenger.sendSystemPrompt(
+ tabId = tabId,
+ followUp = listOf(
+ FollowUp(
+ pillText = message("amazonqDoc.prompt.folder.change"),
+ type = FollowUpTypes.MODIFY_DEFAULT_SOURCE_FOLDER,
+ status = FollowUpStatusType.Info,
+ )
+ ),
+ )
+ }
+
+ is ZipFileError -> {
+ messenger.sendError(
+ tabId = tabId,
+ errMessage = err.message,
+ retries = 0,
+ conversationId = session?.conversationIdUnsafe
+ )
+ }
+
+ is MonthlyConversationLimitError -> {
+ messenger.sendUpdatePlaceholder(tabId, newPlaceholder = message("amazonqFeatureDev.placeholder.after_monthly_limit"))
+ messenger.sendChatInputEnabledMessage(tabId, enabled = true)
+ messenger.sendMonthlyLimitError(tabId = tabId)
+ }
+
+ is DocException -> {
+ messenger.sendError(
+ tabId = tabId,
+ errMessage = err.message,
+ retries = retriesRemaining(session),
+ conversationId = session?.conversationIdUnsafe
+ )
+ }
+
+ is CodeIterationLimitException -> {
+ messenger.sendUpdatePlaceholder(tabId, newPlaceholder = message("amazonqFeatureDev.placeholder.after_monthly_limit"))
+ messenger.sendChatInputEnabledMessage(tabId, enabled = true)
+ messenger.sendError(
+ tabId = tabId,
+ errMessage = err.message,
+ retries = retriesRemaining(session),
+ conversationId = session?.conversationIdUnsafe,
+ showDefaultMessage = true,
+ )
+
+ val filePaths: List = when (val state = session?.sessionState) {
+ is PrepareDocGenerationState -> state.filePaths ?: emptyList()
+ else -> emptyList()
+ }
+
+ val deletedFiles: List = when (val state = session?.sessionState) {
+ is PrepareDocGenerationState -> state.deletedFiles ?: emptyList()
+ else -> emptyList()
+ }
+
+ val followUp = if (filePaths.size == 0 && deletedFiles.size == 0) {
+ listOf(
+ FollowUp(
+ pillText = message("amazonqDoc.prompt.reject.new_task"),
+ type = FollowUpTypes.NEW_TASK,
+ status = FollowUpStatusType.Info
+ ),
+ FollowUp(
+ pillText = message("amazonqDoc.prompt.reject.close_session"),
+ type = FollowUpTypes.CLOSE_SESSION,
+ status = FollowUpStatusType.Info
+ )
+ )
+ } else {
+ listOf(
+ FollowUp(
+ pillText = message("amazonqDoc.prompt.review.accept"),
+ prompt = message("amazonqDoc.prompt.review.accept"),
+ status = FollowUpStatusType.Success,
+ type = FollowUpTypes.ACCEPT_CHANGES,
+ icon = FollowUpIcons.Ok,
+ ),
+ FollowUp(
+ pillText = message("general.reject"),
+ prompt = message("general.reject"),
+ status = FollowUpStatusType.Error,
+ type = FollowUpTypes.REJECT_CHANGES,
+ icon = FollowUpIcons.Cancel,
+ ),
+ )
+ }
+
+ messenger.sendSystemPrompt(
+ tabId = tabId,
+ followUp = followUp,
+ )
+ }
+
+ else -> {
+ var msg = createUserFacingErrorMessage("$FEATURE_NAME request failed: ${err.message ?: err.cause?.message}")
+ val isDenyListedError = denyListedErrors.any { msg?.contains(it) ?: false }
+ val defaultMessage: String = when (session?.sessionState?.phase) {
+ SessionStatePhase.CODEGEN -> {
+ if (isDenyListedError || retriesRemaining(session) > 0) {
+ message("amazonqFeatureDev.code_generation.error_message")
+ } else {
+ message("amazonqFeatureDev.code_generation.no_retries.error_message")
+ }
+ }
+
+ else -> message("amazonqFeatureDev.error_text")
+ }
+
+ messenger.sendError(
+ tabId = tabId,
+ errMessage = defaultMessage,
+ retries = retriesRemaining(session),
+ conversationId = session?.conversationIdUnsafe
+ )
+ }
+ }
+ }
+
+ private suspend fun handleChat(
+ tabId: String,
+ message: String,
+ ) {
+ var session: DocSession? = null
+ try {
+ logger.debug { "$FEATURE_NAME: Processing message: $message" }
+ session = getSessionInfo(tabId)
+
+ val credentialState = authController.getAuthNeededStates(context.project).amazonQ
+ if (credentialState != null) {
+ messenger.sendAuthNeededException(
+ tabId = tabId,
+ triggerId = UUID.randomUUID().toString(),
+ credentialState = credentialState,
+ )
+ session.isAuthenticating = true
+ return
+ }
+ docGenerationTask.userIdentity = session.getUserIdentity()
+ session.preloader(message, messenger)
+
+ when (session.sessionState.phase) {
+ SessionStatePhase.CODEGEN -> {
+ onCodeGeneration(session, message, tabId, mode)
+ }
+ else -> null
+ }
+
+ val filePaths: List = when (val state = session.sessionState) {
+ is PrepareDocGenerationState -> state.filePaths
+ else -> emptyList()
+ }
+
+ if (filePaths.isNotEmpty()) {
+ processOpenDiff(
+ message = IncomingDocMessage.OpenDiff(tabId = tabId, filePath = filePaths[0].zipFilePath, deleted = false)
+ )
+ }
+ } catch (err: Exception) {
+ processErrorChatMessage(err, session, tabId)
+
+ // Lock the chat input until they explicitly click one of the follow-ups
+ messenger.sendChatInputEnabledMessage(tabId, enabled = false)
+ }
+ }
+
+ private suspend fun onDocsGeneration(followUpMessage: IncomingDocMessage.FollowupClicked) {
+ messenger.sendUpdatePromptProgress(tabId = followUpMessage.tabId, inProgress(progress = 10, message("amazonqDoc.progress_message.scanning")))
+
+ val session = getSessionInfo(followUpMessage.tabId)
+
+ messenger.sendAnswer(
+ message = docGenerationProgressMessage(DocGenerationStep.UPLOAD_TO_S3, this.mode),
+ messageType = DocMessageType.AnswerPart,
+ tabId = followUpMessage.tabId,
+ )
+
+ try {
+ val sessionMessage: String = when (mode) {
+ Mode.CREATE -> message("amazonqDoc.session.create")
+ else -> message("amazonqDoc.session.sync")
+ }
+
+ session.send(sessionMessage)
+
+ val filePaths: List = when (val state = session.sessionState) {
+ is PrepareDocGenerationState -> state.filePaths ?: emptyList()
+ else -> emptyList()
+ }
+
+ val deletedFiles: List = when (val state = session.sessionState) {
+ is PrepareDocGenerationState -> state.deletedFiles ?: emptyList()
+ else -> emptyList()
+ }
+
+ val references: List = when (val state = session.sessionState) {
+ is PrepareDocGenerationState -> state.references ?: emptyList()
+ else -> emptyList()
+ }
+
+ if (session.sessionState.token
+ ?.token
+ ?.isCancellationRequested() == true
+ ) {
+ return
+ }
+
+ if (filePaths.isEmpty() && deletedFiles.isEmpty()) {
+ handleEmptyFiles(followUpMessage, session)
+ return
+ }
+
+ messenger.sendAnswer(
+ message = docGenerationProgressMessage(DocGenerationStep.COMPLETE, mode),
+ messageType = DocMessageType.AnswerPart,
+ tabId = followUpMessage.tabId,
+ )
+
+ messenger.sendCodeResult(
+ tabId = followUpMessage.tabId,
+ filePaths = filePaths,
+ deletedFiles = deletedFiles,
+ uploadId = session.conversationId,
+ references = references
+ )
+
+ messenger.sendAnswer(
+ messageType = DocMessageType.Answer,
+ tabId = followUpMessage.tabId,
+ message = message("amazonqDoc.prompt.review.message")
+ )
+
+ messenger.sendAnswer(
+ messageType = DocMessageType.SystemPrompt,
+ tabId = followUpMessage.tabId,
+ followUp = getFollowUpOptions(session.sessionState.phase)
+ )
+
+ processOpenDiff(
+ message = IncomingDocMessage.OpenDiff(tabId = followUpMessage.tabId, filePath = filePaths[0].zipFilePath, deleted = false)
+ )
+ } catch (err: Exception) {
+ processErrorChatMessage(err, session, tabId = followUpMessage.tabId)
+
+ // Lock the chat input until they explicitly click one of the follow-ups
+ messenger.sendChatInputEnabledMessage(tabId = followUpMessage.tabId, enabled = false)
+ } finally {
+ messenger.sendUpdatePlaceholder(
+ tabId = followUpMessage.tabId,
+ newPlaceholder = message("amazonqDoc.prompt.placeholder")
+ )
+
+ messenger.sendChatInputEnabledMessage(followUpMessage.tabId, false)
+
+ if (session.sessionState.token
+ ?.token
+ ?.isCancellationRequested() == true
+ ) {
+ session.sessionState.token = CancellationTokenSource()
+ } else {
+ messenger.sendAsyncEventProgress(tabId = followUpMessage.tabId, inProgress = false) // Finish processing the event
+ messenger.sendChatInputEnabledMessage(tabId = followUpMessage.tabId, enabled = false) // Lock chat input until a follow-up is clicked.
+ }
+ }
+ }
+
+ private suspend fun handleEmptyFiles(
+ message: IncomingDocMessage.FollowupClicked,
+ session: DocSession,
+ ) {
+ messenger.sendAnswer(
+ message = message("amazonqDoc.error.generating"),
+ messageType = DocMessageType.Answer,
+ tabId = message.tabId,
+ canBeVoted = true
+ )
+
+ messenger.sendAnswer(
+ messageType = DocMessageType.SystemPrompt,
+ tabId = message.tabId,
+ followUp = if (retriesRemaining(session) > 0) {
+ listOf(
+ FollowUp(
+ pillText = message("amazonqFeatureDev.follow_up.retry"),
+ type = FollowUpTypes.RETRY,
+ status = FollowUpStatusType.Warning
+ )
+ )
+ } else {
+ emptyList()
+ }
+ )
+
+ // Lock the chat input until they explicitly click retry
+ messenger.sendChatInputEnabledMessage(tabId = message.tabId, enabled = false)
+ }
+
+ private suspend fun retryRequests(tabId: String) {
+ var session: DocSession? = null
+ docGenerationTask = DocGenerationTask()
+ try {
+ messenger.sendAsyncEventProgress(
+ tabId = tabId,
+ inProgress = true,
+ )
+ session = getSessionInfo(tabId)
+
+ // Decrease retries before making this request, just in case this one fails as well
+ session.decreaseRetries()
+
+ // Sending an empty message will re-run the last state with the previous values
+ handleChat(
+ tabId = tabId,
+ message = session.latestMessage
+ )
+ } catch (err: Exception) {
+ logger.error(err) { "Failed to retry request: ${err.message}" }
+ val message = createUserFacingErrorMessage("Failed to retry request: ${err.message}")
+ messenger.sendError(
+ tabId = tabId,
+ errMessage = message ?: message("amazonqFeatureDev.exception.retry_request_failed"),
+ retries = retriesRemaining(session),
+ conversationId = session?.conversationIdUnsafe,
+ )
+ } finally {
+ // Finish processing the event
+ messenger.sendAsyncEventProgress(
+ tabId = tabId,
+ inProgress = false,
+ )
+ }
+ }
+
+ private fun isFolderPathInProjectModules(project: Project, folderPath: String): Boolean {
+ val path = Paths.get(folderPath)
+ val virtualFile = LocalFileSystem.getInstance().findFileByIoFile(path.toFile()) ?: return false
+
+ val projectFileIndex = ProjectRootManager.getInstance(project).fileIndex
+
+ return projectFileIndex.isInProject(virtualFile)
+ }
+
+ private suspend fun modifyDefaultSourceFolder(tabId: String) {
+ val session = getSessionInfo(tabId)
+ val currentSourceFolder = session.context.selectedSourceFolder
+ val projectRoot = session.context.projectRoot
+
+ withContext(EDT) {
+ val selectedFolder = selectFolder(context.project, currentSourceFolder)
+
+ // No folder was selected
+ if (selectedFolder == null) {
+ logger.info { "Cancelled dialog and not selected any folder" }
+ return@withContext
+ }
+
+ val isFolderPathInProject = isFolderPathInProjectModules(context.project, selectedFolder.path)
+
+ if (!isFolderPathInProject) {
+ logger.info { "Selected folder not in workspace: ${selectedFolder.path}" }
+
+ messenger.sendAnswer(
+ tabId = tabId,
+ messageType = DocMessageType.Answer,
+ message = message("amazonqFeatureDev.follow_up.incorrect_source_folder"),
+ followUp = listOf(
+ FollowUp(
+ pillText = message("amazonqDoc.prompt.folder.change"),
+ type = FollowUpTypes.MODIFY_DEFAULT_SOURCE_FOLDER,
+ status = FollowUpStatusType.Info,
+ )
+ ),
+ snapToTop = true
+ )
+
+ messenger.sendChatInputEnabledMessage(tabId, enabled = false)
+
+ return@withContext
+ }
+
+ if (selectedFolder.path == projectRoot.path) {
+ docGenerationTask.folderLevel = DocGenerationFolderLevel.ENTIRE_WORKSPACE
+ } else {
+ docGenerationTask.folderLevel = DocGenerationFolderLevel.SUB_FOLDER
+ }
+
+ logger.info { "Selected correct folder inside workspace: ${selectedFolder.path}" }
+
+ session.context.selectedSourceFolder = selectedFolder
+
+ promptForDocTarget(tabId)
+
+ messenger.sendChatInputEnabledMessage(tabId, enabled = false)
+
+ messenger.sendUpdatePlaceholder(tabId = tabId, newPlaceholder = message("amazonqDoc.prompt.placeholder"))
+ }
+ }
+
+ private fun sendDocGenerationTelemetry(tabId: String) {
+ val session = getSessionInfo(tabId)
+ var filePaths: List = emptyList()
+
+ when (val state = session.sessionState) {
+ is PrepareDocGenerationState -> {
+ filePaths = state.filePaths
+ }
+ }
+ docGenerationTask.conversationId = session.conversationId
+ val (totalAddedChars, totalAddedLines, totalAddedFiles) = session.countAddedContent(filePaths, docGenerationTask.interactionType)
+ docGenerationTask.numberOfAddChars = totalAddedChars
+ docGenerationTask.numberOfAddLines = totalAddedLines
+ docGenerationTask.numberOfAddFiles = totalAddedFiles
+
+ val docGenerationEvent = docGenerationTask.docGenerationEventBase()
+ session.sendDocGenerationEvent(docGenerationEvent)
+ }
+
+ fun getProject() = context.project
+
+ private fun getSessionInfo(tabId: String) = chatSessionStorage.getSession(tabId, context.project)
+
+ fun retriesRemaining(session: DocSession?): Int = session?.retries ?: DEFAULT_RETRY_LIMIT
+
+ companion object {
+ private val logger = getLogger()
+ }
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/controller/DocControllerExtensions.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/controller/DocControllerExtensions.kt
new file mode 100644
index 0000000000..f16503e2c0
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/controller/DocControllerExtensions.kt
@@ -0,0 +1,141 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqDoc.controller
+
+import com.intellij.notification.NotificationAction
+import software.aws.toolkits.jetbrains.services.amazonqDoc.inProgress
+import software.aws.toolkits.jetbrains.services.amazonqDoc.messages.DocMessageType
+import software.aws.toolkits.jetbrains.services.amazonqDoc.messages.FollowUp
+import software.aws.toolkits.jetbrains.services.amazonqDoc.messages.FollowUpStatusType
+import software.aws.toolkits.jetbrains.services.amazonqDoc.messages.FollowUpTypes
+import software.aws.toolkits.jetbrains.services.amazonqDoc.messages.sendAnswer
+import software.aws.toolkits.jetbrains.services.amazonqDoc.messages.sendAsyncEventProgress
+import software.aws.toolkits.jetbrains.services.amazonqDoc.messages.sendChatInputEnabledMessage
+import software.aws.toolkits.jetbrains.services.amazonqDoc.messages.sendCodeResult
+import software.aws.toolkits.jetbrains.services.amazonqDoc.messages.sendSystemPrompt
+import software.aws.toolkits.jetbrains.services.amazonqDoc.messages.sendUpdatePlaceholder
+import software.aws.toolkits.jetbrains.services.amazonqDoc.messages.sendUpdatePromptProgress
+import software.aws.toolkits.jetbrains.services.amazonqDoc.session.DocSession
+import software.aws.toolkits.jetbrains.services.amazonqDoc.session.PrepareDocGenerationState
+import software.aws.toolkits.jetbrains.services.amazonqDoc.util.getFollowUpOptions
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.messages.sendSystemPrompt
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.CodeReferenceGenerated
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.DeletedFileInfo
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.NewFileZipInfo
+import software.aws.toolkits.jetbrains.utils.notifyInfo
+import software.aws.toolkits.resources.message
+
+suspend fun DocController.onCodeGeneration(session: DocSession, message: String, tabId: String, mode: Mode) {
+ try {
+ messenger.sendAsyncEventProgress(tabId, inProgress = true)
+ messenger.sendUpdatePromptProgress(tabId, inProgress(progress = 10, message("amazonqDoc.progress_message.scanning")))
+ messenger.sendAnswer(
+ message = docGenerationProgressMessage(DocGenerationStep.UPLOAD_TO_S3, this.mode),
+ messageType = DocMessageType.AnswerPart,
+ tabId = tabId,
+ )
+
+ val sessionMessage = if (mode == Mode.CREATE) {
+ message(
+ "amazonqDoc.session.create"
+ )
+ } else if (mode == Mode.EDIT) message else message("amazonqDoc.session.sync")
+
+ session.send(sessionMessage) // Trigger code generation
+
+ if (session.sessionState.token
+ ?.token
+ ?.isCancellationRequested() == true
+ ) {
+ return
+ }
+
+ val state = session.sessionState
+
+ var filePaths: List = emptyList()
+ var deletedFiles: List = emptyList()
+ var references: List = emptyList()
+ var uploadId = ""
+ var remainingIterations: Int? = null
+ var totalIterations: Int? = null
+
+ when (state) {
+ is PrepareDocGenerationState -> {
+ filePaths = state.filePaths
+ deletedFiles = state.deletedFiles
+ references = state.references
+ uploadId = state.uploadId
+ remainingIterations = state.codeGenerationRemainingIterationCount
+ totalIterations = state.codeGenerationTotalIterationCount
+ }
+ }
+
+ // Atm this is the only possible path as codegen is mocked to return empty.
+ if (filePaths.size or deletedFiles.size == 0) {
+ messenger.sendAnswer(
+ tabId = tabId,
+ messageType = DocMessageType.Answer,
+ message = message("amazonqFeatureDev.code_generation.no_file_changes")
+ )
+ messenger.sendSystemPrompt(
+ tabId = tabId,
+ followUp = if (retriesRemaining(session) > 0) {
+ listOf(
+ FollowUp(
+ pillText = message("amazonqFeatureDev.follow_up.retry"),
+ type = FollowUpTypes.RETRY,
+ status = FollowUpStatusType.Warning
+ )
+ )
+ } else {
+ emptyList()
+ }
+ )
+ messenger.sendChatInputEnabledMessage(tabId = tabId, enabled = false) // Lock chat input until retry is clicked.
+ return
+ }
+
+ messenger.sendAnswer(
+ message = docGenerationProgressMessage(DocGenerationStep.COMPLETE, mode),
+ messageType = DocMessageType.AnswerPart,
+ tabId = tabId,
+ )
+
+ messenger.sendCodeResult(tabId = tabId, uploadId = uploadId, filePaths = filePaths, deletedFiles = deletedFiles, references = references)
+
+ if (remainingIterations != null && totalIterations != null) {
+ messenger.sendAnswer(
+ tabId = tabId,
+ messageType = DocMessageType.Answer,
+ message = if (this.mode === Mode.CREATE) {
+ message("amazonqDoc.answer.readmeCreated")
+ } else {
+ "${message("amazonqDoc.answer.readmeUpdated")} ${message("amazonqDoc.answer.codeResult")}"
+ }
+ )
+ }
+
+ messenger.sendSystemPrompt(tabId = tabId, followUp = getFollowUpOptions(session.sessionState.phase))
+
+ messenger.sendUpdatePlaceholder(tabId = tabId, newPlaceholder = message("amazonqFeatureDev.placeholder.after_code_generation"))
+ } finally {
+ messenger.sendAsyncEventProgress(tabId = tabId, inProgress = false) // Finish processing the event
+ messenger.sendChatInputEnabledMessage(tabId = tabId, enabled = false) // Lock chat input until a follow-up is clicked.
+
+ if (toolWindow != null && !toolWindow.isVisible) {
+ notifyInfo(
+ title = message("amazonqFeatureDev.code_generation.notification_title"),
+ content = message("amazonqFeatureDev.code_generation.notification_message"),
+ project = getProject(),
+ notificationActions = listOf(openChatNotificationAction())
+ )
+ }
+ }
+}
+
+private fun DocController.openChatNotificationAction() = NotificationAction.createSimple(
+ message("amazonqFeatureDev.code_generation.notification_open_link")
+) {
+ toolWindow?.show()
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/controller/DocGenerationTask.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/controller/DocGenerationTask.kt
new file mode 100644
index 0000000000..5080ab5b4f
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/controller/DocGenerationTask.kt
@@ -0,0 +1,62 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqDoc.controller
+
+import software.amazon.awssdk.services.codewhispererruntime.model.DocGenerationEvent
+import software.amazon.awssdk.services.codewhispererruntime.model.DocGenerationFolderLevel
+import software.amazon.awssdk.services.codewhispererruntime.model.DocGenerationInteractionType
+import software.amazon.awssdk.services.codewhispererruntime.model.DocGenerationUserDecision
+import software.aws.toolkits.core.utils.debug
+import software.aws.toolkits.core.utils.getLogger
+
+class DocGenerationTask {
+ // Telemetry fields
+ var conversationId: String? = null
+ var numberOfAddChars: Int? = null
+ var numberOfAddLines: Int? = null
+ var numberOfAddFiles: Int? = null
+ var userDecision: DocGenerationUserDecision? = null
+ var interactionType: DocGenerationInteractionType? = null
+ var userIdentity: String? = null
+ var numberOfNavigation = 0
+ var folderLevel: DocGenerationFolderLevel? = DocGenerationFolderLevel.ENTIRE_WORKSPACE
+ fun docGenerationEventBase(): DocGenerationEvent {
+ val undefinedProps = this::class.java.declaredFields
+ .filter { it.get(this) == null }
+ .map { it.name }
+
+ if (undefinedProps.isNotEmpty()) {
+ val undefinedValue = undefinedProps.joinToString(", ")
+ logger.debug { "DocGenerationEvent has undefined properties: $undefinedValue" }
+ }
+
+ return DocGenerationEvent.builder()
+ .conversationId(conversationId)
+ .numberOfAddChars(numberOfAddChars)
+ .numberOfAddLines(numberOfAddLines)
+ .numberOfAddFiles(numberOfAddFiles)
+ .userDecision(userDecision)
+ .interactionType(interactionType)
+ .userIdentity(userIdentity)
+ .numberOfNavigation(numberOfNavigation)
+ .folderLevel(folderLevel)
+ .build()
+ }
+
+ fun reset() {
+ conversationId = null
+ numberOfAddChars = null
+ numberOfAddLines = null
+ numberOfAddFiles = null
+ userDecision = null
+ interactionType = null
+ userIdentity = null
+ numberOfNavigation = 0
+ folderLevel = null
+ }
+
+ companion object {
+ private val logger = getLogger()
+ }
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/messages/DocMessage.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/messages/DocMessage.kt
new file mode 100644
index 0000000000..61c8bc0109
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/messages/DocMessage.kt
@@ -0,0 +1,287 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqDoc.messages
+
+import com.fasterxml.jackson.annotation.JsonProperty
+import com.fasterxml.jackson.annotation.JsonValue
+import software.aws.toolkits.jetbrains.services.amazonq.auth.AuthFollowUpType
+import software.aws.toolkits.jetbrains.services.amazonq.messages.AmazonQMessage
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.DeletedFileInfo
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.NewFileZipInfo
+import software.aws.toolkits.jetbrains.services.cwc.messages.CodeReference
+import java.time.Instant
+import java.util.UUID
+
+sealed interface DocBaseMessage : AmazonQMessage
+
+// === UI -> App Messages ===
+sealed interface IncomingDocMessage : DocBaseMessage {
+
+ data class ChatPrompt(
+ val chatMessage: String,
+ val command: String,
+ @JsonProperty("tabID") val tabId: String,
+ ) : IncomingDocMessage
+
+ data class NewTabCreated(
+ val command: String,
+ @JsonProperty("tabID") val tabId: String,
+ ) : IncomingDocMessage
+
+ data class AuthFollowUpWasClicked(
+ @JsonProperty("tabID") val tabId: String,
+ val authType: AuthFollowUpType,
+ ) : IncomingDocMessage
+
+ data class TabRemoved(
+ val command: String,
+ @JsonProperty("tabID") val tabId: String,
+ ) : IncomingDocMessage
+
+ data class FollowupClicked(
+ val followUp: FollowUp,
+ @JsonProperty("tabID") val tabId: String,
+ val messageId: String?,
+ val command: String,
+ val tabType: String,
+ ) : IncomingDocMessage
+
+ data class ChatItemVotedMessage(
+ @JsonProperty("tabID") val tabId: String,
+ val messageId: String,
+ val vote: String,
+ ) : IncomingDocMessage
+
+ data class ChatItemFeedbackMessage(
+ @JsonProperty("tabID") val tabId: String,
+ val selectedOption: String,
+ val comment: String?,
+ val messageId: String,
+ ) : IncomingDocMessage
+
+ data class ClickedLink(
+ @JsonProperty("tabID") val tabId: String,
+ val command: String,
+ val messageId: String?,
+ val link: String,
+ ) : IncomingDocMessage
+
+ data class InsertCodeAtCursorPosition(
+ @JsonProperty("tabID") val tabId: String,
+ val code: String,
+ val insertionTargetType: String?,
+ val codeReference: List?,
+ ) : IncomingDocMessage
+
+ data class OpenDiff(
+ @JsonProperty("tabID") val tabId: String,
+ val filePath: String,
+ val deleted: Boolean,
+ ) : IncomingDocMessage
+
+ data class FileClicked(
+ @JsonProperty("tabID") val tabId: String,
+ val filePath: String,
+ val messageId: String,
+ val actionName: String,
+ ) : IncomingDocMessage
+
+ data class StopDocGeneration(
+ @JsonProperty("tabID") val tabId: String,
+ ) : IncomingDocMessage
+}
+
+// === UI -> App Messages ===
+
+sealed class UiMessage(
+ open val tabId: String?,
+ open val type: String,
+) : DocBaseMessage {
+ val time = Instant.now().epochSecond
+ val sender = "docChat"
+}
+
+enum class DocMessageType(
+ @field:JsonValue val json: String,
+) {
+ Answer("answer"),
+ AnswerPart("answer-part"),
+ AnswerStream("answer-stream"),
+ SystemPrompt("system-prompt"),
+}
+
+data class DocMessage(
+ @JsonProperty("tabID") override val tabId: String,
+ @JsonProperty("triggerID") val triggerId: String,
+ val messageType: DocMessageType,
+ val messageId: String,
+ val message: String? = null,
+ val followUps: List? = null,
+ val canBeVoted: Boolean,
+ val snapToTop: Boolean,
+
+) : UiMessage(
+ tabId = tabId,
+ type = "chatMessage",
+)
+
+data class AsyncEventProgressMessage(
+ @JsonProperty("tabID") override val tabId: String,
+ val message: String? = null,
+ val inProgress: Boolean,
+) : UiMessage(
+ tabId = tabId,
+ type = "asyncEventProgressMessage"
+)
+
+data class UpdatePlaceholderMessage(
+ @JsonProperty("tabID") override val tabId: String,
+ val newPlaceholder: String,
+) : UiMessage(
+ tabId = tabId,
+ type = "updatePlaceholderMessage"
+)
+
+data class FileComponent(
+ @JsonProperty("tabID") override val tabId: String,
+ val filePaths: List,
+ val deletedFiles: List,
+ val messageId: String,
+) : UiMessage(
+ tabId = tabId,
+ type = "updateFileComponent"
+)
+
+data class ChatInputEnabledMessage(
+ @JsonProperty("tabID") override val tabId: String,
+ val enabled: Boolean,
+) : UiMessage(
+ tabId = tabId,
+ type = "chatInputEnabledMessage"
+)
+data class ErrorMessage(
+ @JsonProperty("tabID") override val tabId: String,
+ val title: String,
+ val message: String,
+) : UiMessage(
+ tabId = tabId,
+ type = "errorMessage",
+)
+
+data class FolderConfirmationMessage(
+ @JsonProperty("tabID") override val tabId: String,
+ val folderPath: String,
+ val message: String,
+ val followUps: List?,
+) : UiMessage(
+ tabId = tabId,
+ type = "folderConfirmationMessage"
+)
+
+// this should come from mynah?
+data class ChatItemButton(
+ val id: String,
+ val text: String,
+ val icon: String,
+ val keepCardAfterClick: Boolean,
+ val disabled: Boolean,
+ val waitMandatoryFormItems: Boolean,
+)
+
+data class ProgressField(
+ val status: String,
+ val text: String,
+ val value: Int,
+ var actions: List,
+)
+
+data class AuthenticationUpdateMessage(
+ val authenticatingTabIDs: List,
+ val featureDevEnabled: Boolean,
+ val codeTransformEnabled: Boolean,
+ val codeScanEnabled: Boolean,
+ val codeTestEnabled: Boolean,
+ val docEnabled: Boolean,
+ val message: String? = null,
+ val messageId: String = UUID.randomUUID().toString(),
+) : UiMessage(
+ null,
+ type = "authenticationUpdateMessage",
+)
+
+data class AuthNeededException(
+ @JsonProperty("tabID") override val tabId: String,
+ @JsonProperty("triggerID") val triggerId: String,
+ val authType: AuthFollowUpType,
+ val message: String,
+) : UiMessage(
+ tabId = tabId,
+ type = "authNeededException",
+)
+
+data class CodeResultMessage(
+ @JsonProperty("tabID") override val tabId: String,
+ val conversationId: String,
+ val filePaths: List,
+ val deletedFiles: List,
+ val references: List,
+) : UiMessage(
+ tabId = tabId,
+ type = "codeResultMessage"
+)
+
+data class FollowUp(
+ val type: FollowUpTypes,
+ val pillText: String,
+ val prompt: String? = null,
+ val disabled: Boolean? = false,
+ val description: String? = null,
+ val status: FollowUpStatusType? = null,
+ val icon: FollowUpIcons? = null,
+)
+
+enum class FollowUpIcons(
+ @field:JsonValue val json: String,
+) {
+ Ok("ok"),
+ Refresh("refresh"),
+ Cancel("cancel"),
+ Info("info"),
+ Error("error"),
+}
+
+enum class FollowUpStatusType(
+ @field:JsonValue val json: String,
+) {
+ Info("info"),
+ Success("success"),
+ Warning("warning"),
+ Error("error"),
+}
+
+enum class FollowUpTypes(
+ @field:JsonValue val json: String,
+) {
+ RETRY("Retry"),
+ MODIFY_DEFAULT_SOURCE_FOLDER("ModifyDefaultSourceFolder"),
+ DEV_EXAMPLES("DevExamples"),
+ INSERT_CODE("InsertCode"),
+ PROVIDE_FEEDBACK_AND_REGENERATE_CODE("ProvideFeedbackAndRegenerateCode"),
+ NEW_TASK("NewTask"),
+ CLOSE_SESSION("CloseSession"),
+ CREATE_DOCUMENTATION("CreateDocumentation"),
+ UPDATE_DOCUMENTATION("UpdateDocumentation"),
+ CANCEL_FOLDER_SELECTION("CancelFolderSelection"),
+ PROCEED_FOLDER_SELECTION("ProceedFolderSelection"),
+ ACCEPT_CHANGES("AcceptChanges"),
+ MAKE_CHANGES("MakeChanges"),
+ REJECT_CHANGES("RejectChanges"),
+ SYNCHRONIZE_DOCUMENTATION("SynchronizeDocumentation"),
+ EDIT_DOCUMENTATION("EditDocumentation"),
+}
+
+// Util classes
+data class ReducedCodeReference(
+ val information: String,
+)
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/messages/DocMessagePublisherExtensions.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/messages/DocMessagePublisherExtensions.kt
new file mode 100644
index 0000000000..963b6d95be
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/messages/DocMessagePublisherExtensions.kt
@@ -0,0 +1,215 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqDoc.messages
+
+import software.aws.toolkits.jetbrains.services.amazonq.auth.AuthNeededState
+import software.aws.toolkits.jetbrains.services.amazonq.messages.MessagePublisher
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.messages.ProgressField
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.messages.PromptProgressMessage
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.CodeReferenceGenerated
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.DeletedFileInfo
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.NewFileZipInfo
+import software.aws.toolkits.jetbrains.services.cwc.messages.CodeReference
+import software.aws.toolkits.jetbrains.services.cwc.messages.RecommendationContentSpan
+import software.aws.toolkits.resources.message
+import java.util.UUID
+
+suspend fun MessagePublisher.sendAnswer(
+ tabId: String,
+ message: String? = null,
+ messageType: DocMessageType,
+ followUp: List? = null,
+ canBeVoted: Boolean? = false,
+ snapToTop: Boolean? = false,
+) {
+ val chatMessage =
+ DocMessage(
+ tabId = tabId,
+ triggerId = UUID.randomUUID().toString(),
+ messageId = UUID.randomUUID().toString(),
+ messageType = messageType,
+ message = message,
+ followUps = followUp,
+ canBeVoted = canBeVoted ?: false,
+ snapToTop = snapToTop ?: false
+ )
+ this.publish(chatMessage)
+}
+
+suspend fun MessagePublisher.sendAnswerPart(
+ tabId: String,
+ message: String? = null,
+ canBeVoted: Boolean? = null,
+) {
+ this.sendAnswer(
+ tabId = tabId,
+ message = message,
+ messageType = DocMessageType.AnswerPart,
+ canBeVoted = canBeVoted
+ )
+}
+
+suspend fun MessagePublisher.sendSystemPrompt(
+ tabId: String,
+ followUp: List,
+) {
+ this.sendAnswer(
+ tabId = tabId,
+ messageType = DocMessageType.SystemPrompt,
+ followUp = followUp
+ )
+}
+
+suspend fun MessagePublisher.updateFileComponent(tabId: String, filePaths: List, deletedFiles: List, messageId: String) {
+ val fileComponentMessage = FileComponent(
+ tabId = tabId,
+ filePaths = filePaths,
+ deletedFiles = deletedFiles,
+ messageId = messageId,
+ )
+ this.publish(fileComponentMessage)
+}
+
+suspend fun MessagePublisher.sendAsyncEventProgress(tabId: String, inProgress: Boolean, message: String? = null) {
+ val asyncEventProgressMessage = AsyncEventProgressMessage(
+ tabId = tabId,
+ message = message,
+ inProgress = inProgress,
+ )
+ this.publish(asyncEventProgressMessage)
+}
+
+suspend fun MessagePublisher.sendUpdatePlaceholder(tabId: String, newPlaceholder: String) {
+ val updatePlaceholderMessage = UpdatePlaceholderMessage(
+ tabId = tabId,
+ newPlaceholder = newPlaceholder
+ )
+ this.publish(updatePlaceholderMessage)
+}
+
+suspend fun MessagePublisher.sendAuthNeededException(tabId: String, triggerId: String, credentialState: AuthNeededState) {
+ val message = AuthNeededException(
+ tabId = tabId,
+ triggerId = triggerId,
+ authType = credentialState.authType,
+ message = credentialState.message,
+ )
+ this.publish(message)
+}
+
+suspend fun MessagePublisher.sendAuthenticationInProgressMessage(tabId: String) {
+ this.sendAnswer(
+ tabId = tabId,
+ messageType = DocMessageType.Answer,
+ message = message("amazonqFeatureDev.follow_instructions_for_authentication")
+ )
+}
+suspend fun MessagePublisher.sendChatInputEnabledMessage(tabId: String, enabled: Boolean) {
+ val chatInputEnabledMessage = ChatInputEnabledMessage(
+ tabId,
+ enabled,
+ )
+ this.publish(chatInputEnabledMessage)
+}
+
+suspend fun MessagePublisher.sendError(tabId: String, errMessage: String?, retries: Int, conversationId: String? = null, showDefaultMessage: Boolean? = false) {
+ val conversationIdText = if (conversationId == null) "" else "\n\nConversation ID: **$conversationId**"
+
+ if (retries == 0) {
+ this.sendAnswer(
+ tabId = tabId,
+ messageType = DocMessageType.Answer,
+ message = if (showDefaultMessage == true) errMessage else message("amazonqFeatureDev.no_retries.error_text") + conversationIdText,
+ )
+
+ this.sendAnswer(
+ tabId = tabId,
+ messageType = DocMessageType.SystemPrompt,
+ )
+ return
+ }
+
+ this.sendAnswer(
+ tabId = tabId,
+ messageType = DocMessageType.Answer,
+ message = errMessage + conversationIdText,
+ )
+
+ this.sendAnswer(
+ tabId = tabId,
+ messageType = DocMessageType.SystemPrompt,
+ followUp = listOf(
+ FollowUp(
+ pillText = message("amazonqFeatureDev.follow_up.retry"),
+ type = FollowUpTypes.RETRY,
+ status = FollowUpStatusType.Warning
+ )
+ ),
+ )
+}
+
+suspend fun MessagePublisher.sendMonthlyLimitError(tabId: String) {
+ this.sendAnswer(
+ tabId = tabId,
+ messageType = DocMessageType.Answer,
+ message = message("amazonqFeatureDev.exception.monthly_limit_error")
+ )
+ this.sendUpdatePlaceholder(tabId = tabId, newPlaceholder = message("amazonqFeatureDev.placeholder.after_monthly_limit"))
+}
+
+suspend fun MessagePublisher.initialExamples(tabId: String) {
+ this.sendAnswer(
+ tabId = tabId,
+ messageType = DocMessageType.Answer,
+ message = message("amazonqFeatureDev.example_text"),
+ )
+}
+
+suspend fun MessagePublisher.sendCodeResult(
+ tabId: String,
+ uploadId: String,
+ filePaths: List,
+ deletedFiles: List,
+ references: List,
+) {
+ val refs = references.map { ref ->
+ CodeReference(
+ licenseName = ref.licenseName,
+ repository = ref.repository,
+ url = ref.url,
+ recommendationContentSpan = RecommendationContentSpan(
+ ref.recommendationContentSpan?.start ?: 0,
+ ref.recommendationContentSpan?.end ?: 0,
+ ),
+ information = "Reference code under **${ref.licenseName}** license from repository [${ref.repository}](${ref.url})"
+ )
+ }
+
+ this.publish(
+ CodeResultMessage(
+ tabId = tabId,
+ conversationId = uploadId,
+ filePaths = filePaths,
+ deletedFiles = deletedFiles,
+ references = refs
+ )
+ )
+}
+
+suspend fun MessagePublisher.sendFolderConfirmationMessage(
+ tabId: String,
+ message: String,
+ folderPath: String,
+ followUps: List,
+) {
+ this.publish(
+ FolderConfirmationMessage(tabId = tabId, folderPath = folderPath, message = message, followUps = followUps)
+ )
+}
+
+suspend fun MessagePublisher.sendUpdatePromptProgress(tabId: String, progressField: ProgressField?) {
+ this.publish(
+ PromptProgressMessage(tabId, progressField)
+ )
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/session/DocGenerationState.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/session/DocGenerationState.kt
new file mode 100644
index 0000000000..e6389341cb
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/session/DocGenerationState.kt
@@ -0,0 +1,241 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqDoc.session
+
+import kotlinx.coroutines.delay
+import software.amazon.awssdk.services.codewhispererruntime.model.CodeGenerationWorkflowStatus
+import software.aws.toolkits.core.utils.getLogger
+import software.aws.toolkits.core.utils.warn
+import software.aws.toolkits.jetbrains.common.session.Intent
+import software.aws.toolkits.jetbrains.common.session.SessionState
+import software.aws.toolkits.jetbrains.common.session.SessionStateConfig
+import software.aws.toolkits.jetbrains.services.amazonq.messages.MessagePublisher
+import software.aws.toolkits.jetbrains.services.amazonqDoc.FEATURE_NAME
+import software.aws.toolkits.jetbrains.services.amazonqDoc.controller.DocGenerationStep
+import software.aws.toolkits.jetbrains.services.amazonqDoc.controller.Mode
+import software.aws.toolkits.jetbrains.services.amazonqDoc.controller.docGenerationProgressMessage
+import software.aws.toolkits.jetbrains.services.amazonqDoc.docServiceError
+import software.aws.toolkits.jetbrains.services.amazonqDoc.inProgress
+import software.aws.toolkits.jetbrains.services.amazonqDoc.messages.sendAnswerPart
+import software.aws.toolkits.jetbrains.services.amazonqDoc.messages.sendUpdatePromptProgress
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.CodeGenerationResult
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.Interaction
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.SessionStateAction
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.SessionStatePhase
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.registerDeletedFiles
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.registerNewFiles
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.util.CancellationTokenSource
+import software.aws.toolkits.resources.message
+
+private val logger = getLogger()
+
+class DocGenerationState(
+ override val tabID: String,
+ override var approach: String,
+ val config: SessionStateConfig,
+ val uploadId: String,
+ val currentIteration: Int,
+ val messenger: MessagePublisher,
+ var codeGenerationRemainingIterationCount: Int? = null,
+ var codeGenerationTotalIterationCount: Int? = null,
+ override val phase: SessionStatePhase?,
+ override var token: CancellationTokenSource?,
+) : SessionState {
+ override suspend fun interact(action: SessionStateAction): SessionStateInteraction {
+ try {
+ val response = config.amazonQCodeGenService.startTaskAssistCodeGeneration(
+ conversationId = config.conversationId,
+ uploadId = uploadId,
+ message = action.msg,
+ intent = Intent.DOC
+ )
+ val mode = if (action.msg == message("amazonqDoc.session.create")) Mode.CREATE else null
+ val codeGenerationResult = generateCode(codeGenerationId = response.codeGenerationId(), mode, token)
+ codeGenerationRemainingIterationCount = codeGenerationResult.codeGenerationRemainingIterationCount
+ codeGenerationTotalIterationCount = codeGenerationResult.codeGenerationTotalIterationCount
+
+ val nextState = PrepareDocGenerationState(
+ tabID = tabID,
+ approach = approach,
+ config = config,
+ filePaths = codeGenerationResult.newFiles,
+ deletedFiles = codeGenerationResult.deletedFiles,
+ references = codeGenerationResult.references,
+ currentIteration = currentIteration + 1,
+ uploadId = uploadId,
+ messenger = messenger,
+ codeGenerationRemainingIterationCount = codeGenerationRemainingIterationCount,
+ codeGenerationTotalIterationCount = codeGenerationTotalIterationCount,
+ token = token
+ )
+
+ // It is not needed to interact right away with the PrepareCodeGeneration.
+ // returns therefore a SessionStateInteraction object to be handled by the controller.
+ return SessionStateInteraction(
+ nextState = nextState,
+ interaction = Interaction(content = "", interactionSucceeded = true)
+ )
+ } catch (e: Exception) {
+ logger.warn(e) { "$FEATURE_NAME: Code generation failed: ${e.message}" }
+ throw e
+ }
+ }
+}
+
+fun getFileSummaryPercentage(input: String): Double {
+ // Split the input string by newline characters
+ val lines = input.split("\n")
+
+ // Find the line containing "summarized:"
+ val summaryLine = lines.find { it.contains("summarized:") }
+
+ // If the line is not found, return -1.0
+ if (summaryLine == null) {
+ return -1.0
+ }
+
+ // Extract the numbers from the summary line
+ val (summarized, total) = summaryLine.split(":")[1].trim().split(" of ").map { it.toDouble() }
+
+ // Calculate the percentage
+ val percentage = (summarized / total) * 100
+
+ return percentage
+}
+
+private suspend fun DocGenerationState.generateCode(codeGenerationId: String, mode: Mode?, token: CancellationTokenSource?): CodeGenerationResult {
+ val pollCount = 180
+ val requestDelay = 10000L
+
+ repeat(pollCount) {
+ if (token?.token?.isCancellationRequested() == true) {
+ return CodeGenerationResult(emptyList(), emptyList(), emptyList())
+ }
+
+ val codeGenerationResultState = config.amazonQCodeGenService.getTaskAssistCodeGeneration(
+ conversationId = config.conversationId,
+ codeGenerationId = codeGenerationId,
+ )
+
+ when (codeGenerationResultState.codeGenerationStatus().status()) {
+ CodeGenerationWorkflowStatus.COMPLETE -> {
+ val codeGenerationStreamResult = config.amazonQCodeGenService.exportTaskAssistArchiveResult(
+ conversationId = config.conversationId
+ )
+
+ val newFileInfo = registerNewFiles(newFileContents = codeGenerationStreamResult.newFileContents)
+ val deletedFileInfo = registerDeletedFiles(deletedFiles = codeGenerationStreamResult.deletedFiles.orEmpty())
+
+ messenger.sendUpdatePromptProgress(tabId = tabID, progressField = null)
+
+ return CodeGenerationResult(
+ newFiles = newFileInfo,
+ deletedFiles = deletedFileInfo,
+ references = codeGenerationStreamResult.references,
+ codeGenerationRemainingIterationCount = codeGenerationResultState.codeGenerationRemainingIterationCount(),
+ codeGenerationTotalIterationCount = codeGenerationResultState.codeGenerationTotalIterationCount()
+ )
+ }
+
+ CodeGenerationWorkflowStatus.IN_PROGRESS -> {
+ if (codeGenerationResultState.codeGenerationStatusDetail() != null) {
+ val progress = getFileSummaryPercentage(codeGenerationResultState.codeGenerationStatusDetail())
+
+ messenger.sendUpdatePromptProgress(
+ tabID,
+ inProgress(
+ progress.toInt(),
+ message(if (progress >= 100) "amazonqDoc.inprogress_message.generating" else "amazonqDoc.progress_message.summarizing")
+ )
+ )
+
+ messenger.sendAnswerPart(
+ tabId = tabID,
+ message = docGenerationProgressMessage(
+ if (progress < 100) {
+ if (progress < 20) {
+ DocGenerationStep.CREATE_KNOWLEDGE_GRAPH
+ } else {
+ DocGenerationStep.SUMMARIZING_FILES
+ }
+ } else {
+ DocGenerationStep.GENERATING_ARTIFACTS
+ },
+ mode
+ )
+ )
+ }
+
+ delay(requestDelay)
+ }
+
+ CodeGenerationWorkflowStatus.FAILED -> {
+ messenger.sendUpdatePromptProgress(tabId = tabID, progressField = null)
+
+ when (true) {
+ codeGenerationResultState.codeGenerationStatusDetail()?.contains(
+ "README_TOO_LARGE"
+ ),
+ -> docServiceError(message("amazonqDoc.exception.readme_too_large"))
+
+ codeGenerationResultState.codeGenerationStatusDetail()?.contains(
+ "README_UPDATE_TOO_LARGE"
+ ),
+ -> docServiceError(message("amazonqDoc.exception.readme_update_too_large"))
+
+ codeGenerationResultState.codeGenerationStatusDetail()?.contains(
+ "WORKSPACE_TOO_LARGE"
+ ),
+ -> docServiceError(message("amazonqDoc.exception.content_length_error"))
+
+ codeGenerationResultState.codeGenerationStatusDetail()?.contains(
+ "WORKSPACE_EMPTY"
+ ),
+ -> docServiceError(message("amazonqDoc.exception.workspace_empty"))
+
+ codeGenerationResultState.codeGenerationStatusDetail()?.contains(
+ "PROMPT_UNRELATED"
+ ),
+ -> docServiceError(message("amazonqDoc.exception.prompt_unrelated"))
+
+ codeGenerationResultState.codeGenerationStatusDetail()?.contains(
+ "PROMPT_TOO_VAGUE"
+ ),
+ -> docServiceError(message("amazonqDoc.exception.prompt_too_vague"))
+
+ codeGenerationResultState.codeGenerationStatusDetail()?.contains(
+ "PromptRefusal"
+ ),
+ -> docServiceError(message("amazonqFeatureDev.exception.prompt_refusal"))
+
+ codeGenerationResultState.codeGenerationStatusDetail()?.contains(
+ "Guardrails"
+ ),
+ -> docServiceError(message("amazonqDoc.error_text"))
+
+ codeGenerationResultState.codeGenerationStatusDetail()?.contains(
+ "EmptyPatch"
+ ),
+ -> {
+ if (codeGenerationResultState.codeGenerationStatusDetail()?.contains("NO_CHANGE_REQUIRED") == true) {
+ docServiceError(message("amazonqDoc.exception.no_change_required"))
+ }
+ docServiceError(message("amazonqDoc.error_text"))
+ }
+
+ codeGenerationResultState.codeGenerationStatusDetail()?.contains(
+ "Throttling"
+ ),
+ -> docServiceError(message("amazonqFeatureDev.exception.throttling"))
+
+ else -> docServiceError(message("amazonqDoc.error_text"))
+ }
+ }
+
+ else -> error("Unknown status: ${codeGenerationResultState.codeGenerationStatus().status()}")
+ }
+ }
+
+ return CodeGenerationResult(emptyList(), emptyList(), emptyList())
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/session/DocSession.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/session/DocSession.kt
new file mode 100644
index 0000000000..c34ccb1589
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/session/DocSession.kt
@@ -0,0 +1,235 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqDoc.session
+
+import com.intellij.openapi.diagnostic.logger
+import com.intellij.openapi.project.Project
+import com.intellij.openapi.vfs.VfsUtil
+import software.amazon.awssdk.services.codewhispererruntime.model.DocGenerationEvent
+import software.amazon.awssdk.services.codewhispererruntime.model.DocGenerationInteractionType
+import software.amazon.awssdk.services.codewhispererruntime.model.SendTelemetryEventResponse
+import software.aws.toolkits.core.utils.debug
+import software.aws.toolkits.core.utils.getLogger
+import software.aws.toolkits.core.utils.warn
+import software.aws.toolkits.jetbrains.common.clients.AmazonQCodeGenerateClient
+import software.aws.toolkits.jetbrains.common.session.ConversationNotStartedState
+import software.aws.toolkits.jetbrains.common.session.SessionState
+import software.aws.toolkits.jetbrains.common.session.SessionStateConfigData
+import software.aws.toolkits.jetbrains.common.util.AmazonQCodeGenService
+import software.aws.toolkits.jetbrains.common.util.getDiffCharsAndLines
+import software.aws.toolkits.jetbrains.common.util.resolveAndCreateOrUpdateFile
+import software.aws.toolkits.jetbrains.common.util.resolveAndDeleteFile
+import software.aws.toolkits.jetbrains.services.amazonq.FeatureDevSessionContext
+import software.aws.toolkits.jetbrains.services.amazonq.messages.MessagePublisher
+import software.aws.toolkits.jetbrains.services.amazonqDoc.CODE_GENERATION_RETRY_LIMIT
+import software.aws.toolkits.jetbrains.services.amazonqDoc.FEATURE_NAME
+import software.aws.toolkits.jetbrains.services.amazonqDoc.MAX_PROJECT_SIZE_BYTES
+import software.aws.toolkits.jetbrains.services.amazonqDoc.conversationIdNotFound
+import software.aws.toolkits.jetbrains.services.amazonqDoc.messages.sendAsyncEventProgress
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.DeletedFileInfo
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.Interaction
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.NewFileZipInfo
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.SessionStateAction
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.util.CancellationTokenSource
+import software.aws.toolkits.jetbrains.services.codewhisperer.util.content
+
+private val logger = getLogger()
+
+class DocSession(val tabID: String, val project: Project) {
+ var context: FeatureDevSessionContext
+ val sessionStartTime = System.currentTimeMillis()
+
+ var state: SessionState?
+ var preloaderFinished: Boolean = false
+ var localConversationId: String? = null
+ var localLatestMessage: String = ""
+ var task: String = ""
+ val proxyClient: AmazonQCodeGenerateClient
+ val amazonQCodeGenService: AmazonQCodeGenService
+
+ // retry session state vars
+ private var codegenRetries: Int
+
+ // Used to keep track of whether the current session/tab is currently authenticating/needs authenticating
+ var isAuthenticating: Boolean
+
+ init {
+ context = FeatureDevSessionContext(project, MAX_PROJECT_SIZE_BYTES)
+ proxyClient = AmazonQCodeGenerateClient.getInstance(project)
+ amazonQCodeGenService = AmazonQCodeGenService(proxyClient, project)
+ state = ConversationNotStartedState("", tabID, token = null)
+ isAuthenticating = false
+ codegenRetries = CODE_GENERATION_RETRY_LIMIT
+ }
+
+ fun conversationIDLog(conversationId: String) = "$FEATURE_NAME Conversation ID: $conversationId"
+
+ /**
+ * Preload any events that have to run before a chat message can be sent
+ */
+ suspend fun preloader(msg: String, messenger: MessagePublisher) {
+ if (!preloaderFinished) {
+ setupConversation(msg, messenger)
+ preloaderFinished = true
+ messenger.sendAsyncEventProgress(tabId = this.tabID, inProgress = true)
+ }
+ }
+
+ /**
+ * Starts a conversation with the backend and uploads the repo for the LLMs to be able to use it.
+ */
+ fun setupConversation(msg: String, messenger: MessagePublisher) {
+ // Store the initial message when setting up the conversation so that if it fails we can retry with this message
+ localLatestMessage = msg
+
+ localConversationId = amazonQCodeGenService.createConversation()
+ logger().info(conversationIDLog(this.conversationId))
+
+ val sessionStateConfig = getSessionStateConfig().copy(conversationId = this.conversationId)
+ state = PrepareDocGenerationState(
+ tabID = sessionState.tabID,
+ approach = sessionState.approach,
+ config = sessionStateConfig,
+ filePaths = emptyList(),
+ deletedFiles = emptyList(),
+ references = emptyList(),
+ currentIteration = 0, // first code gen iteration
+ uploadId = "", // There is no code gen uploadId so far
+ messenger = messenger,
+ token = CancellationTokenSource()
+ )
+ }
+
+ /**
+ * Triggered by the Insert code follow-up button to apply code changes.
+ */
+ fun insertChanges(filePaths: List, deletedFiles: List) {
+ val selectedSourceFolder = context.selectedSourceFolder.toNioPath()
+
+ filePaths.forEach { resolveAndCreateOrUpdateFile(selectedSourceFolder, it.zipFilePath, it.fileContent) }
+
+ deletedFiles.forEach { resolveAndDeleteFile(selectedSourceFolder, it.zipFilePath) }
+
+ // Taken from https://intellij-support.jetbrains.com/hc/en-us/community/posts/206118439-Refresh-after-external-changes-to-project-structure-and-sources
+ VfsUtil.markDirtyAndRefresh(true, true, true, context.selectedSourceFolder)
+ }
+
+ data class AddedContent(
+ val totalAddedChars: Int,
+ val totalAddedLines: Int,
+ val totalAddedFiles: Int,
+ )
+
+ fun countAddedContent(filePaths: List, interactionType: DocGenerationInteractionType? = null): AddedContent {
+ var totalAddedChars = 0
+ var totalAddedLines = 0
+ var totalAddedFiles = 0
+
+ filePaths.filter { !it.rejected }.forEach { filePath ->
+ val existingFile = VfsUtil.findRelativeFile(filePath.zipFilePath, context.selectedSourceFolder)
+ val content = filePath.fileContent
+ totalAddedFiles += 1
+
+ if (existingFile != null && interactionType == DocGenerationInteractionType.UPDATE_README) {
+ val existingContent = existingFile.content()
+ val (addedChars, addedLines) = getDiffCharsAndLines(existingContent, content)
+ totalAddedChars += addedChars
+ totalAddedLines += addedLines
+ } else {
+ totalAddedChars += content.length
+ totalAddedLines += content.split('\n').size
+ }
+ }
+
+ return AddedContent(
+ totalAddedChars = totalAddedChars,
+ totalAddedLines = totalAddedLines,
+ totalAddedFiles = totalAddedFiles
+ )
+ }
+
+ suspend fun send(msg: String): Interaction {
+ // When the task/"thing to do" hasn't been set yet, we want it to be the incoming message
+ if (task.isEmpty() && msg.isNotEmpty()) {
+ task = msg
+ }
+
+ localLatestMessage = msg
+
+ return nextInteraction(msg)
+ }
+
+ private suspend fun nextInteraction(msg: String): Interaction {
+ var action = SessionStateAction(
+ task = task,
+ msg = msg,
+ )
+
+ val resp = sessionState.interact(action)
+ if (resp.nextState != null) {
+ // Approach may have been changed after the interaction
+ val newApproach = sessionState.approach
+
+ // Move to the next state
+ state = resp.nextState
+
+ // If approach was changed then we need to set it in the next state and this state
+ sessionState.approach = newApproach
+ }
+
+ return resp.interaction
+ }
+
+ fun getSessionStateConfig(): SessionStateConfigData = SessionStateConfigData(
+ conversationId = this.conversationId,
+ repoContext = this.context,
+ amazonQCodeGenService = this.amazonQCodeGenService,
+ )
+
+ val conversationId: String
+ get() {
+ if (localConversationId == null) {
+ conversationIdNotFound()
+ } else {
+ return localConversationId as String
+ }
+ }
+
+ val conversationIdUnsafe: String?
+ get() = localConversationId
+
+ val sessionState: SessionState
+ get() {
+ if (state == null) {
+ throw Error("State should be initialized before it's read")
+ } else {
+ return state as SessionState
+ }
+ }
+
+ val latestMessage: String
+ get() = this.localLatestMessage
+
+ val retries: Int
+ get() = codegenRetries
+
+ fun decreaseRetries() {
+ codegenRetries -= 1
+ }
+
+ fun sendDocGenerationEvent(docGenerationEvent: DocGenerationEvent) {
+ val sendDocGenerationEventResponse: SendTelemetryEventResponse
+ try {
+ sendDocGenerationEventResponse = proxyClient.sendDocGenerationTelemetryEvent(docGenerationEvent)
+ val requestId = sendDocGenerationEventResponse.responseMetadata().requestId()
+ logger.debug {
+ "${FEATURE_NAME}: succesfully sent doc generation telemetry: ConversationId: $conversationId RequestId: $requestId"
+ }
+ } catch (e: Exception) {
+ logger.warn(e) { "${FEATURE_NAME}: failed to send doc generation telemetry" }
+ }
+ }
+
+ fun getUserIdentity(): String = proxyClient.connection().id
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/session/PrepareDocGenerationState.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/session/PrepareDocGenerationState.kt
new file mode 100644
index 0000000000..2ebf98c465
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/session/PrepareDocGenerationState.kt
@@ -0,0 +1,83 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqDoc.session
+
+import software.aws.toolkits.core.utils.getLogger
+import software.aws.toolkits.core.utils.warn
+import software.aws.toolkits.jetbrains.common.session.SessionState
+import software.aws.toolkits.jetbrains.common.session.SessionStateConfig
+import software.aws.toolkits.jetbrains.services.amazonq.messages.MessagePublisher
+import software.aws.toolkits.jetbrains.services.amazonqDoc.FEATURE_NAME
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.CodeReferenceGenerated
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.DeletedFileInfo
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.NewFileZipInfo
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.SessionStateAction
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.SessionStatePhase
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.util.CancellationTokenSource
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.util.deleteUploadArtifact
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.util.uploadArtifactToS3
+import software.aws.toolkits.jetbrains.settings.CodeWhispererSettings
+
+private val logger = getLogger()
+
+class PrepareDocGenerationState(
+ override var tabID: String,
+ override var approach: String,
+ private var config: SessionStateConfig,
+ val filePaths: List,
+ val deletedFiles: List,
+ val references: List,
+ var uploadId: String,
+ private val currentIteration: Int,
+ private var messenger: MessagePublisher,
+ var codeGenerationRemainingIterationCount: Int? = null,
+ var codeGenerationTotalIterationCount: Int? = null,
+ override var token: CancellationTokenSource?,
+) : SessionState {
+ override val phase = SessionStatePhase.CODEGEN
+ override suspend fun interact(action: SessionStateAction): SessionStateInteraction {
+ var zipFileLength: Long? = null
+ val nextState: SessionState
+ try {
+ val isAutoBuildFeatureEnabled = CodeWhispererSettings.getInstance().isAutoBuildFeatureEnabled(this.config.repoContext.getWorkspaceRoot())
+ val repoZipResult = config.repoContext.getProjectZip(isAutoBuildFeatureEnabled = isAutoBuildFeatureEnabled)
+ val zipFileChecksum = repoZipResult.checksum
+ zipFileLength = repoZipResult.contentLength
+ val fileToUpload = repoZipResult.payload
+
+ val uploadUrlResponse = config.amazonQCodeGenService.createUploadUrl(
+ config.conversationId,
+ zipFileChecksum,
+ zipFileLength,
+ uploadId
+ )
+
+ uploadArtifactToS3(
+ uploadUrlResponse.uploadUrl(),
+ fileToUpload,
+ zipFileChecksum,
+ zipFileLength,
+ uploadUrlResponse.kmsKeyArn()
+ )
+ deleteUploadArtifact(fileToUpload)
+
+ this.uploadId = uploadUrlResponse.uploadId()
+
+ nextState = DocGenerationState(
+ tabID = this.tabID,
+ approach = "", // No approach needed,
+ config = this.config,
+ uploadId = this.uploadId,
+ currentIteration = this.currentIteration,
+ messenger = messenger,
+ phase = phase,
+ token = this.token
+ )
+ } catch (e: Exception) {
+ logger.warn(e) { "$FEATURE_NAME: Code uploading failed: ${e.message}" }
+ throw e
+ }
+ return nextState.interact(action)
+ }
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/session/SessionStateTypes.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/session/SessionStateTypes.kt
new file mode 100644
index 0000000000..f4821d9027
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/session/SessionStateTypes.kt
@@ -0,0 +1,27 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqDoc.session
+
+import com.fasterxml.jackson.annotation.JsonProperty
+import software.aws.toolkits.jetbrains.common.session.SessionState
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.CodeReferenceGenerated
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.Interaction
+
+data class SessionStateInteraction(
+ val nextState: T? = null,
+ val interaction: Interaction,
+)
+
+data class DocGenerationStreamResult(
+ @JsonProperty("new_file_contents")
+ var newFileContents: Map,
+ @JsonProperty("deleted_files")
+ var deletedFiles: List?,
+ var references: List,
+)
+
+data class ExportDocTaskAssistResultArchiveStreamResult(
+ @JsonProperty("code_generation_result")
+ var codeGenerationResult: DocGenerationStreamResult,
+)
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/storage/ChatSessionStorage.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/storage/ChatSessionStorage.kt
new file mode 100644
index 0000000000..2344dac5c9
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/storage/ChatSessionStorage.kt
@@ -0,0 +1,26 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqDoc.storage
+
+import com.intellij.openapi.project.Project
+import software.aws.toolkits.jetbrains.services.amazonqDoc.session.DocSession
+
+class ChatSessionStorage {
+ private val sessions = mutableMapOf()
+
+ private fun createSession(tabId: String, project: Project): DocSession {
+ val session = DocSession(tabId, project)
+ sessions[tabId] = session
+ return session
+ }
+
+ @Synchronized fun getSession(tabId: String, project: Project): DocSession = sessions[tabId] ?: createSession(tabId, project)
+
+ fun deleteSession(tabId: String) {
+ sessions.remove(tabId)
+ }
+
+ // Find all sessions that are currently waiting to be authenticated
+ fun getAuthenticatingSessions(): List = this.sessions.values.filter { it.isAuthenticating }
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/util/DocControllerUtil.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/util/DocControllerUtil.kt
new file mode 100644
index 0000000000..b35e2cb1bd
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqDoc/util/DocControllerUtil.kt
@@ -0,0 +1,43 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.amazonqDoc.util
+
+import software.aws.toolkits.jetbrains.services.amazonqDoc.messages.FollowUp
+import software.aws.toolkits.jetbrains.services.amazonqDoc.messages.FollowUpIcons
+import software.aws.toolkits.jetbrains.services.amazonqDoc.messages.FollowUpStatusType
+import software.aws.toolkits.jetbrains.services.amazonqDoc.messages.FollowUpTypes
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.SessionStatePhase
+import software.aws.toolkits.resources.message
+
+fun getFollowUpOptions(phase: SessionStatePhase?): List {
+ when (phase) {
+ SessionStatePhase.CODEGEN -> {
+ return listOf(
+ FollowUp(
+ pillText = message("amazonqDoc.prompt.review.accept"),
+ prompt = message("amazonqDoc.prompt.review.accept"),
+ status = FollowUpStatusType.Success,
+ type = FollowUpTypes.ACCEPT_CHANGES,
+ icon = FollowUpIcons.Ok,
+ ),
+ FollowUp(
+ pillText = message("amazonqDoc.prompt.review.changes"),
+ prompt = message("amazonqDoc.prompt.review.changes"),
+ status = FollowUpStatusType.Info,
+ type = FollowUpTypes.MAKE_CHANGES,
+ icon = FollowUpIcons.Refresh,
+ ),
+ FollowUp(
+ pillText = message("general.reject"),
+ prompt = message("general.reject"),
+ status = FollowUpStatusType.Error,
+ type = FollowUpTypes.REJECT_CHANGES,
+ icon = FollowUpIcons.Cancel,
+ )
+ )
+ }
+
+ else -> return emptyList()
+ }
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/FeatureDevApp.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/FeatureDevApp.kt
index bc5bcfc6a7..7169e39150 100644
--- a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/FeatureDevApp.kt
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/FeatureDevApp.kt
@@ -11,6 +11,9 @@ import software.aws.toolkits.jetbrains.core.credentials.ToolkitConnectionManager
import software.aws.toolkits.jetbrains.services.amazonq.apps.AmazonQApp
import software.aws.toolkits.jetbrains.services.amazonq.apps.AmazonQAppInitContext
import software.aws.toolkits.jetbrains.services.amazonq.messages.AmazonQMessage
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.auth.isCodeScanAvailable
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.auth.isCodeTestAvailable
+import software.aws.toolkits.jetbrains.services.amazonqDoc.auth.isDocAvailable
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.auth.isFeatureDevAvailable
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.controller.FeatureDevController
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.messages.AuthenticationUpdateMessage
@@ -62,7 +65,10 @@ class FeatureDevApp : AmazonQApp {
AuthenticationUpdateMessage(
featureDevEnabled = isFeatureDevAvailable(context.project),
codeTransformEnabled = isCodeTransformAvailable(context.project),
- authenticatingTabIDs = chatSessionStorage.getAuthenticatingSessions().map { it.tabID }
+ codeScanEnabled = isCodeScanAvailable(context.project),
+ codeTestEnabled = isCodeTestAvailable(context.project),
+ docEnabled = isDocAvailable(context.project),
+ authenticatingTabIDs = chatSessionStorage.getAuthenticatingSessions().map { it.tabID },
)
)
}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/FeatureDevConstants.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/FeatureDevConstants.kt
index c7c76f4738..658cdc2ed5 100644
--- a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/FeatureDevConstants.kt
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/FeatureDevConstants.kt
@@ -8,7 +8,7 @@ const val FEATURE_EVALUATION_PRODUCT_NAME = "FeatureDev"
const val FEATURE_NAME = "Amazon Q Developer Agent for software development"
@Suppress("MaxLineLength")
-const val GENERATE_DEV_FILE_PROMPT = "generate a devfile in my repository. Note that you should only use devfile version 2.0.0 and the only supported command is test, so you should bundle all install, build and test commands in “test”. also you can use “public.ecr.aws/aws-mde/universal-image:latest” as universal image if you aren’t sure which image to use. here is an example for a node repository (but don't assume it's always a node project. look at the existing repository structure before generating the devfile): schemaVersion: 2.0.0 components: - name: dev container: image: public.ecr.aws/aws-mde/universal-image:latest commands: - id: test exec: component: dev commandLine: \"npm install && npm run build && npm run test\""
+const val GENERATE_DEV_FILE_PROMPT = "generate a devfile in my repository. Note that you should only use devfile version 2.0.0 and the only supported commands are install, build and test (are all optional). so you may have to bundle some commands together using '&&'. also you can use \"public.ecr.aws/aws-mde/universal-image:latest\" as universal image if you aren’t sure which image to use. here is an example for a node repository (but don't assume it's always a node project. look at the existing repository structure before generating the devfile): schemaVersion: 2.0.0 components: - name: dev container: image: public.ecr.aws/aws-mde/universal-image:latest commands: - id: install exec: component: dev commandLine: \"npm install\" - id: build exec: component: dev commandLine: \"npm run build\" - id: test exec: component: dev commandLine: \"npm run test\""
// Max number of times a user can attempt to retry a code generation request if it fails
const val CODE_GENERATION_RETRY_LIMIT = 3
@@ -41,3 +41,21 @@ enum class FeatureDevOperation(private val operationName: String) {
override fun toString(): String = operationName
}
+
+enum class MetricDataOperationName(private val operationName: String) {
+ StartCodeGeneration("StartCodeGeneration"),
+ EndCodeGeneration("EndCodeGeneration"),
+ ;
+
+ override fun toString(): String = operationName
+}
+
+enum class MetricDataResult(private val resultName: String) {
+ Success("Success"),
+ Fault("Fault"),
+ Error("Error"),
+ LlmFailure("LLMFailure"),
+ ;
+
+ override fun toString(): String = resultName
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/clients/FeatureDevClient.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/clients/FeatureDevClient.kt
index 29a1feb1a0..2d73f5fb4b 100644
--- a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/clients/FeatureDevClient.kt
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/clients/FeatureDevClient.kt
@@ -13,6 +13,7 @@ import software.amazon.awssdk.services.codewhispererruntime.model.ContentChecksu
import software.amazon.awssdk.services.codewhispererruntime.model.CreateTaskAssistConversationRequest
import software.amazon.awssdk.services.codewhispererruntime.model.CreateTaskAssistConversationResponse
import software.amazon.awssdk.services.codewhispererruntime.model.CreateUploadUrlResponse
+import software.amazon.awssdk.services.codewhispererruntime.model.Dimension
import software.amazon.awssdk.services.codewhispererruntime.model.GetTaskAssistCodeGenerationResponse
import software.amazon.awssdk.services.codewhispererruntime.model.IdeCategory
import software.amazon.awssdk.services.codewhispererruntime.model.OperatingSystem
@@ -89,6 +90,33 @@ class FeatureDevClient(
requestBuilder.userContext(featureDevUserContext)
}
+ fun sendFeatureDevMetricData(operationName: String, result: String): SendTelemetryEventResponse =
+ bearerClient().sendTelemetryEvent { requestBuilder ->
+ requestBuilder.telemetryEvent { telemetryEventBuilder ->
+ telemetryEventBuilder.metricData {
+ it
+ .metricName("Operation")
+ .metricValue(1.0)
+ .timestamp(Instant.now())
+ .product("FeatureDev")
+ .dimensions(
+ listOf(
+ Dimension.builder()
+ .name("operationName")
+ .value(operationName)
+ .build(),
+ Dimension.builder()
+ .name("result")
+ .value(result)
+ .build()
+ )
+ )
+ }
+ }
+ requestBuilder.optOutPreference(getTelemetryOptOutPreference())
+ requestBuilder.userContext(featureDevUserContext)
+ }
+
fun sendFeatureDevCodeGenerationEvent(
conversationId: String,
linesOfCodeGenerated: Int,
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/controller/FeatureDevController.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/controller/FeatureDevController.kt
index 58a079d3b6..1c7cbfbfdc 100644
--- a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/controller/FeatureDevController.kt
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/controller/FeatureDevController.kt
@@ -27,6 +27,7 @@ import software.aws.toolkits.core.utils.error
import software.aws.toolkits.core.utils.getLogger
import software.aws.toolkits.core.utils.info
import software.aws.toolkits.core.utils.warn
+import software.aws.toolkits.jetbrains.common.util.selectFolder
import software.aws.toolkits.jetbrains.core.coroutines.EDT
import software.aws.toolkits.jetbrains.services.amazonq.RepoSizeError
import software.aws.toolkits.jetbrains.services.amazonq.apps.AmazonQAppInitContext
@@ -72,7 +73,6 @@ import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.Sessio
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.storage.ChatSessionStorage
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.util.InsertAction
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.util.getFollowUpOptions
-import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.util.selectFolder
import software.aws.toolkits.jetbrains.services.codewhisperer.util.content
import software.aws.toolkits.jetbrains.services.cwc.controller.chat.telemetry.FeedbackComment
import software.aws.toolkits.jetbrains.services.cwc.controller.chat.telemetry.getStartUrl
@@ -311,7 +311,6 @@ class FeatureDevController(
filePaths = filePaths.filter { it.zipFilePath == fileToUpdate },
deletedFiles = deletedFiles.filter { it.zipFilePath == fileToUpdate },
references = references, // Add all references (not attributed per-file)
- messenger
)
AmazonqTelemetry.isAcceptedCodeChanges(
@@ -326,8 +325,6 @@ class FeatureDevController(
deletedFiles.find { it.zipFilePath == fileToUpdate }?.let { it.rejected = !it.rejected }
}
- // FIXME: This is a kludge that is hiding the fact that insertChanges is updating the file tree above this point to
- // an incorrect state. Update the state of the tree view:
messenger.updateFileComponent(message.tabId, filePaths, deletedFiles, messageId)
// Then, if the accepted file is not a deletion, open a diff to show the changes are applied:
@@ -434,14 +431,14 @@ class FeatureDevController(
credentialStartUrl = getStartUrl(project = context.project)
)
- // Caution: insertChanges has multiple responsibilities.
- // The filter here results in rejected files being hidden from the tree after continuing, by design.
- // However, it is critical that we don't hide already-accepted files. Inside insertChanges, it
- // filters to only update the subset of passed files that aren't accepted or rejected already.
session.insertChanges(
- filePaths = filePaths.filter { !it.rejected },
- deletedFiles = deletedFiles.filter { !it.rejected },
- references = references,
+ filePaths = filePaths,
+ deletedFiles = deletedFiles,
+ references = references
+ )
+ session.updateFilesPaths(
+ filePaths = filePaths,
+ deletedFiles = deletedFiles,
messenger
)
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/controller/FeatureDevControllerExtensions.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/controller/FeatureDevControllerExtensions.kt
index 19331d7f61..dc564b6800 100644
--- a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/controller/FeatureDevControllerExtensions.kt
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/controller/FeatureDevControllerExtensions.kt
@@ -6,6 +6,13 @@ package software.aws.toolkits.jetbrains.services.amazonqFeatureDev.controller
import com.intellij.notification.NotificationAction
import software.aws.toolkits.jetbrains.services.amazonq.messages.MessagePublisher
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.CODE_GENERATION_RETRY_LIMIT
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.EmptyPatchException
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.GuardrailsException
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.MetricDataOperationName
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.MetricDataResult
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.NoChangeRequiredException
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.PromptRefusalException
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.ThrottlingException
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.messages.FeatureDevMessageType
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.messages.FollowUp
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.messages.FollowUpStatusType
@@ -21,7 +28,6 @@ import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.Delete
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.NewFileZipInfo
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.PrepareCodeGenerationState
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.Session
-import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.SessionState
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.util.CancellationTokenSource
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.util.InsertAction
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.util.getFollowUpOptions
@@ -57,12 +63,17 @@ suspend fun FeatureDevController.onCodeGeneration(
var totalIterations: Int? = state.codeGenerationTotalIterationCount
if (state.token?.token?.isCancellationRequested() == true) {
- disposeToken(state, messenger, tabId, state.currentIteration?.let { CODE_GENERATION_RETRY_LIMIT.minus(it) }, CODE_GENERATION_RETRY_LIMIT)
+ disposeToken(messenger, tabId, state.codeGenerationRemainingIterationCount, state.codeGenerationTotalIterationCount)
return
}
messenger.sendUpdatePlaceholder(tabId = tabId, newPlaceholder = message("amazonqFeatureDev.placeholder.generating_code"))
+ session.sendMetricDataTelemetry(
+ MetricDataOperationName.StartCodeGeneration,
+ MetricDataResult.Success
+ )
+
session.send(message) // Trigger code generation
state = session.sessionState
@@ -84,7 +95,7 @@ suspend fun FeatureDevController.onCodeGeneration(
}
if (state.token?.token?.isCancellationRequested() == true) {
- disposeToken(state, messenger, tabId, state.currentIteration?.let { CODE_GENERATION_RETRY_LIMIT.minus(it) }, CODE_GENERATION_RETRY_LIMIT)
+ disposeToken(messenger, tabId, state.codeGenerationRemainingIterationCount, state.codeGenerationTotalIterationCount)
return
}
@@ -121,21 +132,48 @@ suspend fun FeatureDevController.onCodeGeneration(
tabId = tabId,
messageType = FeatureDevMessageType.Answer,
message =
- if (remainingIterations == 0) {
- message("amazonqFeatureDev.code_generation.iteration_zero")
- } else {
+ if (remainingIterations > 2) {
+ message("amazonqFeatureDev.code_generation.iteration_counts_ask_to_add_code_or_feedback")
+ } else if (remainingIterations > 0) {
message(
"amazonqFeatureDev.code_generation.iteration_counts",
remainingIterations,
totalIterations,
)
+ } else {
+ message(
+ "amazonqFeatureDev.code_generation.iteration_counts_ask_to_add_code",
+ remainingIterations,
+ totalIterations,
+ )
},
)
}
messenger.sendSystemPrompt(tabId = tabId, followUp = getFollowUpOptions(session.sessionState.phase, InsertAction.ALL))
-
messenger.sendUpdatePlaceholder(tabId = tabId, newPlaceholder = message("amazonqFeatureDev.placeholder.after_code_generation"))
+ } catch (err: Exception) {
+ when (err) {
+ is GuardrailsException, is NoChangeRequiredException, is PromptRefusalException, is ThrottlingException -> {
+ session.sendMetricDataTelemetry(
+ MetricDataOperationName.EndCodeGeneration,
+ MetricDataResult.Error
+ )
+ }
+ is EmptyPatchException -> {
+ session.sendMetricDataTelemetry(
+ MetricDataOperationName.EndCodeGeneration,
+ MetricDataResult.LlmFailure
+ )
+ }
+ else -> {
+ session.sendMetricDataTelemetry(
+ MetricDataOperationName.EndCodeGeneration,
+ MetricDataResult.Fault
+ )
+ }
+ }
+ throw err
} finally {
if (session.sessionState.token
?.token
@@ -155,10 +193,14 @@ suspend fun FeatureDevController.onCodeGeneration(
)
}
}
+
+ session.sendMetricDataTelemetry(
+ MetricDataOperationName.EndCodeGeneration,
+ MetricDataResult.Success
+ )
}
private suspend fun disposeToken(
- state: SessionState,
messenger: MessagePublisher,
tabId: String,
remainingIterations: Int?,
@@ -196,16 +238,25 @@ private suspend fun disposeToken(
return
}
- messenger.sendAnswer(
- tabId = tabId,
- messageType = FeatureDevMessageType.Answer,
- message =
- message(
- "amazonqFeatureDev.code_generation.stopped_code_generation",
- remainingIterations ?: state.currentIteration?.let { CODE_GENERATION_RETRY_LIMIT - it } as Any,
- totalIterations ?: CODE_GENERATION_RETRY_LIMIT,
- ),
- )
+ if (remainingIterations !== null && totalIterations !== null && remainingIterations <= 2) {
+ messenger.sendAnswer(
+ tabId = tabId,
+ messageType = FeatureDevMessageType.Answer,
+ message =
+ message(
+ "amazonqFeatureDev.code_generation.stopped_code_generation",
+ remainingIterations,
+ totalIterations,
+ ),
+ )
+ } else {
+ messenger.sendAnswer(
+ tabId = tabId,
+ messageType = FeatureDevMessageType.Answer,
+ message =
+ message("amazonqFeatureDev.code_generation.stopped_code_generation_no_iteration_count_display"),
+ )
+ }
messenger.sendChatInputEnabledMessage(tabId = tabId, enabled = true)
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/messages/FeatureDevMessage.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/messages/FeatureDevMessage.kt
index 339a0c080b..9d82818c4a 100644
--- a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/messages/FeatureDevMessage.kt
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/messages/FeatureDevMessage.kt
@@ -181,6 +181,9 @@ data class AuthenticationUpdateMessage(
val authenticatingTabIDs: List,
val featureDevEnabled: Boolean,
val codeTransformEnabled: Boolean,
+ val codeScanEnabled: Boolean,
+ val codeTestEnabled: Boolean,
+ val docEnabled: Boolean,
val message: String? = null,
val messageId: String = UUID.randomUUID().toString(),
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/session/CodeGenerationState.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/session/CodeGenerationState.kt
index 5b172e73c7..1872f8df14 100644
--- a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/session/CodeGenerationState.kt
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/session/CodeGenerationState.kt
@@ -26,7 +26,7 @@ import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.util.readFileT
import software.aws.toolkits.jetbrains.services.cwc.controller.chat.telemetry.getStartUrl
import software.aws.toolkits.resources.message
import software.aws.toolkits.telemetry.AmazonqTelemetry
-import software.aws.toolkits.telemetry.Result
+import software.aws.toolkits.telemetry.MetricResult
import java.util.UUID
private val logger = getLogger()
@@ -47,9 +47,9 @@ class CodeGenerationState(
) : SessionState {
override val phase = SessionStatePhase.CODEGEN
- override suspend fun interact(action: SessionStateAction): SessionStateInteraction {
+ override suspend fun interact(action: SessionStateAction): SessionStateInteraction {
val startTime = System.currentTimeMillis()
- var result: Result = Result.Succeeded
+ var result: MetricResult = MetricResult.Succeeded
var failureReason: String? = null
var failureReasonDesc: String? = null
var codeGenerationWorkflowStatus: CodeGenerationWorkflowStatus = CodeGenerationWorkflowStatus.COMPLETE
@@ -84,6 +84,12 @@ class CodeGenerationState(
numberOfFilesGenerated = codeGenerationResult.newFiles.size
codeGenerationRemainingIterationCount = codeGenerationResult.codeGenerationRemainingIterationCount
codeGenerationTotalIterationCount = codeGenerationResult.codeGenerationTotalIterationCount
+ currentIteration =
+ if (codeGenerationRemainingIterationCount != null && codeGenerationTotalIterationCount != null) {
+ codeGenerationTotalIterationCount?.let { total -> codeGenerationRemainingIterationCount?.let { remaining -> total - remaining } }
+ } else {
+ currentIteration?.plus(1)
+ }
runCatching {
var insertedLines = 0
@@ -128,7 +134,7 @@ class CodeGenerationState(
filePaths = codeGenerationResult.newFiles,
deletedFiles = codeGenerationResult.deletedFiles,
references = codeGenerationResult.references,
- currentIteration = currentIteration?.plus(1),
+ currentIteration = currentIteration,
uploadId = uploadId,
messenger = messenger,
codeGenerationRemainingIterationCount = codeGenerationRemainingIterationCount,
@@ -145,7 +151,7 @@ class CodeGenerationState(
)
} catch (e: Exception) {
logger.warn(e) { "$FEATURE_NAME: Code generation failed: ${e.message}" }
- result = Result.Failed
+ result = MetricResult.Failed
failureReason = e.javaClass.simpleName
if (e is FeatureDevException) {
failureReason = e.reason()
@@ -182,10 +188,12 @@ private suspend fun CodeGenerationState.generateCode(
): CodeGenerationResult {
val pollCount = 360
val requestDelay = 5000L
+ var codeGenerationRemainingIterationCount: Int? = null
+ var codeGenerationTotalIterationCount: Int? = null
repeat(pollCount) {
if (token?.token?.isCancellationRequested() == true) {
- return CodeGenerationResult(emptyList(), emptyList(), emptyList())
+ return CodeGenerationResult(emptyList(), emptyList(), emptyList(), codeGenerationRemainingIterationCount, codeGenerationTotalIterationCount)
}
val codeGenerationResultState =
config.featureDevService.getTaskAssistCodeGeneration(
@@ -193,6 +201,9 @@ private suspend fun CodeGenerationState.generateCode(
codeGenerationId = codeGenerationId,
)
+ codeGenerationRemainingIterationCount = codeGenerationResultState.codeGenerationRemainingIterationCount()
+ codeGenerationTotalIterationCount = codeGenerationResultState.codeGenerationTotalIterationCount()
+
when (codeGenerationResultState.codeGenerationStatus().status()) {
CodeGenerationWorkflowStatus.COMPLETE -> {
val codeGenerationStreamResult =
@@ -207,8 +218,8 @@ private suspend fun CodeGenerationState.generateCode(
newFiles = newFileInfo,
deletedFiles = deletedFileInfo,
references = codeGenerationStreamResult.references,
- codeGenerationRemainingIterationCount = codeGenerationResultState.codeGenerationRemainingIterationCount(),
- codeGenerationTotalIterationCount = codeGenerationResultState.codeGenerationTotalIterationCount(),
+ codeGenerationRemainingIterationCount = codeGenerationRemainingIterationCount,
+ codeGenerationTotalIterationCount = codeGenerationTotalIterationCount,
)
}
CodeGenerationWorkflowStatus.IN_PROGRESS -> {
@@ -252,13 +263,14 @@ private suspend fun CodeGenerationState.generateCode(
}
}
- return CodeGenerationResult(emptyList(), emptyList(), emptyList())
+ return CodeGenerationResult(emptyList(), emptyList(), emptyList(), codeGenerationRemainingIterationCount, codeGenerationTotalIterationCount)
}
fun registerNewFiles(newFileContents: Map): List =
newFileContents.map {
NewFileZipInfo(
- zipFilePath = it.key,
+ // Note: When managing file state, we normalize file paths returned from the agent in order to ensure they are handled as relative paths.
+ zipFilePath = it.key.removePrefix("/"),
fileContent = it.value,
rejected = false,
changeApplied = false
@@ -268,7 +280,8 @@ fun registerNewFiles(newFileContents: Map): List
fun registerDeletedFiles(deletedFiles: List): List =
deletedFiles.map {
DeletedFileInfo(
- zipFilePath = it,
+ // Note: When managing file state, we normalize file paths returned from the agent in order to ensure they are handled as relative paths.
+ zipFilePath = it.removePrefix("/"),
rejected = false,
changeApplied = false
)
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/session/ConversationNotStartedState.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/session/ConversationNotStartedState.kt
index 2b687d144d..7bfa85b39a 100644
--- a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/session/ConversationNotStartedState.kt
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/session/ConversationNotStartedState.kt
@@ -9,14 +9,14 @@ class ConversationNotStartedState(
override var approach: String,
override val tabID: String,
override var token: CancellationTokenSource?,
- override var codeGenerationRemainingIterationCount: Int?,
- override var codeGenerationTotalIterationCount: Int?,
+ override var codeGenerationRemainingIterationCount: Int? = null,
+ override var codeGenerationTotalIterationCount: Int? = null,
override var currentIteration: Int?,
override var diffMetricsProcessed: DiffMetricsProcessed,
) : SessionState {
override val phase = SessionStatePhase.INIT
- override suspend fun interact(action: SessionStateAction): SessionStateInteraction {
+ override suspend fun interact(action: SessionStateAction): SessionStateInteraction {
error("Illegal transition between states, restart the conversation")
}
}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/session/PrepareCodeGenerationState.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/session/PrepareCodeGenerationState.kt
index 7f55af07ed..b9dff5445d 100644
--- a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/session/PrepareCodeGenerationState.kt
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/session/PrepareCodeGenerationState.kt
@@ -38,7 +38,7 @@ class PrepareCodeGenerationState(
override var diffMetricsProcessed: DiffMetricsProcessed,
) : SessionState {
override val phase = SessionStatePhase.CODEGEN
- override suspend fun interact(action: SessionStateAction): SessionStateInteraction {
+ override suspend fun interact(action: SessionStateAction): SessionStateInteraction {
val startTime = System.currentTimeMillis()
var result: Result = Result.Succeeded
var failureReason: String? = null
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/session/Session.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/session/Session.kt
index 6fdf12be66..dd1bf4bd7c 100644
--- a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/session/Session.kt
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/session/Session.kt
@@ -6,12 +6,16 @@ package software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session
import com.intellij.openapi.diagnostic.logger
import com.intellij.openapi.project.Project
import com.intellij.openapi.vfs.VfsUtil
+import software.aws.toolkits.jetbrains.common.util.resolveAndCreateOrUpdateFile
+import software.aws.toolkits.jetbrains.common.util.resolveAndDeleteFile
import software.aws.toolkits.jetbrains.services.amazonq.FeatureDevSessionContext
import software.aws.toolkits.jetbrains.services.amazonq.messages.MessagePublisher
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.CODE_GENERATION_RETRY_LIMIT
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.ConversationIdNotFoundException
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.FEATURE_NAME
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.MAX_PROJECT_SIZE_BYTES
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.MetricDataOperationName
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.MetricDataResult
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.clients.FeatureDevClient
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.messages.IncomingFeatureDevMessage
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.messages.sendAsyncEventProgress
@@ -21,8 +25,6 @@ import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.util.FeatureDe
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.util.getChangeIdentifier
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.util.getDiffMetrics
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.util.readFileToString
-import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.util.resolveAndCreateOrUpdateFile
-import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.util.resolveAndDeleteFile
import software.aws.toolkits.jetbrains.services.cwc.controller.ReferenceLogController
import java.util.HashSet
@@ -53,8 +55,6 @@ class Session(val tabID: String, val project: Project) {
approach = "",
tabID = tabID,
token = null,
- codeGenerationRemainingIterationCount = 0,
- codeGenerationTotalIterationCount = CODE_GENERATION_RETRY_LIMIT,
currentIteration = 0,
diffMetricsProcessed = DiffMetricsProcessed(HashSet(), HashSet())
)
@@ -108,6 +108,18 @@ class Session(val tabID: String, val project: Project) {
this._codeResultMessageId = messageId
}
+ suspend fun updateFilesPaths(
+ filePaths: List,
+ deletedFiles: List,
+ messenger: MessagePublisher,
+ disableFileActions: Boolean = false,
+ ) {
+ val codeResultMessageId = this._codeResultMessageId
+ if (codeResultMessageId != null) {
+ messenger.updateFileComponent(this.tabID, filePaths, deletedFiles, codeResultMessageId, disableFileActions)
+ }
+ }
+
/**
* Triggered by the Insert code follow-up button to apply code changes.
*/
@@ -115,11 +127,10 @@ class Session(val tabID: String, val project: Project) {
filePaths: List,
deletedFiles: List,
references: List,
- messenger: MessagePublisher,
) {
- val selectedSourceFolder = context.selectedSourceFolder.toNioPath()
val newFilePaths = filePaths.filter { !it.rejected && !it.changeApplied }
val newDeletedFiles = deletedFiles.filter { !it.rejected && !it.changeApplied }
+ val selectedSourceFolder = context.selectedSourceFolder.toNioPath()
runCatching {
var insertedLines = 0
@@ -156,23 +167,39 @@ class Session(val tabID: String, val project: Project) {
}
}.onFailure { /* Noop on diff telemetry failure */ }
- newFilePaths.forEach {
- resolveAndCreateOrUpdateFile(selectedSourceFolder, it.zipFilePath, it.fileContent)
- it.changeApplied = true
- }
+ insertNewFiles(newFilePaths)
- newDeletedFiles.forEach {
- resolveAndDeleteFile(selectedSourceFolder, it.zipFilePath)
- it.changeApplied = true
- }
+ applyDeleteFiles(newDeletedFiles)
ReferenceLogController.addReferenceLog(references, project)
// Taken from https://intellij-support.jetbrains.com/hc/en-us/community/posts/206118439-Refresh-after-external-changes-to-project-structure-and-sources
VfsUtil.markDirtyAndRefresh(true, true, true, context.selectedSourceFolder)
- val codeResultMessageId = this._codeResultMessageId
- if (codeResultMessageId != null) {
- messenger.updateFileComponent(this.tabID, filePaths, deletedFiles, codeResultMessageId)
+ }
+
+// Suppressing because insertNewFiles needs to be a suspend function in order to be tested
+ @Suppress("RedundantSuspendModifier")
+ suspend fun insertNewFiles(
+ filePaths: List,
+ ) {
+ val selectedSourceFolder = context.selectedSourceFolder.toNioPath()
+
+ filePaths.forEach {
+ resolveAndCreateOrUpdateFile(selectedSourceFolder, it.zipFilePath, it.fileContent)
+ it.changeApplied = true
+ }
+ }
+
+// Suppressing because applyDeleteFiles needs to be a suspend function in order to be tested
+ @Suppress("RedundantSuspendModifier")
+ suspend fun applyDeleteFiles(
+ deletedFiles: List,
+ ) {
+ val selectedSourceFolder = context.selectedSourceFolder.toNioPath()
+
+ deletedFiles.forEach {
+ resolveAndDeleteFile(selectedSourceFolder, it.zipFilePath)
+ it.changeApplied = true
}
}
@@ -185,13 +212,14 @@ class Session(val tabID: String, val project: Project) {
return
}
- val codeResultMessageId = this._codeResultMessageId
- if (codeResultMessageId != null) {
- messenger.updateFileComponent(this.tabID, filePaths, deletedFiles, codeResultMessageId, disableFileActions = true)
- }
+ updateFilesPaths(filePaths, deletedFiles, messenger, disableFileActions = true)
this._codeResultMessageId = null
}
+ fun sendMetricDataTelemetry(operationName: MetricDataOperationName, result: MetricDataResult) {
+ featureDevService.sendFeatureDevMetricData(operationName.toString(), result.toString())
+ }
+
suspend fun send(msg: String): Interaction {
// When the task/"thing to do" hasn't been set yet, we want it to be the incoming message
if (task.isEmpty() && msg.isNotEmpty()) {
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/session/SessionState.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/session/SessionState.kt
index c1248a74dd..936b7a88d3 100644
--- a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/session/SessionState.kt
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/session/SessionState.kt
@@ -14,5 +14,5 @@ interface SessionState {
var currentIteration: Int?
var approach: String
var diffMetricsProcessed: DiffMetricsProcessed
- suspend fun interact(action: SessionStateAction): SessionStateInteraction
+ suspend fun interact(action: SessionStateAction): SessionStateInteraction
}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/session/SessionStateTypes.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/session/SessionStateTypes.kt
index 0567bf980a..a8fd7af3f8 100644
--- a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/session/SessionStateTypes.kt
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/session/SessionStateTypes.kt
@@ -20,8 +20,8 @@ data class Interaction(
val interactionSucceeded: Boolean,
)
-data class SessionStateInteraction(
- val nextState: SessionState? = null,
+data class SessionStateInteraction(
+ val nextState: T? = null,
val interaction: Interaction,
)
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/util/FeatureDevService.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/util/FeatureDevService.kt
index 123105e850..5a8e011aa6 100644
--- a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/util/FeatureDevService.kt
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/util/FeatureDevService.kt
@@ -232,6 +232,19 @@ class FeatureDevService(val proxyClient: FeatureDevClient, val project: Project)
}
}
+ fun sendFeatureDevMetricData(operationName: String, result: String) {
+ val sendFeatureDevTelemetryEventResponse: SendTelemetryEventResponse
+ try {
+ sendFeatureDevTelemetryEventResponse = proxyClient.sendFeatureDevMetricData(operationName, result)
+ val requestId = sendFeatureDevTelemetryEventResponse.responseMetadata().requestId()
+ logger.debug {
+ "$FEATURE_NAME: succesfully sent feature dev metric data: OperationName: $operationName Result: $result RequestId: $requestId"
+ }
+ } catch (e: Exception) {
+ logger.warn(e) { "$FEATURE_NAME: failed to send feature dev metric data" }
+ }
+ }
+
fun sendFeatureDevCodeGenerationEvent(conversationId: String, linesOfCodeGenerated: Int, charactersOfCodeGenerated: Int) {
val sendFeatureDevTelemetryEventResponse: SendTelemetryEventResponse
try {
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/cwc/clients/chat/model/Responses.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/cwc/clients/chat/model/Responses.kt
index 2cb7878e7d..6a246ca45f 100644
--- a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/cwc/clients/chat/model/Responses.kt
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/cwc/clients/chat/model/Responses.kt
@@ -23,6 +23,9 @@ enum class FollowUpType {
Generated,
StopCodeTransform,
NewCodeTransform,
+ CreateDocumentation,
+ NewCodeScan,
+ ViewDiff,
}
data class SuggestedFollowUp(
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/cwc/commands/GenerateUnitTestsAction.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/cwc/commands/GenerateUnitTestsAction.kt
index 2c43378d45..e48cbe4df3 100644
--- a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/cwc/commands/GenerateUnitTestsAction.kt
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/cwc/commands/GenerateUnitTestsAction.kt
@@ -3,20 +3,4 @@
package software.aws.toolkits.jetbrains.services.cwc.commands
-import com.intellij.openapi.actionSystem.ActionUpdateThread
-import com.intellij.openapi.actionSystem.AnActionEvent
-import com.intellij.openapi.actionSystem.CommonDataKeys
-import software.aws.toolkits.jetbrains.core.credentials.AwsBearerTokenConnection
-import software.aws.toolkits.jetbrains.core.credentials.ToolkitConnectionManager
-import software.aws.toolkits.jetbrains.core.credentials.pinning.QConnection
-import software.aws.toolkits.jetbrains.core.credentials.sono.isInternalUser
-
-class GenerateUnitTestsAction : CustomAction(EditorContextCommand.GenerateUnitTests) {
- override fun getActionUpdateThread() = ActionUpdateThread.BGT
-
- override fun update(e: AnActionEvent) {
- val project = e.getData(CommonDataKeys.PROJECT) ?: return
- val connection = ToolkitConnectionManager.getInstance(project).activeConnectionForFeature(QConnection.getInstance()) as? AwsBearerTokenConnection
- e.presentation.isEnabledAndVisible = isInternalUser(connection?.startUrl)
- }
-}
+class GenerateUnitTestsAction : CustomAction(EditorContextCommand.GenerateUnitTests)
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/cwc/commands/codescan/actions/CodeScanCompleteAction.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/cwc/commands/codescan/actions/CodeScanCompleteAction.kt
new file mode 100644
index 0000000000..41160fe348
--- /dev/null
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/cwc/commands/codescan/actions/CodeScanCompleteAction.kt
@@ -0,0 +1,21 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.cwc.commands.codescan.actions
+
+import com.intellij.openapi.actionSystem.AnAction
+import com.intellij.openapi.actionSystem.AnActionEvent
+import com.intellij.openapi.actionSystem.CommonDataKeys
+import com.intellij.openapi.components.service
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.commands.CodeScanMessageListener
+import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererConstants.scanResultsKey
+import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererConstants.scanScopeKey
+
+class CodeScanCompleteAction : AnAction() {
+ override fun actionPerformed(e: AnActionEvent) {
+ val project = e.getData(CommonDataKeys.PROJECT) ?: return
+ val result = e.getData(scanResultsKey)
+ val scope = e.getData(scanScopeKey) ?: return
+ service().onScanResult(result, scope, project)
+ }
+}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/cwc/controller/ChatController.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/cwc/controller/ChatController.kt
index 84ba6543cc..7d7b28b664 100644
--- a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/cwc/controller/ChatController.kt
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/cwc/controller/ChatController.kt
@@ -32,15 +32,16 @@ import kotlinx.coroutines.job
import kotlinx.coroutines.withContext
import migration.software.aws.toolkits.jetbrains.services.codewhisperer.customization.CodeWhispererModelConfigurator
import software.amazon.awssdk.services.codewhispererstreaming.model.UserIntent
+import software.aws.toolkits.core.utils.debug
import software.aws.toolkits.core.utils.getLogger
import software.aws.toolkits.core.utils.info
import software.aws.toolkits.core.utils.warn
import software.aws.toolkits.jetbrains.core.coroutines.EDT
-import software.aws.toolkits.jetbrains.core.credentials.sono.isInternalUser
import software.aws.toolkits.jetbrains.services.amazonq.CHAT_IMPLICIT_PROJECT_CONTEXT_TIMEOUT
import software.aws.toolkits.jetbrains.services.amazonq.apps.AmazonQAppInitContext
import software.aws.toolkits.jetbrains.services.amazonq.auth.AuthController
import software.aws.toolkits.jetbrains.services.amazonq.auth.AuthNeededState
+import software.aws.toolkits.jetbrains.services.amazonq.messages.AmazonQMessage
import software.aws.toolkits.jetbrains.services.amazonq.messages.MessagePublisher
import software.aws.toolkits.jetbrains.services.amazonq.onboarding.OnboardingPageInteraction
import software.aws.toolkits.jetbrains.services.amazonq.onboarding.OnboardingPageInteractionType
@@ -84,6 +85,12 @@ import software.aws.toolkits.telemetry.CwsprChatCommandType
import java.time.Instant
import java.util.UUID
+data class TestCommandMessage(
+ val sender: String = "codetest",
+ val command: String = "test",
+ val type: String = "addAnswer",
+) : AmazonQMessage
+
class ChatController private constructor(
private val context: AmazonQAppInitContext,
private val chatSessionStorage: ChatSessionStorage,
@@ -132,28 +139,25 @@ class ChatController private constructor(
var shouldAddIndexInProgressMessage: Boolean = false
var shouldUseWorkspaceContext: Boolean = false
val startUrl = getStartUrl(context.project)
- val isInternalUser = isInternalUser(startUrl)
if (prompt.contains("@workspace")) {
if (CodeWhispererSettings.getInstance().isProjectContextEnabled()) {
shouldUseWorkspaceContext = true
prompt = prompt.replace("@workspace", "")
val projectContextController = ProjectContextController.getInstance(context.project)
- queryResult = projectContextController.query(prompt, timeout = null)
+ queryResult = projectContextController.queryChat(prompt, timeout = null)
if (!projectContextController.getProjectContextIndexComplete()) shouldAddIndexInProgressMessage = true
logger.info { "project context relevant document count: ${queryResult.size}" }
} else {
sendOpenSettingsMessage(message.tabId)
}
- } else if (
- CodeWhispererSettings.getInstance().isProjectContextEnabled() &&
- isInternalUser &&
- ProjectContextController.getInstance(context.project).getProjectContextIndexComplete()
- ) {
- // if user does not have @workspace in the prompt, but user is Amazon internal
- // add project context by default
- val projectContextController = ProjectContextController.getInstance(context.project)
- queryResult = projectContextController.query(prompt, timeout = CHAT_IMPLICIT_PROJECT_CONTEXT_TIMEOUT)
+ } else if (CodeWhispererSettings.getInstance().isProjectContextEnabled()) {
+ if (ProjectContextController.getInstance(context.project).getProjectContextIndexComplete()) {
+ val projectContextController = ProjectContextController.getInstance(context.project)
+ queryResult = projectContextController.queryChat(prompt, timeout = CHAT_IMPLICIT_PROJECT_CONTEXT_TIMEOUT)
+ } else {
+ logger.debug { "skipping implicit workspace context as index is not ready" }
+ }
}
handleChat(
@@ -161,7 +165,7 @@ class ChatController private constructor(
triggerId = triggerId,
message = prompt,
activeFileContext = contextExtractor.extractContextForTrigger(ExtractionTriggerType.ChatMessage),
- userIntent = intentRecognizer.getUserIntentFromPromptChatMessage(message.chatMessage, startUrl),
+ userIntent = intentRecognizer.getUserIntentFromPromptChatMessage(message.chatMessage),
TriggerType.Click,
projectContextQueryResult = queryResult,
shouldAddIndexInProgressMessage = shouldAddIndexInProgressMessage,
@@ -302,7 +306,7 @@ class ChatController private constructor(
}
override suspend fun processCodeScanIssueAction(message: CodeScanIssueActionMessage) {
- logger.info { "Code Scan Explain issue with Q message received for issue: ${message.issue["title"]}" }
+ logger.info { "Code Review Explain issue with Q message received for issue: ${message.issue["title"]}" }
// Extract context
val fileContext = contextExtractor.extractContextForTrigger(ExtractionTriggerType.CodeScanButton)
val triggerId = UUID.randomUUID().toString()
@@ -340,15 +344,15 @@ class ChatController private constructor(
)
return
}
-
- // Create prompt
- val prompt = if (EditorContextCommand.GenerateUnitTests == message.command) {
- "${message.command.verb} the following part of my code for me: $codeSelection"
+ if (message.command == EditorContextCommand.GenerateUnitTests) {
+ // Publish an event to "codetest" tab with command as "test" and type as "addAnswer"
+ val messageToPublish = TestCommandMessage()
+ context.messagesFromAppToUi.publish(messageToPublish)
} else {
- "${message.command} the following part of my code for me: $codeSelection"
+ // Create prompt
+ val prompt = "${message.command} the following part of my code for me: $codeSelection"
+ processPromptActions(prompt, message, triggerId, fileContext)
}
-
- processPromptActions(prompt, message, triggerId, fileContext)
}
private suspend fun processPromptActions(
@@ -388,7 +392,11 @@ class ChatController private constructor(
}
override suspend fun processLinkClick(message: IncomingCwcMessage.ClickedLink) {
- BrowserUtil.browse(message.link)
+ processLinkClick(message, message.link)
+ }
+
+ private suspend fun processLinkClick(message: IncomingCwcMessage, link: String) {
+ BrowserUtil.browse(link)
telemetryHelper.recordInteractWithMessage(message)
}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/cwc/controller/chat/userIntent/UserIntentRecognizer.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/cwc/controller/chat/userIntent/UserIntentRecognizer.kt
index 39725dfc46..6d910aa729 100644
--- a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/cwc/controller/chat/userIntent/UserIntentRecognizer.kt
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/cwc/controller/chat/userIntent/UserIntentRecognizer.kt
@@ -4,7 +4,6 @@
package software.aws.toolkits.jetbrains.services.cwc.controller.chat.userIntent
import software.amazon.awssdk.services.codewhispererstreaming.model.UserIntent
-import software.aws.toolkits.jetbrains.core.credentials.sono.isInternalUser
import software.aws.toolkits.jetbrains.services.amazonq.onboarding.OnboardingPageInteraction
import software.aws.toolkits.jetbrains.services.amazonq.onboarding.OnboardingPageInteractionType
import software.aws.toolkits.jetbrains.services.cwc.clients.chat.model.FollowUpType
@@ -21,12 +20,12 @@ class UserIntentRecognizer {
EditorContextCommand.SendToPrompt -> null
}
- fun getUserIntentFromPromptChatMessage(prompt: String, startUrl: String?) = when {
+ fun getUserIntentFromPromptChatMessage(prompt: String) = when {
prompt.startsWith("Explain") -> UserIntent.EXPLAIN_CODE_SELECTION
prompt.startsWith("Refactor") -> UserIntent.SUGGEST_ALTERNATE_IMPLEMENTATION
prompt.startsWith("Fix") -> UserIntent.APPLY_COMMON_BEST_PRACTICES
prompt.startsWith("Optimize") -> UserIntent.IMPROVE_CODE
- prompt.startsWith("Generate unit tests") && isInternalUser(startUrl) -> UserIntent.GENERATE_UNIT_TESTS
+ prompt.startsWith("Generate unit tests") -> UserIntent.GENERATE_UNIT_TESTS
else -> null
}
@@ -41,6 +40,9 @@ class UserIntentRecognizer {
FollowUpType.Generated -> null
FollowUpType.StopCodeTransform -> null
FollowUpType.NewCodeTransform -> null
+ FollowUpType.CreateDocumentation -> null
+ FollowUpType.NewCodeScan -> null
+ FollowUpType.ViewDiff -> UserIntent.GENERATE_UNIT_TESTS
}
fun getUserIntentFromOnboardingPageInteraction(interaction: OnboardingPageInteraction) = when (interaction.type) {
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/cwc/editor/context/focusArea/FocusAreaContextExtractor.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/cwc/editor/context/focusArea/FocusAreaContextExtractor.kt
index b494ddbae4..255b21aa71 100644
--- a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/cwc/editor/context/focusArea/FocusAreaContextExtractor.kt
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/cwc/editor/context/focusArea/FocusAreaContextExtractor.kt
@@ -180,7 +180,7 @@ class FocusAreaContextExtractor(private val fqnWebviewAdapter: FqnWebviewAdapter
val startOffset = 0.coerceAtLeast(offset - halfMaxCharacters)
val endOffset = fileText.length.coerceAtMost(offset + halfMaxCharacters)
- // Adjust the start and end offsets if necessary to ensure a total of 10k characters
+ // Adjust the start and end offsets if necessary to ensure a total of 40k characters
val excessCharacters = maxCharacters - (endOffset - startOffset)
val adjustedStartOffset = 0.coerceAtLeast(startOffset - excessCharacters)
val adjustedEndOffset = fileText.length.coerceAtMost(endOffset + excessCharacters)
@@ -198,7 +198,7 @@ class FocusAreaContextExtractor(private val fqnWebviewAdapter: FqnWebviewAdapter
}
companion object {
- const val MAX_LENGTH = 10000
+ const val MAX_LENGTH = 40000
}
}
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/cwc/inline/InlineChatController.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/cwc/inline/InlineChatController.kt
index 22a3dd4640..bcadef007a 100644
--- a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/cwc/inline/InlineChatController.kt
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/cwc/inline/InlineChatController.kt
@@ -630,7 +630,7 @@ class InlineChatController(
tabId = tabId,
message = message,
activeFileContext = fileContext,
- userIntent = intentRecognizer.getUserIntentFromPromptChatMessage(message, null),
+ userIntent = intentRecognizer.getUserIntentFromPromptChatMessage(message),
triggerType = TriggerType.Inline,
customization = CodeWhispererModelConfigurator.getInstance().activeCustomization(project),
relevantTextDocuments = emptyList(),
diff --git a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/cwc/messages/CwcMessage.kt b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/cwc/messages/CwcMessage.kt
index 958db5076b..85b3075c96 100644
--- a/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/cwc/messages/CwcMessage.kt
+++ b/plugins/amazonq/chat/jetbrains-community/src/software/aws/toolkits/jetbrains/services/cwc/messages/CwcMessage.kt
@@ -125,6 +125,10 @@ sealed interface IncomingCwcMessage : CwcMessage {
val type: FocusType,
) : IncomingCwcMessage
+ data class OpenUserGuide(
+ val userGuideLink: String,
+ ) : IncomingCwcMessage
+
data class ClickedLink(
@JsonProperty("command") val type: LinkType,
@JsonProperty("tabID") override val tabId: String,
@@ -200,6 +204,7 @@ data class FollowUp(
val type: FollowUpType,
val pillText: String,
val prompt: String,
+ val status: String? = null,
)
data class Suggestion(
diff --git a/plugins/amazonq/chat/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/amazonq/workspace/context/ProjectContextProviderTest.kt b/plugins/amazonq/chat/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/amazonq/workspace/context/ProjectContextProviderTest.kt
index 8853bb4a68..700f3a2024 100644
--- a/plugins/amazonq/chat/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/amazonq/workspace/context/ProjectContextProviderTest.kt
+++ b/plugins/amazonq/chat/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/amazonq/workspace/context/ProjectContextProviderTest.kt
@@ -41,6 +41,7 @@ import software.aws.toolkits.jetbrains.services.amazonq.project.EncoderServer
import software.aws.toolkits.jetbrains.services.amazonq.project.IndexRequest
import software.aws.toolkits.jetbrains.services.amazonq.project.IndexUpdateMode
import software.aws.toolkits.jetbrains.services.amazonq.project.InlineBm25Chunk
+import software.aws.toolkits.jetbrains.services.amazonq.project.InlineContextTarget
import software.aws.toolkits.jetbrains.services.amazonq.project.LspMessage
import software.aws.toolkits.jetbrains.services.amazonq.project.ProjectContextProvider
import software.aws.toolkits.jetbrains.services.amazonq.project.QueryChatRequest
@@ -237,13 +238,13 @@ class ProjectContextProviderTest {
@Test
fun `queryInline should send correct encrypted request to lsp`() = runTest {
sut = ProjectContextProvider(project, encoderServer, this)
- sut.queryInline("foo", "Foo.java")
+ sut.queryInline("foo", "Foo.java", InlineContextTarget.CODEMAP)
advanceUntilIdle()
- val request = QueryInlineCompletionRequest("foo", "Foo.java")
+ val request = QueryInlineCompletionRequest("foo", "Foo.java", "codemap")
val requestJson = mapper.writeValueAsString(request)
- assertThat(mapper.readTree(requestJson)).isEqualTo(mapper.readTree("""{ "query": "foo", "filePath": "Foo.java" }"""))
+ assertThat(mapper.readTree(requestJson)).isEqualTo(mapper.readTree("""{ "query": "foo", "filePath": "Foo.java", "target": "codemap" }"""))
val encryptedRequest = encoderServer.encrypt(requestJson)
wireMock.verify(
@@ -315,7 +316,7 @@ class ProjectContextProviderTest {
)
assertThrows {
- sut.queryInline("foo", "filepath")
+ sut.queryInline("foo", "filepath", InlineContextTarget.CODEMAP)
advanceUntilIdle()
}
}
@@ -326,7 +327,7 @@ class ProjectContextProviderTest {
fun `query inline should return deserialized bm25 chunks`() = runTest {
sut = ProjectContextProvider(project, encoderServer, this)
advanceUntilIdle()
- val r = sut.queryInline("foo", "filepath")
+ val r = sut.queryInline("foo", "filepath", InlineContextTarget.CODEMAP)
assertThat(r).hasSize(3)
assertThat(r[0]).isEqualTo(
InlineBm25Chunk(
@@ -374,7 +375,7 @@ class ProjectContextProviderTest {
// it won't throw if it's executed within TestDispatcher context
withContext(getCoroutineBgContext()) {
- sut.queryInline("foo", "bar")
+ sut.queryInline("foo", "bar", InlineContextTarget.CODEMAP)
}
advanceUntilIdle()
diff --git a/plugins/amazonq/chat/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/FeatureDevSessionContextTest.kt b/plugins/amazonq/chat/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/FeatureDevSessionContextTest.kt
index a46cb90886..e02e21218d 100644
--- a/plugins/amazonq/chat/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/FeatureDevSessionContextTest.kt
+++ b/plugins/amazonq/chat/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/FeatureDevSessionContextTest.kt
@@ -12,20 +12,34 @@ import org.mockito.kotlin.mock
import org.mockito.kotlin.whenever
import software.aws.toolkits.jetbrains.services.amazonq.FeatureDevSessionContext
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.FeatureDevTestBase
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.SessionStateConfig
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.util.FeatureDevService
+import software.aws.toolkits.jetbrains.settings.CodeWhispererSettings
+import software.aws.toolkits.jetbrains.utils.rules.HeavyJavaCodeInsightTestFixtureRule
+import software.aws.toolkits.jetbrains.utils.rules.addFileToModule
+import java.util.zip.ZipFile
+
+class FeatureDevSessionContextTest : FeatureDevTestBase(HeavyJavaCodeInsightTestFixtureRule()) {
+
+ private fun addFilesToProjectModule(vararg path: String) {
+ val module = projectRule.module
+ path.forEach { projectRule.fixture.addFileToModule(module, it, it) }
+ }
-class FeatureDevSessionContextTest : FeatureDevTestBase() {
@Rule
@JvmField
val ruleChain = RuleChain(projectRule, disposableRule)
private lateinit var featureDevSessionContext: FeatureDevSessionContext
private lateinit var featureDevService: FeatureDevService
+ private lateinit var config: SessionStateConfig
@Before
fun setUp() {
+ val conversationId = "test-conversation"
featureDevService = mock()
whenever(featureDevService.project).thenReturn(projectRule.project)
featureDevSessionContext = FeatureDevSessionContext(featureDevService.project, 1024)
+ config = SessionStateConfig(conversationId, featureDevSessionContext, featureDevService)
}
@Test
@@ -40,6 +54,7 @@ class FeatureDevSessionContextTest : FeatureDevTestBase() {
fun testWithValidFile() {
val ktFile = mock()
whenever(ktFile.extension).thenReturn("kt")
+ whenever(ktFile.path).thenReturn("code.kt")
assertTrue(featureDevSessionContext.isFileExtensionAllowed(ktFile))
}
@@ -49,4 +64,60 @@ class FeatureDevSessionContextTest : FeatureDevTestBase() {
whenever(txtFile.extension).thenReturn("mp4")
assertFalse(featureDevSessionContext.isFileExtensionAllowed(txtFile))
}
+
+ @Test
+ fun testAllowedFilePath() {
+ val allowedPaths = listOf("build.gradle", "gradle.properties", ".mvn/wrapper/maven-wrapper.properties")
+ allowedPaths.forEach({
+ val txtFile = mock()
+ whenever(txtFile.path).thenReturn(it)
+ whenever(txtFile.extension).thenReturn(it.split(".").last())
+ assertTrue(featureDevSessionContext.isFileExtensionAllowed(txtFile))
+ })
+ }
+
+ @Test
+ fun testZipProject() {
+ addFilesToProjectModule(
+ ".gradle/cached.jar",
+ "src/MyClass.java",
+ "gradlew",
+ "gradlew.bat",
+ "README.md",
+ "settings.gradle",
+ "build.gradle",
+ "gradle/wrapper/gradle-wrapper.properties",
+ "gradle/wrapper/gradle-wrapper.jar",
+ ".idea/ref",
+ )
+ val module = projectRule.module
+ projectRule.fixture.addFileToModule(module, "/.gitignore", "node_modules\n.idea\n.vscode\n.DS_Store\ngradle/wrapper/gradle-wrapper.jar")
+
+ val isAutoBuildFeatureEnabled = CodeWhispererSettings.getInstance().isAutoBuildFeatureEnabled(config.repoContext.getWorkspaceRoot())
+ val zipResult = featureDevSessionContext.getProjectZip(isAutoBuildFeatureEnabled = isAutoBuildFeatureEnabled)
+ val zipPath = zipResult.payload.path
+
+ val zippedFiles = mutableSetOf()
+ ZipFile(zipPath).use { zipFile ->
+ for (entry in zipFile.entries()) {
+ if (!entry.name.endsWith("/")) {
+ zippedFiles.add(entry.name)
+ }
+ }
+ }
+
+ val expectedFiles = setOf(
+ "src/MyClass.java",
+ "gradlew",
+ "gradlew.bat",
+ "README.md",
+ "settings.gradle",
+ "build.gradle",
+ "gradle/wrapper/gradle-wrapper.properties",
+ "gradle/wrapper/gradle-wrapper.jar",
+ ".gitignore",
+ )
+
+ assertTrue(zippedFiles == expectedFiles)
+ }
}
diff --git a/plugins/amazonq/chat/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/controller/FeatureDevControllerTest.kt b/plugins/amazonq/chat/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/controller/FeatureDevControllerTest.kt
index afb1d3a2c1..d463e48f5a 100644
--- a/plugins/amazonq/chat/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/controller/FeatureDevControllerTest.kt
+++ b/plugins/amazonq/chat/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/controller/FeatureDevControllerTest.kt
@@ -22,20 +22,30 @@ import org.junit.After
import org.junit.Before
import org.junit.Rule
import org.junit.Test
+import org.junit.jupiter.api.assertThrows
import org.mockito.kotlin.any
import org.mockito.kotlin.doNothing
import org.mockito.kotlin.doReturn
+import org.mockito.kotlin.inOrder
import org.mockito.kotlin.mock
import org.mockito.kotlin.reset
import org.mockito.kotlin.spy
import org.mockito.kotlin.times
import org.mockito.kotlin.whenever
+import software.aws.toolkits.jetbrains.common.util.selectFolder
import software.aws.toolkits.jetbrains.services.amazonq.FeatureDevSessionContext
import software.aws.toolkits.jetbrains.services.amazonq.apps.AmazonQAppInitContext
import software.aws.toolkits.jetbrains.services.amazonq.auth.AuthController
import software.aws.toolkits.jetbrains.services.amazonq.auth.AuthNeededStates
import software.aws.toolkits.jetbrains.services.amazonq.messages.MessagePublisher
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.EmptyPatchException
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.FeatureDevTestBase
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.GuardrailsException
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.MetricDataOperationName
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.MetricDataResult
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.NoChangeRequiredException
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.PromptRefusalException
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.ThrottlingException
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.clients.FeatureDevClient
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.messages.FeatureDevMessageType
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.messages.FollowUp
@@ -49,6 +59,7 @@ import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.messages.sendC
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.messages.sendSystemPrompt
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.messages.sendUpdatePlaceholder
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.messages.updateFileComponent
+import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.CodeGenerationState
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.DeletedFileInfo
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.DiffMetricsProcessed
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.session.Interaction
@@ -62,7 +73,6 @@ import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.util.Cancellat
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.util.FeatureDevService
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.util.InsertAction
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.util.getFollowUpOptions
-import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.util.selectFolder
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.util.uploadArtifactToS3
import software.aws.toolkits.resources.message
import software.aws.toolkits.telemetry.AmazonqTelemetry
@@ -241,7 +251,9 @@ class FeatureDevControllerTest : FeatureDevTestBase() {
),
)
- doReturn(Unit).`when`(spySession).insertChanges(any(), any(), any(), any())
+ doReturn(Unit).whenever(spySession).insertChanges(any(), any(), any())
+ doReturn(Unit).whenever(spySession).insertNewFiles(any())
+ doReturn(Unit).whenever(spySession).applyDeleteFiles(any())
spySession.preloader(messenger)
controller.processFollowupClickedMessage(message)
@@ -249,7 +261,7 @@ class FeatureDevControllerTest : FeatureDevTestBase() {
mockitoVerify(
spySession,
times(1),
- ).insertChanges(listOf(newFileContents[0]), listOf(deletedFiles[0]), testReferences, messenger) // insert changes for only non rejected files
+ ).insertChanges(newFileContents, deletedFiles, testReferences) // updates for all files
coVerifyOrder {
AmazonqTelemetry.isAcceptedCodeChanges(
amazonqNumberOfFilesAccepted = 2.0, // it should be 2 files per test setup
@@ -257,6 +269,16 @@ class FeatureDevControllerTest : FeatureDevTestBase() {
enabled = true,
createTime = any(),
)
+
+ // insert changes for only non rejected files
+ spySession.insertNewFiles(listOf(newFileContents[0]))
+ spySession.applyDeleteFiles(listOf(deletedFiles[0]))
+
+ spySession.updateFilesPaths(
+ filePaths = newFileContents,
+ deletedFiles = deletedFiles,
+ messenger
+ )
messenger.sendAnswer(
tabId = testTabId,
message = message("amazonqFeatureDev.code_generation.updated_code"),
@@ -412,6 +434,114 @@ class FeatureDevControllerTest : FeatureDevTestBase() {
}
}
+ @Test
+ fun `test handleChat onCodeGeneration sends success metrics`() = runTest {
+ val mockSession = mock()
+ val featureDevService = mockk()
+ val repoContext = mock()
+ val sessionStateConfig = SessionStateConfig(testConversationId, repoContext, featureDevService)
+ val mockInteraction = mock()
+ whenever(mockSession.send(userMessage)).thenReturn(mockInteraction)
+ whenever(mockSession.sessionState).thenReturn(
+ PrepareCodeGenerationState(
+ testTabId,
+ CancellationTokenSource(),
+ "test-command",
+ sessionStateConfig,
+ newFileContents,
+ deletedFiles,
+ testReferences,
+ testUploadId,
+ 0,
+ messenger,
+ diffMetricsProcessed = DiffMetricsProcessed(HashSet(), HashSet()),
+ ),
+ )
+
+ controller.onCodeGeneration(mockSession, userMessage, testTabId)
+
+ val mockInOrder = inOrder(mockSession)
+
+ mockInOrder.verify(mockSession).sendMetricDataTelemetry(
+ MetricDataOperationName.StartCodeGeneration,
+ MetricDataResult.Success
+
+ )
+ mockInOrder.verify(mockSession).sendMetricDataTelemetry(
+ MetricDataOperationName.EndCodeGeneration,
+ MetricDataResult.Success
+ )
+ }
+
+ @Test
+ fun `test handleChat onCodeGeneration sends correct failure metrics for different errors`() = runTest {
+ data class ErrorTestCase(
+ val error: Exception,
+ val expectedMetricResult: MetricDataResult,
+ )
+
+ val testCases = listOf(
+ ErrorTestCase(
+ EmptyPatchException("EmptyPatchException", "Empty patch"),
+ MetricDataResult.LlmFailure
+ ),
+ ErrorTestCase(
+ GuardrailsException(operation = "GenerateCode", desc = "Failed guardrails"),
+ MetricDataResult.Error
+ ),
+ ErrorTestCase(
+ PromptRefusalException(operation = "GenerateCode", desc = "Prompt refused"),
+ MetricDataResult.Error
+ ),
+ ErrorTestCase(
+ NoChangeRequiredException(operation = "GenerateCode", desc = "No changes needed"),
+ MetricDataResult.Error
+ ),
+ ErrorTestCase(
+ ThrottlingException(operation = "GenerateCode", desc = "Request throttled"),
+ MetricDataResult.Error
+ ),
+ ErrorTestCase(
+ RuntimeException("Unknown error"),
+ MetricDataResult.Fault
+ )
+ )
+
+ testCases.forEach { (error, expectedResult) ->
+ val mockSession = mock()
+ whenever(mockSession.send(userMessage)).thenThrow(error)
+ whenever(mockSession.sessionState).thenReturn(
+ CodeGenerationState(
+ testTabId,
+ "",
+ mock(),
+ testUploadId,
+ 0,
+ 0.0,
+ messenger,
+ token = CancellationTokenSource(),
+ diffMetricsProcessed = DiffMetricsProcessed(HashSet(), HashSet())
+ )
+ )
+
+ assertThrows {
+ controller.onCodeGeneration(mockSession, userMessage, testTabId)
+ }
+
+ val mockInOrder = inOrder(mockSession)
+
+ mockInOrder.verify(mockSession).sendMetricDataTelemetry(
+ MetricDataOperationName.StartCodeGeneration,
+ MetricDataResult.Success
+
+ )
+ mockInOrder.verify(mockSession).sendMetricDataTelemetry(
+ MetricDataOperationName.EndCodeGeneration,
+ expectedResult
+ )
+ }
+ }
+
@Test
fun `test processFileClicked handles file rejection`() =
runTest {
@@ -469,7 +599,7 @@ class FeatureDevControllerTest : FeatureDevTestBase() {
),
)
doReturn(testConversationId).`when`(spySession).conversationId
- doReturn(Unit).`when`(spySession).insertChanges(any(), any(), any(), any())
+ doReturn(Unit).`when`(spySession).insertChanges(any(), any(), any())
mockkObject(AmazonqTelemetry)
every {
@@ -492,7 +622,7 @@ class FeatureDevControllerTest : FeatureDevTestBase() {
mockitoVerify(
spySession,
times(1),
- ).insertChanges(listOf(newFileContents[0]), listOf(), testReferences, messenger)
+ ).insertChanges(listOf(newFileContents[0]), listOf(), testReferences)
// Does not continue automatically, because files are remaining:
mockitoVerify(
@@ -526,7 +656,7 @@ class FeatureDevControllerTest : FeatureDevTestBase() {
),
)
doReturn(testConversationId).`when`(spySession).conversationId
- doReturn(Unit).`when`(spySession).insertChanges(any(), any(), any(), any())
+ doReturn(Unit).`when`(spySession).insertChanges(any(), any(), any())
mockkObject(AmazonqTelemetry)
every {
@@ -571,7 +701,7 @@ class FeatureDevControllerTest : FeatureDevTestBase() {
whenever(featureDevClient.sendFeatureDevTelemetryEvent(any())).thenReturn(exampleSendTelemetryEventResponse)
whenever(chatSessionStorage.getSession(any(), any())).thenReturn(spySession)
- mockkStatic("software.aws.toolkits.jetbrains.services.amazonqFeatureDev.util.FileUtilsKt")
+ mockkStatic("software.aws.toolkits.jetbrains.common.util.FileUtilsKt")
every { selectFolder(any(), any()) } returns null
spySession.preloader(messenger)
@@ -602,7 +732,7 @@ class FeatureDevControllerTest : FeatureDevTestBase() {
whenever(featureDevClient.sendFeatureDevTelemetryEvent(any())).thenReturn(exampleSendTelemetryEventResponse)
whenever(chatSessionStorage.getSession(any(), any())).thenReturn(spySession)
- mockkStatic("software.aws.toolkits.jetbrains.services.amazonqFeatureDev.util.FileUtilsKt")
+ mockkStatic("software.aws.toolkits.jetbrains.common.util.FileUtilsKt")
every { selectFolder(any(), any()) } returns LightVirtualFile("/path")
spySession.preloader(messenger)
@@ -639,7 +769,7 @@ class FeatureDevControllerTest : FeatureDevTestBase() {
whenever(chatSessionStorage.getSession(any(), any())).thenReturn(spySession)
val folder = LightVirtualFile("${spySession.context.projectRoot.name}/path/to/sub/folder")
- mockkStatic("software.aws.toolkits.jetbrains.services.amazonqFeatureDev.util.FileUtilsKt")
+ mockkStatic("software.aws.toolkits.jetbrains.common.util.FileUtilsKt")
every { selectFolder(any(), any()) } returns folder
spySession.preloader(messenger)
diff --git a/plugins/amazonq/chat/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/session/SessionTest.kt b/plugins/amazonq/chat/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/session/SessionTest.kt
index 550058231c..104cd06461 100644
--- a/plugins/amazonq/chat/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/session/SessionTest.kt
+++ b/plugins/amazonq/chat/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/amazonqFeatureDev/session/SessionTest.kt
@@ -23,11 +23,11 @@ import org.mockito.kotlin.mock
import org.mockito.kotlin.times
import org.mockito.kotlin.verify
import org.mockito.kotlin.whenever
+import software.aws.toolkits.jetbrains.common.util.resolveAndCreateOrUpdateFile
+import software.aws.toolkits.jetbrains.common.util.resolveAndDeleteFile
import software.aws.toolkits.jetbrains.services.amazonq.messages.MessagePublisher
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.FeatureDevTestBase
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.clients.FeatureDevClient
-import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.util.resolveAndCreateOrUpdateFile
-import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.util.resolveAndDeleteFile
import software.aws.toolkits.jetbrains.services.cwc.controller.ReferenceLogController
import kotlin.io.path.Path
@@ -72,7 +72,7 @@ class SessionTest : FeatureDevTestBase() {
mockkObject(ReferenceLogController)
every { ReferenceLogController.addReferenceLog(any(), any()) } just runs
- mockkStatic("software.aws.toolkits.jetbrains.services.amazonqFeatureDev.util.FileUtilsKt")
+ mockkStatic("software.aws.toolkits.jetbrains.common.util.FileUtilsKt")
every { resolveAndDeleteFile(any(), any()) } just runs
every { resolveAndCreateOrUpdateFile(any(), any(), any()) } just runs
@@ -83,7 +83,7 @@ class SessionTest : FeatureDevTestBase() {
whenever(session.context.selectedSourceFolder.toNioPath()).thenReturn(Path(""))
runBlocking {
- session.insertChanges(mockNewFile, mockDeletedFile, emptyList(), messenger)
+ session.insertChanges(mockNewFile, mockDeletedFile, emptyList())
}
verify(exactly = 1) { resolveAndDeleteFile(any(), "deletedTest.ts") }
diff --git a/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/ArtifactHandler.kt b/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/ArtifactHandler.kt
index b0608d1f07..22bc863767 100644
--- a/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/ArtifactHandler.kt
+++ b/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/ArtifactHandler.kt
@@ -31,7 +31,6 @@ import software.aws.toolkits.jetbrains.services.amazonq.CODE_TRANSFORM_TROUBLESH
import software.aws.toolkits.jetbrains.services.amazonq.CODE_TRANSFORM_TROUBLESHOOT_DOC_DOWNLOAD_EXPIRED
import software.aws.toolkits.jetbrains.services.codemodernizer.client.GumbyClient
import software.aws.toolkits.jetbrains.services.codemodernizer.commands.CodeTransformMessageListener
-import software.aws.toolkits.jetbrains.services.codemodernizer.constants.buildStartNewTransformFollowup
import software.aws.toolkits.jetbrains.services.codemodernizer.constants.createViewDiffButton
import software.aws.toolkits.jetbrains.services.codemodernizer.constants.getDownloadedArtifactTextFromType
import software.aws.toolkits.jetbrains.services.codemodernizer.constants.viewSummaryButton
@@ -56,7 +55,6 @@ import software.aws.toolkits.jetbrains.utils.notifyStickyInfo
import software.aws.toolkits.jetbrains.utils.notifyStickyWarn
import software.aws.toolkits.resources.message
import software.aws.toolkits.telemetry.CodeTransformArtifactType
-import software.aws.toolkits.telemetry.CodeTransformVCSViewerSrcComponents
import java.io.File
import java.nio.file.Files
import java.nio.file.Path
@@ -93,17 +91,17 @@ class ArtifactHandler(
private var totalPatchFiles: Int = 0
private var sharedPatchIndex: Int = 0
- internal suspend fun displayDiff(job: JobId, source: CodeTransformVCSViewerSrcComponents) {
+ internal suspend fun displayDiff(job: JobId) {
if (isCurrentlyDownloading.get()) return
when (val result = downloadArtifact(job, TransformationDownloadArtifactType.CLIENT_INSTRUCTIONS)) {
is DownloadArtifactResult.Success -> {
if (result.artifact !is CodeModernizerArtifact) return notifyUnableToApplyPatch("")
totalPatchFiles = result.artifact.patches.size
if (result.artifact.description == null) {
- displayDiffUsingPatch(result.artifact.patches.first(), totalPatchFiles, null, job, source)
+ displayDiffUsingPatch(result.artifact.patches.first(), totalPatchFiles, null, job)
} else {
val diffDescription = result.artifact.description[getCurrentPatchIndex()]
- displayDiffUsingPatch(result.artifact.patches[getCurrentPatchIndex()], totalPatchFiles, diffDescription, job, source)
+ displayDiffUsingPatch(result.artifact.patches[getCurrentPatchIndex()], totalPatchFiles, diffDescription, job)
}
}
is DownloadArtifactResult.ParseZipFailure -> notifyUnableToApplyPatch(result.failureReason.errorMessage)
@@ -285,7 +283,6 @@ class ArtifactHandler(
totalPatchFiles: Int,
diffDescription: PatchInfo?,
jobId: JobId,
- source: CodeTransformVCSViewerSrcComponents,
) {
withContext(EDT) {
val dialog = ApplyPatchDifferentiatedDialog(
@@ -312,15 +309,13 @@ class ArtifactHandler(
dialog.isModal = true
if (dialog.showAndGet()) {
- telemetry.submitSelection("Submit-${diffDescription?.name}")
- telemetry.viewArtifact(CodeTransformArtifactType.ClientInstructions, jobId, "Submit", source)
+ telemetry.submitSelection("Submit-${diffDescription?.name}", jobId.toString())
if (diffDescription == null) {
val resultContent = CodeTransformChatMessageContent(
type = CodeTransformChatMessageType.PendingAnswer,
message = message("codemodernizer.chat.message.changes_applied"),
)
codeTransformChatHelper?.updateLastPendingMessage(resultContent)
- codeTransformChatHelper?.addNewMessage(buildStartNewTransformFollowup())
} else {
if (getCurrentPatchIndex() < totalPatchFiles) {
val message = "I applied the changes in diff patch ${getCurrentPatchIndex() + 1} of $totalPatchFiles. " +
@@ -347,11 +342,11 @@ class ArtifactHandler(
)
}
} else {
- codeTransformChatHelper?.addNewMessage(buildStartNewTransformFollowup())
+ // no-op; start a new transformation button already visible at this point
}
}
} else {
- telemetry.viewArtifact(CodeTransformArtifactType.ClientInstructions, jobId, "Cancel", source)
+ telemetry.submitSelection("Cancel", jobId.toString())
}
}
}
@@ -464,9 +459,9 @@ class ArtifactHandler(
)
}
- fun displayDiffAction(jobId: JobId, source: CodeTransformVCSViewerSrcComponents) = runReadAction {
+ fun displayDiffAction(jobId: JobId) = runReadAction {
projectCoroutineScope(project).launch {
- displayDiff(jobId, source)
+ displayDiff(jobId)
}
}
diff --git a/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/CodeModernizerManager.kt b/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/CodeModernizerManager.kt
index 4552cde103..01df64995a 100644
--- a/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/CodeModernizerManager.kt
+++ b/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/CodeModernizerManager.kt
@@ -84,7 +84,6 @@ import software.aws.toolkits.resources.message
import software.aws.toolkits.telemetry.CodeTransformBuildSystem
import software.aws.toolkits.telemetry.CodeTransformCancelSrcComponents
import software.aws.toolkits.telemetry.CodeTransformPreValidationError
-import software.aws.toolkits.telemetry.CodeTransformVCSViewerSrcComponents
import java.io.File
import java.nio.file.Path
import java.time.Instant
@@ -165,6 +164,7 @@ class CodeModernizerManager(private val project: Project) : PersistentStateCompo
return if (javaModules.isNotEmpty()) {
ValidationResult(
true,
+ metadata = "found ${javaModules.size} modules with SQL"
)
} else {
ValidationResult(
@@ -630,7 +630,7 @@ class CodeModernizerManager(private val project: Project) : PersistentStateCompo
is CodeModernizerJobCompletedResult.JobPartiallySucceeded -> {
notifyStickyInfo(
message("codemodernizer.notification.info.modernize_partial_complete.title"),
- message("codemodernizer.notification.info.modernize_partial_complete.content", result.targetJavaVersion.description),
+ message("codemodernizer.notification.info.modernize_partial_complete.content"),
project,
listOf(displaySummaryNotificationAction(result.jobId), displayFeedbackNotificationAction()),
)
@@ -770,12 +770,6 @@ class CodeModernizerManager(private val project: Project) : PersistentStateCompo
codeTransformationSession?.tryOpenTransformationPlanEditor()
}
- fun showDiff() {
- val job = codeTransformationSession?.getActiveJobId() ?: return
- // Use "TreeViewHeader" for Hub
- artifactHandler.displayDiffAction(job, CodeTransformVCSViewerSrcComponents.TreeViewHeader)
- }
-
fun handleCredentialsChanged() {
codeTransformationSession?.dispose()
codeModernizerBottomWindowPanelManager.reset()
diff --git a/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/CodeModernizerSession.kt b/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/CodeModernizerSession.kt
index 0e9d3b1ebc..50ec1a935d 100644
--- a/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/CodeModernizerSession.kt
+++ b/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/CodeModernizerSession.kt
@@ -8,6 +8,7 @@ import com.intellij.openapi.application.runInEdt
import com.intellij.serviceContainer.AlreadyDisposedException
import com.intellij.util.io.HttpRequests
import kotlinx.coroutines.delay
+import kotlinx.coroutines.withContext
import org.apache.commons.codec.digest.DigestUtils
import software.amazon.awssdk.core.exception.SdkClientException
import software.amazon.awssdk.services.codewhispererruntime.model.ResumeTransformationResponse
@@ -20,11 +21,13 @@ import software.amazon.awssdk.services.codewhispererruntime.model.Transformation
import software.amazon.awssdk.services.codewhispererruntime.model.TransformationUserActionStatus
import software.amazon.awssdk.services.codewhispererstreaming.model.TransformationDownloadArtifactType
import software.amazon.awssdk.services.ssooidc.model.SsoOidcException
+import software.aws.toolkits.core.utils.Waiters.waitUntil
import software.aws.toolkits.core.utils.error
import software.aws.toolkits.core.utils.exists
import software.aws.toolkits.core.utils.getLogger
import software.aws.toolkits.core.utils.info
import software.aws.toolkits.core.utils.warn
+import software.aws.toolkits.jetbrains.core.coroutines.getCoroutineBgContext
import software.aws.toolkits.jetbrains.services.codemodernizer.client.GumbyClient
import software.aws.toolkits.jetbrains.services.codemodernizer.commands.CodeTransformMessageListener
import software.aws.toolkits.jetbrains.services.codemodernizer.model.CodeModernizerException
@@ -55,7 +58,10 @@ import java.io.File
import java.io.FileInputStream
import java.io.IOException
import java.net.ConnectException
+import java.net.SocketTimeoutException
+import java.net.UnknownHostException
import java.nio.file.Path
+import java.time.Duration
import java.time.Instant
import java.util.Base64
import java.util.concurrent.CancellationException
@@ -142,7 +148,7 @@ class CodeModernizerSession(
*
* Based on [CodeWhispererCodeScanSession]
*/
- fun createModernizationJob(copyResult: MavenCopyCommandsResult?): CodeModernizerStartJobResult {
+ suspend fun createModernizationJob(copyResult: MavenCopyCommandsResult?): CodeModernizerStartJobResult {
LOG.info { "Compressing local project" }
val payload: File?
var payloadSize = 0
@@ -377,8 +383,12 @@ class CodeModernizerSession(
/**
* Adapted from [CodeWhispererCodeScanSession]
*/
- fun uploadPayload(payload: File): String {
- val sha256checksum: String = Base64.getEncoder().encodeToString(DigestUtils.sha256(FileInputStream(payload)))
+ suspend fun uploadPayload(payload: File): String {
+ val sha256checksum: String = Base64.getEncoder().encodeToString(
+ withContext(getCoroutineBgContext()) {
+ DigestUtils.sha256(FileInputStream(payload))
+ }
+ )
if (isDisposed.get()) {
throw AlreadyDisposedException("Disposed when about to create upload URL")
}
@@ -394,17 +404,24 @@ class CodeModernizerSession(
throw AlreadyDisposedException("Disposed when about to upload project artifact to s3")
}
val uploadStartTime = Instant.now()
- try {
+ waitUntil(
+ exceptionsToIgnore = setOf(
+ UnknownHostException::class,
+ SocketTimeoutException::class,
+ HttpRequests.HttpStatusException::class,
+ ConnectException::class,
+ IOException::class,
+ ),
+ maxDuration = Duration.ofMinutes(5)
+ ) {
clientAdaptor.uploadArtifactToS3(
createUploadUrlResponse.uploadUrl(),
payload,
sha256checksum,
createUploadUrlResponse.kmsKeyArn().orEmpty(),
) { shouldStop.get() }
- } catch (e: Exception) {
- LOG.error { "Unexpected error when uploading project artifact to S3: $e" }
- throw e // pass along error to callee
}
+ LOG.info { "Upload to S3 succeeded" }
if (!shouldStop.get()) {
LOG.info { "Uploaded artifact. Latency: ${calculateTotalLatency(uploadStartTime, Instant.now())}ms" }
}
@@ -493,10 +510,7 @@ class CodeModernizerSession(
message("codemodernizer.notification.warn.unknown_status_response")
)
- result.state == TransformationStatus.PARTIALLY_COMPLETED -> CodeModernizerJobCompletedResult.JobPartiallySucceeded(
- jobId,
- sessionContext.targetJavaVersion
- )
+ result.state == TransformationStatus.PARTIALLY_COMPLETED -> CodeModernizerJobCompletedResult.JobPartiallySucceeded(jobId)
result.state == TransformationStatus.FAILED -> {
if (!passedStart) {
diff --git a/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/CodeTransformChatApp.kt b/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/CodeTransformChatApp.kt
index 22e0a1a2dd..1c32f2f07c 100644
--- a/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/CodeTransformChatApp.kt
+++ b/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/CodeTransformChatApp.kt
@@ -15,6 +15,9 @@ import software.aws.toolkits.jetbrains.services.amazonq.apps.AmazonQApp
import software.aws.toolkits.jetbrains.services.amazonq.apps.AmazonQAppInitContext
import software.aws.toolkits.jetbrains.services.amazonq.auth.AuthController
import software.aws.toolkits.jetbrains.services.amazonq.messages.AmazonQMessage
+import software.aws.toolkits.jetbrains.services.amazonqCodeScan.auth.isCodeScanAvailable
+import software.aws.toolkits.jetbrains.services.amazonqCodeTest.auth.isCodeTestAvailable
+import software.aws.toolkits.jetbrains.services.amazonqDoc.auth.isDocAvailable
import software.aws.toolkits.jetbrains.services.amazonqFeatureDev.auth.isFeatureDevAvailable
import software.aws.toolkits.jetbrains.services.codemodernizer.commands.CodeTransformActionMessage
import software.aws.toolkits.jetbrains.services.codemodernizer.commands.CodeTransformMessageListener
@@ -106,7 +109,10 @@ class CodeTransformChatApp : AmazonQApp {
AuthenticationUpdateMessage(
featureDevEnabled = isFeatureDevAvailable(context.project),
codeTransformEnabled = isCodeTransformAvailable(context.project),
- authenticatingTabIDs = chatSessionStorage.getAuthenticatingSessions().map { it.tabId }
+ codeScanEnabled = isCodeScanAvailable(context.project),
+ codeTestEnabled = isCodeTestAvailable(context.project),
+ docEnabled = isDocAvailable(context.project),
+ authenticatingTabIDs = chatSessionStorage.getAuthenticatingSessions().map { it.tabId },
)
)
@@ -164,6 +170,7 @@ class CodeTransformChatApp : AmazonQApp {
is IncomingCodeTransformMessage.CodeTransformSelectSQLMetadata -> inboundAppMessagesHandler.processCodeTransformSelectSQLMetadataAction(message)
is IncomingCodeTransformMessage.CodeTransformSelectSQLModuleSchema ->
inboundAppMessagesHandler.processCodeTransformSelectSQLModuleSchemaAction(message)
+
is IncomingCodeTransformMessage.CodeTransformCancel -> inboundAppMessagesHandler.processCodeTransformCancelAction(message)
is IncomingCodeTransformMessage.CodeTransformStop -> inboundAppMessagesHandler.processCodeTransformStopAction(message.tabId)
is IncomingCodeTransformMessage.ChatPrompt -> inboundAppMessagesHandler.processChatPromptMessage(message)
diff --git a/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/CodeTransformTelemetryManager.kt b/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/CodeTransformTelemetryManager.kt
index 3f2d7a6d31..0f247685f0 100644
--- a/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/CodeTransformTelemetryManager.kt
+++ b/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/CodeTransformTelemetryManager.kt
@@ -11,7 +11,6 @@ import software.amazon.awssdk.services.codewhispererruntime.model.Transformation
import software.aws.toolkits.jetbrains.services.codemodernizer.model.CustomerSelection
import software.aws.toolkits.jetbrains.services.codemodernizer.model.JobId
import software.aws.toolkits.jetbrains.services.codemodernizer.model.ValidationResult
-import software.aws.toolkits.jetbrains.services.codemodernizer.state.CodeModernizerSessionState
import software.aws.toolkits.jetbrains.services.codemodernizer.state.CodeTransformTelemetryState
import software.aws.toolkits.jetbrains.services.codemodernizer.utils.calculateTotalLatency
import software.aws.toolkits.jetbrains.services.codemodernizer.utils.getAuthType
@@ -24,8 +23,8 @@ import software.aws.toolkits.telemetry.CodeTransformCancelSrcComponents
import software.aws.toolkits.telemetry.CodeTransformJavaSourceVersionsAllowed
import software.aws.toolkits.telemetry.CodeTransformJavaTargetVersionsAllowed
import software.aws.toolkits.telemetry.CodeTransformPreValidationError
-import software.aws.toolkits.telemetry.CodeTransformVCSViewerSrcComponents
import software.aws.toolkits.telemetry.CodetransformTelemetry
+import software.aws.toolkits.telemetry.MetricResult
import software.aws.toolkits.telemetry.Result
import java.time.Instant
import java.util.Base64
@@ -36,7 +35,6 @@ import java.util.Base64
@Service(Service.Level.PROJECT)
class CodeTransformTelemetryManager(private val project: Project) {
private val sessionId get() = CodeTransformTelemetryState.instance.getSessionId()
- private val currentJobStatus get() = CodeModernizerSessionState.getInstance(project).currentJobStatus.toString()
fun initiateTransform(telemetryErrorMessage: String? = null) {
CodetransformTelemetry.initiateTransform(
@@ -60,18 +58,20 @@ class CodeTransformTelemetryManager(private val project: Project) {
codeTransformPreValidationError = validationError,
codeTransformBuildSystem = validationResult.buildSystem,
codeTransformSessionId = sessionId,
+ codeTransformMetadata = validationResult.metadata,
result = if (validationResult.valid) Result.Succeeded else Result.Failed,
reason = if (validationResult.valid) null else validationResult.invalidTelemetryReason.additionalInfo,
)
}
- fun submitSelection(userChoice: String, customerSelection: CustomerSelection? = null, telemetryErrorMessage: String? = null) {
+ fun submitSelection(userChoice: String, jobId: String? = null, customerSelection: CustomerSelection? = null, telemetryErrorMessage: String? = null) {
CodetransformTelemetry.submitSelection(
- // TODO: remove below 2 lines (JavaSource / JavaTarget) once BI is updated to use source / target
+ // TODO: remove the below 2 lines (JavaSource / JavaTarget) once BI is updated to use source / target
codeTransformJavaSourceVersionsAllowed = CodeTransformJavaSourceVersionsAllowed.from(customerSelection?.sourceJavaVersion?.name.orEmpty()),
codeTransformJavaTargetVersionsAllowed = CodeTransformJavaTargetVersionsAllowed.from(customerSelection?.targetJavaVersion?.name.orEmpty()),
codeTransformSessionId = sessionId,
codeTransformProjectId = customerSelection?.let { getProjectHash(it) },
+ codeTransformJobId = jobId,
source = if (userChoice == "Confirm-Java") customerSelection?.sourceJavaVersion?.name.orEmpty() else customerSelection?.sourceVendor.orEmpty(),
target = if (userChoice == "Confirm-Java") customerSelection?.targetJavaVersion?.name.orEmpty() else customerSelection?.targetVendor.orEmpty(),
userChoice = userChoice,
@@ -128,25 +128,6 @@ class CodeTransformTelemetryManager(private val project: Project) {
)
}
- fun viewArtifact(
- artifactType: CodeTransformArtifactType,
- jobId: JobId,
- userChoice: String,
- source: CodeTransformVCSViewerSrcComponents,
- telemetryErrorMessage: String? = null,
- ) {
- CodetransformTelemetry.viewArtifact(
- codeTransformArtifactType = artifactType,
- codeTransformVCSViewerSrcComponents = source,
- codeTransformSessionId = sessionId,
- codeTransformJobId = jobId.id,
- codeTransformStatus = currentJobStatus,
- userChoice = userChoice,
- result = if (telemetryErrorMessage.isNullOrEmpty()) Result.Succeeded else Result.Failed,
- reason = telemetryErrorMessage,
- )
- }
-
fun getProjectHash(customerSelection: CustomerSelection) = Base64.getEncoder().encodeToString(
DigestUtils.sha256(customerSelection.configurationFile?.toNioPath()?.toAbsolutePath().toString())
)
@@ -196,12 +177,12 @@ class CodeTransformTelemetryManager(private val project: Project) {
fun logHil(jobId: String, metaData: HilTelemetryMetaData, success: Boolean, reason: String) {
CodetransformTelemetry.humanInTheLoop(
- project,
- jobId,
- metaData.toString(),
- sessionId,
- reason,
- success,
+ project = project,
+ codeTransformJobId = jobId,
+ codeTransformMetadata = metaData.toString(),
+ codeTransformSessionId = sessionId,
+ reason = reason,
+ result = if (success) MetricResult.Succeeded else MetricResult.Failed,
)
}
diff --git a/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/client/GumbyClient.kt b/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/client/GumbyClient.kt
index 2a92dc33d0..097b9231d4 100644
--- a/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/client/GumbyClient.kt
+++ b/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/client/GumbyClient.kt
@@ -148,6 +148,7 @@ class GumbyClient(private val project: Project) {
var result: CodeWhispererRuntimeResponse? = null
try {
result = apiCall()
+ LOG.info { "$apiName request ID: ${result.responseMetadata()?.requestId()}" }
return result
} catch (e: Exception) {
LOG.error(e) { "$apiName failed: ${e.message}" }
diff --git a/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/constants/CodeTransformChatItems.kt b/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/constants/CodeTransformChatItems.kt
index 8a8358c41b..9c03fbdf51 100644
--- a/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/constants/CodeTransformChatItems.kt
+++ b/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/constants/CodeTransformChatItems.kt
@@ -431,6 +431,11 @@ fun buildTransformStoppedChatContent() = CodeTransformChatMessageContent(
type = CodeTransformChatMessageType.FinalizedAnswer,
)
+fun buildTransformFailedChatContent(failureReason: String) = CodeTransformChatMessageContent(
+ message = message("codemodernizer.chat.message.transform_failed", failureReason),
+ type = CodeTransformChatMessageType.FinalizedAnswer,
+)
+
fun buildUserSQLConversionSelectionSummaryChatContent(moduleName: String, schema: String) = CodeTransformChatMessageContent(
type = CodeTransformChatMessageType.Prompt,
message = getUserSQLConversionSelectionFormattedMarkdown(moduleName, schema)
@@ -537,7 +542,7 @@ fun buildTransformResumingChatContent() = CodeTransformChatMessageContent(
type = CodeTransformChatMessageType.PendingAnswer,
)
-fun buildTransformResultChatContent(result: CodeModernizerJobCompletedResult, totalPatchFiles: Int): CodeTransformChatMessageContent {
+fun buildTransformResultChatContent(result: CodeModernizerJobCompletedResult, totalPatchFiles: Int? = null): CodeTransformChatMessageContent {
val resultMessage = when (result) {
is CodeModernizerJobCompletedResult.JobAbortedZipTooLarge -> {
"${message(
@@ -595,6 +600,7 @@ fun buildTransformResultChatContent(result: CodeModernizerJobCompletedResult, to
} else {
null
},
+ followUps = listOf(startNewTransformFollowUp),
)
}
diff --git a/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/controller/CodeTransformChatController.kt b/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/controller/CodeTransformChatController.kt
index a31615d4e2..2be8c8ade1 100644
--- a/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/controller/CodeTransformChatController.kt
+++ b/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/controller/CodeTransformChatController.kt
@@ -25,6 +25,7 @@ import software.aws.toolkits.jetbrains.services.amazonq.auth.AuthController
import software.aws.toolkits.jetbrains.services.amazonq.auth.AuthFollowUpType
import software.aws.toolkits.jetbrains.services.codemodernizer.ArtifactHandler
import software.aws.toolkits.jetbrains.services.codemodernizer.CodeModernizerManager
+import software.aws.toolkits.jetbrains.services.codemodernizer.CodeModernizerManager.Companion.LOG
import software.aws.toolkits.jetbrains.services.codemodernizer.CodeTransformTelemetryManager
import software.aws.toolkits.jetbrains.services.codemodernizer.EXPLAINABILITY_V1
import software.aws.toolkits.jetbrains.services.codemodernizer.HilTelemetryMetaData
@@ -61,6 +62,7 @@ import software.aws.toolkits.jetbrains.services.codemodernizer.constants.buildSt
import software.aws.toolkits.jetbrains.services.codemodernizer.constants.buildTransformAwaitUserInputChatContent
import software.aws.toolkits.jetbrains.services.codemodernizer.constants.buildTransformBeginChatContent
import software.aws.toolkits.jetbrains.services.codemodernizer.constants.buildTransformDependencyErrorChatContent
+import software.aws.toolkits.jetbrains.services.codemodernizer.constants.buildTransformFailedChatContent
import software.aws.toolkits.jetbrains.services.codemodernizer.constants.buildTransformFindingLocalAlternativeDependencyChatContent
import software.aws.toolkits.jetbrains.services.codemodernizer.constants.buildTransformInProgressChatContent
import software.aws.toolkits.jetbrains.services.codemodernizer.constants.buildTransformResultChatContent
@@ -111,7 +113,6 @@ import software.aws.toolkits.jetbrains.services.codemodernizer.utils.unzipFile
import software.aws.toolkits.jetbrains.services.codemodernizer.utils.validateSctMetadata
import software.aws.toolkits.jetbrains.services.cwc.messages.ChatMessageType
import software.aws.toolkits.resources.message
-import software.aws.toolkits.telemetry.CodeTransformVCSViewerSrcComponents
class CodeTransformChatController(
private val context: AmazonQAppInitContext,
@@ -183,7 +184,7 @@ class CodeTransformChatController(
private suspend fun getUserObjective(tabId: String) {
codeTransformChatHelper.addNewMessage(buildChooseTransformationObjectiveChatContent())
codeTransformChatHelper.sendChatInputEnabledMessage(tabId, true)
- codeTransformChatHelper.sendUpdatePlaceholderMessage(tabId, message("codemodernizer.chat.message.choose_objective"))
+ codeTransformChatHelper.sendUpdatePlaceholderMessage(tabId, message("codemodernizer.chat.message.choose_objective_placeholder"))
}
private suspend fun validateAndReplyOnError(transformationType: CodeTransformType): ValidationResult? {
@@ -208,6 +209,7 @@ class CodeTransformChatController(
}
private suspend fun handleSQLConversion() {
+ telemetry.submitSelection("sql conversion")
this.validateAndReplyOnError(CodeTransformType.SQL_CONVERSION) ?: return
codeTransformChatHelper.addNewMessage(
buildUserInputSQLConversionMetadataChatContent()
@@ -215,6 +217,7 @@ class CodeTransformChatController(
}
private suspend fun handleLanguageUpgrade() {
+ telemetry.submitSelection("language upgrade")
val validationResult = this.validateAndReplyOnError(CodeTransformType.LANGUAGE_UPGRADE) ?: return
codeTransformChatHelper.updateLastPendingMessage(
buildLanguageUpgradeProjectValidChatContent()
@@ -269,7 +272,7 @@ class CodeTransformChatController(
override suspend fun processCodeTransformCancelAction(message: IncomingCodeTransformMessage.CodeTransformCancel) {
if (!checkForAuth(message.tabId)) {
- telemetry.submitSelection("Cancel", null, "User is not authenticated")
+ telemetry.submitSelection("Cancel", null, null, "User is not authenticated")
return
}
@@ -284,7 +287,7 @@ class CodeTransformChatController(
override suspend fun processCodeTransformStartAction(message: IncomingCodeTransformMessage.CodeTransformStart) {
if (!checkForAuth(message.tabId)) {
- telemetry.submitSelection("Confirm", null, "User is not authenticated")
+ telemetry.submitSelection("Confirm", null, null, "User is not authenticated")
return
}
@@ -307,7 +310,7 @@ class CodeTransformChatController(
codeModernizerManager.createCodeModernizerSession(selection, context.project)
// Publish metric to capture user selection before local build starts
- telemetry.submitSelection("Confirm-Java", selection)
+ telemetry.submitSelection("Confirm-Java", null, selection)
codeTransformChatHelper.run {
addNewMessage(buildUserInputSkipTestsFlagChatIntroContent())
@@ -339,7 +342,7 @@ class CodeTransformChatController(
unzipFile(selectedZipFile.toNioPath(), extractedZip.toPath(), true)
- val sctFile = extractedZip.listFiles { file -> file.name.endsWith(".sct") }.firstOrNull()
+ val sctFile = extractedZip.listFiles { file -> file.name.endsWith(".sct") }?.firstOrNull()
val metadataValidationResult = validateSctMetadata(sctFile)
@@ -369,7 +372,7 @@ class CodeTransformChatController(
sqlMetadataZip = extractedZip,
)
codeModernizerManager.createCodeModernizerSession(selection, context.project)
- telemetry.submitSelection("Confirm-SQL", selection)
+ telemetry.submitSelection("Confirm-SQL", null, selection)
}
}
@@ -398,6 +401,7 @@ class CodeTransformChatController(
EXPLAINABILITY_V1
)
}
+ telemetry.submitSelection(message.oneOrMultipleDiffsSelection)
codeTransformChatHelper.addNewMessage(buildUserOneOrMultipleDiffsSelectionChatContent(message.oneOrMultipleDiffsSelection))
codeTransformChatHelper.addNewMessage(buildCompileLocalInProgressChatContent())
codeModernizerManager.codeTransformationSession?.let {
@@ -457,7 +461,7 @@ class CodeTransformChatController(
}
override suspend fun processCodeTransformStopAction(tabId: String) {
- if (!checkForAuth(tabId)) {
+ if (!checkForAuth(tabId) || !codeModernizerManager.isModernizationJobActive()) {
return
}
@@ -489,7 +493,6 @@ class CodeTransformChatController(
override suspend fun processCodeTransformViewDiff(message: IncomingCodeTransformMessage.CodeTransformViewDiff) {
artifactHandler.displayDiffAction(
CodeModernizerSessionState.getInstance(context.project).currentJobId as JobId,
- CodeTransformVCSViewerSrcComponents.Chat
)
}
@@ -665,9 +668,24 @@ class CodeTransformChatController(
codeTransformChatHelper.addNewMessage(buildStartNewTransformFollowup())
}
+ private suspend fun handleCodeTransformJobFailed(failureReason: String) {
+ codeTransformChatHelper.updateLastPendingMessage(buildTransformFailedChatContent(failureReason))
+ codeTransformChatHelper.addNewMessage(buildStartNewTransformFollowup())
+ }
+
+ private suspend fun handleCodeTransformJobFailedPreBuild(result: CodeModernizerJobCompletedResult.JobFailedInitialBuild) {
+ codeTransformChatHelper.addNewMessage(
+ buildTransformResultChatContent(result)
+ )
+ artifactHandler.showBuildLog(CodeModernizerSessionState.getInstance(context.project).currentJobId as JobId)
+ }
+
private suspend fun handleCodeTransformResult(result: CodeModernizerJobCompletedResult) {
+ LOG.info { "CodeModernizerJobCompletedResult: $result" }
when (result) {
is CodeModernizerJobCompletedResult.Stopped, CodeModernizerJobCompletedResult.JobAbortedBeforeStarting -> handleCodeTransformStoppedByUser()
+ is CodeModernizerJobCompletedResult.JobFailed -> handleCodeTransformJobFailed(result.failureReason)
+ is CodeModernizerJobCompletedResult.JobFailedInitialBuild -> handleCodeTransformJobFailedPreBuild(result)
else -> {
if (result is CodeModernizerJobCompletedResult.ZipUploadFailed && result.failureReason is UploadFailureReason.CREDENTIALS_EXPIRED) {
return
@@ -676,6 +694,7 @@ class CodeTransformChatController(
CodeModernizerSessionState.getInstance(context.project).currentJobId as JobId,
TransformationDownloadArtifactType.CLIENT_INSTRUCTIONS
)
+ LOG.info { "Download result: $downloadResult" }
when (downloadResult) {
is DownloadArtifactResult.Success -> {
if (downloadResult.artifact !is CodeModernizerArtifact) return artifactHandler.notifyUnableToApplyPatch("")
diff --git a/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/messages/CodeTransformMessage.kt b/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/messages/CodeTransformMessage.kt
index 96ca9db9cd..9a32e82aff 100644
--- a/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/messages/CodeTransformMessage.kt
+++ b/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/messages/CodeTransformMessage.kt
@@ -189,7 +189,10 @@ data class AuthenticationUpdateMessage(
val authenticatingTabIDs: List,
val featureDevEnabled: Boolean,
val codeTransformEnabled: Boolean,
+ val codeScanEnabled: Boolean,
val message: String? = null,
+ val codeTestEnabled: Boolean,
+ val docEnabled: Boolean,
) : CodeTransformUiMessage(
null,
type = "authenticationUpdateMessage",
diff --git a/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/model/CodeModernizerArtifact.kt b/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/model/CodeModernizerArtifact.kt
index 555570dd8c..86a3903569 100644
--- a/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/model/CodeModernizerArtifact.kt
+++ b/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/model/CodeModernizerArtifact.kt
@@ -19,6 +19,7 @@ import software.aws.toolkits.core.utils.warn
import software.aws.toolkits.jetbrains.services.codemodernizer.TransformationSummary
import software.aws.toolkits.jetbrains.services.codemodernizer.utils.unzipFile
import java.io.File
+import java.util.UUID
import kotlin.io.path.ExperimentalPathApi
import kotlin.io.path.Path
import kotlin.io.path.isDirectory
@@ -39,7 +40,7 @@ open class CodeModernizerArtifact(
companion object {
private const val MAX_SUPPORTED_VERSION = 1.0
- private val tempDir = createTempDirectory("codeTransformArtifacts", null)
+ private var tempDir = createTempDirectory("codeTransformArtifacts", null)
private const val MANIFEST_FILE_NAME = "manifest.json"
private const val SUMMARY_FILE_NAME = "summary.md"
private const val METRICS_FILE_NAME = "metrics.json"
@@ -52,6 +53,7 @@ open class CodeModernizerArtifact(
* If anything goes wrong during this process an exception is thrown.
*/
fun create(zipPath: String): CodeModernizerArtifact {
+ tempDir = createTempDirectory("codeTransformArtifacts-", UUID.randomUUID().toString())
val path = Path(zipPath)
if (path.exists()) {
if (!unzipFile(path, tempDir.toPath())) {
diff --git a/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/model/CodeModernizerJobCompletedResult.kt b/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/model/CodeModernizerJobCompletedResult.kt
index 8bb51b3777..516594d3ec 100644
--- a/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/model/CodeModernizerJobCompletedResult.kt
+++ b/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/model/CodeModernizerJobCompletedResult.kt
@@ -3,15 +3,13 @@
package software.aws.toolkits.jetbrains.services.codemodernizer.model
-import com.intellij.openapi.projectRoots.JavaSdkVersion
-
sealed class CodeModernizerJobCompletedResult {
data class RetryableFailure(val jobId: JobId, val failureReason: String) : CodeModernizerJobCompletedResult()
data class UnableToCreateJob(val failureReason: String, val retryable: Boolean = false) : CodeModernizerJobCompletedResult()
data class JobFailed(val jobId: JobId, val failureReason: String) : CodeModernizerJobCompletedResult()
data class ZipUploadFailed(val failureReason: UploadFailureReason) : CodeModernizerJobCompletedResult()
data class JobCompletedSuccessfully(val jobId: JobId) : CodeModernizerJobCompletedResult()
- data class JobPartiallySucceeded(val jobId: JobId, val targetJavaVersion: JavaSdkVersion) : CodeModernizerJobCompletedResult()
+ data class JobPartiallySucceeded(val jobId: JobId) : CodeModernizerJobCompletedResult()
data class JobPaused(val jobId: JobId, val downloadArtifactId: String) : CodeModernizerJobCompletedResult()
diff --git a/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/model/CodeModernizerSessionContext.kt b/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/model/CodeModernizerSessionContext.kt
index bd75a61c4f..b3c0fc9fc4 100644
--- a/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/model/CodeModernizerSessionContext.kt
+++ b/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/model/CodeModernizerSessionContext.kt
@@ -58,7 +58,7 @@ const val INVALID_SUFFIX_SHA = "sha1"
const val INVALID_SUFFIX_REPOSITORIES = "repositories"
const val ORACLE_DB = "ORACLE"
const val AURORA_DB = "AURORA_POSTGRESQL"
-const val RDS_DB = "RDS_POSTGRESQL"
+const val RDS_DB = "POSTGRESQL"
data class CodeModernizerSessionContext(
val project: Project,
var configurationFile: VirtualFile? = null, // used to ZIP module
diff --git a/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/model/ValidationResult.kt b/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/model/ValidationResult.kt
index e4dd4a4bae..e01ae3762a 100644
--- a/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/model/ValidationResult.kt
+++ b/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/model/ValidationResult.kt
@@ -6,10 +6,12 @@ package software.aws.toolkits.jetbrains.services.codemodernizer.model
import com.intellij.openapi.vfs.VirtualFile
import software.aws.toolkits.telemetry.CodeTransformBuildSystem
+// TODO: combine a lot of these fields into the 'metadata' field
data class ValidationResult(
val valid: Boolean,
val invalidTelemetryReason: InvalidTelemetryReason = InvalidTelemetryReason(),
val validatedBuildFiles: List = emptyList(),
val buildSystem: CodeTransformBuildSystem = CodeTransformBuildSystem.Unknown,
val buildSystemVersion: String = "",
+ val metadata: String? = null,
)
diff --git a/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/panels/CodeModernizerBanner.kt b/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/panels/CodeModernizerBanner.kt
index 0d96887f16..8afe80e61e 100644
--- a/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/panels/CodeModernizerBanner.kt
+++ b/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/panels/CodeModernizerBanner.kt
@@ -5,7 +5,6 @@ package software.aws.toolkits.jetbrains.services.codemodernizer.panels
import com.intellij.icons.AllIcons
import com.intellij.openapi.project.Project
-import com.intellij.serviceContainer.AlreadyDisposedException
import com.intellij.ui.JBColor
import com.intellij.ui.border.CustomLineBorder
import com.intellij.ui.components.ActionLink
@@ -13,9 +12,8 @@ import com.intellij.ui.components.JBLabel
import com.intellij.util.ui.JBInsets
import com.intellij.util.ui.JBUI
import icons.AwsIcons
-import software.aws.toolkits.core.utils.getLogger
-import software.aws.toolkits.core.utils.warn
import software.aws.toolkits.jetbrains.services.codemodernizer.CodeModernizerManager
+import software.aws.toolkits.jetbrains.services.codemodernizer.model.JobId
import software.aws.toolkits.jetbrains.services.codewhisperer.layout.CodeWhispererLayoutConfig
import software.aws.toolkits.jetbrains.services.codewhisperer.layout.CodeWhispererLayoutConfig.addHorizontalGlue
import software.aws.toolkits.jetbrains.ui.feedback.CodeTransformFeedbackDialog
@@ -42,11 +40,13 @@ class CodeModernizerBanner(val project: Project) : JPanel(BorderLayout()) {
border = BorderFactory.createEmptyBorder(0, 5, 0, 0)
}
+ private val infoLabelJobId = JBLabel().apply {
+ foreground = JBColor.GRAY
+ border = BorderFactory.createEmptyBorder(0, 5, 0, 0)
+ }
+
private val infoPanel = JPanel(GridBagLayout())
- val showDiffAction = ActionLink(message("codemodernizer.toolwindow.banner.action.diff")) {
- CodeModernizerManager.getInstance(project).showDiff()
- }
val showPlanAction = ActionLink(message("codemodernizer.toolwindow.banner.action.plan")) {
CodeModernizerManager.getInstance(project).showTransformationPlan()
}
@@ -84,6 +84,7 @@ class CodeModernizerBanner(val project: Project) : JPanel(BorderLayout()) {
)
}
add(infoLabelRunningTime, CodeWhispererLayoutConfig.kebabMenuConstraints)
+ add(infoLabelJobId, CodeWhispererLayoutConfig.kebabMenuConstraints)
}
infoPanel.revalidate()
infoPanel.repaint()
@@ -114,19 +115,18 @@ class CodeModernizerBanner(val project: Project) : JPanel(BorderLayout()) {
}
fun updateRunningTime(runTime: Duration?) {
- try {
- if (runTime == null) {
- infoLabelRunningTime.text = ""
- } else {
- val timeTaken = runTime.toKotlinDuration().inWholeSeconds.seconds.toString()
- infoLabelRunningTime.text = message(
- "codemodernizer.toolwindow.transformation.progress.running_time",
- timeTaken
- )
- }
- } catch (exception: AlreadyDisposedException) {
- LOG.warn { "Disposed when about to create the loading panel" }
- return
+ infoLabelRunningTime.text = if (runTime != null) {
+ message("codemodernizer.toolwindow.transformation.progress.running_time", runTime.toKotlinDuration().inWholeSeconds.seconds.toString())
+ } else {
+ ""
+ }
+ }
+
+ fun updateJobId(jobId: JobId?) {
+ infoLabelJobId.text = if (jobId != null) {
+ message("codemodernizer.toolwindow.transformation.progress.job_id", jobId.id)
+ } else {
+ ""
}
}
@@ -134,8 +134,4 @@ class CodeModernizerBanner(val project: Project) : JPanel(BorderLayout()) {
currentlyShownOptions.clear()
buildContent()
}
-
- companion object {
- private val LOG = getLogger()
- }
}
diff --git a/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/panels/managers/CodeModernizerBottomWindowPanelManager.kt b/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/panels/managers/CodeModernizerBottomWindowPanelManager.kt
index 52460512c9..4d73038a30 100644
--- a/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/panels/managers/CodeModernizerBottomWindowPanelManager.kt
+++ b/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/panels/managers/CodeModernizerBottomWindowPanelManager.kt
@@ -88,6 +88,16 @@ class CodeModernizerBottomWindowPanelManager(private val project: Project) : JPa
}
}
+ private fun updateJobId() {
+ try {
+ val jobId = CodeModernizerSessionState.getInstance(project).currentJobId
+ banner.updateJobId(jobId)
+ } catch (e: AlreadyDisposedException) {
+ LOG.warn { "Disposed when about to update the jobId" }
+ return
+ }
+ }
+
fun setJobStartingUI() = setUI {
add(BorderLayout.CENTER, fullSizeLoadingPanel)
banner.clearActions()
@@ -123,6 +133,7 @@ class CodeModernizerBottomWindowPanelManager(private val project: Project) : JPa
fun setJobRunningUI() = setUI {
add(BorderLayout.CENTER, buildProgressSplitterPanelManager)
banner.updateContent(message("codemodernizer.toolwindow.banner.job_is_running"), AllIcons.General.BalloonInformation)
+ updateJobId()
buildProgressSplitterPanelManager.apply {
reset()
statusTreePanel.setDefaultUI()
diff --git a/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/utils/CodeTransformModuleUtils.kt b/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/utils/CodeTransformModuleUtils.kt
index fef9674314..07317adef3 100644
--- a/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/utils/CodeTransformModuleUtils.kt
+++ b/plugins/amazonq/codetransform/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codemodernizer/utils/CodeTransformModuleUtils.kt
@@ -44,9 +44,8 @@ fun Module.tryGetJdkLanguageLevelJdk(): JavaSdkVersion? {
// search for Strings that indicate embedded Oracle SQL statements are present
fun containsSQL(contentRoot: VirtualFile): Boolean {
val patterns = listOf(
- "oracle.jdbc.OracleDriver",
- "jdbc:oracle:thin:@",
- "jdbc:oracle:oci:@",
+ "oracle.jdbc.",
+ "jdbc:oracle:",
"jdbc:odbc:",
)
diff --git a/plugins/amazonq/codetransform/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codemodernizer/CodeTransformChatTest.kt b/plugins/amazonq/codetransform/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codemodernizer/CodeTransformChatTest.kt
new file mode 100644
index 0000000000..e1eed07547
--- /dev/null
+++ b/plugins/amazonq/codetransform/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codemodernizer/CodeTransformChatTest.kt
@@ -0,0 +1,60 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.codemodernizer
+
+import org.junit.Assert.assertEquals
+import org.junit.Assert.assertNotNull
+import org.junit.Assert.assertNull
+import org.junit.Assert.fail
+import org.junit.Test
+import software.aws.toolkits.jetbrains.services.codemodernizer.constants.buildTransformResultChatContent
+import software.aws.toolkits.jetbrains.services.codemodernizer.messages.CodeTransformButtonId
+import software.aws.toolkits.jetbrains.services.codemodernizer.model.CodeModernizerJobCompletedResult
+import software.aws.toolkits.jetbrains.services.codemodernizer.model.JobId
+import software.aws.toolkits.resources.message
+
+class CodeTransformChatTest {
+
+ @Test
+ fun `test that transform result chat item includes view build log button and message when pre-build fails`() {
+ val result = CodeModernizerJobCompletedResult.JobFailedInitialBuild(JobId("dummy-job-id-123"), "Build failed in Java 8 sandbox", true)
+ val chatItem = buildTransformResultChatContent(result)
+ assertEquals(chatItem.message, message("codemodernizer.chat.message.result.fail_initial_build"))
+ assertNotNull(chatItem.buttons)
+ assertEquals(chatItem.buttons?.size ?: fail("buttons is null"), 1)
+ assertEquals(chatItem.buttons?.get(0)?.id ?: fail("buttons is null"), CodeTransformButtonId.ViewBuildLog.id)
+ }
+
+ @Test
+ fun `test that transform result chat item includes view summary button and view diff button with correct label when job fully succeeded with 5 patch files`() {
+ val result = CodeModernizerJobCompletedResult.JobCompletedSuccessfully(JobId("dummy-job-id-123"))
+ val chatItem = buildTransformResultChatContent(result, 5)
+ assertEquals(chatItem.message, message("codemodernizer.chat.message.result.success.multiple_diffs"))
+ assertNotNull(chatItem.buttons)
+ assertEquals(chatItem.buttons?.size ?: fail("buttons is null"), 2)
+ assertEquals(chatItem.buttons?.get(0)?.id ?: fail("buttons is null"), CodeTransformButtonId.ViewDiff.id)
+ assertEquals(chatItem.buttons?.get(0)?.text ?: fail("buttons is null"), "View diff 1/5")
+ assertEquals(chatItem.buttons?.get(1)?.id ?: fail("buttons is null"), CodeTransformButtonId.ViewSummary.id)
+ }
+
+ @Test
+ fun `test that transform result chat item includes view summary button and view diff button with correct label when job partially succeeded with 1 patch file`() {
+ val result = CodeModernizerJobCompletedResult.JobPartiallySucceeded(JobId("dummy-job-id-123"))
+ val chatItem = buildTransformResultChatContent(result, 1)
+ assertEquals(chatItem.message, message("codemodernizer.chat.message.result.partially_success"))
+ assertNotNull(chatItem.buttons)
+ assertEquals(chatItem.buttons?.size ?: fail("buttons is null"), 2)
+ assertEquals(chatItem.buttons?.get(0)?.id ?: fail("buttons is null"), CodeTransformButtonId.ViewDiff.id)
+ assertEquals(chatItem.buttons?.get(0)?.text ?: fail("buttons is null"), "View diff")
+ assertEquals(chatItem.buttons?.get(1)?.id ?: fail("buttons is null"), CodeTransformButtonId.ViewSummary.id)
+ }
+
+ @Test
+ fun `test that transform result chat item does not include any buttons when job failed with known reason`() {
+ val result = CodeModernizerJobCompletedResult.JobFailed(JobId("dummy-job-id-123"), message("codemodernizer.file.invalid_pom_version"))
+ val chatItem = buildTransformResultChatContent(result)
+ assertEquals(chatItem.message, message("codemodernizer.chat.message.result.fail_with_known_reason", result.failureReason))
+ assertNull(chatItem.buttons)
+ }
+}
diff --git a/plugins/amazonq/codetransform/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codemodernizer/CodeWhispererCodeModernizerSessionTest.kt b/plugins/amazonq/codetransform/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codemodernizer/CodeWhispererCodeModernizerSessionTest.kt
index f50e5e9c73..4623edb2ff 100644
--- a/plugins/amazonq/codetransform/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codemodernizer/CodeWhispererCodeModernizerSessionTest.kt
+++ b/plugins/amazonq/codetransform/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codemodernizer/CodeWhispererCodeModernizerSessionTest.kt
@@ -67,7 +67,6 @@ import software.aws.toolkits.jetbrains.utils.rules.HeavyJavaCodeInsightTestFixtu
import software.aws.toolkits.jetbrains.utils.rules.addFileToModule
import java.io.File
import java.io.FileInputStream
-import java.io.IOException
import java.net.ConnectException
import java.util.Base64
import java.util.zip.ZipFile
@@ -410,7 +409,7 @@ class CodeWhispererCodeModernizerSessionTest : CodeWhispererCodeModernizerTestBa
}
@Test
- fun `CodeModernizer can create modernization job`() {
+ fun `CodeModernizer can create modernization job`() = runTest {
doReturn(ZipCreationResult.Succeeded(File("./tst-resources/codemodernizer/test.txt")))
.whenever(testSessionContextSpy).createZipWithModuleFiles(any())
doReturn(exampleCreateUploadUrlResponse).whenever(clientAdaptorSpy).createGumbyUploadUrl(any())
@@ -425,7 +424,7 @@ class CodeWhispererCodeModernizerSessionTest : CodeWhispererCodeModernizerTestBa
}
@Test
- fun `CodeModernizer cannot upload payload due to already disposed`() {
+ fun `CodeModernizer cannot upload payload due to already disposed`() = runTest {
doReturn(ZipCreationResult.Succeeded(File("./tst-resources/codemodernizer/test.txt")))
.whenever(testSessionContextSpy).createZipWithModuleFiles(any())
doReturn(exampleCreateUploadUrlResponse).whenever(clientAdaptorSpy).createGumbyUploadUrl(any())
@@ -435,7 +434,7 @@ class CodeWhispererCodeModernizerSessionTest : CodeWhispererCodeModernizerTestBa
}
@Test
- fun `CodeModernizer returns credentials expired when SsoOidcException during upload`() {
+ fun `CodeModernizer returns credentials expired when SsoOidcException during upload`() = runTest {
setupConnection(BearerTokenAuthState.AUTHORIZED)
doReturn(ZipCreationResult.Succeeded(File("./tst-resources/codemodernizer/test.txt")))
.whenever(testSessionContextSpy).createZipWithModuleFiles(any())
@@ -445,7 +444,7 @@ class CodeWhispererCodeModernizerSessionTest : CodeWhispererCodeModernizerTestBa
}
@Test
- fun `CodeModernizer returns credentials expired when expired before upload`() {
+ fun `CodeModernizer returns credentials expired when expired before upload`() = runTest {
listOf(BearerTokenAuthState.NEEDS_REFRESH, BearerTokenAuthState.NOT_AUTHENTICATED).forEach {
setupConnection(it)
val result = testSessionSpy.createModernizationJob(MavenCopyCommandsResult.Success(File("./mock/path/")))
@@ -454,12 +453,11 @@ class CodeWhispererCodeModernizerSessionTest : CodeWhispererCodeModernizerTestBa
}
@Test
- fun `CodeModernizer cannot upload payload due to presigned url issue`() {
+ fun `CodeModernizer cannot upload payload due to presigned url issue`() = runTest {
doReturn(ZipCreationResult.Succeeded(File("./tst-resources/codemodernizer/test.txt")))
.whenever(testSessionContextSpy).createZipWithModuleFiles(any())
doReturn(exampleCreateUploadUrlResponse).whenever(clientAdaptorSpy).createGumbyUploadUrl(any())
- doAnswer { throw HttpRequests.HttpStatusException("mock error", 403, "mock url") }
- .whenever(clientAdaptorSpy).uploadArtifactToS3(any(), any(), any(), any(), any())
+ doAnswer { throw HttpRequests.HttpStatusException("mock error", 403, "mock url") }.whenever(testSessionSpy).uploadPayload(any())
val result = testSessionSpy.createModernizationJob(MavenCopyCommandsResult.Success(File("./mock/path/")))
assertEquals(CodeModernizerStartJobResult.ZipUploadFailed(UploadFailureReason.PRESIGNED_URL_EXPIRED), result)
verify(testSessionStateSpy, times(1)).putJobHistory(any(), eq(TransformationStatus.FAILED), any(), any())
@@ -467,12 +465,11 @@ class CodeWhispererCodeModernizerSessionTest : CodeWhispererCodeModernizerTestBa
}
@Test
- fun `CodeModernizer cannot upload payload due to other status code`() {
+ fun `CodeModernizer cannot upload payload due to other status code`() = runTest {
doReturn(ZipCreationResult.Succeeded(File("./tst-resources/codemodernizer/test.txt")))
.whenever(testSessionContextSpy).createZipWithModuleFiles(any())
doReturn(exampleCreateUploadUrlResponse).whenever(clientAdaptorSpy).createGumbyUploadUrl(any())
- doAnswer { throw HttpRequests.HttpStatusException("mock error", 407, "mock url") }
- .whenever(clientAdaptorSpy).uploadArtifactToS3(any(), any(), any(), any(), any())
+ doAnswer { throw HttpRequests.HttpStatusException("mock error", 407, "mock url") }.whenever(testSessionSpy).uploadPayload(any())
val result = testSessionSpy.createModernizationJob(MavenCopyCommandsResult.Success(File("./mock/path/")))
assertEquals(CodeModernizerStartJobResult.ZipUploadFailed(UploadFailureReason.HTTP_ERROR(407)), result)
verify(testSessionStateSpy, times(1)).putJobHistory(any(), eq(TransformationStatus.FAILED), any(), any())
@@ -480,23 +477,23 @@ class CodeWhispererCodeModernizerSessionTest : CodeWhispererCodeModernizerTestBa
}
@Test
- fun `CodeModernizer cannot upload payload due to unknown issue`() {
+ fun `CodeModernizer cannot upload payload due to unknown client-side issue`() = runTest {
doReturn(ZipCreationResult.Succeeded(File("./tst-resources/codemodernizer/test.txt")))
.whenever(testSessionContextSpy).createZipWithModuleFiles(any())
doReturn(exampleCreateUploadUrlResponse).whenever(clientAdaptorSpy).createGumbyUploadUrl(any())
- doAnswer { throw IOException("mock exception") }.whenever(clientAdaptorSpy).uploadArtifactToS3(any(), any(), any(), any(), any())
+ doAnswer { throw Exception("mock client-side exception") }.whenever(clientAdaptorSpy).uploadArtifactToS3(any(), any(), any(), any(), any())
val result = testSessionSpy.createModernizationJob(MavenCopyCommandsResult.Success(File("./mock/path/")))
- assertEquals(CodeModernizerStartJobResult.ZipUploadFailed(UploadFailureReason.OTHER("mock exception")), result)
+ assertEquals(CodeModernizerStartJobResult.ZipUploadFailed(UploadFailureReason.OTHER("mock client-side exception")), result)
verify(testSessionStateSpy, times(1)).putJobHistory(any(), eq(TransformationStatus.FAILED), any(), any())
assertEquals(testSessionStateSpy.currentJobStatus, TransformationStatus.FAILED)
}
@Test
- fun `CodeModernizer cannot upload payload due to connection refused`() {
+ fun `CodeModernizer cannot upload payload due to connection refused`() = runTest {
doReturn(ZipCreationResult.Succeeded(File("./tst-resources/codemodernizer/test.txt")))
.whenever(testSessionContextSpy).createZipWithModuleFiles(any())
doReturn(exampleCreateUploadUrlResponse).whenever(clientAdaptorSpy).createGumbyUploadUrl(any())
- doAnswer { throw ConnectException("mock exception") }.whenever(clientAdaptorSpy).uploadArtifactToS3(any(), any(), any(), any(), any())
+ doAnswer { throw ConnectException("mock exception") }.whenever(testSessionSpy).uploadPayload(any())
val result = testSessionSpy.createModernizationJob(MavenCopyCommandsResult.Success(File("./mock/path/")))
assertEquals(CodeModernizerStartJobResult.ZipUploadFailed(UploadFailureReason.CONNECTION_REFUSED), result)
verify(testSessionStateSpy, times(1)).putJobHistory(any(), eq(TransformationStatus.FAILED), any(), any())
@@ -532,7 +529,7 @@ class CodeWhispererCodeModernizerSessionTest : CodeWhispererCodeModernizerTestBa
doNothing().whenever(testSessionStateSpy).updateJobHistory(any(), any(), any())
val result = testSessionSpy.pollUntilJobCompletion(CodeTransformType.LANGUAGE_UPGRADE, jobId) { _, _ -> }
- assertEquals(CodeModernizerJobCompletedResult.JobPartiallySucceeded(jobId, testSessionContextSpy.targetJavaVersion), result)
+ assertEquals(CodeModernizerJobCompletedResult.JobPartiallySucceeded(jobId), result)
verify(clientAdaptorSpy, times(4)).getCodeModernizationJob(any())
verify(clientAdaptorSpy, atLeastOnce()).getCodeModernizationPlan(any())
}
@@ -549,7 +546,7 @@ class CodeWhispererCodeModernizerSessionTest : CodeWhispererCodeModernizerTestBa
}
@Test
- fun `test uploadPayload()`() {
+ fun `test uploadPayload()`() = runTest {
val s3endpoint = "http://127.0.0.1:${wireMock.port()}"
val gumbyUploadUrlResponse = CreateUploadUrlResponse.builder()
.uploadUrl(s3endpoint)
diff --git a/plugins/amazonq/codetransform/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codemodernizer/CodeWhispererCodeModernizerTest.kt b/plugins/amazonq/codetransform/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codemodernizer/CodeWhispererCodeModernizerTest.kt
index e8de6cfc5c..633879f487 100644
--- a/plugins/amazonq/codetransform/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codemodernizer/CodeWhispererCodeModernizerTest.kt
+++ b/plugins/amazonq/codetransform/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codemodernizer/CodeWhispererCodeModernizerTest.kt
@@ -35,7 +35,6 @@ import software.aws.toolkits.jetbrains.services.codemodernizer.model.ValidationR
import software.aws.toolkits.jetbrains.services.codemodernizer.utils.filterOnlyParentFiles
import software.aws.toolkits.jetbrains.services.codemodernizer.utils.unzipFile
import software.aws.toolkits.telemetry.CodeTransformPreValidationError
-import software.aws.toolkits.telemetry.CodeTransformVCSViewerSrcComponents
import kotlin.io.path.Path
import kotlin.io.path.createTempDirectory
import kotlin.io.path.exists
@@ -55,7 +54,7 @@ class CodeWhispererCodeModernizerTest : CodeWhispererCodeModernizerTestBase() {
doNothing().whenever(handler).notifyUnableToApplyPatch(any())
val result = DownloadArtifactResult.ParseZipFailure(expectedError)
doReturn(result).whenever(handler).downloadArtifact(any(), eq(TransformationDownloadArtifactType.CLIENT_INSTRUCTIONS), eq(false))
- handler.displayDiff(jobId, CodeTransformVCSViewerSrcComponents.ToastNotification)
+ handler.displayDiff(jobId)
verify(handler, times(1)).notifyUnableToApplyPatch(any())
}
@@ -136,15 +135,14 @@ class CodeWhispererCodeModernizerTest : CodeWhispererCodeModernizerTestBase() {
doAnswer {
mockDialog.showAndGet()
mockDialog
- }.whenever(handler).displayDiffUsingPatch(any(), any(), any(), any(), any())
- handler.displayDiff(jobId, CodeTransformVCSViewerSrcComponents.Chat)
+ }.whenever(handler).displayDiffUsingPatch(any(), any(), any(), any())
+ handler.displayDiff(jobId)
verify(handler, never()).notifyUnableToApplyPatch(any())
verify(handler, times(1)).displayDiffUsingPatch(
testCodeModernizerArtifact.patches[0],
testCodeModernizerArtifact.patches.size,
testCodeModernizerArtifact.description?.get(0),
jobId,
- CodeTransformVCSViewerSrcComponents.Chat
)
}
diff --git a/plugins/amazonq/codetransform/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codemodernizer/panels/managers/CodeModernizerBottomWindowPanelManagerTest.kt b/plugins/amazonq/codetransform/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codemodernizer/panels/managers/CodeModernizerBottomWindowPanelManagerTest.kt
index 7b37fd3016..960d0dea35 100644
--- a/plugins/amazonq/codetransform/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codemodernizer/panels/managers/CodeModernizerBottomWindowPanelManagerTest.kt
+++ b/plugins/amazonq/codetransform/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codemodernizer/panels/managers/CodeModernizerBottomWindowPanelManagerTest.kt
@@ -32,6 +32,7 @@ class CodeModernizerBottomWindowPanelManagerTest : PanelTestBase() {
val layout = codeModernizerBottomWindowPanelManagerMock.layout as BorderLayout
assertThat(codeModernizerBottomWindowPanelManagerMock.toolbar.component.isVisible).isTrue()
assertThat(codeModernizerBottomWindowPanelManagerMock.fullSizeLoadingPanel.isVisible).isTrue()
+ assertThat(codeModernizerBottomWindowPanelManagerMock.banner.isVisible).isTrue()
assertThat(BorderLayout.WEST).isEqualTo(layout.getConstraints(codeModernizerBottomWindowPanelManagerMock.toolbar.component))
assertThat(BorderLayout.NORTH).isEqualTo(layout.getConstraints(codeModernizerBottomWindowPanelManagerMock.banner))
assertThat(layout.getLayoutComponent(BorderLayout.EAST)).isNull()
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/it/software/aws/toolkits/jetbrains/services/codewhisperer/CodeWhispererIntegrationTestBase.kt b/plugins/amazonq/codewhisperer/jetbrains-community/it/software/aws/toolkits/jetbrains/services/codewhisperer/CodeWhispererIntegrationTestBase.kt
index 779efabcea..5ff82bdae5 100644
--- a/plugins/amazonq/codewhisperer/jetbrains-community/it/software/aws/toolkits/jetbrains/services/codewhisperer/CodeWhispererIntegrationTestBase.kt
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/it/software/aws/toolkits/jetbrains/services/codewhisperer/CodeWhispererIntegrationTestBase.kt
@@ -98,7 +98,8 @@ open class CodeWhispererIntegrationTestBase(val projectRule: CodeInsightTestFixt
)
scanManager = spy(CodeWhispererCodeScanManager.getInstance(projectRule.project))
- doNothing().whenever(scanManager).addCodeScanUI(any())
+ doNothing().whenever(scanManager).buildCodeScanUI()
+ doNothing().whenever(scanManager).showCodeScanUI()
projectRule.project.replaceService(CodeWhispererCodeScanManager::class.java, scanManager, disposableRule.disposable)
telemetryServiceSpy = spy(CodeWhispererTelemetryService.getInstance())
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/resources/META-INF/plugin-codewhisperer.xml b/plugins/amazonq/codewhisperer/jetbrains-community/resources/META-INF/plugin-codewhisperer.xml
index 0c3e05ea1f..dcfb006872 100644
--- a/plugins/amazonq/codewhisperer/jetbrains-community/resources/META-INF/plugin-codewhisperer.xml
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/resources/META-INF/plugin-codewhisperer.xml
@@ -38,6 +38,7 @@
+
@@ -111,13 +112,13 @@
+
-
-
+
+
+
+
+ class="software.aws.toolkits.jetbrains.services.cwc.commands.codescan.actions.ExplainCodeIssueAction"/>
+
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/AmazonQCodeFixSession.kt b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/AmazonQCodeFixSession.kt
new file mode 100644
index 0000000000..a1452f74cf
--- /dev/null
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/AmazonQCodeFixSession.kt
@@ -0,0 +1,236 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.codewhisperer.codescan
+import com.intellij.openapi.project.Project
+import kotlinx.coroutines.delay
+import kotlinx.coroutines.ensureActive
+import kotlinx.coroutines.withTimeout
+import software.amazon.awssdk.services.codewhispererruntime.model.CodeFixJobStatus
+import software.amazon.awssdk.services.codewhispererruntime.model.CodeFixUploadContext
+import software.amazon.awssdk.services.codewhispererruntime.model.CreateUploadUrlRequest
+import software.amazon.awssdk.services.codewhispererruntime.model.CreateUploadUrlResponse
+import software.amazon.awssdk.services.codewhispererruntime.model.GetCodeFixJobRequest
+import software.amazon.awssdk.services.codewhispererruntime.model.GetCodeFixJobResponse
+import software.amazon.awssdk.services.codewhispererruntime.model.Position
+import software.amazon.awssdk.services.codewhispererruntime.model.Range
+import software.amazon.awssdk.services.codewhispererruntime.model.StartCodeFixJobRequest
+import software.amazon.awssdk.services.codewhispererruntime.model.StartCodeFixJobResponse
+import software.amazon.awssdk.services.codewhispererruntime.model.UploadContext
+import software.amazon.awssdk.services.codewhispererruntime.model.UploadIntent
+import software.aws.toolkits.core.utils.createTemporaryZipFile
+import software.aws.toolkits.core.utils.debug
+import software.aws.toolkits.core.utils.error
+import software.aws.toolkits.core.utils.getLogger
+import software.aws.toolkits.core.utils.info
+import software.aws.toolkits.core.utils.putNextEntry
+import software.aws.toolkits.jetbrains.services.codewhisperer.credentials.CodeWhispererClientAdaptor
+import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererConstants
+import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererZipUploadManager
+import software.aws.toolkits.jetbrains.services.codewhisperer.util.getTelemetryErrorMessage
+import software.aws.toolkits.resources.message
+import java.io.File
+import java.nio.file.Path
+import java.time.Instant
+import kotlin.coroutines.CoroutineContext
+import kotlin.coroutines.coroutineContext
+import kotlin.io.path.pathString
+
+class AmazonQCodeFixSession(val project: Project) {
+ private fun now() = Instant.now().toEpochMilli()
+
+ private val clientAdaptor get() = CodeWhispererClientAdaptor.getInstance(project)
+
+ suspend fun runCodeFixWorkflow(issue: CodeWhispererCodeScanIssue): CodeFixResponse {
+ val currentCoroutineContext = coroutineContext
+
+ try {
+ currentCoroutineContext.ensureActive()
+
+ /**
+ * * Step 1: Generate zip
+ * */
+ val sourceZip = withTimeout(CodeWhispererConstants.CODE_FIX_CREATE_PAYLOAD_TIMEOUT_IN_SECONDS) {
+ zipFile(issue.file.toNioPath())
+ }
+
+ /**
+ * * Step 2: Create upload URL and upload the zip
+ * */
+
+ currentCoroutineContext.ensureActive()
+ val codeFixName = issue.findingId
+ if (!sourceZip.exists()) {
+ return CodeFixResponse(
+ getCodeFixJobResponse = null,
+ failureResponse = message("codewhisperer.codefix.invalid_zip_error")
+ )
+ }
+
+ val sourceZipUploadResponse =
+ CodeWhispererZipUploadManager.getInstance(project).createUploadUrlAndUpload(
+ sourceZip,
+ "SourceCode",
+ CodeWhispererConstants.UploadTaskType.CODE_FIX,
+ codeFixName,
+ CodeWhispererConstants.FeatureName.CODE_REVIEW
+ )
+
+ /**
+ * * Step 3: Create code fix
+ * */
+ currentCoroutineContext.ensureActive()
+ val issueRange = Range.builder().start(
+ Position.builder()
+ .line(issue.startLine)
+ .character(0)
+ .build()
+ )
+ .end(
+ Position.builder()
+ .line(issue.endLine)
+ .character(0)
+ .build()
+ )
+ .build()
+
+ val createCodeFixResponse = createCodeFixJob(
+ sourceZipUploadResponse.uploadId(),
+ issueRange,
+ issue.recommendation.text,
+ codeFixName,
+ issue.ruleId
+ )
+ val codeFixJobId = createCodeFixResponse.jobId()
+ LOG.info { "Code fix job created: $codeFixJobId" }
+ /**
+ * * Step 4: polling for code fix
+ */
+ currentCoroutineContext.ensureActive()
+ val jobStatus = pollCodeFixJobStatus(createCodeFixResponse.jobId(), currentCoroutineContext)
+ if (jobStatus == CodeFixJobStatus.FAILED) {
+ LOG.debug { "Code fix generation failed." }
+ return CodeFixResponse(
+ getCodeFixJobResponse = null,
+ failureResponse = message("codewhisperer.codefix.create_code_fix_error")
+ )
+ }
+ /**
+ * Step 5: Process and render code fix results
+ */
+ currentCoroutineContext.ensureActive()
+ LOG.debug { "Code fix job succeeded and start processing result." }
+ val getCodeFixJobResponse = getCodeFixJob(createCodeFixResponse.jobId())
+ return CodeFixResponse(
+ getCodeFixJobResponse = getCodeFixJobResponse,
+ failureResponse = null,
+ jobId = codeFixJobId
+ )
+ } catch (e: Exception) {
+ LOG.error(e) { "Code scan session failed: ${e.message}" }
+ val timeoutMessage = message("codewhisperer.codefix.code_fix_job_timed_out")
+ return CodeFixResponse(
+ getCodeFixJobResponse = null,
+ failureResponse = when {
+ e is CodeWhispererCodeFixException && e.message == timeoutMessage -> timeoutMessage
+ else -> message("codewhisperer.codefix.create_code_fix_error")
+ }
+ )
+ }
+ }
+
+ fun createUploadUrl(md5Content: String, artifactType: String, codeFixName: String): CreateUploadUrlResponse = try {
+ clientAdaptor.createUploadUrl(
+ CreateUploadUrlRequest.builder()
+ .contentMd5(md5Content)
+ .artifactType(artifactType)
+ .uploadIntent(UploadIntent.CODE_FIX_GENERATION)
+ .uploadContext(UploadContext.fromCodeFixUploadContext(CodeFixUploadContext.builder().codeFixName(codeFixName).build()))
+ .build()
+ )
+ } catch (e: Exception) {
+ LOG.debug { "Create Upload URL failed: ${e.message}" }
+ val errorMessage = getTelemetryErrorMessage(e, featureUseCase = CodeWhispererConstants.FeatureName.CODE_REVIEW)
+ throw codeScanServerException("CreateUploadUrlException: $errorMessage")
+ }
+
+ private fun createCodeFixJob(
+ uploadId: String,
+ snippetRange: Range,
+ description: String,
+ codeFixName: String? = null,
+ ruleId: String? = null,
+ ): StartCodeFixJobResponse {
+ val request = StartCodeFixJobRequest.builder()
+ .uploadId(uploadId)
+ .snippetRange(snippetRange)
+ .codeFixName(codeFixName)
+ .ruleId(ruleId)
+ .description(description)
+ .build()
+
+ return try {
+ val response = clientAdaptor.startCodeFixJob(request)
+ LOG.debug {
+ "Code Fix Request id: ${response.responseMetadata().requestId()} " +
+ "and Code Fix Job id: ${response.jobId()}"
+ }
+ response
+ } catch (e: Exception) {
+ LOG.error { "Failed creating code fix job ${e.message}" }
+ throw CodeWhispererCodeFixException(message("codewhisperer.codefix.create_code_fix_error"))
+ }
+ }
+
+ private suspend fun pollCodeFixJobStatus(jobId: String, currentCoroutineContext: CoroutineContext): CodeFixJobStatus {
+ val pollingStartTime = now()
+ delay(CodeWhispererConstants.CODE_FIX_POLLING_INTERVAL_IN_SECONDS)
+ var status = CodeFixJobStatus.IN_PROGRESS
+ while (true) {
+ currentCoroutineContext.ensureActive()
+
+ val request = GetCodeFixJobRequest.builder()
+ .jobId(jobId)
+ .build()
+
+ val response = clientAdaptor.getCodeFixJob(request)
+ LOG.debug { "Request id: ${response.responseMetadata().requestId()}" }
+
+ if (response.jobStatus() != CodeFixJobStatus.IN_PROGRESS) {
+ status = response.jobStatus()
+ LOG.debug { "Code fix job status: ${status.name}" }
+ LOG.debug { "Complete polling code fix job status." }
+ break
+ }
+
+ currentCoroutineContext.ensureActive()
+ delay(CodeWhispererConstants.CODE_FIX_POLLING_INTERVAL_IN_SECONDS)
+
+ val elapsedTime = (now() - pollingStartTime) / CodeWhispererConstants.TOTAL_MILLIS_IN_SECOND // In seconds
+ if (elapsedTime > CodeWhispererConstants.CODE_FIX_TIMEOUT_IN_SECONDS) {
+ LOG.debug { "Code fix job status: ${status.name}" }
+ LOG.debug { "Code fix job failed. Amazon Q timed out." }
+ throw CodeWhispererCodeFixException(message("codewhisperer.codefix.code_fix_job_timed_out"))
+ }
+ }
+ return status
+ }
+
+ private fun getCodeFixJob(jobId: String): GetCodeFixJobResponse {
+ val response = clientAdaptor.getCodeFixJob(GetCodeFixJobRequest.builder().jobId(jobId).build())
+ return response
+ }
+ private fun zipFile(file: Path): File = createTemporaryZipFile {
+ try {
+ LOG.debug { "Selected file for truncation: $file" }
+ it.putNextEntry(file.toString(), file)
+ } catch (e: Exception) {
+ cannotFindFile("Zipping error: ${e.message}", file.pathString)
+ }
+ }.toFile()
+
+ companion object {
+ private val LOG = getLogger()
+ }
+ data class CodeFixResponse(val getCodeFixJobResponse: GetCodeFixJobResponse? = null, val failureResponse: String? = null, val jobId: String? = null)
+}
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/CodeWhispererCodeScanException.kt b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/CodeWhispererCodeScanException.kt
index 9407aaecf0..0380ca5fae 100644
--- a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/CodeWhispererCodeScanException.kt
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/CodeWhispererCodeScanException.kt
@@ -7,6 +7,8 @@ import software.aws.toolkits.resources.message
open class CodeWhispererCodeScanException(override val message: String?) : RuntimeException()
+open class CodeWhispererCodeFixException(override val message: String?) : RuntimeException()
+
open class CodeWhispererCodeScanServerException(override val message: String?) : RuntimeException()
internal fun noFileOpenError(): Nothing =
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/CodeWhispererCodeScanHighlightingFilesPanel.kt b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/CodeWhispererCodeScanHighlightingFilesPanel.kt
index 3bd8e425d9..59f6a7ad25 100644
--- a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/CodeWhispererCodeScanHighlightingFilesPanel.kt
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/CodeWhispererCodeScanHighlightingFilesPanel.kt
@@ -31,7 +31,7 @@ internal class CodeWhispererCodeScanHighlightingFilesPanel(private val project:
init {
removeAll()
- val scannedFilesTreeNodeRoot = DefaultMutableTreeNode("CodeWhisperer scanned files for security scan")
+ val scannedFilesTreeNodeRoot = DefaultMutableTreeNode("Amazon Q reviewed files for code issues")
files.forEach {
scannedFilesTreeNodeRoot.add(DefaultMutableTreeNode(it))
}
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/CodeWhispererCodeScanIssueDetailsPanel.kt b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/CodeWhispererCodeScanIssueDetailsPanel.kt
new file mode 100644
index 0000000000..3a1f7791a8
--- /dev/null
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/CodeWhispererCodeScanIssueDetailsPanel.kt
@@ -0,0 +1,323 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.codewhisperer.codescan
+
+import com.intellij.icons.AllIcons
+import com.intellij.ide.BrowserUtil
+import com.intellij.ide.ui.laf.darcula.ui.DarculaButtonUI
+import com.intellij.openapi.application.ApplicationManager
+import com.intellij.openapi.application.runInEdt
+import com.intellij.openapi.editor.colors.EditorColorsManager
+import com.intellij.openapi.fileEditor.FileEditorManager
+import com.intellij.openapi.fileEditor.OpenFileDescriptor
+import com.intellij.openapi.ide.CopyPasteManager
+import com.intellij.openapi.project.Project
+import com.intellij.ui.components.JBScrollPane
+import com.intellij.util.Alarm
+import kotlinx.coroutines.CoroutineScope
+import kotlinx.coroutines.launch
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.context.CodeScanIssueDetailsDisplayType
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils.additionBackgroundColor
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils.additionForegroundColor
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils.applySuggestedFix
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils.codeBlockBackgroundColor
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils.codeBlockBorderColor
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils.codeBlockForegroundColor
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils.deletionBackgroundColor
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils.deletionForegroundColor
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils.explainIssue
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils.getCodeScanIssueDetailsHtml
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils.getSeverityIcon
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils.metaBackgroundColor
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils.metaForegroundColor
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils.openDiff
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils.sendCodeFixGeneratedTelemetryToServiceAPI
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils.truncateIssueTitle
+import software.aws.toolkits.jetbrains.services.codewhisperer.telemetry.CodeWhispererTelemetryService
+import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererColorUtil.getHexString
+import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererConstants
+import software.aws.toolkits.jetbrains.settings.CodeWhispererSettings
+import software.aws.toolkits.resources.message
+import software.aws.toolkits.telemetry.Component
+import software.aws.toolkits.telemetry.MetricResult
+import java.awt.BorderLayout
+import java.awt.Dimension
+import java.awt.datatransfer.StringSelection
+import javax.swing.BorderFactory
+import javax.swing.Box
+import javax.swing.BoxLayout
+import javax.swing.JButton
+import javax.swing.JEditorPane
+import javax.swing.JLabel
+import javax.swing.JPanel
+import javax.swing.ScrollPaneConstants
+import javax.swing.event.HyperlinkEvent
+import javax.swing.text.html.HTMLEditorKit
+
+internal class CodeWhispererCodeScanIssueDetailsPanel(
+ private val project: Project,
+ issue: CodeWhispererCodeScanIssue,
+ private val defaultScope: CoroutineScope,
+) : JPanel(BorderLayout()) {
+ private val kit = HTMLEditorKit()
+ private val doc = kit.createDefaultDocument()
+ private val amazonQCodeFixSession = AmazonQCodeFixSession(project)
+ private val codeScanManager = CodeWhispererCodeScanManager.getInstance(project)
+
+ private suspend fun handleGenerateFix(issue: CodeWhispererCodeScanIssue, isRegenerate: Boolean = false) {
+ editorPane.text = getCodeScanIssueDetailsHtml(
+ issue, CodeScanIssueDetailsDisplayType.DetailsPane, CodeWhispererConstants.FixGenerationState.GENERATING,
+ project = project
+ )
+ editorPane.revalidate()
+ editorPane.repaint()
+ runInEdt {
+ editorPane.scrollToReference("fixLoadingSection")
+ }
+
+ val codeFixResponse: AmazonQCodeFixSession.CodeFixResponse = amazonQCodeFixSession.runCodeFixWorkflow(issue)
+ if (codeFixResponse.failureResponse != null) {
+ editorPane.apply {
+ text = getCodeScanIssueDetailsHtml(
+ issue, CodeScanIssueDetailsDisplayType.DetailsPane, CodeWhispererConstants.FixGenerationState.FAILED,
+ project = project
+ )
+ revalidate()
+ repaint()
+ runInEdt {
+ scrollToReference("fixFailureSection")
+ }
+ }
+ CodeWhispererTelemetryService.getInstance().sendCodeScanIssueGenerateFix(
+ Component.Webview,
+ issue,
+ isRegenerate,
+ MetricResult.Failed,
+ codeFixResponse.failureResponse
+ )
+ } else {
+ val isReferenceAllowed = CodeWhispererSettings.getInstance().isIncludeCodeWithReference()
+ var suggestedFix = SuggestedFix(
+ code = "",
+ description = ""
+ )
+ codeFixResponse.getCodeFixJobResponse?.suggestedFix()?.let {
+ it.codeDiff()?.let { codeDiff ->
+ // TODO: enable later
+ if (it.references() == null || it.references()?.isEmpty() == true) {
+ suggestedFix = SuggestedFix(
+ code = codeDiff.split("\n").drop(2).joinToString("\n"), // drop first two comment lines
+ description = it.description(),
+ codeFixJobId = codeFixResponse.jobId,
+ references = it.references(),
+ )
+ }
+ }
+ }
+
+ val showReferenceWarning = !isReferenceAllowed && suggestedFix.references.isNotEmpty()
+ if (suggestedFix.code.isNotEmpty() && !showReferenceWarning) {
+ issue.suggestedFixes = listOf(suggestedFix)
+ codeScanManager.updateIssue(issue)
+ }
+
+ editorPane.apply {
+ text = getCodeScanIssueDetailsHtml(
+ issue, CodeScanIssueDetailsDisplayType.DetailsPane, project = project,
+ showReferenceWarning = showReferenceWarning
+ )
+ revalidate()
+ repaint()
+ runInEdt {
+ scrollToReference("codeFixActions")
+ }
+ }
+
+ buttonPane.apply {
+ removeAll()
+ if (issue.suggestedFixes.isNotEmpty()) add(applyFixButton)
+ add(regenerateFixButton)
+ add(explainIssueButton)
+ add(ignoreIssueButton)
+ add(ignoreIssuesButton)
+ add(Box.createHorizontalGlue())
+ revalidate()
+ repaint()
+ }
+ ApplicationManager.getApplication().executeOnPooledThread {
+ if (suggestedFix.code.isNotBlank()) {
+ sendCodeFixGeneratedTelemetryToServiceAPI(issue, false)
+ }
+ CodeWhispererTelemetryService.getInstance().sendCodeScanIssueGenerateFix(Component.Webview, issue, isRegenerate, MetricResult.Succeeded)
+ }
+ }
+ }
+
+ private val editorPane = JEditorPane().apply {
+ contentType = "text/html"
+ putClientProperty(JEditorPane.HONOR_DISPLAY_PROPERTIES, true)
+ border = BorderFactory.createCompoundBorder(
+ BorderFactory.createEmptyBorder(),
+ BorderFactory.createEmptyBorder(3, 10, 8, 11)
+ )
+ val editorColorsScheme = EditorColorsManager.getInstance().globalScheme
+ background = editorColorsScheme.defaultBackground
+ isEditable = false
+ addHyperlinkListener { he ->
+ if (he.eventType == HyperlinkEvent.EventType.ACTIVATED) {
+ when {
+ he.description.startsWith("amazonq://issue/openDiff-") -> {
+ openDiff(issue)
+ }
+ he.description.startsWith("amazonq://issue/copyDiff-") -> {
+ text = getCodeScanIssueDetailsHtml(
+ issue,
+ CodeScanIssueDetailsDisplayType.DetailsPane,
+ CodeWhispererConstants.FixGenerationState.COMPLETED,
+ true,
+ project = project
+ )
+ CopyPasteManager.getInstance().setContents(StringSelection(issue.suggestedFixes.first().code))
+ val alarm = Alarm()
+ alarm.addRequest({
+ ApplicationManager.getApplication().invokeLater {
+ text = getCodeScanIssueDetailsHtml(
+ issue,
+ CodeScanIssueDetailsDisplayType.DetailsPane,
+ CodeWhispererConstants.FixGenerationState.COMPLETED,
+ false,
+ project = project
+ )
+ }
+ }, 500)
+ }
+ he.description.startsWith("amazonq://issue/openFile-") -> {
+ runInEdt {
+ FileEditorManager.getInstance(project).openTextEditor(
+ OpenFileDescriptor(project, issue.file),
+ true
+ )
+ }
+ }
+ else -> {
+ BrowserUtil.browse(he.url)
+ }
+ }
+ }
+ }
+ editorKit = kit
+ document = doc
+ text = getCodeScanIssueDetailsHtml(issue, CodeScanIssueDetailsDisplayType.DetailsPane, project = project)
+ caretPosition = 0
+ }
+
+ private val scrollPane = JBScrollPane(editorPane).apply {
+ verticalScrollBarPolicy = ScrollPaneConstants.VERTICAL_SCROLLBAR_AS_NEEDED
+ horizontalScrollBarPolicy = ScrollPaneConstants.HORIZONTAL_SCROLLBAR_AS_NEEDED
+ }
+ private val severityLabel = JLabel(truncateIssueTitle(issue.title)).apply {
+ icon = getSeverityIcon(issue)
+ horizontalTextPosition = JLabel.LEFT
+ font = font.deriveFont(16f)
+ }
+ private val applyFixButton = JButton(message("codewhisperer.codescan.apply_fix_button_label")).apply {
+ putClientProperty(DarculaButtonUI.DEFAULT_STYLE_KEY, true)
+ addActionListener {
+ applySuggestedFix(project, issue)
+ }
+ }
+ private val generateFixButton = JButton(message("codewhisperer.codescan.generate_fix_button_label")).apply {
+ putClientProperty(DarculaButtonUI.DEFAULT_STYLE_KEY, true)
+ addActionListener {
+ defaultScope.launch {
+ handleGenerateFix(issue)
+ }
+ }
+ }
+ private val regenerateFixButton = JButton(message("codewhisperer.codescan.regenerate_fix_button_label")).apply {
+ putClientProperty(DarculaButtonUI.DEFAULT_STYLE_KEY, true)
+ addActionListener {
+ defaultScope.launch {
+ handleGenerateFix(issue, isRegenerate = true)
+ }
+ }
+ }
+ private val explainIssueButton = JButton(message("codewhisperer.codescan.explain_button_label")).apply {
+ addActionListener {
+ explainIssue(issue)
+ }
+ }
+ private val ignoreIssueButton = JButton(message("codewhisperer.codescan.ignore_button")).apply {
+ addActionListener {
+ codeScanManager.ignoreSingleIssue(issue)
+ ApplicationManager.getApplication().executeOnPooledThread {
+ CodeWhispererTelemetryService.getInstance().sendCodeScanIssueIgnore(Component.Webview, issue, false)
+ }
+ }
+ }
+ private val ignoreIssuesButton = JButton(message("codewhisperer.codescan.ignore_all_button")).apply {
+ addActionListener {
+ codeScanManager.ignoreAllIssues(issue)
+ ApplicationManager.getApplication().executeOnPooledThread {
+ CodeWhispererTelemetryService.getInstance().sendCodeScanIssueIgnore(Component.Webview, issue, true)
+ }
+ }
+ }
+ private val closeDetailsButton = JButton(AllIcons.Actions.CloseDarkGrey).apply {
+ border = null
+ margin = null
+ isContentAreaFilled = false
+ putClientProperty(DarculaButtonUI.DEFAULT_STYLE_KEY, true)
+ addActionListener {
+ hideIssueDetails()
+ }
+ }
+ private val titlePane = JPanel().apply {
+ layout = BoxLayout(this, BoxLayout.X_AXIS)
+ preferredSize = Dimension(this.width, 30)
+ add(Box.createHorizontalStrut(10))
+ add(severityLabel)
+ add(Box.createHorizontalGlue())
+ add(closeDetailsButton)
+ }
+ private val buttonPane = JPanel().apply {
+ layout = BoxLayout(this, BoxLayout.X_AXIS)
+ preferredSize = Dimension(this.width, 30)
+ if (issue.suggestedFixes.isNotEmpty()) add(applyFixButton)
+ if (issue.suggestedFixes.isNotEmpty()) add(regenerateFixButton) else add(generateFixButton)
+ add(explainIssueButton)
+ add(ignoreIssueButton)
+ add(ignoreIssuesButton)
+ add(Box.createHorizontalGlue())
+ }
+ private fun hideIssueDetails() {
+ isVisible = false
+ revalidate()
+ repaint()
+ }
+
+ init {
+ removeAll()
+ kit.styleSheet.apply {
+ addRule("h1, h3 { margin-bottom: 0 }")
+ addRule("th { text-align: left; }")
+ addRule(".code-block { background-color: ${codeBlockBackgroundColor.getHexString()}; border: 1px solid ${codeBlockBorderColor.getHexString()}; }")
+ addRule(".code-block pre { margin: 0; }")
+ addRule(".code-block div { color: ${codeBlockForegroundColor.getHexString()}; }")
+ addRule(
+ ".code-block div.deletion { background-color: ${deletionBackgroundColor.getHexString()}; color: ${deletionForegroundColor.getHexString()}; }"
+ )
+ addRule(
+ ".code-block div.addition { background-color: ${additionBackgroundColor.getHexString()}; color: ${additionForegroundColor.getHexString()}; }"
+ )
+ addRule(".code-block div.meta { background-color: ${metaBackgroundColor.getHexString()}; color: ${metaForegroundColor.getHexString()}; }")
+ }
+
+ add(BorderLayout.NORTH, titlePane)
+ add(BorderLayout.CENTER, scrollPane)
+ add(BorderLayout.SOUTH, buttonPane)
+ isVisible = true
+ revalidate()
+ }
+}
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/CodeWhispererCodeScanManager.kt b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/CodeWhispererCodeScanManager.kt
index 78481866a8..1f715b32d3 100644
--- a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/CodeWhispererCodeScanManager.kt
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/CodeWhispererCodeScanManager.kt
@@ -7,11 +7,21 @@ import com.intellij.analysis.problemsView.toolWindow.ProblemsView
import com.intellij.codeHighlighting.HighlightDisplayLevel
import com.intellij.codeInspection.util.InspectionMessage
import com.intellij.icons.AllIcons
+import com.intellij.lang.Commenter
+import com.intellij.lang.Language
+import com.intellij.lang.LanguageCommenters
+import com.intellij.openapi.actionSystem.ActionManager
+import com.intellij.openapi.actionSystem.AnActionEvent
+import com.intellij.openapi.actionSystem.CommonDataKeys
+import com.intellij.openapi.actionSystem.impl.SimpleDataContext
import com.intellij.openapi.application.ApplicationInfo
import com.intellij.openapi.application.ApplicationManager
+import com.intellij.openapi.application.readAction
import com.intellij.openapi.application.runInEdt
import com.intellij.openapi.application.runReadAction
+import com.intellij.openapi.command.WriteCommandAction
import com.intellij.openapi.components.service
+import com.intellij.openapi.editor.Document
import com.intellij.openapi.editor.EditorFactory
import com.intellij.openapi.editor.ex.RangeHighlighterEx
import com.intellij.openapi.editor.impl.DocumentMarkupModel
@@ -20,13 +30,13 @@ import com.intellij.openapi.editor.markup.HighlighterTargetArea
import com.intellij.openapi.editor.markup.MarkupModel
import com.intellij.openapi.fileEditor.FileDocumentManager
import com.intellij.openapi.fileEditor.FileEditorManager
-import com.intellij.openapi.fileEditor.impl.FileDocumentManagerImpl
import com.intellij.openapi.project.Project
-import com.intellij.openapi.ui.MessageDialogBuilder
+import com.intellij.openapi.project.guessProjectDir
import com.intellij.openapi.util.Disposer
import com.intellij.openapi.util.TextRange
import com.intellij.openapi.vfs.VirtualFile
import com.intellij.openapi.vfs.isFile
+import com.intellij.psi.PsiDocumentManager
import com.intellij.refactoring.suggested.range
import com.intellij.ui.content.ContentManagerEvent
import com.intellij.ui.content.ContentManagerListener
@@ -51,6 +61,7 @@ import software.aws.toolkits.core.utils.error
import software.aws.toolkits.core.utils.getLogger
import software.aws.toolkits.core.utils.info
import software.aws.toolkits.core.utils.warn
+import software.aws.toolkits.jetbrains.core.coroutines.EDT
import software.aws.toolkits.jetbrains.core.coroutines.getCoroutineUiContext
import software.aws.toolkits.jetbrains.core.coroutines.projectCoroutineScope
import software.aws.toolkits.jetbrains.core.credentials.ToolkitConnectionManager
@@ -59,6 +70,8 @@ import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.listeners
import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.listeners.CodeWhispererCodeScanEditorMouseMotionListener
import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.listeners.CodeWhispererCodeScanFileListener
import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.sessionconfig.CodeScanSessionConfig
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils.AmazonQCodeReviewGitUtils.isInsideWorkTree
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils.IssueSeverity
import software.aws.toolkits.jetbrains.services.codewhisperer.credentials.CodeWhispererClientAdaptor
import software.aws.toolkits.jetbrains.services.codewhisperer.editor.CodeWhispererEditorUtil.overlaps
import software.aws.toolkits.jetbrains.services.codewhisperer.explorer.CodeWhispererExplorerActionManager
@@ -66,13 +79,19 @@ import software.aws.toolkits.jetbrains.services.codewhisperer.explorer.isUserBui
import software.aws.toolkits.jetbrains.services.codewhisperer.language.CodeWhispererProgrammingLanguage
import software.aws.toolkits.jetbrains.services.codewhisperer.language.languages.CodeWhispererPlainText
import software.aws.toolkits.jetbrains.services.codewhisperer.language.languages.CodeWhispererUnknownLanguage
+import software.aws.toolkits.jetbrains.services.codewhisperer.language.programmingLanguage
+import software.aws.toolkits.jetbrains.services.codewhisperer.model.CodeScanResponseContext
import software.aws.toolkits.jetbrains.services.codewhisperer.model.CodeScanTelemetryEvent
import software.aws.toolkits.jetbrains.services.codewhisperer.telemetry.CodeWhispererTelemetryService
import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererColorUtil.INACTIVE_TEXT_COLOR
import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererConstants
import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererConstants.ISSUE_HIGHLIGHT_TEXT_ATTRIBUTES
+import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererConstants.amazonqIgnoreNextLine
+import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererConstants.scanResultsKey
+import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererConstants.scanScopeKey
import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererUtil.promptReAuth
import software.aws.toolkits.jetbrains.services.codewhisperer.util.runIfIdcConnectionOrTelemetryEnabled
+import software.aws.toolkits.jetbrains.settings.CodeWhispererSettings
import software.aws.toolkits.jetbrains.utils.isQConnected
import software.aws.toolkits.jetbrains.utils.isQExpired
import software.aws.toolkits.jetbrains.utils.isRunningOnRemoteBackend
@@ -89,8 +108,9 @@ import kotlin.coroutines.CoroutineContext
private val LOG = getLogger()
class CodeWhispererCodeScanManager(val project: Project) {
+ private val defaultScope = projectCoroutineScope(project)
private val codeScanResultsPanel by lazy {
- CodeWhispererCodeScanResultsView(project)
+ CodeWhispererCodeScanResultsView(project, defaultScope)
}
private val codeScanIssuesContent by lazy {
val contentManager = getProblemsWindow().contentManager
@@ -108,16 +128,24 @@ class CodeWhispererCodeScanManager(val project: Project) {
}
}
- private val fileNodeLookup = mutableMapOf()
+ private var autoScanIssues = emptyList()
+ private var ondemandScanIssues = emptyList()
+ private fun getCombinedScanIssues() = getUniqueIssues(autoScanIssues + ondemandScanIssues)
+ private val severityNodeLookup = mapOf(
+ IssueSeverity.CRITICAL.displayName to DefaultMutableTreeNode(IssueSeverity.CRITICAL.displayName),
+ IssueSeverity.HIGH.displayName to DefaultMutableTreeNode(IssueSeverity.HIGH.displayName),
+ IssueSeverity.MEDIUM.displayName to DefaultMutableTreeNode(IssueSeverity.MEDIUM.displayName),
+ IssueSeverity.LOW.displayName to DefaultMutableTreeNode(IssueSeverity.LOW.displayName),
+ IssueSeverity.INFO.displayName to DefaultMutableTreeNode(IssueSeverity.INFO.displayName)
+ )
private val scanNodesLookup = mutableMapOf>()
+ private val selectedSeverityValues = IssueSeverity.entries.associate { it.displayName to true }.toMutableMap()
private val documentListener = CodeWhispererCodeScanDocumentListener(project)
private val editorMouseListener = CodeWhispererCodeScanEditorMouseMotionListener(project)
private val fileListener = CodeWhispererCodeScanFileListener(project)
- private val isProjectScanInProgress = AtomicBoolean(false)
-
- private val defaultScope = projectCoroutineScope(project)
+ private val isOnDemandScanInProgress = AtomicBoolean(false)
private lateinit var codeScanJob: Job
private lateinit var debouncedCodeScanJob: Job
@@ -126,33 +154,154 @@ class CodeWhispererCodeScanManager(val project: Project) {
* Returns true if the code scan is in progress.
* This function will return true for a cancelled code scan job which is in cancellation state.
*/
- fun isProjectScanInProgress(): Boolean = isProjectScanInProgress.get()
+ fun isOnDemandScanInProgress(): Boolean = isOnDemandScanInProgress.get()
/**
* Code scan job is active when the [Job] is started and is in active state.
*/
- fun isCodeScanJobActive(): Boolean = this::codeScanJob.isInitialized && codeScanJob.isActive && isProjectScanInProgress()
+ fun isCodeScanJobActive(): Boolean = this::codeScanJob.isInitialized && codeScanJob.isActive && isOnDemandScanInProgress()
- fun getRunActionButtonIcon(): Icon = if (isProjectScanInProgress()) AllIcons.Process.Step_1 else AllIcons.Actions.Execute
+ fun getRunActionButtonIcon(): Icon = if (isOnDemandScanInProgress()) AllIcons.Process.Step_1 else AllIcons.Actions.Execute
- fun getActionButtonIconForExplorerNode(): Icon = if (isProjectScanInProgress()) AllIcons.Actions.Suspend else AllIcons.Actions.Execute
+ fun getActionButtonIconForExplorerNode(): Icon = if (isOnDemandScanInProgress()) AllIcons.Actions.Suspend else AllIcons.Actions.Execute
- fun getActionButtonText(): String = if (!isProjectScanInProgress()) {
+ fun getActionButtonText(): String = if (!isOnDemandScanInProgress()) {
message(
- "codewhisperer.codescan.run_scan"
+ "codewhisperer.codescan.run_scan",
+ INACTIVE_TEXT_COLOR
)
} else {
message("codewhisperer.codescan.stop_scan")
}
+ private fun isIgnoredIssueTitle(title: String) = getIgnoredIssueTitles().contains(title)
+
+ fun isIgnoredIssue(title: String, document: Document, file: VirtualFile, startLine: Int) = isIgnoredIssueTitle(title) ||
+ detectSingleIssueIgnored(document, file, startLine)
+
+ private fun getIgnoredIssueTitles() = CodeWhispererSettings.getInstance().getIgnoredCodeReviewIssues().split(";").toMutableSet()
+
+ fun ignoreAllIssues(issue: CodeWhispererCodeScanIssue) {
+ val ignoredIssueTitles = getIgnoredIssueTitles()
+ ignoredIssueTitles.add(issue.title)
+ CodeWhispererSettings.getInstance().setIgnoredCodeReviewIssues(ignoredIssueTitles.joinToString(separator = ";"))
+ // update the in memory copy and UI
+ ondemandScanIssues = ondemandScanIssues.filter { it.title != issue.title }
+ autoScanIssues = autoScanIssues.filter { it.title != issue.title }
+ updateCodeScanIssuesTree()
+ }
+
+ private fun detectSingleIssueIgnored(document: Document, file: VirtualFile, startLine: Int): Boolean = runReadAction {
+ try {
+ if (startLine == 0) return@runReadAction false
+ val commenter = getLanguageCommenter(document, project)
+ val linePrefix: String? = commenter?.lineCommentPrefix ?: file.programmingLanguage().lineCommentPrefix()
+ val blockPrefix: String? = commenter?.blockCommentPrefix ?: file.programmingLanguage().blockCommentPrefix()
+ val blockSuffix: String? = commenter?.blockCommentSuffix ?: file.programmingLanguage().blockCommentSuffix()
+
+ for (i in (startLine - 1) downTo 0) {
+ val lineStart = document.getLineStartOffset(i)
+ val lineEnd = document.getLineEndOffset(i)
+ val targetRange = TextRange(lineStart, lineEnd)
+ val lineText = document.getText(targetRange)
+ if (lineText.isEmpty()) {
+ continue
+ }
+ if (linePrefix != null &&
+ lineText.trimIndent().startsWith(linePrefix) &&
+ lineText.contains(amazonqIgnoreNextLine)
+ ) {
+ return@runReadAction true
+ }
+ if (blockPrefix != null &&
+ blockSuffix != null &&
+ lineText.trimIndent().startsWith(blockPrefix) &&
+ lineText.contains(amazonqIgnoreNextLine) &&
+ lineText.trimEnd().endsWith(blockSuffix)
+ ) {
+ return@runReadAction true
+ }
+ return@runReadAction false
+ }
+ return@runReadAction false
+ } catch (e: Exception) {
+ LOG.warn { "Failed to detect if scan issue is ignored: ${e.stackTraceToString()}" }
+ return@runReadAction false
+ }
+ }
+
+ fun ignoreSingleIssue(issue: CodeWhispererCodeScanIssue) {
+ val document = issue.document
+ var commentString: String? = null
+ var insertOffset: Int? = null
+ try {
+ runReadAction {
+ if (!issue.isVisible) return@runReadAction
+ val lineNumber = issue.startLine
+ val issueRange = TextRange(document.getLineStartOffset(lineNumber - 1), document.getLineEndOffset(lineNumber - 1))
+ val lineContent = document.getText(issueRange)
+ val indentation = lineContent.takeWhile { it.isWhitespace() }
+ insertOffset = document.getLineStartOffset(lineNumber - 1)
+ val commenter = getLanguageCommenter(document, project)
+ val linePrefix: String? = commenter?.lineCommentPrefix ?: issue.file.programmingLanguage().lineCommentPrefix()
+ val blockPrefix: String? = commenter?.blockCommentPrefix ?: issue.file.programmingLanguage().blockCommentPrefix()
+ val blockSuffix: String? = commenter?.blockCommentSuffix ?: issue.file.programmingLanguage().blockCommentSuffix()
+ if (linePrefix != null) {
+ commentString = "$indentation$linePrefix $amazonqIgnoreNextLine\n"
+ } else if (blockPrefix != null && blockSuffix != null) {
+ commentString = "$indentation$blockPrefix $amazonqIgnoreNextLine $blockSuffix\n"
+ }
+ }
+ val finalOffset = insertOffset ?: return
+ val finalCommentString = commentString ?: return
+ ApplicationManager.getApplication().invokeLater {
+ WriteCommandAction.runWriteCommandAction(project) {
+ document.insertString(finalOffset, finalCommentString)
+ }
+ }
+ } catch (e: Exception) {
+ LOG.warn { "Failed to insert ignore comment ${e.stackTraceToString()}" }
+ }
+ ondemandScanIssues = ondemandScanIssues.filter { it.findingId != issue.findingId }
+ autoScanIssues = autoScanIssues.filter { it.findingId != issue.findingId }
+ removeIssueByFindingId(issue, issue.findingId)
+ }
+
+ private fun getLanguageCommenter(document: Document, project: Project): Commenter? {
+ // TODO: need to implement fall back for languages not supported by IDE
+ val language: Language = document
+ .let { PsiDocumentManager.getInstance(project).getPsiFile(it) }
+ ?.language ?: return null
+ return LanguageCommenters.INSTANCE.forLanguage(language)
+ }
+
+ fun isSeveritySelected(severity: String): Boolean = selectedSeverityValues[severity] ?: true
+ fun setSeveritySelected(severity: String, selected: Boolean) {
+ selectedSeverityValues[severity] = selected
+ updateCodeScanIssuesTree()
+ }
+
+ /**
+ * Returns true if there are any code scan issues.
+ */
+ fun hasCodeScanIssues(): Boolean = getCombinedScanIssues().isNotEmpty()
+
+ /**
+ * Clears all filters and updates the code scan issues tree.
+ */
+ fun clearFilters() {
+ selectedSeverityValues.keys.forEach { selectedSeverityValues[it] = true }
+ updateCodeScanIssuesTree()
+ }
+
/**
* Triggers a code scan and displays results in the new tab in problems view panel.
*/
- fun runCodeScan(scope: CodeWhispererConstants.CodeAnalysisScope, isPluginStarting: Boolean = false) {
+ fun runCodeScan(scope: CodeWhispererConstants.CodeAnalysisScope, isPluginStarting: Boolean = false, initiatedByChat: Boolean = false) {
if (!isQConnected(project)) return
// Return if a scan is already in progress.
- if (isProjectScanInProgress() && scope == CodeWhispererConstants.CodeAnalysisScope.PROJECT) return
+ if (isOnDemandScanInProgress()) return
val connectionExpired = if (isPluginStarting) {
isQExpired(project)
@@ -160,24 +309,30 @@ class CodeWhispererCodeScanManager(val project: Project) {
promptReAuth(project)
}
if (connectionExpired) {
- isProjectScanInProgress.set(false)
+ isOnDemandScanInProgress.set(false)
return
}
// If scope is project
if (scope == CodeWhispererConstants.CodeAnalysisScope.PROJECT) {
- // Prepare for a project code scan
- beforeCodeScan()
// launch code scan coroutine
- codeScanJob = launchCodeScanCoroutine(CodeWhispererConstants.CodeAnalysisScope.PROJECT)
+ try {
+ codeScanJob = launchCodeScanCoroutine(CodeWhispererConstants.CodeAnalysisScope.PROJECT, initiatedByChat)
+ } catch (e: CancellationException) {
+ notifyChat(codeScanResponse = null, scope = scope)
+ }
} else {
- if (CodeWhispererExplorerActionManager.getInstance().isAutoEnabledForCodeScan() and !isUserBuilderId(project)) {
+ if (CodeWhispererExplorerActionManager.getInstance().isAutoEnabledForCodeScan() and !isUserBuilderId(project) || initiatedByChat) {
// cancel if a file scan is running.
- if (!isProjectScanInProgress() && this::codeScanJob.isInitialized && codeScanJob.isActive) {
+ if (!isOnDemandScanInProgress() && this::codeScanJob.isInitialized && codeScanJob.isActive) {
codeScanJob.cancel()
}
// Add File Scan
- codeScanJob = launchCodeScanCoroutine(CodeWhispererConstants.CodeAnalysisScope.FILE)
+ try {
+ codeScanJob = launchCodeScanCoroutine(CodeWhispererConstants.CodeAnalysisScope.FILE, initiatedByChat)
+ } catch (e: CancellationException) {
+ notifyChat(codeScanResponse = null, scope = scope)
+ }
}
}
}
@@ -200,25 +355,37 @@ class CodeWhispererCodeScanManager(val project: Project) {
}
}
- fun stopCodeScan() {
+ fun stopCodeScan(scope: CodeWhispererConstants.CodeAnalysisScope) {
// Return if code scan job is not active.
- if (!codeScanJob.isActive) return
- if (isProjectScanInProgress() && confirmCancelCodeScan()) {
- LOG.info { "Security scan stopped by user..." }
+ if (!codeScanJob.isActive) {
+ notifyChat(codeScanResponse = null, scope = scope)
+ return
+ }
+ // TODO: need to check if we need to ask for user's confirmation again
+ if (isOnDemandScanInProgress()) {
+ LOG.info { "Code Review stopped by user..." }
// Checking `codeScanJob.isActive` to ensure that the job is not already completed by the time user confirms.
if (codeScanJob.isActive) {
codeScanResultsPanel.setStoppingCodeScan()
+ notifyChat(codeScanResponse = null, scope = scope)
codeScanJob.cancel(CancellationException("User requested cancellation"))
}
}
}
- private fun confirmCancelCodeScan(): Boolean = MessageDialogBuilder
- .okCancel(message("codewhisperer.codescan.stop_scan"), message("codewhisperer.codescan.stop_scan_confirm_message"))
- .yesText(message("codewhisperer.codescan.stop_scan_confirm_button"))
- .ask(project)
-
- private fun launchCodeScanCoroutine(scope: CodeWhispererConstants.CodeAnalysisScope) = projectCoroutineScope(project).launch {
+ private suspend fun isInValidFile(
+ selectedFile: VirtualFile?,
+ language: CodeWhispererProgrammingLanguage,
+ codeScanSessionConfig: CodeScanSessionConfig,
+ ): Boolean =
+ selectedFile == null ||
+ !language.isAutoFileScanSupported() ||
+ !selectedFile.isFile ||
+ selectedFile.fileSystem.protocol == "remoteDeploymentFS" ||
+ readAction { codeScanSessionConfig.fileIndex.isInLibrarySource(selectedFile) }
+
+ private fun launchCodeScanCoroutine(scope: CodeWhispererConstants.CodeAnalysisScope, initiatedByChat: Boolean) = projectCoroutineScope(project).launch {
+ if (scope == CodeWhispererConstants.CodeAnalysisScope.PROJECT || initiatedByChat) beforeCodeScan()
var codeScanStatus: Result = Result.Failed
val startTime = Instant.now().toEpochMilli()
var codeScanResponseContext = defaultCodeScanResponseContext()
@@ -232,18 +399,14 @@ class CodeWhispererCodeScanManager(val project: Project) {
} else {
FileEditorManager.getInstance(project).selectedEditor?.file
}
- val codeScanSessionConfig = CodeScanSessionConfig.create(file, project, scope)
+ val codeScanSessionConfig = CodeScanSessionConfig.create(file, project, scope, initiatedByChat)
val selectedFile = codeScanSessionConfig.getSelectedFile()
language = codeScanSessionConfig.getProgrammingLanguage()
if (scope == CodeWhispererConstants.CodeAnalysisScope.FILE &&
- (
- selectedFile == null || !language.isAutoFileScanSupported() || !selectedFile.isFile ||
- runReadAction { (codeScanSessionConfig.fileIndex.isInLibrarySource(selectedFile)) } ||
- selectedFile.fileSystem.protocol == "remoteDeploymentFS"
- )
+ isInValidFile(selectedFile, language, codeScanSessionConfig) && !initiatedByChat
) {
skipTelemetry = true
- LOG.debug { "Language is unknown or plaintext, skipping code scan." }
+ LOG.debug { "Language is unknown or plaintext, skipping code review." }
codeScanStatus = Result.Cancelled
return@launch
} else {
@@ -257,41 +420,51 @@ class CodeWhispererCodeScanManager(val project: Project) {
codeScanResponseContext = codeScanResponse.responseContext
when (codeScanResponse) {
is CodeScanResponse.Success -> {
- val issues = codeScanResponse.issues
+ if (initiatedByChat) {
+ ondemandScanIssues = if (scope == CodeWhispererConstants.CodeAnalysisScope.PROJECT) {
+ codeScanResponse.issues
+ } else {
+ ondemandScanIssues + codeScanResponse.issues
+ }
+ } else {
+ autoScanIssues = codeScanResponse.issues
+ }
coroutineContext.ensureActive()
renderResponseOnUIThread(
- issues,
+ getCombinedScanIssues(),
codeScanResponse.responseContext.payloadContext.scannedFiles,
scope
)
codeScanStatus = Result.Succeeded
+ if (initiatedByChat) {
+ notifyChat(getChatMessageResponse(codeScanResponse, scope), scope = scope)
+ }
}
is CodeScanResponse.Failure -> {
if (codeScanResponse.failureReason !is TimeoutCancellationException && codeScanResponse.failureReason is CancellationException) {
codeScanStatus = Result.Cancelled
}
+ if (initiatedByChat) {
+ notifyChat(codeScanResponse, scope = scope)
+ }
throw codeScanResponse.failureReason
}
}
- LOG.info { "Security scan completed for jobID: ${codeScanResponseContext.codeScanJobId}." }
+ LOG.info { "Code review completed for jobID: ${codeScanResponseContext.codeScanJobId}." }
}
}
} catch (e: Error) {
- if (scope == CodeWhispererConstants.CodeAnalysisScope.PROJECT) {
- isProjectScanInProgress.set(false)
- }
- val errorMessage = handleError(coroutineContext, e, scope)
+ afterCodeScan(scope, initiatedByChat)
+ val errorMessage = handleError(coroutineContext, e)
codeScanResponseContext = codeScanResponseContext.copy(reason = errorMessage)
} catch (e: Exception) {
- if (scope == CodeWhispererConstants.CodeAnalysisScope.PROJECT) {
- isProjectScanInProgress.set(false)
- }
+ afterCodeScan(scope, initiatedByChat)
val errorMessage = handleException(coroutineContext, e, scope)
codeScanResponseContext = codeScanResponseContext.copy(reason = errorMessage)
} finally {
// After code scan
- afterCodeScan(scope)
+ afterCodeScan(scope, initiatedByChat)
if (!skipTelemetry) {
launch {
val duration = (Instant.now().toEpochMilli() - startTime).toDouble()
@@ -302,16 +475,74 @@ class CodeWhispererCodeScanManager(val project: Project) {
codeScanStatus,
codeScanResponseContext.payloadContext.srcPayloadSize.toDouble() ?: 0.0,
connection,
- scope
+ scope,
+ initiatedByChat
)
)
- sendCodeScanTelemetryToServiceAPI(project, language, codeScanResponseContext.codeScanJobId, scope)
+ sendCodeScanTelemetryToServiceAPI(project, language, codeScanResponseContext, scope)
}
}
}
}
- fun handleError(coroutineContext: CoroutineContext, e: Error, scope: CodeWhispererConstants.CodeAnalysisScope): String {
+ private fun getChatMessageResponse(originalResponse: CodeScanResponse.Success, scope: CodeWhispererConstants.CodeAnalysisScope): CodeScanResponse {
+ if (originalResponse.issues.isEmpty()) return originalResponse
+ val chatIssues = if (scope == CodeWhispererConstants.CodeAnalysisScope.PROJECT) {
+ getCombinedScanIssues()
+ } else {
+ getCombinedScanIssues().filter { issue -> issue.file == originalResponse.issues.first().file }
+ }
+ val finalResponse = originalResponse.copy(issues = chatIssues)
+ return finalResponse
+ }
+
+ private fun refreshUi() {
+ val codeScanTreeModel = CodeWhispererCodeScanTreeModel(codeScanTreeNodeRoot)
+ val totalIssuesCount = codeScanTreeModel.getTotalIssuesCount()
+ if (totalIssuesCount > 0) {
+ codeScanIssuesContent.displayName =
+ message("codewhisperer.codescan.scan_display_with_issues", totalIssuesCount, INACTIVE_TEXT_COLOR)
+ } else {
+ codeScanIssuesContent.displayName = message("codewhisperer.codescan.scan_display")
+ }
+ codeScanResultsPanel.refreshUIWithUpdatedModel(codeScanTreeModel)
+ }
+
+ fun updateIssue(updatedIssue: CodeWhispererCodeScanIssue) {
+ autoScanIssues.find { it.findingId == updatedIssue.findingId }?.let { oldIssue ->
+ val updatedList = autoScanIssues.toMutableList()
+ val index = autoScanIssues.indexOf(oldIssue)
+ updatedList[index] = updatedIssue
+ autoScanIssues = updatedList
+ return
+ }
+ ondemandScanIssues.find { it.findingId == updatedIssue.findingId }?.let { oldIssue ->
+ val updatedList = ondemandScanIssues.toMutableList()
+ val index = ondemandScanIssues.indexOf(oldIssue)
+ updatedList[index] = updatedIssue
+ ondemandScanIssues = updatedList
+ }
+ }
+
+ fun removeIssue(issue: CodeWhispererCodeScanIssue) {
+ autoScanIssues = autoScanIssues.filter { it.findingId != issue.findingId }
+ ondemandScanIssues = ondemandScanIssues.filter { it.findingId != issue.findingId }
+ }
+
+ fun removeIssueByFindingId(issue: CodeWhispererCodeScanIssue, findingId: String) {
+ scanNodesLookup[issue.file]?.forEach { node ->
+ val issueNode = node.userObject as CodeWhispererCodeScanIssue
+ if (issueNode.findingId == findingId) {
+ issueNode.rangeHighlighter?.textAttributes = null
+ issueNode.rangeHighlighter?.dispose()
+ node.removeFromParent()
+ removeIssue(issue)
+ }
+ }
+ refreshUi()
+ }
+
+ fun handleError(coroutineContext: CoroutineContext, e: Error): String {
val errorMessage = when (e) {
is NoClassDefFoundError -> {
if (e.cause?.message?.contains("com.intellij.openapi.compiler.CompilerPaths") == true) {
@@ -326,9 +557,7 @@ class CodeWhispererCodeScanManager(val project: Project) {
if (!coroutineContext.isActive) {
codeScanResultsPanel.setDefaultUI()
} else {
- if (scope == CodeWhispererConstants.CodeAnalysisScope.PROJECT) {
- codeScanResultsPanel.showError(errorMessage)
- }
+ codeScanResultsPanel.showError(errorMessage)
}
return errorMessage
@@ -345,9 +574,13 @@ class CodeWhispererCodeScanManager(val project: Project) {
}
private fun getCodeScanServerExceptionMessage(e: CodeWhispererCodeScanServerException): String? =
- e.message?.takeIf { it.startsWith("UploadArtifactToS3Exception:") }
- ?.let { message("codewhisperer.codescan.upload_to_s3_failed") }
-
+ when {
+ e.message?.startsWith("UploadArtifactToS3Exception:") == true ->
+ message("codewhisperer.codescan.upload_to_s3_failed")
+ e.message?.startsWith("You've reached") == true ->
+ message("codewhisperer.codescan.quota_exceeded")
+ else -> null
+ }
fun handleException(coroutineContext: CoroutineContext, e: Exception, scope: CodeWhispererConstants.CodeAnalysisScope): String {
val errorMessage = when (e) {
is CodeWhispererException -> e.awsErrorDetails().errorMessage() ?: message("codewhisperer.codescan.run_scan_error")
@@ -365,9 +598,7 @@ class CodeWhispererCodeScanManager(val project: Project) {
if (!coroutineContext.isActive) {
codeScanResultsPanel.setDefaultUI()
} else {
- if (scope == CodeWhispererConstants.CodeAnalysisScope.PROJECT) {
- codeScanResultsPanel.showError(errorMessage)
- }
+ codeScanResultsPanel.showError(errorMessage)
}
if (
@@ -379,7 +610,7 @@ class CodeWhispererCodeScanManager(val project: Project) {
if (scope == CodeWhispererConstants.CodeAnalysisScope.PROJECT) {
LOG.error(e) {
- "Failed to run security scan and display results. Caused by: $errorMessage, status code: $errorCode, " +
+ "Failed to run code review and display results. Caused by: $errorMessage, status code: $errorCode, " +
"exception: ${e::class.simpleName}, request ID: $requestId " +
"Jetbrains IDE: ${ApplicationInfo.getInstance().fullApplicationName}, " +
"IDE version: ${ApplicationInfo.getInstance().apiVersion}, "
@@ -394,7 +625,10 @@ class CodeWhispererCodeScanManager(val project: Project) {
message("codewhisperer.codescan.file_too_large") -> message("codewhisperer.codescan.file_too_large_telemetry")
else -> e.message
}
- is CodeWhispererCodeScanServerException -> e.message
+ is CodeWhispererCodeScanServerException -> when (e.message) {
+ message("testgen.error.maximum_generations_reach") -> message("codewhisperer.codescan.quota_exceeded")
+ else -> e.message
+ }
is WaiterTimeoutException, is TimeoutCancellationException -> message("codewhisperer.codescan.scan_timed_out")
is CancellationException -> message("codewhisperer.codescan.cancelled_by_user_exception")
else -> e.message
@@ -406,21 +640,24 @@ class CodeWhispererCodeScanManager(val project: Project) {
/**
* The initial landing UI for the code scan results view panel.
* This method adds code content to the problems view if not already added.
- * When [setSelected] is true, code scan panel is set to be in focus.
*/
- fun addCodeScanUI(setSelected: Boolean = false) = runInEdt {
- reset()
+ fun buildCodeScanUI() = runInEdt {
val problemsWindow = getProblemsWindow()
if (!problemsWindow.contentManager.contents.contains(codeScanIssuesContent)) {
problemsWindow.contentManager.addContent(codeScanIssuesContent)
- }
- codeScanIssuesContent.displayName = message("codewhisperer.codescan.scan_display")
- if (setSelected) {
- problemsWindow.contentManager.setSelectedContent(codeScanIssuesContent)
- problemsWindow.show()
+ codeScanIssuesContent.displayName = message("codewhisperer.codescan.scan_display")
}
}
+ /**
+ * This method shows the code content panel in problems view
+ */
+ fun showCodeScanUI() = runInEdt {
+ val problemsWindow = getProblemsWindow()
+ problemsWindow.contentManager.setSelectedContent(codeScanIssuesContent)
+ problemsWindow.show()
+ }
+
fun removeCodeScanUI() = runInEdt {
val problemsWindow = getProblemsWindow()
if (problemsWindow.contentManager.contents.contains(codeScanIssuesContent)) {
@@ -455,6 +692,7 @@ class CodeWhispererCodeScanManager(val project: Project) {
getScanTree().model.valueForPathChanged(TreePath(node.path), newIssue)
node.userObject = newIssue
}
+ updateIssue(newIssue)
}
}
}
@@ -464,9 +702,14 @@ class CodeWhispererCodeScanManager(val project: Project) {
scanNodesLookup[file]?.forEach { node ->
val issue = node.userObject as CodeWhispererCodeScanIssue
if (document.getLineNumber(editedTextRange.startOffset) <= issue.startLine) {
- issue.startLine = issue.startLine + lineOffset
- issue.endLine = issue.endLine + lineOffset
- issue.suggestedFixes = issue.suggestedFixes.map { fix -> offsetSuggestedFix(fix, lineOffset) }
+ val newIssue = issue.copy()
+ newIssue.startLine = issue.startLine + lineOffset
+ newIssue.endLine = issue.endLine + lineOffset
+ newIssue.suggestedFixes = issue.suggestedFixes.map { fix -> offsetSuggestedFix(fix, lineOffset) }.toMutableList()
+ synchronized(node) {
+ node.userObject = newIssue
+ }
+ updateIssue(issue)
}
}
}
@@ -492,8 +735,9 @@ class CodeWhispererCodeScanManager(val project: Project) {
synchronized(codeScanTreeNodeRoot) {
codeScanTreeNodeRoot.removeAllChildren()
}
- // Remove previous document listeners before starting a new scan.
- fileNodeLookup.clear()
+ severityNodeLookup.onEach { (_, node) ->
+ node.removeAllChildren()
+ }
// Erase all range highlighter before cleaning up.
scanNodesLookup.apply {
forEach { (_, nodes) ->
@@ -518,31 +762,80 @@ class CodeWhispererCodeScanManager(val project: Project) {
}
}
- private fun beforeCodeScan() {
- isProjectScanInProgress.set(true)
- addCodeScanUI(setSelected = true)
+ private suspend fun beforeCodeScan() {
+ isOnDemandScanInProgress.set(true)
// Show in progress indicator
+ buildCodeScanUI()
codeScanResultsPanel.showInProgressIndicator()
- (FileDocumentManagerImpl.getInstance() as FileDocumentManagerImpl).saveAllDocuments(false)
+ withContext(EDT) {
+ FileDocumentManager.getInstance().saveAllDocuments()
+ }
}
- private fun afterCodeScan(scope: CodeWhispererConstants.CodeAnalysisScope) {
- if (scope == CodeWhispererConstants.CodeAnalysisScope.PROJECT) {
- isProjectScanInProgress.set(false)
+ private fun getUniqueIssues(codeScanResponse: List): List {
+ val uniqueIssues = codeScanResponse.distinctBy { issue ->
+ Triple(issue.file.path, issue.title, issue.startLine)
+ }
+ val uniqueIssueList: MutableList = mutableListOf()
+
+ uniqueIssues.forEach { issue ->
+ val isValid = runReadAction {
+ FileDocumentManager.getInstance().getDocument(issue.file)?.let { document ->
+ val documentLines = document.getText().split("\n")
+ val (startLine, endLine) = issue.run { startLine to endLine }
+ checkIssueCodeSnippet(issue.codeSnippet, startLine, endLine, documentLines)
+ } ?: false
+ }
+ if (isValid) {
+ uniqueIssueList.add(issue)
+ }
+ }
+ return uniqueIssueList
+ }
+
+ private fun afterCodeScan(scope: CodeWhispererConstants.CodeAnalysisScope, initiatedByChat: Boolean) {
+ if (initiatedByChat || scope == CodeWhispererConstants.CodeAnalysisScope.PROJECT) {
+ isOnDemandScanInProgress.set(false)
+ showCodeScanUI()
}
}
private fun sendCodeScanTelemetryToServiceAPI(
project: Project,
programmingLanguage: CodeWhispererProgrammingLanguage,
- codeScanJobId: String?,
+ codeScanResponseContext: CodeScanResponseContext,
scope: CodeWhispererConstants.CodeAnalysisScope,
) {
runIfIdcConnectionOrTelemetryEnabled(project) {
try {
- val response = CodeWhispererClientAdaptor.getInstance(project)
- .sendCodeScanTelemetry(programmingLanguage, codeScanJobId, scope)
- LOG.debug { "Successfully sent code scan telemetry. RequestId: ${response.responseMetadata().requestId()} for ${scope.value} scan" }
+ val client = CodeWhispererClientAdaptor.getInstance(project)
+ val response = client.sendCodeScanTelemetry(programmingLanguage, codeScanResponseContext.codeScanJobId, scope)
+ LOG.debug { "Successfully sent code review telemetry. RequestId: ${response.responseMetadata().requestId()} for ${scope.value} scan" }
+
+ if (codeScanResponseContext.reason == "Succeeded") {
+ val codeScanSuccessResponse = client.sendCodeScanSucceededTelemetry(
+ programmingLanguage,
+ codeScanResponseContext.codeScanJobId,
+ scope,
+ codeScanResponseContext.codeScanTotalIssues
+ )
+ LOG.debug {
+ "Successfully sent code review succeeded telemetry. RequestId: ${
+ codeScanSuccessResponse.responseMetadata().requestId()
+ } for ${scope.value} review"
+ }
+ } else {
+ val codeScanFailureResponse = client.sendCodeScanFailedTelemetry(
+ programmingLanguage,
+ codeScanResponseContext.codeScanJobId,
+ scope
+ )
+ LOG.debug {
+ "Successfully sent code review failed telemetry. RequestId: ${
+ codeScanFailureResponse.responseMetadata().requestId()
+ } for ${scope.value} review"
+ }
+ }
} catch (e: Exception) {
val requestId = if (e is CodeWhispererRuntimeException) e.requestId() else null
LOG.debug {
@@ -552,7 +845,7 @@ class CodeWhispererCodeScanManager(val project: Project) {
}
}
- private val codeScanTreeNodeRoot = DefaultMutableTreeNode("CodeWhisperer Code scan results")
+ private val codeScanTreeNodeRoot = DefaultMutableTreeNode("Amazon Q code review results")
/**
* Creates a CodeWhisperer code scan issues tree.
@@ -562,112 +855,68 @@ class CodeWhispererCodeScanManager(val project: Project) {
* [scanNodesLookup] for receiving the editor events and updating the corresponding scan nodes.
*/
private fun createCodeScanIssuesTree(codeScanIssues: List): DefaultMutableTreeNode {
- LOG.debug { "Rendering response from the scan API" }
+ LOG.debug { "Rendering response from the code review API" }
synchronized(codeScanTreeNodeRoot) {
codeScanTreeNodeRoot.removeAllChildren()
}
- // clear file node lookup and scan node lookup
- synchronized(fileNodeLookup) {
- fileNodeLookup.clear()
- }
synchronized(scanNodesLookup) {
scanNodesLookup.clear()
}
-
- codeScanIssues.forEach { issue ->
- val fileNode = synchronized(fileNodeLookup) {
- fileNodeLookup.getOrPut(issue.file) {
- val node = DefaultMutableTreeNode(issue.file)
- synchronized(codeScanTreeNodeRoot) {
- codeScanTreeNodeRoot.add(node)
- }
- node
- }
+ synchronized(severityNodeLookup) {
+ severityNodeLookup.forEach { (_, node) ->
+ node.removeAllChildren()
}
-
- val scanNode = DefaultMutableTreeNode(issue)
- fileNode.add(scanNode)
- scanNodesLookup.getOrPut(issue.file) {
- mutableListOf()
- }.add(scanNode)
}
- return codeScanTreeNodeRoot
- }
- /**
- * Updates a CodeWhisperer code scan issues tree for a file
- * For a given file, looks up its file node:
- * 1. Remove all its existing children scan nodes
- * 2. add the new issues as new scan nodes
- * [scanNodesLookup] for receiving the editor events and updating the corresponding scan nodes.
- */
- fun updateFileIssues(file: VirtualFile, newIssues: List): DefaultMutableTreeNode {
- val fileNode = synchronized(fileNodeLookup) {
- fileNodeLookup.getOrPut(file) {
- val node = DefaultMutableTreeNode(file)
+ severityNodeLookup.forEach { (severity, node) ->
+ if (selectedSeverityValues[severity] == true) {
synchronized(codeScanTreeNodeRoot) {
codeScanTreeNodeRoot.add(node)
}
- node
}
}
- // Remove the underline for old issues
- scanNodesLookup[file]?.forEach { node ->
- val issue = node.userObject as CodeWhispererCodeScanIssue
- issue.rangeHighlighter?.textAttributes = null
- issue.rangeHighlighter?.dispose()
- }
- // Remove the old scan nodes from the file node.
- synchronized(fileNode) {
- fileNode.removeAllChildren()
- }
- // Remove the entry for the file from the scan nodes lookup.
- synchronized(scanNodesLookup) {
- if (scanNodesLookup.containsKey(file)) {
- scanNodesLookup.remove(file)
- }
- }
-
- // Add new issues to the file node.
- newIssues.forEach { issue ->
- val scanNode = DefaultMutableTreeNode(issue)
- fileNode.add(scanNode)
- scanNodesLookup.getOrPut(issue.file) {
- mutableListOf()
- }.add(scanNode)
- }
- if (fileNode.childCount == 0) {
- fileNode.removeFromParent()
- fileNodeLookup.remove(file)
+ codeScanIssues.forEach { issue ->
+ if (selectedSeverityValues[issue.severity] == true) {
+ val scanNode = DefaultMutableTreeNode(issue)
+ severityNodeLookup[issue.severity]?.add(scanNode)
+ scanNodesLookup.getOrPut(issue.file) {
+ mutableListOf()
+ }.add(scanNode)
+ }
}
return codeScanTreeNodeRoot
}
- fun removeIssueByFindingId(file: VirtualFile, findingId: String) {
- scanNodesLookup[file]?.forEach { node ->
- val issue = node.userObject as CodeWhispererCodeScanIssue
- if (issue.findingId == findingId) {
- issue.rangeHighlighter?.textAttributes = null
- issue.rangeHighlighter?.dispose()
- node.removeFromParent()
- }
- }
- fileNodeLookup[file]?.let {
- if (it.childCount == 0) {
- it.removeFromParent()
- fileNodeLookup.remove(file)
+ private fun checkIssueCodeSnippet(codeSnippet: List, startLine: Int, endLine: Int, documentLines: List): Boolean = try {
+ codeSnippet
+ .asSequence()
+ .filter { it.number in startLine..endLine }
+ .all { codeBlock ->
+ val lineNumber = codeBlock.number - 1
+ val documentLine = documentLines.getOrNull(lineNumber) ?: return@all false
+
+ when {
+ codeBlock.content.trim().replace(" ", "").all { it == '*' } ->
+ documentLine.length == codeBlock.content.length
+
+ else ->
+ documentLine == codeBlock.content
+ }
}
- }
+ } catch (e: Exception) {
+ false
+ }
+
+ private fun updateCodeScanIssuesTree() {
+ val codeScanTreeNodeRoot = createCodeScanIssuesTree(getCombinedScanIssues())
val codeScanTreeModel = CodeWhispererCodeScanTreeModel(codeScanTreeNodeRoot)
val totalIssuesCount = codeScanTreeModel.getTotalIssuesCount()
if (totalIssuesCount > 0) {
codeScanIssuesContent.displayName =
message("codewhisperer.codescan.scan_display_with_issues", totalIssuesCount, INACTIVE_TEXT_COLOR)
- } else {
- codeScanIssuesContent.displayName = message("codewhisperer.codescan.scan_display")
}
codeScanResultsPanel.refreshUIWithUpdatedModel(codeScanTreeModel)
}
@@ -678,15 +927,7 @@ class CodeWhispererCodeScanManager(val project: Project) {
scope: CodeWhispererConstants.CodeAnalysisScope,
) {
withContext(getCoroutineUiContext()) {
- var root: DefaultMutableTreeNode? = null
- when (scope) {
- CodeWhispererConstants.CodeAnalysisScope.FILE -> {
- val file = scannedFiles.first()
- root = updateFileIssues(file, issues)
- } else -> {
- root = createCodeScanIssuesTree(issues)
- }
- }
+ val root = createCodeScanIssuesTree(issues)
val codeScanTreeModel = CodeWhispererCodeScanTreeModel(root)
val totalIssuesCount = codeScanTreeModel.getTotalIssuesCount()
if (totalIssuesCount > 0) {
@@ -697,12 +938,32 @@ class CodeWhispererCodeScanManager(val project: Project) {
}
}
+ private fun notifyChat(codeScanResponse: CodeScanResponse?, scope: CodeWhispererConstants.CodeAnalysisScope) {
+ // We can't use CodeScanMessageListener directly since it causes circular dependency between plugin-amazonq and plugin-core
+ // Workaround: send an action performed event to notify Q of scan result
+ val dataContext = SimpleDataContext.builder()
+ .add(scanResultsKey, codeScanResponse)
+ .add(CommonDataKeys.PROJECT, project)
+ .add(scanScopeKey, scope)
+ .build()
+ val actionEvent = AnActionEvent.createFromDataContext("", null, dataContext)
+ ActionManager.getInstance().getAction("aws.amazonq.codeScanComplete").actionPerformed(actionEvent)
+ }
+
@TestOnly
suspend fun testRenderResponseOnUIThread(issues: List, scannedFiles: List) {
assert(ApplicationManager.getApplication().isUnitTestMode)
renderResponseOnUIThread(issues, scannedFiles, CodeWhispererConstants.CodeAnalysisScope.PROJECT)
}
+ fun isInsideWorkTree(): Boolean {
+ val projectDir = project.guessProjectDir() ?: run {
+ LOG.error { "Failed to guess project directory" }
+ return false
+ }
+ return isInsideWorkTree(projectDir)
+ }
+
companion object {
fun getInstance(project: Project): CodeWhispererCodeScanManager = project.service()
}
@@ -735,6 +996,7 @@ data class CodeWhispererCodeScanIssue(
val issueSeverity: HighlightDisplayLevel = HighlightDisplayLevel.WARNING,
val isInvalid: Boolean = false,
var rangeHighlighter: RangeHighlighterEx? = null,
+ var isVisible: Boolean = true,
) {
override fun toString(): String = title
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/CodeWhispererCodeScanResultsView.kt b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/CodeWhispererCodeScanResultsView.kt
index bb066970ea..3bb9a755c0 100644
--- a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/CodeWhispererCodeScanResultsView.kt
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/CodeWhispererCodeScanResultsView.kt
@@ -10,13 +10,17 @@ import com.intellij.openapi.actionSystem.ActionToolbar
import com.intellij.openapi.project.Project
import com.intellij.openapi.vfs.VirtualFile
import com.intellij.ui.AnimatedIcon
+import com.intellij.ui.ClickListener
import com.intellij.ui.OnePixelSplitter
import com.intellij.ui.ScrollPaneFactory
import com.intellij.ui.border.CustomLineBorder
import com.intellij.ui.components.ActionLink
import com.intellij.ui.treeStructure.Tree
import com.intellij.util.ui.JBUI
+import icons.AwsIcons
+import kotlinx.coroutines.CoroutineScope
import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.listeners.CodeWhispererCodeScanTreeMouseListener
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils.IssueSeverity
import software.aws.toolkits.jetbrains.services.codewhisperer.layout.CodeWhispererLayoutConfig
import software.aws.toolkits.jetbrains.services.codewhisperer.layout.CodeWhispererLayoutConfig.addHorizontalGlue
import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererColorUtil.INACTIVE_TEXT_COLOR
@@ -26,27 +30,44 @@ import java.awt.BorderLayout
import java.awt.Component
import java.awt.GridBagConstraints
import java.awt.GridBagLayout
+import java.awt.event.MouseEvent
import java.time.Instant
import java.time.format.DateTimeFormatter
import javax.swing.BorderFactory
-import javax.swing.JButton
+import javax.swing.Icon
import javax.swing.JLabel
import javax.swing.JPanel
import javax.swing.JTree
import javax.swing.tree.DefaultMutableTreeNode
import javax.swing.tree.TreeCellRenderer
+import javax.swing.tree.TreePath
/**
* Create a Code Scan results view that displays the code scan results.
*/
-internal class CodeWhispererCodeScanResultsView(private val project: Project) : JPanel(BorderLayout()) {
+internal class CodeWhispererCodeScanResultsView(private val project: Project, private val defaultScope: CoroutineScope) : JPanel(BorderLayout()) {
private val codeScanTree: Tree = Tree().apply {
isRootVisible = false
CodeWhispererCodeScanTreeMouseListener(project).installOn(this)
+ object : ClickListener() {
+ override fun onClick(event: MouseEvent, clickCount: Int): Boolean {
+ val issueNode = (event.source as Tree).selectionPath?.lastPathComponent as? DefaultMutableTreeNode
+ val issue = issueNode?.userObject as? CodeWhispererCodeScanIssue ?: return false
+ showIssueDetails(issue, defaultScope)
+ return true
+ }
+ }.installOn(this)
cellRenderer = ColoredTreeCellRenderer()
}
+ private fun expandItems() {
+ val criticalTreePath = TreePath(arrayOf(codeScanTree.model.root, codeScanTree.model.getChild(codeScanTree.model.root, 0)))
+ val highTreePath = TreePath(arrayOf(codeScanTree.model.root, codeScanTree.model.getChild(codeScanTree.model.root, 1)))
+ codeScanTree.expandPath(criticalTreePath)
+ codeScanTree.expandPath(highTreePath)
+ }
+
private val scrollPane = ScrollPaneFactory.createScrollPane(codeScanTree, true)
private val splitter = OnePixelSplitter(CODE_SCAN_SPLITTER_PROPORTION_KEY, 1.0f).apply {
firstComponent = scrollPane
@@ -70,6 +91,20 @@ internal class CodeWhispererCodeScanResultsView(private val project: Project) :
showScannedFiles(scannedFiles)
}
}
+
+ private val filtersAppliedToResultsLabel = JLabel(message("codewhisperer.codescan.scan_results_hidden_by_filters")).apply {
+ border = BorderFactory.createEmptyBorder(7, 7, 7, 7)
+ }
+ private val clearFiltersLink = ActionLink(message("codewhisperer.codescan.clear_filters")).apply {
+ addActionListener {
+ CodeWhispererCodeScanManager.getInstance(project).clearFilters()
+ }
+ }
+ private val filtersAppliedIndicator = JPanel(GridBagLayout()).apply {
+ add(filtersAppliedToResultsLabel, GridBagConstraints())
+ add(clearFiltersLink, GridBagConstraints().apply { gridy = 1 })
+ }
+
private val learnMoreLabelLink = ActionLink().apply {
border = BorderFactory.createEmptyBorder(0, 7, 0, 0)
}
@@ -89,15 +124,9 @@ internal class CodeWhispererCodeScanResultsView(private val project: Project) :
private val progressIndicatorLabel = JLabel(message("codewhisperer.codescan.scan_in_progress"), AnimatedIcon.Default(), JLabel.CENTER).apply {
border = BorderFactory.createEmptyBorder(7, 7, 7, 7)
}
- private val stopCodeScanButton = JButton(message("codewhisperer.codescan.stop_scan")).apply {
- addActionListener {
- CodeWhispererCodeScanManager.getInstance(project).stopCodeScan()
- }
- }
private val progressIndicator = JPanel(GridBagLayout()).apply {
add(progressIndicatorLabel, GridBagConstraints())
- add(stopCodeScanButton, GridBagConstraints().apply { gridy = 1 })
}
// Results panel containing info label and progressIndicator/scrollPane
@@ -126,6 +155,7 @@ internal class CodeWhispererCodeScanResultsView(private val project: Project) :
model = scanTreeModel
repaint()
}
+ expandItems()
if (scope == CodeWhispererConstants.CodeAnalysisScope.PROJECT) {
this.scannedFiles = scannedFiles
@@ -146,15 +176,34 @@ internal class CodeWhispererCodeScanResultsView(private val project: Project) :
}
fun refreshUIWithUpdatedModel(scanTreeModel: CodeWhispererCodeScanTreeModel) {
- codeScanTree.apply {
- model = scanTreeModel
- repaint()
+ changeInfoLabelToDisplayScanCompleted(scannedFiles.size)
+ val codeScanManager = CodeWhispererCodeScanManager.getInstance(project)
+ if (scanTreeModel.getTotalIssuesCount() == 0 && codeScanManager.hasCodeScanIssues()) {
+ resultsPanel.apply {
+ if (components.contains(splitter)) remove(splitter)
+ add(BorderLayout.CENTER, filtersAppliedIndicator)
+ revalidate()
+ repaint()
+ }
+ } else {
+ codeScanTree.apply {
+ model = scanTreeModel
+ repaint()
+ }
+ expandItems()
+ resultsPanel.apply {
+ if (components.contains(filtersAppliedIndicator)) remove(filtersAppliedIndicator)
+ add(BorderLayout.CENTER, splitter)
+ splitter.proportion = 1.0f
+ splitter.secondComponent = null
+ revalidate()
+ repaint()
+ }
}
}
fun setStoppingCodeScan() {
completeInfoLabel.isVisible = false
- stopCodeScanButton.isVisible = false
resultsPanel.apply {
if (components.contains(splitter)) remove(splitter)
progressIndicatorLabel.apply {
@@ -197,7 +246,6 @@ internal class CodeWhispererCodeScanResultsView(private val project: Project) :
fun showInProgressIndicator() {
completeInfoLabel.isVisible = false
- stopCodeScanButton.isVisible = true
progressIndicatorLabel.text = message("codewhisperer.codescan.scan_in_progress")
resultsPanel.apply {
if (components.contains(splitter)) remove(splitter)
@@ -241,6 +289,20 @@ internal class CodeWhispererCodeScanResultsView(private val project: Project) :
}
}
+ private fun showIssueDetails(issue: CodeWhispererCodeScanIssue, defaultScope: CoroutineScope) {
+ val issueDetailsViewPanel = CodeWhispererCodeScanIssueDetailsPanel(project, issue, defaultScope)
+ issueDetailsViewPanel.apply {
+ isVisible = true
+ revalidate()
+ }
+ splitter.apply {
+ secondComponent = issueDetailsViewPanel
+ proportion = 0.5f
+ revalidate()
+ repaint()
+ }
+ }
+
private fun changeInfoLabelToDisplayScanCompleted(numScannedFiles: Int) {
completeInfoLabel.isVisible = true
infoLabelPrefix.icon = AllIcons.Actions.Commit
@@ -265,6 +327,15 @@ internal class CodeWhispererCodeScanResultsView(private val project: Project) :
}
private class ColoredTreeCellRenderer : TreeCellRenderer {
+ private fun getSeverityIcon(severity: String): Icon? = when (severity) {
+ IssueSeverity.LOW.displayName -> AwsIcons.Resources.CodeWhisperer.SEVERITY_INITIAL_LOW
+ IssueSeverity.MEDIUM.displayName -> AwsIcons.Resources.CodeWhisperer.SEVERITY_INITIAL_MEDIUM
+ IssueSeverity.HIGH.displayName -> AwsIcons.Resources.CodeWhisperer.SEVERITY_INITIAL_HIGH
+ IssueSeverity.CRITICAL.displayName -> AwsIcons.Resources.CodeWhisperer.SEVERITY_INITIAL_CRITICAL
+ IssueSeverity.INFO.displayName -> AwsIcons.Resources.CodeWhisperer.SEVERITY_INITIAL_INFO
+ else -> null
+ }
+
override fun getTreeCellRendererComponent(
tree: JTree?,
value: Any?,
@@ -278,18 +349,23 @@ internal class CodeWhispererCodeScanResultsView(private val project: Project) :
val cell = JLabel()
synchronized(value) {
when (val obj = value.userObject) {
+ is String -> {
+ cell.text = message("codewhisperer.codescan.severity_issues_count", obj, value.childCount, INACTIVE_TEXT_COLOR)
+ cell.icon = this.getSeverityIcon(obj)
+ }
is VirtualFile -> {
cell.text = message("codewhisperer.codescan.file_name_issues_count", obj.name, obj.path, value.childCount, INACTIVE_TEXT_COLOR)
cell.icon = obj.fileType.icon
}
is CodeWhispererCodeScanIssue -> {
- val cellText = "${obj.title}: ${obj.description.text}"
+ val cellText = obj.title.trimEnd('.')
+ val cellDescription = "${obj.file.name} ${obj.displayTextRange()}"
if (obj.isInvalid) {
- cell.text = message("codewhisperer.codescan.scan_recommendation_invalid", obj.title, obj.displayTextRange(), INACTIVE_TEXT_COLOR)
+ cell.text = message("codewhisperer.codescan.scan_recommendation_invalid", obj.title, cellDescription, INACTIVE_TEXT_COLOR)
cell.toolTipText = message("codewhisperer.codescan.scan_recommendation_invalid.tooltip_text")
cell.icon = AllIcons.General.Information
} else {
- cell.text = message("codewhisperer.codescan.scan_recommendation", cellText, obj.displayTextRange(), INACTIVE_TEXT_COLOR)
+ cell.text = message("codewhisperer.codescan.scan_recommendation", cellText, cellDescription, INACTIVE_TEXT_COLOR)
cell.toolTipText = cellText
cell.icon = obj.issueSeverity.icon
}
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/CodeWhispererCodeScanSession.kt b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/CodeWhispererCodeScanSession.kt
index 45f629206e..2550d830e4 100644
--- a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/CodeWhispererCodeScanSession.kt
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/CodeWhispererCodeScanSession.kt
@@ -12,13 +12,11 @@ import com.intellij.openapi.application.runReadAction
import com.intellij.openapi.fileEditor.FileDocumentManager
import com.intellij.openapi.project.Project
import com.intellij.openapi.vfs.LocalFileSystem
-import com.intellij.util.io.HttpRequests
import kotlinx.coroutines.delay
import kotlinx.coroutines.ensureActive
import kotlinx.coroutines.isActive
import kotlinx.coroutines.time.withTimeout
import kotlinx.coroutines.withContext
-import org.apache.commons.codec.digest.DigestUtils
import software.amazon.awssdk.services.codewhisperer.model.ArtifactType
import software.amazon.awssdk.services.codewhisperer.model.CodeScanFindingsSchema
import software.amazon.awssdk.services.codewhisperer.model.CodeScanStatus
@@ -29,24 +27,18 @@ import software.amazon.awssdk.services.codewhisperer.model.GetCodeScanRequest
import software.amazon.awssdk.services.codewhisperer.model.GetCodeScanResponse
import software.amazon.awssdk.services.codewhisperer.model.ListCodeScanFindingsRequest
import software.amazon.awssdk.services.codewhisperer.model.ListCodeScanFindingsResponse
-import software.amazon.awssdk.services.codewhispererruntime.model.CodeAnalysisUploadContext
-import software.amazon.awssdk.services.codewhispererruntime.model.CreateUploadUrlRequest
import software.amazon.awssdk.services.codewhispererruntime.model.CreateUploadUrlResponse
-import software.amazon.awssdk.services.codewhispererruntime.model.UploadContext
-import software.amazon.awssdk.services.codewhispererruntime.model.UploadIntent
-import software.amazon.awssdk.utils.IoUtils
+import software.amazon.awssdk.services.codewhispererruntime.model.Reference
import software.aws.toolkits.core.utils.Waiters.waitUntil
import software.aws.toolkits.core.utils.debug
-import software.aws.toolkits.core.utils.error
import software.aws.toolkits.core.utils.getLogger
import software.aws.toolkits.core.utils.info
-import software.aws.toolkits.jetbrains.core.AwsClientManager
import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.sessionconfig.CodeScanSessionConfig
import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.sessionconfig.PayloadContext
import software.aws.toolkits.jetbrains.services.codewhisperer.credentials.CodeWhispererClientAdaptor
import software.aws.toolkits.jetbrains.services.codewhisperer.explorer.CodeWhispererExplorerActionManager
import software.aws.toolkits.jetbrains.services.codewhisperer.model.CodeScanResponseContext
-import software.aws.toolkits.jetbrains.services.codewhisperer.model.CodeScanServiceInvocationContext
+import software.aws.toolkits.jetbrains.services.codewhisperer.model.CreateUploadUrlServiceInvocationContext
import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererConstants
import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererConstants.CODE_SCAN_POLLING_INTERVAL_IN_SECONDS
import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererConstants.FILE_SCANS_THROTTLING_MESSAGE
@@ -56,17 +48,14 @@ import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhisperer
import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererConstants.TOTAL_BYTES_IN_KB
import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererConstants.TOTAL_MILLIS_IN_SECOND
import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererUtil.notifyErrorCodeWhispererUsageLimit
+import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererZipUploadManager
+import software.aws.toolkits.jetbrains.services.codewhisperer.util.getTelemetryErrorMessage
import software.aws.toolkits.jetbrains.utils.assertIsNonDispatchThread
import software.aws.toolkits.resources.message
import software.aws.toolkits.telemetry.CodewhispererLanguage
-import java.io.File
-import java.io.FileInputStream
-import java.io.IOException
-import java.net.HttpURLConnection
import java.nio.file.Path
import java.time.Duration
import java.time.Instant
-import java.util.Base64
import java.util.UUID
import kotlin.coroutines.coroutineContext
@@ -104,7 +93,7 @@ class CodeWhispererCodeScanSession(val sessionContext: CodeScanSessionContext) {
"Total size of source payload in KB: ${payloadContext.srcPayloadSize * 1.0 / TOTAL_BYTES_IN_KB} \n" +
"Total size of build payload in KB: ${(payloadContext.buildPayloadSize ?: 0) * 1.0 / TOTAL_BYTES_IN_KB} \n" +
"Total size of source zip file in KB: ${payloadContext.srcZipFileSize * 1.0 / TOTAL_BYTES_IN_KB} \n" +
- "Total number of lines scanned: ${payloadContext.totalLines} \n" +
+ "Total number of lines reviewed: ${payloadContext.totalLines} \n" +
"Total number of files included in payload: ${payloadContext.totalFiles} \n" +
"Total time taken for creating payload: ${payloadContext.totalTimeInMilliseconds * 1.0 / TOTAL_MILLIS_IN_SECOND} seconds\n" +
"Payload context language: ${payloadContext.language}"
@@ -116,7 +105,21 @@ class CodeWhispererCodeScanSession(val sessionContext: CodeScanSessionContext) {
currentCoroutineContext.ensureActive()
val artifactsUploadStartTime = now()
val codeScanName = UUID.randomUUID().toString()
- val sourceZipUploadResponse = createUploadUrlAndUpload(sourceZip, "SourceCode", codeScanName)
+
+ val taskType = if (sessionContext.codeAnalysisScope == CodeWhispererConstants.CodeAnalysisScope.PROJECT) {
+ CodeWhispererConstants.UploadTaskType.SCAN_PROJECT
+ } else {
+ CodeWhispererConstants.UploadTaskType.SCAN_FILE
+ }
+
+ val sourceZipUploadResponse =
+ CodeWhispererZipUploadManager.getInstance(sessionContext.project).createUploadUrlAndUpload(
+ sourceZip,
+ "SourceCode",
+ taskType,
+ codeScanName,
+ CodeWhispererConstants.FeatureName.CODE_REVIEW
+ )
if (isProjectScope()) {
LOG.debug {
"Successfully uploaded source zip to s3: " +
@@ -137,7 +140,7 @@ class CodeWhispererCodeScanSession(val sessionContext: CodeScanSessionContext) {
val createCodeScanResponse = createCodeScan(payloadContext.language.toString(), codeScanName)
if (isProjectScope()) {
LOG.debug {
- "Successfully created security scan with " +
+ "Successfully created code review with " +
"status: ${createCodeScanResponse.status()} " +
"for request id: ${createCodeScanResponse.responseMetadata().requestId()}"
}
@@ -146,7 +149,7 @@ class CodeWhispererCodeScanSession(val sessionContext: CodeScanSessionContext) {
if (codeScanStatus == CodeScanStatus.FAILED) {
if (isProjectScope()) {
LOG.debug {
- "CodeWhisperer service error occurred. Something went wrong when creating a security scan: $createCodeScanResponse " +
+ "CodeWhisperer service error occurred. Something went wrong when creating a code review: $createCodeScanResponse " +
"Status: ${createCodeScanResponse.status()} for request id: ${createCodeScanResponse.responseMetadata().requestId()}"
}
}
@@ -169,13 +172,13 @@ class CodeWhispererCodeScanSession(val sessionContext: CodeScanSessionContext) {
currentCoroutineContext.ensureActive()
val elapsedTime = (now() - startTime) * 1.0 / TOTAL_MILLIS_IN_SECOND
if (isProjectScope()) {
- LOG.debug { "Waiting for security scan to complete. Elapsed time: $elapsedTime sec." }
+ LOG.debug { "Waiting for code review to complete. Elapsed time: $elapsedTime sec." }
}
val getCodeScanResponse = getCodeScan(jobId)
codeScanStatus = getCodeScanResponse.status()
if (isProjectScope()) {
LOG.debug {
- "Get security scan status: ${getCodeScanResponse.status()}, " +
+ "Get code review status: ${getCodeScanResponse.status()}, " +
"request id: ${getCodeScanResponse.responseMetadata().requestId()}"
}
}
@@ -183,7 +186,7 @@ class CodeWhispererCodeScanSession(val sessionContext: CodeScanSessionContext) {
if (codeScanStatus == CodeScanStatus.FAILED) {
if (isProjectScope()) {
LOG.debug {
- "CodeWhisperer service error occurred. Something went wrong fetching results for security scan: $getCodeScanResponse " +
+ "CodeWhisperer service error occurred. Something went wrong fetching results for code review: $getCodeScanResponse " +
"Status: ${getCodeScanResponse.status()} for request id: ${getCodeScanResponse.responseMetadata().requestId()}"
}
}
@@ -192,7 +195,7 @@ class CodeWhispererCodeScanSession(val sessionContext: CodeScanSessionContext) {
}
}
- LOG.debug { "Security scan completed successfully by CodeWhisperer." }
+ LOG.debug { "Code review completed successfully by Amazon Q." }
// 6. Return the results from the ListCodeScan API.
currentCoroutineContext.ensureActive()
@@ -213,10 +216,10 @@ class CodeWhispererCodeScanSession(val sessionContext: CodeScanSessionContext) {
}
if (isProjectScope()) {
- LOG.debug { "Rendering response to display security scan results." }
+ LOG.debug { "Rendering response to display code review results." }
}
currentCoroutineContext.ensureActive()
- val issues = mapToCodeScanIssues(documents)
+ val issues = mapToCodeScanIssues(documents, sessionContext.project).filter { it.isVisible }
codeScanResponseContext = codeScanResponseContext.copy(codeScanTotalIssues = issues.count())
codeScanResponseContext = codeScanResponseContext.copy(codeScanIssuesWithFixes = issues.count { it.suggestedFixes.isNotEmpty() })
codeScanResponseContext = codeScanResponseContext.copy(reason = "Succeeded")
@@ -228,16 +231,16 @@ class CodeWhispererCodeScanSession(val sessionContext: CodeScanSessionContext) {
if (awsError != null) {
if (awsError.errorCode() == "ThrottlingException" && awsError.errorMessage() != null) {
if (awsError.errorMessage()!!.contains(PROJECT_SCANS_THROTTLING_MESSAGE)) {
- LOG.info { "Project Scans limit reached" }
+ LOG.info { "Project reviews limit reached" }
notifyErrorCodeWhispererUsageLimit(sessionContext.project, true)
} else if (awsError.errorMessage()!!.contains(FILE_SCANS_THROTTLING_MESSAGE)) {
- LOG.info { "File Scans limit reached" }
+ LOG.info { "File reviews limit reached" }
CodeWhispererExplorerActionManager.getInstance().setMonthlyQuotaForCodeScansExceeded(true)
}
}
}
- LOG.error(e) {
- "Failed to run security scan and display results. Caused by: ${e.message}, status code: ${awsError?.errorCode()}, " +
+ LOG.debug(e) {
+ "Failed to run code review and display results. Caused by: ${e.message}, status code: ${awsError?.errorCode()}, " +
"exception: ${e::class.simpleName}, request ID: ${exception?.requestId()}" +
"Jetbrains IDE: ${ApplicationInfo.getInstance().fullApplicationName}, " +
"IDE version: ${ApplicationInfo.getInstance().apiVersion}, "
@@ -246,101 +249,32 @@ class CodeWhispererCodeScanSession(val sessionContext: CodeScanSessionContext) {
}
}
- /**
- * Creates an upload URL and uplaods the zip file to the presigned URL
- */
- fun createUploadUrlAndUpload(zipFile: File, artifactType: String, codeScanName: String): CreateUploadUrlResponse {
- // Throw error if zipFile is invalid.
- if (!zipFile.exists()) {
- invalidSourceZipError()
- }
- val fileMd5: String = Base64.getEncoder().encodeToString(DigestUtils.md5(FileInputStream(zipFile)))
- val createUploadUrlResponse = createUploadUrl(fileMd5, artifactType, codeScanName)
- val url = createUploadUrlResponse.uploadUrl()
- if (isProjectScope()) {
- LOG.debug { "Uploading $artifactType using the presigned URL." }
- }
- uploadArtifactToS3(
- url,
- createUploadUrlResponse.uploadId(),
- zipFile,
- fileMd5,
- createUploadUrlResponse.kmsKeyArn(),
- createUploadUrlResponse.requestHeaders()
- )
- return createUploadUrlResponse
- }
-
- fun createUploadUrl(md5Content: String, artifactType: String, codeScanName: String): CreateUploadUrlResponse = try {
- clientAdaptor.createUploadUrl(
- CreateUploadUrlRequest.builder()
- .contentMd5(md5Content)
- .artifactType(artifactType)
- .uploadIntent(getUploadIntent(sessionContext.codeAnalysisScope))
- .uploadContext(UploadContext.fromCodeAnalysisUploadContext(CodeAnalysisUploadContext.builder().codeScanName(codeScanName).build()))
- .build()
- )
- } catch (e: Exception) {
- LOG.debug { "Create Upload URL failed: ${e.message}" }
- val errorMessage = getTelemetryErrorMessage(e)
- throw codeScanServerException("CreateUploadUrlException: $errorMessage")
- }
-
- private fun getUploadIntent(scope: CodeWhispererConstants.CodeAnalysisScope): UploadIntent = when (scope) {
- CodeWhispererConstants.CodeAnalysisScope.FILE -> UploadIntent.AUTOMATIC_FILE_SECURITY_SCAN
- CodeWhispererConstants.CodeAnalysisScope.PROJECT -> UploadIntent.FULL_PROJECT_SECURITY_SCAN
- }
-
- @Throws(IOException::class)
- fun uploadArtifactToS3(url: String, uploadId: String, fileToUpload: File, md5: String, kmsArn: String?, requestHeaders: Map?) {
- try {
- val uploadIdJson = """{"uploadId":"$uploadId"}"""
- HttpRequests.put(url, "application/zip").userAgent(AwsClientManager.getUserAgent()).tuner {
- if (requestHeaders.isNullOrEmpty()) {
- it.setRequestProperty(CONTENT_MD5, md5)
- it.setRequestProperty(CONTENT_TYPE, APPLICATION_ZIP)
- it.setRequestProperty(SERVER_SIDE_ENCRYPTION, AWS_KMS)
- if (kmsArn?.isNotEmpty() == true) {
- it.setRequestProperty(SERVER_SIDE_ENCRYPTION_AWS_KMS_KEY_ID, kmsArn)
- }
- it.setRequestProperty(SERVER_SIDE_ENCRYPTION_CONTEXT, Base64.getEncoder().encodeToString(uploadIdJson.toByteArray()))
- } else {
- requestHeaders.forEach { entry ->
- it.setRequestProperty(entry.key, entry.value)
- }
- }
- }.connect {
- val connection = it.connection as HttpURLConnection
- connection.setFixedLengthStreamingMode(fileToUpload.length())
- IoUtils.copy(fileToUpload.inputStream(), connection.outputStream)
- }
- } catch (e: Exception) {
- LOG.debug { "Artifact failed to upload in the S3 bucket: ${e.message}" }
- val errorMessage = getTelemetryErrorMessage(e)
- throw codeScanServerException("UploadArtifactToS3Exception: $errorMessage")
- }
- }
-
fun createCodeScan(language: String, codeScanName: String): CreateCodeScanResponse {
val artifactsMap = mapOf(
ArtifactType.SOURCE_CODE to urlResponse[ArtifactType.SOURCE_CODE]?.uploadId(),
ArtifactType.BUILT_JARS to urlResponse[ArtifactType.BUILT_JARS]?.uploadId()
).filter { (_, v) -> v != null }
+ val scope = when {
+ sessionContext.codeAnalysisScope == CodeWhispererConstants.CodeAnalysisScope.FILE &&
+ !sessionContext.sessionConfig.isInitiatedByChat() -> CodeWhispererConstants.CodeAnalysisScope.FILE
+ else -> CodeWhispererConstants.CodeAnalysisScope.PROJECT
+ }
+
try {
return clientAdaptor.createCodeScan(
CreateCodeScanRequest.builder()
.clientToken(clientToken.toString())
.programmingLanguage { it.languageName(language) }
.artifacts(artifactsMap)
- .scope(sessionContext.codeAnalysisScope.value)
+ .scope(scope.value)
.codeScanName(codeScanName)
.build()
)
} catch (e: Exception) {
- LOG.debug { "Creating security scan failed: ${e.message}" }
- val errorMessage = getTelemetryErrorMessage(e)
- throw codeScanServerException("CreateCodeScanException: $errorMessage")
+ LOG.debug { "Creating code review failed: ${e.message}" }
+ val errorMessage = getTelemetryErrorMessage(e, featureUseCase = CodeWhispererConstants.FeatureName.CODE_REVIEW)
+ throw codeScanServerException(errorMessage)
}
}
@@ -351,9 +285,9 @@ class CodeWhispererCodeScanSession(val sessionContext: CodeScanSessionContext) {
.build()
)
} catch (e: Exception) {
- LOG.debug { "Getting security scan failed: ${e.message}" }
- val errorMessage = getTelemetryErrorMessage(e)
- throw codeScanServerException("GetCodeScanException: $errorMessage")
+ LOG.debug { "Getting code review failed: ${e.message}" }
+ val errorMessage = getTelemetryErrorMessage(e, featureUseCase = CodeWhispererConstants.FeatureName.CODE_REVIEW)
+ throw codeScanServerException("GetCodeReviewException: $errorMessage")
}
fun listCodeScanFindings(jobId: String, nextToken: String?): ListCodeScanFindingsResponse = try {
@@ -365,20 +299,20 @@ class CodeWhispererCodeScanSession(val sessionContext: CodeScanSessionContext) {
.build()
)
} catch (e: Exception) {
- LOG.debug { "Listing security scan failed: ${e.message}" }
- val errorMessage = getTelemetryErrorMessage(e)
- throw codeScanServerException("ListCodeScanFindingsException: $errorMessage")
+ LOG.debug { "Listing code review failed: ${e.message}" }
+ val errorMessage = getTelemetryErrorMessage(e, featureUseCase = CodeWhispererConstants.FeatureName.CODE_REVIEW)
+ throw codeScanServerException("ListCodeReviewFindingsException: $errorMessage")
}
- fun mapToCodeScanIssues(recommendations: List): List {
+ fun mapToCodeScanIssues(recommendations: List, project: Project): List {
val scanRecommendations = recommendations.flatMap { MAPPER.readValue>(it) }
if (isProjectScope()) {
- LOG.debug { "Total code scan issues returned from service: ${scanRecommendations.size}" }
+ LOG.debug { "Total code review issues returned from service: ${scanRecommendations.size}" }
}
return scanRecommendations.mapNotNull { recommendation ->
val file = try {
LocalFileSystem.getInstance().findFileByIoFile(
- Path.of(sessionContext.sessionConfig.projectRoot.path, recommendation.filePath).toFile()
+ Path.of(sessionContext.sessionConfig.projectRoot.path, recommendation.filePath.substringAfter(sessionContext.project.name)).toFile()
)
} catch (e: Exception) {
LOG.debug { "Cannot find file at location ${recommendation.filePath}" }
@@ -389,48 +323,31 @@ class CodeWhispererCodeScanSession(val sessionContext: CodeScanSessionContext) {
runReadAction {
FileDocumentManager.getInstance().getDocument(file)
}?.let { document ->
-
- val documentLines = document.getText().split("\n")
- val (startLine, endLine) = recommendation.run { startLine to endLine }
- var shouldDisplayIssue = true
-
- for (codeBlock in recommendation.codeSnippet) {
- val lineNumber = codeBlock.number - 1
- if (codeBlock.number in startLine..endLine) {
- val documentLine = documentLines.getOrNull(lineNumber)
- if (documentLine != codeBlock.content) {
- shouldDisplayIssue = false
- break
- }
- }
- }
-
- if (shouldDisplayIssue) {
- val endLineInDocument = minOf(maxOf(0, recommendation.endLine - 1), document.lineCount - 1)
- val endCol = document.getLineEndOffset(endLineInDocument) - document.getLineStartOffset(endLineInDocument) + 1
-
- CodeWhispererCodeScanIssue(
- startLine = recommendation.startLine,
- startCol = 1,
- endLine = recommendation.endLine,
- endCol = endCol,
- file = file,
- project = sessionContext.project,
- title = recommendation.title,
- description = recommendation.description,
- detectorId = recommendation.detectorId,
- detectorName = recommendation.detectorName,
- findingId = recommendation.findingId,
- ruleId = recommendation.ruleId,
- relatedVulnerabilities = recommendation.relatedVulnerabilities,
- severity = recommendation.severity,
- recommendation = recommendation.remediation.recommendation,
- suggestedFixes = recommendation.remediation.suggestedFixes,
- codeSnippet = recommendation.codeSnippet
- )
- } else {
- null
- }
+ val endLineInDocument = minOf(maxOf(0, recommendation.endLine - 1), document.lineCount - 1)
+ val endCol = document.getLineEndOffset(endLineInDocument) - document.getLineStartOffset(endLineInDocument) + 1
+ val manager = CodeWhispererCodeScanManager.getInstance(project)
+ val isIssueIgnored = manager.isIgnoredIssue(recommendation.title, document, file, recommendation.startLine - 1)
+
+ CodeWhispererCodeScanIssue(
+ startLine = recommendation.startLine,
+ startCol = 1,
+ endLine = recommendation.endLine,
+ endCol = endCol,
+ file = file,
+ project = sessionContext.project,
+ title = recommendation.title,
+ description = recommendation.description,
+ detectorId = recommendation.detectorId,
+ detectorName = recommendation.detectorName,
+ findingId = recommendation.findingId,
+ ruleId = recommendation.ruleId,
+ relatedVulnerabilities = recommendation.relatedVulnerabilities,
+ severity = recommendation.severity,
+ recommendation = recommendation.remediation.recommendation,
+ suggestedFixes = recommendation.remediation.suggestedFixes,
+ codeSnippet = recommendation.codeSnippet,
+ isVisible = !isIssueIgnored,
+ )
}
} else {
null
@@ -443,18 +360,6 @@ class CodeWhispererCodeScanSession(val sessionContext: CodeScanSessionContext) {
}
}
- fun getTelemetryErrorMessage(e: Exception): String = when {
- e.message?.contains("Resource not found.") == true -> "Resource not found."
- e.message?.contains("Service returned HTTP status code 407") == true -> "Service returned HTTP status code 407"
- e.message?.contains("Improperly formed request") == true -> "Improperly formed request"
- e.message?.contains("Service returned HTTP status code 403") == true -> "Service returned HTTP status code 403"
- e.message?.contains("invalid_grant: Invalid token provided") == true -> "invalid_grant: Invalid token provided"
- e.message?.contains("Connect timed out") == true -> "Unable to execute HTTP request: Connect timed out" // Error: Connect to host failed
- e.message?.contains("Encountered an unexpected error when processing the request, please try again.") == true ->
- "Encountered an unexpected error when processing the request, please try again."
- else -> e.message ?: message("codewhisperer.codescan.run_scan_error_telemetry")
- }
-
private fun isProjectScope(): Boolean = sessionContext.codeAnalysisScope == CodeWhispererConstants.CodeAnalysisScope.PROJECT
companion object {
@@ -508,7 +413,7 @@ data class Remediation(val recommendation: Recommendation, val suggestedFixes: L
data class Recommendation(val text: String, val url: String?)
-data class SuggestedFix(val description: String, val code: String)
+data class SuggestedFix(val description: String, val code: String, val codeFixJobId: String? = null, val references: MutableList = mutableListOf())
data class CodeLine(val number: Int, val content: String)
@@ -520,6 +425,6 @@ data class CodeScanSessionContext(
internal fun defaultPayloadContext() = PayloadContext(CodewhispererLanguage.Unknown, 0, 0, 0, listOf(), 0, 0)
-internal fun defaultServiceInvocationContext() = CodeScanServiceInvocationContext(0, 0)
+internal fun defaultCreateUploadUrlServiceInvocationContext() = CreateUploadUrlServiceInvocationContext()
-internal fun defaultCodeScanResponseContext() = CodeScanResponseContext(defaultPayloadContext(), defaultServiceInvocationContext())
+internal fun defaultCodeScanResponseContext() = CodeScanResponseContext(defaultPayloadContext(), defaultCreateUploadUrlServiceInvocationContext())
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/actions/CodeWhispererCodeScanFilterGroup.kt b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/actions/CodeWhispererCodeScanFilterGroup.kt
new file mode 100644
index 0000000000..25f31a9f15
--- /dev/null
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/actions/CodeWhispererCodeScanFilterGroup.kt
@@ -0,0 +1,39 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.codewhisperer.codescan.actions
+
+import com.intellij.openapi.actionSystem.ActionGroup
+import com.intellij.openapi.actionSystem.ActionUpdateThread
+import com.intellij.openapi.actionSystem.AnAction
+import com.intellij.openapi.actionSystem.AnActionEvent
+import com.intellij.openapi.actionSystem.ex.CheckboxAction
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.CodeWhispererCodeScanManager
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils.IssueSeverity
+
+class CodeWhispererCodeScanFilterGroup : ActionGroup() {
+ override fun getChildren(e: AnActionEvent?): Array =
+ IssueSeverity.entries.map { FilterBySeverityAction(e, it.displayName) }.toTypedArray()
+
+ private class FilterBySeverityAction(event: AnActionEvent?, severity: String) : CheckboxAction() {
+ override fun getActionUpdateThread() = ActionUpdateThread.BGT
+ private val severity = severity
+
+ override fun isSelected(event: AnActionEvent): Boolean {
+ val project = event.project ?: return false
+ return CodeWhispererCodeScanManager.getInstance(project).isSeveritySelected(severity)
+ }
+
+ override fun setSelected(event: AnActionEvent, state: Boolean) {
+ val project = event.project
+ if (project != null) {
+ CodeWhispererCodeScanManager.getInstance(project).setSeveritySelected(severity, state)
+ }
+ }
+
+ override fun update(e: AnActionEvent) {
+ super.update(e)
+ e.presentation.text = severity
+ }
+ }
+}
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/actions/CodeWhispererCodeScanRunAction.kt b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/actions/CodeWhispererCodeScanRunAction.kt
index da115041d3..c0a86ca2c4 100644
--- a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/actions/CodeWhispererCodeScanRunAction.kt
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/actions/CodeWhispererCodeScanRunAction.kt
@@ -4,16 +4,19 @@
package software.aws.toolkits.jetbrains.services.codewhisperer.codescan.actions
import com.intellij.icons.AllIcons
+import com.intellij.openapi.actionSystem.ActionManager
import com.intellij.openapi.actionSystem.ActionUpdateThread
import com.intellij.openapi.actionSystem.AnActionEvent
+import com.intellij.openapi.actionSystem.impl.SimpleDataContext
import com.intellij.openapi.project.DumbAwareAction
import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.CodeWhispererCodeScanManager
import software.aws.toolkits.jetbrains.services.codewhisperer.explorer.isCodeWhispererEnabled
-import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererConstants
+import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererColorUtil.INACTIVE_TEXT_COLOR
+import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererConstants.runScanKey
import software.aws.toolkits.resources.message
class CodeWhispererCodeScanRunAction : DumbAwareAction(
- message("codewhisperer.codescan.run_scan"),
+ message("codewhisperer.codescan.run_scan", INACTIVE_TEXT_COLOR),
null,
AllIcons.Actions.Execute
) {
@@ -27,7 +30,11 @@ class CodeWhispererCodeScanRunAction : DumbAwareAction(
}
override fun actionPerformed(event: AnActionEvent) {
- val project = event.project ?: return
- CodeWhispererCodeScanManager.getInstance(project).runCodeScan(CodeWhispererConstants.CodeAnalysisScope.PROJECT)
+ val dataContext = SimpleDataContext.builder()
+ .setParent(event.dataContext)
+ .add(runScanKey, true)
+ .build()
+ val actionEvent = AnActionEvent.createFromDataContext("", null, dataContext)
+ ActionManager.getInstance().getAction("q.openchat").actionPerformed(actionEvent)
}
}
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/actions/CodeWhispererStopCodeScanAction.kt b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/actions/CodeWhispererStopCodeScanAction.kt
deleted file mode 100644
index 06dfed3793..0000000000
--- a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/actions/CodeWhispererStopCodeScanAction.kt
+++ /dev/null
@@ -1,30 +0,0 @@
-// Copyright 2023 Amazon.com, Inc. or its affiliates. All Rights Reserved.
-// SPDX-License-Identifier: Apache-2.0
-
-package software.aws.toolkits.jetbrains.services.codewhisperer.codescan.actions
-
-import com.intellij.icons.AllIcons
-import com.intellij.openapi.actionSystem.ActionUpdateThread
-import com.intellij.openapi.actionSystem.AnActionEvent
-import com.intellij.openapi.project.DumbAwareAction
-import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.CodeWhispererCodeScanManager
-import software.aws.toolkits.resources.message
-
-class CodeWhispererStopCodeScanAction : DumbAwareAction(
- message("codewhisperer.codescan.stop_scan"),
- null,
- AllIcons.Actions.Suspend
-) {
- override fun getActionUpdateThread() = ActionUpdateThread.BGT
-
- override fun update(event: AnActionEvent) {
- val project = event.project ?: return
- val scanManager = CodeWhispererCodeScanManager.getInstance(project)
- event.presentation.isEnabled = scanManager.isCodeScanJobActive()
- }
-
- override fun actionPerformed(event: AnActionEvent) {
- val project = event.project ?: return
- CodeWhispererCodeScanManager.getInstance(project).stopCodeScan()
- }
-}
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/context/CodeScanIssueDetailsDisplayType.kt b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/context/CodeScanIssueDetailsDisplayType.kt
new file mode 100644
index 0000000000..4e148d08a3
--- /dev/null
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/context/CodeScanIssueDetailsDisplayType.kt
@@ -0,0 +1,8 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.codewhisperer.codescan.context
+
+enum class CodeScanIssueDetailsDisplayType {
+ EditorPopup, DetailsPane
+}
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/listeners/CodeWhispererCodeScanDocumentListener.kt b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/listeners/CodeWhispererCodeScanDocumentListener.kt
index 4ac4bc7ada..265a573275 100644
--- a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/listeners/CodeWhispererCodeScanDocumentListener.kt
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/listeners/CodeWhispererCodeScanDocumentListener.kt
@@ -45,6 +45,7 @@ internal class CodeWhispererCodeScanDocumentListener(val project: Project) : Doc
}
issue.rangeHighlighter?.textAttributes = null
issue.rangeHighlighter?.dispose()
+ scanManager.removeIssue(issue)
}
scanManager.updateScanNodes(file)
if (activeEditor != null && activeEditor.file == file &&
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/listeners/CodeWhispererCodeScanEditorMouseMotionListener.kt b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/listeners/CodeWhispererCodeScanEditorMouseMotionListener.kt
index 587da3f963..5e993efc54 100644
--- a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/listeners/CodeWhispererCodeScanEditorMouseMotionListener.kt
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/listeners/CodeWhispererCodeScanEditorMouseMotionListener.kt
@@ -6,50 +6,51 @@ package software.aws.toolkits.jetbrains.services.codewhisperer.codescan.listener
import com.intellij.icons.AllIcons
import com.intellij.ide.BrowserUtil
import com.intellij.ide.ui.laf.darcula.ui.DarculaButtonUI
-import com.intellij.openapi.actionSystem.ActionManager
-import com.intellij.openapi.actionSystem.AnActionEvent
-import com.intellij.openapi.actionSystem.CommonDataKeys
-import com.intellij.openapi.actionSystem.DataKey
-import com.intellij.openapi.actionSystem.impl.SimpleDataContext
-import com.intellij.openapi.command.WriteCommandAction
+import com.intellij.openapi.application.ApplicationManager
import com.intellij.openapi.editor.event.EditorMouseEvent
import com.intellij.openapi.editor.event.EditorMouseEventArea
import com.intellij.openapi.editor.event.EditorMouseMotionListener
import com.intellij.openapi.fileEditor.FileDocumentManager
+import com.intellij.openapi.ide.CopyPasteManager
import com.intellij.openapi.project.Project
import com.intellij.openapi.ui.popup.JBPopup
import com.intellij.openapi.ui.popup.JBPopupFactory
-import com.intellij.psi.PsiDocumentManager
-import com.intellij.ui.JBColor
import com.intellij.ui.awt.RelativePoint
import com.intellij.ui.components.JBScrollPane
-import icons.AwsIcons
-import software.amazon.awssdk.services.codewhispererruntime.model.CodeWhispererRuntimeException
-import software.aws.toolkits.core.utils.convertMarkdownToHTML
+import com.intellij.util.Alarm
import software.aws.toolkits.core.utils.debug
-import software.aws.toolkits.core.utils.error
import software.aws.toolkits.core.utils.getLogger
-import software.aws.toolkits.jetbrains.ToolkitPlaces
import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.CodeWhispererCodeScanIssue
import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.CodeWhispererCodeScanManager
-import software.aws.toolkits.jetbrains.services.codewhisperer.credentials.CodeWhispererClientAdaptor
-import software.aws.toolkits.jetbrains.services.codewhisperer.explorer.CodeWhispererExplorerActionManager
-import software.aws.toolkits.jetbrains.services.codewhisperer.language.CodeWhispererProgrammingLanguage
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.context.CodeScanIssueDetailsDisplayType
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils.additionBackgroundColor
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils.additionForegroundColor
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils.applySuggestedFix
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils.codeBlockBackgroundColor
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils.codeBlockBorderColor
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils.codeBlockForegroundColor
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils.deletionBackgroundColor
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils.deletionForegroundColor
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils.explainIssue
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils.getCodeScanIssueDetailsHtml
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils.getSeverityIcon
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils.metaBackgroundColor
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils.metaForegroundColor
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils.openDiff
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils.sendCodeRemediationTelemetryToServiceApi
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils.truncateIssueTitle
import software.aws.toolkits.jetbrains.services.codewhisperer.language.programmingLanguage
import software.aws.toolkits.jetbrains.services.codewhisperer.telemetry.CodeWhispererTelemetryService
import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererColorUtil.getHexString
-import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererConstants.CODE_SCAN_ISSUE_TITLE_MAX_LENGTH
-import software.aws.toolkits.jetbrains.services.codewhisperer.util.runIfIdcConnectionOrTelemetryEnabled
-import software.aws.toolkits.jetbrains.utils.applyPatch
-import software.aws.toolkits.jetbrains.utils.notifyError
-import software.aws.toolkits.jetbrains.utils.pluginAwareExecuteOnPooledThread
+import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererConstants
import software.aws.toolkits.resources.message
-import software.aws.toolkits.telemetry.Result
+import software.aws.toolkits.telemetry.CodeFixAction
+import software.aws.toolkits.telemetry.MetricResult
import java.awt.Dimension
+import java.awt.datatransfer.StringSelection
import javax.swing.BorderFactory
import javax.swing.Box
import javax.swing.BoxLayout
-import javax.swing.Icon
import javax.swing.JButton
import javax.swing.JEditorPane
import javax.swing.JLabel
@@ -64,114 +65,10 @@ class CodeWhispererCodeScanEditorMouseMotionListener(private val project: Projec
*/
private var currentPopupContext: ScanIssuePopupContext? = null
- private val codeBlockBackgroundColor = JBColor.namedColor("Editor.background", JBColor(0xf7f8fa, 0x2b2d30))
- private val codeBlockForegroundColor = JBColor.namedColor("Editor.foreground", JBColor(0x808080, 0xdfe1e5))
- private val codeBlockBorderColor = JBColor.namedColor("borderColor", JBColor(0xebecf0, 0x1e1f22))
- private val deletionBackgroundColor = JBColor.namedColor("FileColor.Rose", JBColor(0xf5c2c2, 0x511e1e))
- private val deletionForegroundColor = JBColor.namedColor("Label.errorForeground", JBColor(0xb63e3e, 0xfc6479))
- private val additionBackgroundColor = JBColor.namedColor("FileColor.Green", JBColor(0xdde9c1, 0x394323))
- private val additionForegroundColor = JBColor.namedColor("Label.successForeground", JBColor(0x42a174, 0xacc49e))
- private val metaBackgroundColor = JBColor.namedColor("FileColor.Blue", JBColor(0xeaf6ff, 0x4f556b))
- private val metaForegroundColor = JBColor.namedColor("Label.infoForeground", JBColor(0x808080, 0x8C8C8C))
-
private fun hidePopup() {
currentPopupContext?.popup?.cancel()
currentPopupContext = null
}
- private val issueDataKey = DataKey.create>("amazonq.codescan.explainissue")
-
- private fun getHtml(issue: CodeWhispererCodeScanIssue): String {
- // not sure why service team allows multiple remediations, but we only show one
- val suggestedFix = issue.suggestedFixes.firstOrNull()
-
- val cweLinks = if (issue.relatedVulnerabilities.isNotEmpty()) {
- issue.relatedVulnerabilities.joinToString(", ") { cwe ->
- "$cwe "
- }
- } else {
- "-"
- }
-
- val detectorLibraryLink = issue.recommendation.url?.let { "${issue.detectorName} " } ?: "-"
- val detectorSection = """
-
-
-
-
-
- ${message("codewhisperer.codescan.cwe_label")}
- ${message("codewhisperer.codescan.fix_available_label")}
- ${message("codewhisperer.codescan.detector_library_label")}
-
-
-
-
- $cweLinks
- ${if (suggestedFix != null) "Yes " else "No "}
- $detectorLibraryLink
-
-
-
- """.trimIndent()
-
- // add a link sections
- val explainButton = "${message(
- "codewhisperer.codescan.explain_button_label"
- )} "
- val linksSection = """
-
- •$explainButton
-
- """.trimIndent()
-
- val suggestedFixSection = suggestedFix?.let {
- val isFixDescriptionAvailable = it.description.isNotBlank() &&
- it.description.trim() != "Suggested remediation:"
- """
- |
- |
- |
- |## ${message("codewhisperer.codescan.suggested_fix_label")}
- |
- |```diff
- |${it.code}
- |```
- |
- |${
- if (isFixDescriptionAvailable) {
- "|### ${
- message(
- "codewhisperer.codescan.suggested_fix_description"
- )
- }\n${it.description}"
- } else {
- ""
- }
- }
- """.trimMargin()
- }
-
- return convertMarkdownToHTML(
- """
- |$linksSection
- |
- |${issue.recommendation.text}
- |
- |$detectorSection
- |
- |${suggestedFixSection.orEmpty()}
- """.trimMargin()
- )
- }
-
- private fun getSeverityIcon(issue: CodeWhispererCodeScanIssue): Icon? = when (issue.severity) {
- "Info" -> AwsIcons.Resources.CodeWhisperer.SEVERITY_INFO
- "Low" -> AwsIcons.Resources.CodeWhisperer.SEVERITY_LOW
- "Medium" -> AwsIcons.Resources.CodeWhisperer.SEVERITY_MEDIUM
- "High" -> AwsIcons.Resources.CodeWhisperer.SEVERITY_HIGH
- "Critical" -> AwsIcons.Resources.CodeWhisperer.SEVERITY_CRITICAL
- else -> null
- }
private fun showPopup(issues: List, e: EditorMouseEvent, issueIndex: Int = 0) {
if (issues.isEmpty()) {
@@ -182,7 +79,7 @@ class CodeWhispererCodeScanEditorMouseMotionListener(private val project: Projec
}
val issue = issues[issueIndex]
- val content = getHtml(issue)
+ val content = getCodeScanIssueDetailsHtml(issue, CodeScanIssueDetailsDisplayType.EditorPopup, project = project)
val kit = HTMLEditorKit()
kit.styleSheet.apply {
addRule("h1, h3 { margin-bottom: 0 }")
@@ -210,19 +107,34 @@ class CodeWhispererCodeScanEditorMouseMotionListener(private val project: Projec
addHyperlinkListener { he ->
if (he.eventType == HyperlinkEvent.EventType.ACTIVATED) {
when {
- he.description.startsWith("amazonq://issue/explain-") -> {
- val issueItemMap = mutableMapOf()
- issueItemMap["title"] = issue.title
- issueItemMap["description"] = issue.description.markdown
- issueItemMap["code"] = issue.codeText
- val myDataContext = SimpleDataContext.builder().add(issueDataKey, issueItemMap).add(CommonDataKeys.PROJECT, issue.project).build()
- val actionEvent = AnActionEvent.createFromInputEvent(
- he.inputEvent,
- ToolkitPlaces.EDITOR_PSI_REFERENCE,
- null,
- myDataContext
+ he.description.startsWith("amazonq://issue/openDiff-") -> {
+ openDiff(issue)
+ }
+ he.description.startsWith("amazonq://issue/copyDiff-") -> {
+ text = getCodeScanIssueDetailsHtml(
+ issue,
+ CodeScanIssueDetailsDisplayType.DetailsPane,
+ CodeWhispererConstants.FixGenerationState.COMPLETED,
+ true,
+ project = project
)
- ActionManager.getInstance().getAction("aws.amazonq.explainCodeScanIssue").actionPerformed(actionEvent)
+ CopyPasteManager.getInstance().setContents(StringSelection(issue.suggestedFixes.first().code))
+ val alarm = Alarm()
+ alarm.addRequest({
+ ApplicationManager.getApplication().invokeLater {
+ text = getCodeScanIssueDetailsHtml(
+ issue,
+ CodeScanIssueDetailsDisplayType.DetailsPane,
+ CodeWhispererConstants.FixGenerationState.COMPLETED,
+ false,
+ project = project
+ )
+ }
+ }, 500)
+ ApplicationManager.getApplication().executeOnPooledThread {
+ CodeWhispererTelemetryService.getInstance()
+ .sendCodeScanIssueApplyFixEvent(issue, MetricResult.Succeeded, codeFixAction = CodeFixAction.CopyDiff)
+ }
}
else -> {
BrowserUtil.browse(he.url)
@@ -240,7 +152,7 @@ class CodeWhispererCodeScanEditorMouseMotionListener(private val project: Projec
verticalScrollBarPolicy = ScrollPaneConstants.VERTICAL_SCROLLBAR_AS_NEEDED
horizontalScrollBarPolicy = ScrollPaneConstants.HORIZONTAL_SCROLLBAR_AS_NEEDED
}
- val label = JLabel(truncateTitle(issue.title)).apply {
+ val label = JLabel(truncateIssueTitle(issue.title)).apply {
icon = getSeverityIcon(issue)
horizontalTextPosition = JLabel.LEFT
}
@@ -249,7 +161,7 @@ class CodeWhispererCodeScanEditorMouseMotionListener(private val project: Projec
putClientProperty(DarculaButtonUI.DEFAULT_STYLE_KEY, true)
}
button.addActionListener {
- handleApplyFix(issue)
+ applySuggestedFix(project, issue)
button.isVisible = false
}
val nextButton = JButton(AllIcons.Actions.ArrowExpand).apply {
@@ -267,12 +179,21 @@ class CodeWhispererCodeScanEditorMouseMotionListener(private val project: Projec
}
}
+ val explainButton = JButton(
+ message("codewhisperer.codescan.explain_button_label")
+ ).apply {
+ toolTipText = message("codewhisperer.codescan.apply_fix_button_tooltip")
+ addActionListener {
+ hidePopup()
+ explainIssue(issue)
+ }
+ }
+
val titlePane = JPanel().apply {
layout = BoxLayout(this, BoxLayout.X_AXIS)
preferredSize = Dimension(this.width, 30)
- add(Box.createHorizontalGlue())
- add(label)
- add(Box.createHorizontalGlue())
+
+ // Add buttons first if they exist
if (issues.size > 1) {
add(prevButton)
add(JLabel("${issueIndex + 1} of ${issues.size}"))
@@ -282,6 +203,12 @@ class CodeWhispererCodeScanEditorMouseMotionListener(private val project: Projec
if (issue.suggestedFixes.isNotEmpty()) {
add(button)
}
+ add(explainButton)
+
+ // Add glue before and after label to center it
+ add(Box.createHorizontalGlue())
+ add(label)
+ add(Box.createHorizontalGlue())
}
val containerPane = JPanel().apply {
@@ -300,6 +227,7 @@ class CodeWhispererCodeScanEditorMouseMotionListener(private val project: Projec
CodeWhispererTelemetryService.getInstance().sendCodeScanIssueHoverEvent(issue)
sendCodeRemediationTelemetryToServiceApi(
+ project,
issue.file.programmingLanguage(),
"CODESCAN_ISSUE_HOVER",
issue.detectorId,
@@ -312,43 +240,6 @@ class CodeWhispererCodeScanEditorMouseMotionListener(private val project: Projec
)
}
- private fun sendCodeRemediationTelemetryToServiceApi(
- language: CodeWhispererProgrammingLanguage?,
- codeScanRemediationEventType: String?,
- detectorId: String?,
- findingId: String?,
- ruleId: String?,
- component: String?,
- reason: String?,
- result: String?,
- includesFix: Boolean?,
- ) {
- runIfIdcConnectionOrTelemetryEnabled(project) {
- pluginAwareExecuteOnPooledThread {
- try {
- val response = CodeWhispererClientAdaptor.getInstance(project)
- .sendCodeScanRemediationTelemetry(
- language,
- codeScanRemediationEventType,
- detectorId,
- findingId,
- ruleId,
- component,
- reason,
- result,
- includesFix
- )
- LOG.debug { "Successfully sent code scan remediation telemetry. RequestId: ${response.responseMetadata().requestId()}" }
- } catch (e: Exception) {
- val requestId = if (e is CodeWhispererRuntimeException) e.requestId() else null
- LOG.debug {
- "Failed to send code scan remediation telemetry. RequestId: $requestId, ErrorMessage: ${e.message}"
- }
- }
- }
- }
- }
-
override fun mouseMoved(e: EditorMouseEvent) {
val scanManager = CodeWhispererCodeScanManager.getInstance(project)
if (e.area != EditorMouseEventArea.EDITING_AREA || !e.isOverText) {
@@ -381,52 +272,4 @@ class CodeWhispererCodeScanEditorMouseMotionListener(private val project: Projec
companion object {
private val LOG = getLogger()
}
-
- private fun handleApplyFix(issue: CodeWhispererCodeScanIssue) {
- try {
- WriteCommandAction.runWriteCommandAction(issue.project) {
- val document = FileDocumentManager.getInstance().getDocument(issue.file) ?: return@runWriteCommandAction
-
- val documentContent = document.text
- val updatedContent = applyPatch(issue.suggestedFixes[0].code, documentContent, issue.file.name)
- document.replaceString(document.getLineStartOffset(0), document.getLineEndOffset(document.lineCount - 1), updatedContent)
- PsiDocumentManager.getInstance(issue.project).commitDocument(document)
- CodeWhispererTelemetryService.getInstance().sendCodeScanIssueApplyFixEvent(issue, Result.Succeeded)
- hidePopup()
- if (CodeWhispererExplorerActionManager.getInstance().isAutoEnabledForCodeScan()) {
- CodeWhispererCodeScanManager.getInstance(issue.project).removeIssueByFindingId(issue.file, issue.findingId)
- }
- }
- sendCodeRemediationTelemetryToServiceApi(
- issue.file.programmingLanguage(),
- "CODESCAN_ISSUE_APPLY_FIX",
- issue.detectorId,
- issue.findingId,
- issue.ruleId,
- null,
- null,
- Result.Succeeded.toString(),
- issue.suggestedFixes.isNotEmpty()
- )
- } catch (err: Error) {
- notifyError(message("codewhisperer.codescan.fix_applied_fail", err))
- LOG.error { "Apply fix command failed. $err" }
- CodeWhispererTelemetryService.getInstance().sendCodeScanIssueApplyFixEvent(issue, Result.Failed, err.message)
- sendCodeRemediationTelemetryToServiceApi(
- issue.file.programmingLanguage(),
- "CODESCAN_ISSUE_APPLY_FIX",
- issue.detectorId,
- issue.findingId,
- issue.ruleId,
- null,
- err.message,
- Result.Failed.toString(),
- issue.suggestedFixes.isNotEmpty()
- )
- }
- }
-
- private fun truncateTitle(title: String): String = title.takeUnless { it.length <= CODE_SCAN_ISSUE_TITLE_MAX_LENGTH }?.let {
- it.substring(0, CODE_SCAN_ISSUE_TITLE_MAX_LENGTH - 3) + "..."
- } ?: title
}
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/sessionconfig/CodeScanSessionConfig.kt b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/sessionconfig/CodeScanSessionConfig.kt
index 4360731fd3..e02a32e278 100644
--- a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/sessionconfig/CodeScanSessionConfig.kt
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/sessionconfig/CodeScanSessionConfig.kt
@@ -16,6 +16,7 @@ import com.intellij.openapi.vfs.isFile
import kotlinx.coroutines.runBlocking
import software.aws.toolkits.core.utils.createTemporaryZipFile
import software.aws.toolkits.core.utils.debug
+import software.aws.toolkits.core.utils.error
import software.aws.toolkits.core.utils.getLogger
import software.aws.toolkits.core.utils.putNextEntry
import software.aws.toolkits.jetbrains.services.amazonq.FeatureDevSessionContext
@@ -24,9 +25,13 @@ import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.cannotFin
import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.fileTooLarge
import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.noFileOpenError
import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.noSupportedFilesError
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils.AmazonQCodeReviewGitUtils.getUnstagedFiles
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils.AmazonQCodeReviewGitUtils.isGitRoot
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils.AmazonQCodeReviewGitUtils.runGitDiffHead
import software.aws.toolkits.jetbrains.services.codewhisperer.language.CodeWhispererProgrammingLanguage
import software.aws.toolkits.jetbrains.services.codewhisperer.language.languages.CodeWhispererUnknownLanguage
import software.aws.toolkits.jetbrains.services.codewhisperer.language.programmingLanguage
+import software.aws.toolkits.jetbrains.services.codewhisperer.telemetry.CodeWhispererTelemetryService
import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererConstants.CODE_SCAN_CREATE_PAYLOAD_TIMEOUT_IN_SECONDS
import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererConstants.CodeAnalysisScope
import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererConstants.DEFAULT_CODE_SCAN_TIMEOUT_IN_SECONDS
@@ -46,6 +51,7 @@ class CodeScanSessionConfig(
private val selectedFile: VirtualFile?,
private val project: Project,
private val scope: CodeAnalysisScope,
+ private val initiatedByChat: Boolean,
) {
var projectRoot = project.basePath?.let { Path.of(it) }?.toFile()?.toVirtualFile() ?: run {
project.guessProjectDir() ?: error("Cannot guess base directory for project ${project.name}")
@@ -56,6 +62,8 @@ class CodeScanSessionConfig(
val fileIndex = ProjectRootManager.getInstance(project).fileIndex
+ fun isInitiatedByChat(): Boolean = initiatedByChat
+
/**
* Timeout for the overall job - "Run Security Scan".
*/
@@ -99,7 +107,7 @@ class CodeScanSessionConfig(
else -> when (scope) {
CodeAnalysisScope.PROJECT -> getProjectPayloadMetadata()
CodeAnalysisScope.FILE -> if (selectedFile.path.startsWith(projectRoot.path)) {
- getFilePayloadMetadata(selectedFile)
+ getFilePayloadMetadata(selectedFile, true)
} else {
projectRoot = selectedFile.parent
getFilePayloadMetadata(selectedFile)
@@ -116,7 +124,7 @@ class CodeScanSessionConfig(
}
// Copy all the included source files to the source zip
- val srcZip = zipFiles(payloadMetadata.sourceFiles.map { Path.of(it) })
+ val srcZip = zipFiles(payloadMetadata.sourceFiles.map { Path.of(it) }, payloadMetadata.codeDiff)
val payloadContext = PayloadContext(
payloadMetadata.language,
payloadMetadata.linesScanned,
@@ -130,19 +138,46 @@ class CodeScanSessionConfig(
return Payload(payloadContext, srcZip)
}
- private fun getFilePayloadMetadata(file: VirtualFile): PayloadMetadata {
+ private fun getFilePayloadMetadata(file: VirtualFile, getCodeDiff: Boolean? = false): PayloadMetadata {
try {
+ val gitDiffContent = if (initiatedByChat && getCodeDiff == true) {
+ getFileGitDiffContent(file)
+ } else {
+ null
+ }
return PayloadMetadata(
setOf(file.path),
file.length,
countLinesInVirtualFile(file).toLong(),
- file.programmingLanguage().toTelemetryType()
+ file.programmingLanguage().toTelemetryType(),
+ gitDiffContent
)
} catch (e: Exception) {
cannotFindFile("File payload creation error: ${e.message}", file.path)
}
}
+ private fun getFileGitDiffContent(file: VirtualFile): String {
+ if (!file.exists()) {
+ LOG.debug { "File does not exist: ${file.path}" }
+ return ""
+ }
+ try {
+ val projectRootNio = projectRoot.toNioPath()
+ val fileNio = file.toNioPath()
+
+ return buildString {
+ append("+++ b/")
+ append(project.name)
+ append('/')
+ append(fileNio.relativeTo(projectRootNio).toString().replace(File.separator, "/"))
+ }
+ } catch (e: Exception) {
+ LOG.debug(e) { "Failed to create git diff" }
+ return ""
+ }
+ }
+
/**
* Timeout for creating the payload [createPayload]
*/
@@ -157,16 +192,32 @@ class CodeScanSessionConfig(
}
}
- private fun zipFiles(files: List): File = createTemporaryZipFile {
+ private fun zipFiles(files: List, codeDiff: String? = null): File = createTemporaryZipFile {
files.forEach { file ->
try {
- val relativePath = file.relativeTo(projectRoot.toNioPath())
+ val relativePath = "${project.name}/${file.relativeTo(projectRoot.toNioPath())}"
+ if (relativePath.contains("../") || relativePath.contains("..\\")) {
+ CodeWhispererTelemetryService.getInstance().sendInvalidZipEvent(file, projectRoot.toNioPath(), relativePath)
+ }
LOG.debug { "Selected file for truncation: $file" }
it.putNextEntry(relativePath.toString(), file)
} catch (e: Exception) {
cannotFindFile("Zipping error: ${e.message}", file.pathString)
}
}
+
+ codeDiff?.takeIf { diff ->
+ initiatedByChat && diff.isNotEmpty()
+ }?.let { diff ->
+ try {
+ LOG.debug { "Adding Code.Diff file to zip" }
+ diff.byteInputStream(Charsets.UTF_8).buffered().use { inputStream ->
+ it.putNextEntry("codeDiff/code.diff", inputStream)
+ }
+ } catch (e: Exception) {
+ LOG.error(e) { "Failed to add Code.Diff" }
+ }
+ }
}.toFile()
fun getProjectPayloadMetadata(): PayloadMetadata {
@@ -176,6 +227,7 @@ class CodeScanSessionConfig(
var currentTotalFileSize = 0L
var currentTotalLines = 0L
val languageCounts = mutableMapOf()
+ var gitDiffContent = ""
moduleLoop@ for (module in project.modules) {
val changeListManager = ChangeListManager.getInstance(module.project)
@@ -207,6 +259,26 @@ class CodeScanSessionConfig(
}
}
} else {
+ try {
+ if (isGitRoot(current)) {
+ LOG.debug { "$current is git directory" }
+ gitDiffContent = buildString {
+ append(runGitDiffHead(project.name, current))
+ getUnstagedFiles(current).takeIf { it.isNotEmpty() }?.let { unstagedFiles ->
+ unstagedFiles
+ .asSequence()
+ .map { relativePath -> runGitDiffHead(project.name, current, relativePath, true) }
+ .filter { it.isNotEmpty() }
+ .forEach { diff ->
+ if (isNotEmpty()) append('\n')
+ append(diff)
+ }
+ }
+ }
+ }
+ } catch (e: Exception) {
+ LOG.debug { "Error parsing the git diff for repository $current" }
+ }
// Directory case: only traverse if not ignored
if (!changeListManager.isIgnoredFile(current) &&
runBlocking { !featureDevSessionContext.ignoreFile(current) } &&
@@ -231,21 +303,19 @@ class CodeScanSessionConfig(
noSupportedFilesError()
}
programmingLanguage = maxCountLanguage
- return PayloadMetadata(files, currentTotalFileSize, currentTotalLines, maxCountLanguage.toTelemetryType())
- }
-
- fun getPath(root: String, relativePath: String = ""): Path? = try {
- Path.of(root, relativePath).normalize()
- } catch (e: Exception) {
- LOG.debug { "Cannot find file at path $relativePath relative to the root $root" }
- null
+ return PayloadMetadata(files, currentTotalFileSize, currentTotalLines, maxCountLanguage.toTelemetryType(), gitDiffContent)
}
fun File.toVirtualFile() = LocalFileSystem.getInstance().findFileByIoFile(this)
companion object {
private val LOG = getLogger()
- fun create(file: VirtualFile?, project: Project, scope: CodeAnalysisScope): CodeScanSessionConfig = CodeScanSessionConfig(file, project, scope)
+ fun create(file: VirtualFile?, project: Project, scope: CodeAnalysisScope, initiatedByChat: Boolean): CodeScanSessionConfig = CodeScanSessionConfig(
+ file,
+ project,
+ scope,
+ initiatedByChat
+ )
}
}
@@ -270,4 +340,5 @@ data class PayloadMetadata(
val payloadSize: Long,
val linesScanned: Long,
val language: CodewhispererLanguage,
+ val codeDiff: String? = null,
)
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/utils/AmazonQCodeReviewGitUtils.kt b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/utils/AmazonQCodeReviewGitUtils.kt
new file mode 100644
index 0000000000..c4cab3fef3
--- /dev/null
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/utils/AmazonQCodeReviewGitUtils.kt
@@ -0,0 +1,141 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils
+
+import com.intellij.execution.configurations.GeneralCommandLine
+import com.intellij.execution.util.ExecUtil
+import com.intellij.openapi.util.SystemInfo
+import com.intellij.openapi.vfs.LocalFileSystem
+import com.intellij.openapi.vfs.VirtualFile
+import software.aws.toolkits.core.utils.debug
+import software.aws.toolkits.core.utils.getLogger
+import java.io.File
+
+object AmazonQCodeReviewGitUtils {
+ private val LOG = getLogger()
+ private const val PROCESS_TIMEOUT_MS = 5000L
+
+ /**
+ * Executes a git command and returns the process output
+ */
+ private fun executeGitCommand(
+ workDir: File,
+ vararg parameters: String,
+ timeoutMs: Long = PROCESS_TIMEOUT_MS,
+ ): Pair {
+ val commandLine = GeneralCommandLine().apply {
+ workDirectory = workDir
+ exePath = "git"
+ addParameters(*parameters)
+ }
+
+ return try {
+ val output = ExecUtil.execAndGetOutput(commandLine, timeoutMs.toInt())
+ if (output.exitCode != 0) {
+ LOG.debug { "Git command failed with exit code ${output.exitCode}: ${output.stderr}" }
+ }
+ Pair(output.stdout.trim(), output.stderr.trim())
+ } catch (e: Exception) {
+ LOG.debug(e) { "Git command failed: ${commandLine.commandLineString}" }
+ Pair("", e.message ?: "Unknown error")
+ }
+ }
+
+ fun runGitDiffHead(
+ projectName: String,
+ root: VirtualFile,
+ relativeFilePath: String? = null,
+ newFile: Boolean? = false,
+ ): String {
+ if (!root.exists()) {
+ LOG.debug { "Root directory does not exist: ${root.path}" }
+ return ""
+ }
+
+ val prefixes = arrayOf(
+ "--src-prefix=a/$projectName/",
+ "--dst-prefix=b/$projectName/"
+ )
+ val ref = if (SystemInfo.isWindows) "NUL" else "/dev/null"
+
+ val parameters = when {
+ relativeFilePath == null -> arrayOf("diff", "HEAD") + prefixes
+ newFile == true -> arrayOf("diff", "--no-index", *prefixes, ref, relativeFilePath)
+ else -> arrayOf("diff", "HEAD", *prefixes, relativeFilePath)
+ }
+
+ val (output, error) = executeGitCommand(File(root.path), *parameters)
+
+ return when {
+ error.contains("Authentication failed") -> {
+ LOG.debug { "Git Authentication Failed" }
+ throw RuntimeException("Git Authentication Failed")
+ }
+ error.isNotEmpty() -> {
+ LOG.debug { "Git command failed: $error" }
+ ""
+ }
+ else -> output
+ }
+ }
+
+ fun isGitRoot(file: VirtualFile): Boolean {
+ if (!file.exists()) return false
+
+ val workDir = if (file.isDirectory) File(file.path) else File(file.parent.path)
+ val (output, _) = executeGitCommand(workDir, "rev-parse", "--git-dir")
+
+ return output == ".git"
+ }
+
+ fun getGitRepositoryRoot(file: VirtualFile): VirtualFile? {
+ if (!file.exists()) return null
+
+ val workDir = if (file.isDirectory) {
+ File(file.path)
+ } else {
+ File(file.parent.path)
+ }
+
+ if (!workDir.exists() || !workDir.isDirectory) {
+ LOG.debug { "Invalid working directory: ${workDir.path}" }
+ return null
+ }
+
+ val (output, error) = executeGitCommand(workDir, "rev-parse", "--show-toplevel")
+
+ return when {
+ error.isNotEmpty() -> {
+ LOG.debug { "Failed to get git root: $error" }
+ null
+ }
+ output.isEmpty() -> null
+ else -> LocalFileSystem.getInstance().findFileByPath(output)
+ }
+ }
+
+ fun getUnstagedFiles(root: VirtualFile): List {
+ if (!root.exists()) return emptyList()
+
+ val (output, error) = executeGitCommand(
+ File(root.path),
+ "ls-files",
+ "--others",
+ "--exclude-standard"
+ )
+
+ return when {
+ error.isNotEmpty() -> {
+ LOG.debug { "Failed to get unstaged files: $error" }
+ emptyList()
+ }
+ else -> output.split("\n").filter { it.isNotEmpty() }
+ }
+ }
+
+ fun isInsideWorkTree(folder: VirtualFile): Boolean {
+ val (output) = executeGitCommand(File(folder.path), "rev-parse", "--is-inside-work-tree")
+ return output == "true"
+ }
+}
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/utils/CodeWhispererCodeScanIssueUtils.kt b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/utils/CodeWhispererCodeScanIssueUtils.kt
new file mode 100644
index 0000000000..f1d581673b
--- /dev/null
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/utils/CodeWhispererCodeScanIssueUtils.kt
@@ -0,0 +1,444 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.codewhisperer.codescan.utils
+
+import com.intellij.diff.DiffContentFactory
+import com.intellij.diff.DiffManager
+import com.intellij.diff.requests.SimpleDiffRequest
+import com.intellij.diff.util.DiffUserDataKeys
+import com.intellij.openapi.actionSystem.ActionManager
+import com.intellij.openapi.actionSystem.AnActionEvent
+import com.intellij.openapi.actionSystem.CommonDataKeys
+import com.intellij.openapi.actionSystem.DataKey
+import com.intellij.openapi.actionSystem.impl.SimpleDataContext
+import com.intellij.openapi.application.ApplicationManager
+import com.intellij.openapi.command.WriteCommandAction
+import com.intellij.openapi.fileEditor.FileDocumentManager
+import com.intellij.openapi.project.Project
+import com.intellij.openapi.project.guessProjectDir
+import com.intellij.openapi.vfs.VfsUtil
+import com.intellij.openapi.vfs.VfsUtilCore
+import com.intellij.openapi.vfs.VirtualFileManager
+import com.intellij.psi.PsiDocumentManager
+import com.intellij.ui.JBColor
+import icons.AwsIcons
+import software.amazon.awssdk.services.codewhispererruntime.model.CodeWhispererRuntimeException
+import software.aws.toolkits.core.utils.convertMarkdownToHTML
+import software.aws.toolkits.core.utils.debug
+import software.aws.toolkits.core.utils.getLogger
+import software.aws.toolkits.jetbrains.ToolkitPlaces
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.CodeWhispererCodeScanHighlightingFilesPanel
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.CodeWhispererCodeScanIssue
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.CodeWhispererCodeScanManager
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.SuggestedFix
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.context.CodeScanIssueDetailsDisplayType
+import software.aws.toolkits.jetbrains.services.codewhisperer.credentials.CodeWhispererClientAdaptor
+import software.aws.toolkits.jetbrains.services.codewhisperer.explorer.CodeWhispererExplorerActionManager
+import software.aws.toolkits.jetbrains.services.codewhisperer.language.CodeWhispererProgrammingLanguage
+import software.aws.toolkits.jetbrains.services.codewhisperer.language.programmingLanguage
+import software.aws.toolkits.jetbrains.services.codewhisperer.telemetry.CodeWhispererTelemetryService
+import software.aws.toolkits.jetbrains.services.codewhisperer.toolwindow.CodeWhispererCodeReferenceManager
+import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererConstants
+import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererConstants.CODE_SCAN_ISSUE_TITLE_MAX_LENGTH
+import software.aws.toolkits.jetbrains.services.codewhisperer.util.runIfIdcConnectionOrTelemetryEnabled
+import software.aws.toolkits.jetbrains.utils.applyPatch
+import software.aws.toolkits.jetbrains.utils.notifyError
+import software.aws.toolkits.jetbrains.utils.pluginAwareExecuteOnPooledThread
+import software.aws.toolkits.resources.message
+import software.aws.toolkits.telemetry.CodeFixAction
+import software.aws.toolkits.telemetry.Result
+import javax.swing.Icon
+
+val codeBlockBackgroundColor = JBColor.namedColor("Editor.background", JBColor(0xf7f8fa, 0x2b2d30))
+val codeBlockForegroundColor = JBColor.namedColor("Editor.foreground", JBColor(0x808080, 0xdfe1e5))
+val codeBlockBorderColor = JBColor.namedColor("borderColor", JBColor(0xebecf0, 0x1e1f22))
+val deletionBackgroundColor = JBColor.namedColor("FileColor.Rose", JBColor(0xf5c2c2, 0x511e1e))
+val deletionForegroundColor = JBColor.namedColor("Label.errorForeground", JBColor(0xb63e3e, 0xfc6479))
+val additionBackgroundColor = JBColor.namedColor("FileColor.Green", JBColor(0xdde9c1, 0x394323))
+val additionForegroundColor = JBColor.namedColor("Label.successForeground", JBColor(0x42a174, 0xacc49e))
+val metaBackgroundColor = JBColor.namedColor("FileColor.Blue", JBColor(0xeaf6ff, 0x4f556b))
+val metaForegroundColor = JBColor.namedColor("Label.infoForeground", JBColor(0x808080, 0x8C8C8C))
+
+private val LOG = getLogger()
+private val explainIssueDataKey = DataKey.create>("amazonq.codescan.explainissue")
+
+enum class IssueSeverity(val displayName: String) {
+ CRITICAL("Critical"),
+ HIGH("High"),
+ MEDIUM("Medium"),
+ LOW("Low"),
+ INFO("Info"),
+}
+
+fun getCodeScanIssueDetailsHtml(
+ issue: CodeWhispererCodeScanIssue,
+ display: CodeScanIssueDetailsDisplayType,
+ fixGenerationState: CodeWhispererConstants.FixGenerationState = CodeWhispererConstants.FixGenerationState.COMPLETED,
+ isCopied: Boolean = false,
+ project: Project,
+ showReferenceWarning: Boolean? = false,
+): String {
+ val suggestedFix = issue.suggestedFixes.firstOrNull()
+
+ val cweLinks = if (issue.relatedVulnerabilities.isNotEmpty()) {
+ issue.relatedVulnerabilities.joinToString(", ") { cwe ->
+ "$cwe "
+ }
+ } else {
+ "-"
+ }
+
+ val projectRoot = project.basePath?.let { VirtualFileManager.getInstance().findFileByUrl(VfsUtilCore.pathToUrl(it)) } ?: project.guessProjectDir()
+ val filePathString = projectRoot?.let { VfsUtil.getRelativePath(issue.file, it) } ?: issue.file.path
+
+ val fileLink = "${ filePathString } [Ln ${issue.startLine}] "
+
+ val detectorLibraryLink = issue.recommendation.url?.let { "${issue.detectorName} " } ?: "-"
+ val detectorSection = """
+
+
+
+
+
+ ${message("codewhisperer.codescan.cwe_label")}
+ ${message("codewhisperer.codescan.detector_library_label")}
+
+
+
+
+ $cweLinks
+ $detectorLibraryLink
+
+
+
+
+
+
+ ${message("codewhisperer.codescan.file_path_label")}
+
+
+
+
+ $fileLink
+
+
+
+ """.trimIndent()
+
+ val suggestedFixSection = if (showReferenceWarning == false) {
+ createSuggestedFixSection(issue, suggestedFix, isCopied)
+ } else {
+ """
+
+ Your settings do not allow code generation with references.
+
+ """.trimIndent()
+ }
+
+ val fixLoadingSection = """
+
+
+
+ ...
+
+
+ """.trimIndent()
+
+ val fixFailureSection = """
+
+
+
+ Amazon Q failed to generate fix. Please try again
+
+
+ """.trimIndent()
+
+ val commonContent = """
+ |${issue.recommendation.text}
+ |
+ |$detectorSection
+ |
+ |${when (fixGenerationState) {
+ CodeWhispererConstants.FixGenerationState.COMPLETED -> suggestedFixSection.orEmpty()
+ CodeWhispererConstants.FixGenerationState.GENERATING -> fixLoadingSection
+ CodeWhispererConstants.FixGenerationState.FAILED -> fixFailureSection
+ }}
+ """.trimMargin()
+
+ if (display == CodeScanIssueDetailsDisplayType.EditorPopup) {
+ return convertMarkdownToHTML(
+ """
+ |$commonContent
+ """.trimMargin()
+ )
+ }
+
+ return convertMarkdownToHTML(commonContent)
+}
+
+private fun createSuggestedFixSection(issue: CodeWhispererCodeScanIssue, suggestedFix: SuggestedFix?, isCopied: Boolean = false): String? = suggestedFix?.let {
+ val isFixDescriptionAvailable = it.description.isNotBlank() &&
+ it.description.trim() != "Suggested remediation:"
+ """
+ |
+ |
+ |### ${message("codewhisperer.codescan.suggested_fix_label")}
+ |
+ |
+ |
+ |
+ |
+ |
+ |```diff
+ |${it.code.trim()}
+ |```
+ |
+ |
+ |
+ |
+ |
+ |${
+ if (isFixDescriptionAvailable) {
+ "|### ${
+ message(
+ "codewhisperer.codescan.suggested_fix_description"
+ )
+ }\n${it.description}"
+ } else {
+ ""
+ }
+ }
+ """.trimMargin()
+}
+
+fun explainIssue(issue: CodeWhispererCodeScanIssue) {
+ val explainIssueContext = mutableMapOf(
+ "title" to issue.title,
+ "description" to issue.description.markdown,
+ "code" to issue.codeText
+ )
+ val actionEvent = AnActionEvent.createFromInputEvent(
+ null,
+ ToolkitPlaces.EDITOR_PSI_REFERENCE,
+ null,
+ SimpleDataContext.builder().add(explainIssueDataKey, explainIssueContext).add(CommonDataKeys.PROJECT, issue.project).build()
+ )
+ ActionManager.getInstance().getAction("aws.amazonq.explainCodeScanIssue").actionPerformed(actionEvent)
+}
+
+fun openDiff(issue: CodeWhispererCodeScanIssue) {
+ val diffContentFactory = DiffContentFactory.getInstance()
+ val document = FileDocumentManager.getInstance().getDocument(issue.file)
+ document?.text?.let { documentContent ->
+ val updatedContent = applyPatch(issue.suggestedFixes[0].code, documentContent, issue.file.name)
+ val (originalContent, suggestedContent) = try {
+ diffContentFactory.create(documentContent) to
+ diffContentFactory.create(updatedContent)
+ } catch (e: Exception) {
+ ApplicationManager.getApplication().executeOnPooledThread {
+ CodeWhispererTelemetryService.getInstance().sendCodeScanIssueApplyFixEvent(
+ issue,
+ Result.Failed,
+ e.message,
+ codeFixAction = CodeFixAction.OpenDiff
+ )
+ }
+ return@let null
+ }
+
+ val request = SimpleDiffRequest(
+ "Amazon Q Code Suggestion Diff",
+ suggestedContent,
+ originalContent,
+ "Suggested fix",
+ "Original code"
+ ).apply {
+ putUserData(DiffUserDataKeys.MERGE_EDITOR_FLAG, true)
+
+ putUserData(DiffUserDataKeys.DO_NOT_IGNORE_WHITESPACES, true)
+
+ putUserData(DiffUserDataKeys.ENABLE_SEARCH_IN_CHANGES, true)
+ putUserData(DiffUserDataKeys.GO_TO_SOURCE_DISABLE, false)
+
+ putUserData(DiffUserDataKeys.ALIGNED_TWO_SIDED_DIFF, true)
+ putUserData(DiffUserDataKeys.FORCE_READ_ONLY_CONTENTS, booleanArrayOf(true, false))
+ putUserData(DiffUserDataKeys.FORCE_READ_ONLY, false)
+ }
+ ApplicationManager.getApplication().invokeLater {
+ DiffManager.getInstance().showDiff(
+ issue.project,
+ request
+ )
+ }
+ }
+ ApplicationManager.getApplication().executeOnPooledThread {
+ CodeWhispererTelemetryService.getInstance().sendCodeScanIssueApplyFixEvent(issue, Result.Succeeded, codeFixAction = CodeFixAction.OpenDiff)
+ }
+}
+
+fun truncateIssueTitle(title: String): String = title.takeUnless { it.length <= CODE_SCAN_ISSUE_TITLE_MAX_LENGTH }?.let {
+ it.substring(0, CODE_SCAN_ISSUE_TITLE_MAX_LENGTH - 3) + "..."
+} ?: title
+
+fun sendCodeRemediationTelemetryToServiceApi(
+ project: Project,
+ language: CodeWhispererProgrammingLanguage?,
+ codeScanRemediationEventType: String?,
+ detectorId: String?,
+ findingId: String?,
+ ruleId: String?,
+ component: String?,
+ reason: String?,
+ result: String?,
+ includesFix: Boolean?,
+) {
+ runIfIdcConnectionOrTelemetryEnabled(project) {
+ pluginAwareExecuteOnPooledThread {
+ try {
+ val response = CodeWhispererClientAdaptor.getInstance(project)
+ .sendCodeScanRemediationTelemetry(
+ language,
+ codeScanRemediationEventType,
+ detectorId,
+ findingId,
+ ruleId,
+ component,
+ reason,
+ result,
+ includesFix
+ )
+ LOG.debug { "Successfully sent code scan remediation telemetry. RequestId: ${response.responseMetadata().requestId()}" }
+ } catch (e: Exception) {
+ val requestId = if (e is CodeWhispererRuntimeException) e.requestId() else null
+ LOG.debug(e) {
+ "Failed to send code scan remediation telemetry. RequestId: $requestId"
+ }
+ }
+ }
+ }
+}
+
+fun applySuggestedFix(project: Project, issue: CodeWhispererCodeScanIssue) {
+ try {
+ val manager = CodeWhispererCodeReferenceManager.getInstance(issue.project)
+ WriteCommandAction.runWriteCommandAction(issue.project) {
+ val document = FileDocumentManager.getInstance().getDocument(issue.file) ?: return@runWriteCommandAction
+
+ val documentContent = document.text
+ val updatedContent = applyPatch(issue.suggestedFixes[0].code, documentContent, issue.file.name)
+ document.replaceString(document.getLineStartOffset(0), document.getLineEndOffset(document.lineCount - 1), updatedContent)
+ PsiDocumentManager.getInstance(issue.project).commitDocument(document)
+ issue.suggestedFixes[0].references.forEach { reference ->
+ LOG.debug { "Applied fix with reference: $reference" }
+ val originalContent = updatedContent.substring(reference.recommendationContentSpan().start(), reference.recommendationContentSpan().end())
+ LOG.debug { "Original content from reference span: $originalContent" }
+ manager.addReferenceLogPanelEntry(reference = reference, null, null, originalContent.split("\n"))
+ }
+ }
+ if (issue.suggestedFixes[0].references.isNotEmpty()) {
+ manager.toolWindow?.show()
+ }
+ if (CodeWhispererExplorerActionManager.getInstance().isAutoEnabledForCodeScan()) {
+ CodeWhispererCodeScanManager.getInstance(issue.project).removeIssueByFindingId(issue, issue.findingId)
+ }
+ ApplicationManager.getApplication().executeOnPooledThread {
+ CodeWhispererTelemetryService.getInstance().sendCodeScanIssueApplyFixEvent(issue, Result.Succeeded, codeFixAction = CodeFixAction.ApplyFix)
+ }
+ sendCodeRemediationTelemetryToServiceApi(
+ project,
+ issue.file.programmingLanguage(),
+ "CODESCAN_ISSUE_APPLY_FIX",
+ issue.detectorId,
+ issue.findingId,
+ issue.ruleId,
+ null,
+ null,
+ Result.Succeeded.toString(),
+ issue.suggestedFixes.isNotEmpty()
+ )
+ sendCodeFixGeneratedTelemetryToServiceAPI(issue, true)
+ } catch (e: Throwable) {
+ notifyError(message("codewhisperer.codescan.fix_applied_fail", e))
+ LOG.debug(e) { "Apply fix command failed." }
+ ApplicationManager.getApplication().executeOnPooledThread {
+ CodeWhispererTelemetryService.getInstance().sendCodeScanIssueApplyFixEvent(issue, Result.Failed, e.message, codeFixAction = CodeFixAction.ApplyFix)
+ sendCodeRemediationTelemetryToServiceApi(
+ project,
+ issue.file.programmingLanguage(),
+ "CODESCAN_ISSUE_APPLY_FIX",
+ issue.detectorId,
+ issue.findingId,
+ issue.ruleId,
+ null,
+ e.message,
+ Result.Failed.toString(),
+ issue.suggestedFixes.isNotEmpty()
+ )
+ }
+ }
+}
+
+fun getSeverityIcon(issue: CodeWhispererCodeScanIssue): Icon? = when (issue.severity) {
+ "Info" -> AwsIcons.Resources.CodeWhisperer.SEVERITY_INFO
+ "Low" -> AwsIcons.Resources.CodeWhisperer.SEVERITY_LOW
+ "Medium" -> AwsIcons.Resources.CodeWhisperer.SEVERITY_MEDIUM
+ "High" -> AwsIcons.Resources.CodeWhisperer.SEVERITY_HIGH
+ "Critical" -> AwsIcons.Resources.CodeWhisperer.SEVERITY_CRITICAL
+ else -> null
+}
+
+fun sendCodeFixGeneratedTelemetryToServiceAPI(
+ issue: CodeWhispererCodeScanIssue,
+ acceptFix: Boolean,
+) {
+ runIfIdcConnectionOrTelemetryEnabled(issue.project) {
+ pluginAwareExecuteOnPooledThread {
+ try {
+ val client = CodeWhispererClientAdaptor.getInstance(issue.project)
+ if (acceptFix) {
+ val acceptFixResponse = client.sendCodeFixAcceptanceTelemetry(
+ issue.file.programmingLanguage(),
+ issue.suggestedFixes.first().codeFixJobId,
+ issue.ruleId,
+ issue.detectorId,
+ issue.findingId,
+ issue.suggestedFixes.first().code.split("\n").size - 1,
+ issue.suggestedFixes.first().code.length
+ )
+ LOG.debug {
+ "Successfully sent code fix acceptance telemetry. RequestId: ${
+ acceptFixResponse.responseMetadata().requestId()
+ }"
+ }
+ } else {
+ val generateFixResponse = client.sendCodeFixGenerationTelemetry(
+ issue.file.programmingLanguage(),
+ issue.suggestedFixes.first().codeFixJobId,
+ issue.ruleId,
+ issue.detectorId,
+ issue.findingId,
+ issue.suggestedFixes.first().code.split("\n").size - 1,
+ issue.suggestedFixes.first().code.length
+ )
+ LOG.debug {
+ "Successfully sent code fix generated telemetry. RequestId: ${
+ generateFixResponse.responseMetadata().requestId()
+ }"
+ }
+ }
+ } catch (e: Exception) {
+ val requestId = if (e is CodeWhispererRuntimeException) e.requestId() else null
+ LOG.debug { "Failed to send code fix telemetry. RequestId: $requestId, ErrorMessage: ${e.message}" }
+ }
+ }
+ }
+}
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codetest/CodeTestException.kt b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codetest/CodeTestException.kt
new file mode 100644
index 0000000000..1c71796463
--- /dev/null
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codetest/CodeTestException.kt
@@ -0,0 +1,35 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.codewhisperer.codetest
+
+import software.aws.toolkits.resources.message
+
+open class CodeTestException(
+ override val message: String?,
+ val code: String? = "DefaultError",
+ val uiMessage: String? = message(
+ "testgen.error.generic_error_message"
+ ),
+) : RuntimeException()
+
+internal fun noFileOpenError(): Nothing =
+ throw CodeTestException(message("codewhisperer.codescan.no_file_open"), "ProjectZipError")
+
+internal fun fileTooLarge(): Nothing =
+ throw CodeTestException(message("codewhisperer.codescan.file_too_large_telemetry"), "ProjectZipError")
+
+internal fun cannotFindFile(errorMessage: String, filepath: String): Nothing =
+ error(message("codewhisperer.codescan.file_not_found", filepath, errorMessage))
+
+internal fun cannotFindValidFile(errorMessage: String): Nothing =
+ throw CodeTestException(errorMessage, "ProjectZipError")
+
+internal fun cannotFindBuildArtifacts(errorMessage: String): Nothing =
+ throw CodeTestException(errorMessage, "ProjectZipError")
+
+internal fun invalidSourceZipError(): Nothing =
+ throw CodeTestException(message("codewhisperer.codescan.invalid_source_zip_telemetry"), "InvalidSourceZipError")
+
+fun testGenStoppedError(): Nothing =
+ throw CodeTestException(message("testgen.message.cancelled"), "TestGenCancelled", message("testgen.message.cancelled"))
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codetest/sessionconfig/CodeTestSessionConfig.kt b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codetest/sessionconfig/CodeTestSessionConfig.kt
new file mode 100644
index 0000000000..e059bcdb0f
--- /dev/null
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/codetest/sessionconfig/CodeTestSessionConfig.kt
@@ -0,0 +1,248 @@
+// Copyright 2022 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.codewhisperer.codetest.sessionconfig
+
+import com.intellij.openapi.application.runReadAction
+import com.intellij.openapi.project.Project
+import com.intellij.openapi.project.guessModuleDir
+import com.intellij.openapi.project.guessProjectDir
+import com.intellij.openapi.project.modules
+import com.intellij.openapi.roots.ProjectRootManager
+import com.intellij.openapi.vcs.changes.ChangeListManager
+import com.intellij.openapi.vfs.LocalFileSystem
+import com.intellij.openapi.vfs.VirtualFile
+import com.intellij.openapi.vfs.isFile
+import kotlinx.coroutines.runBlocking
+import software.aws.toolkits.core.utils.createTemporaryZipFile
+import software.aws.toolkits.core.utils.debug
+import software.aws.toolkits.core.utils.getLogger
+import software.aws.toolkits.core.utils.putNextEntry
+import software.aws.toolkits.jetbrains.services.amazonq.FeatureDevSessionContext
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.sessionconfig.Payload
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.sessionconfig.PayloadContext
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.sessionconfig.PayloadMetadata
+import software.aws.toolkits.jetbrains.services.codewhisperer.codetest.cannotFindBuildArtifacts
+import software.aws.toolkits.jetbrains.services.codewhisperer.codetest.cannotFindFile
+import software.aws.toolkits.jetbrains.services.codewhisperer.codetest.cannotFindValidFile
+import software.aws.toolkits.jetbrains.services.codewhisperer.codetest.fileTooLarge
+import software.aws.toolkits.jetbrains.services.codewhisperer.codetest.noFileOpenError
+import software.aws.toolkits.jetbrains.services.codewhisperer.language.CodeWhispererProgrammingLanguage
+import software.aws.toolkits.jetbrains.services.codewhisperer.language.languages.CodeWhispererUnknownLanguage
+import software.aws.toolkits.jetbrains.services.codewhisperer.language.programmingLanguage
+import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererConstants.CODE_SCAN_CREATE_PAYLOAD_TIMEOUT_IN_SECONDS
+import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererConstants.DEFAULT_CODE_SCAN_TIMEOUT_IN_SECONDS
+import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererConstants.DEFAULT_PAYLOAD_LIMIT_IN_BYTES
+import software.aws.toolkits.resources.message
+import java.io.File
+import java.nio.file.Path
+import java.time.Instant
+import java.util.Stack
+import java.util.zip.ZipEntry
+import kotlin.io.path.name
+import kotlin.io.path.relativeTo
+
+// TODO: share huge duplicates with CodeScanSessionConfig need to abstract to a ZipSessionConfig
+class CodeTestSessionConfig(
+ private val selectedFile: VirtualFile?,
+ private val project: Project,
+ private val buildAndExecuteLogFile: VirtualFile? = null,
+) {
+ val projectRoot = project.basePath?.let { Path.of(it) }?.toFile()?.toVirtualFile() ?: run {
+ project.guessProjectDir() ?: error("Cannot guess base directory for project ${project.name}")
+ }
+
+ private val featureDevSessionContext = FeatureDevSessionContext(project)
+
+ val fileIndex = ProjectRootManager.getInstance(project).fileIndex
+
+ /**
+ * return default timeout
+ */
+ fun overallJobTimeoutInSeconds(): Long = DEFAULT_CODE_SCAN_TIMEOUT_IN_SECONDS
+
+ fun getPayloadLimitInBytes(): Long = DEFAULT_PAYLOAD_LIMIT_IN_BYTES
+
+ private fun willExceedPayloadLimit(currentTotalFileSize: Long, currentFileSize: Long): Boolean =
+ currentTotalFileSize.let { totalSize -> totalSize > (getPayloadLimitInBytes() - currentFileSize) }
+
+ private var programmingLanguage: CodeWhispererProgrammingLanguage = selectedFile?.programmingLanguage() ?: CodeWhispererUnknownLanguage.INSTANCE
+
+ fun getProgrammingLanguage(): CodeWhispererProgrammingLanguage = programmingLanguage
+
+ fun getSelectedFile(): VirtualFile? = selectedFile
+
+ fun createPayload(): Payload {
+ // Fail fast if the selected file is null for UTG
+ if (selectedFile == null) {
+ noFileOpenError()
+ }
+
+ // Fail fast if the selected file size is greater than the payload limit.
+ if (selectedFile.length > getPayloadLimitInBytes()) {
+ fileTooLarge()
+ }
+
+ val start = Instant.now().toEpochMilli()
+
+ LOG.debug { "Creating payload. File selected as root for the context truncation: ${projectRoot.path}" }
+
+ val payloadMetadata: PayloadMetadata = try {
+ getProjectPayloadMetadata()
+ } catch (e: Exception) {
+ val errorMessage = when {
+ e.message?.contains("Illegal repetition near index") == true -> "Illegal repetition near index"
+ else -> e.message
+ }
+ LOG.debug { "Error creating payload metadata: $errorMessage" }
+ cannotFindBuildArtifacts(errorMessage ?: message("testgen.message.failed"))
+ }
+
+ // Copy all the included source files to the source zip
+ val srcZip = zipFiles(payloadMetadata.sourceFiles.map { Path.of(it) })
+ val payloadContext = PayloadContext(
+ payloadMetadata.language,
+ payloadMetadata.linesScanned,
+ payloadMetadata.sourceFiles.size,
+ Instant.now().toEpochMilli() - start,
+ payloadMetadata.sourceFiles.mapNotNull { Path.of(it).toFile().toVirtualFile() },
+ payloadMetadata.payloadSize,
+ srcZip.length()
+ )
+
+ return Payload(payloadContext, srcZip)
+ }
+
+ /**
+ * Timeout for creating the payload [createPayload]
+ */
+ fun createPayloadTimeoutInSeconds(): Long = CODE_SCAN_CREATE_PAYLOAD_TIMEOUT_IN_SECONDS
+
+ private fun countLinesInVirtualFile(virtualFile: VirtualFile): Int {
+ try {
+ val bufferedReader = virtualFile.inputStream.bufferedReader()
+ return bufferedReader.useLines { lines -> lines.count() }
+ } catch (e: Exception) {
+ cannotFindFile("Line count error: ${e.message}", virtualFile.path)
+ }
+ }
+
+ private fun zipFiles(files: List): File = createTemporaryZipFile {
+ files.forEach { file ->
+ try {
+ val relativePath = file.relativeTo(projectRoot.toNioPath())
+ val projectBaseName = projectRoot.name
+ val zipEntryPath = "$projectBaseName/${relativePath.toString().replace("\\", "/")}"
+ LOG.debug { "Adding file to ZIP: $zipEntryPath" }
+ it.putNextEntry(zipEntryPath, file)
+ } catch (e: Exception) {
+ cannotFindFile("Zipping error: ${e.message}", file.toString())
+ }
+ }
+
+ // 2. Add the "utgRequiredArtifactsDir" directory
+ val utgDir = "utgRequiredArtifactsDir"
+ LOG.debug { "Adding directory to ZIP: $utgDir" }
+ val utgEntry = ZipEntry(utgDir)
+ it.putNextEntry(utgEntry)
+
+ // 3. Add the three empty subdirectories
+ val buildAndExecuteLogDir = "buildAndExecuteLogDir"
+ val subDirs = listOf(buildAndExecuteLogDir, "repoMapData", "testCoverageDir")
+ subDirs.forEach { subDir ->
+ val subDirPathString = Path.of(utgDir, subDir).name
+ LOG.debug { "Adding empty directory to ZIP: $subDirPathString" }
+ val zipEntry = ZipEntry(subDirPathString)
+ it.putNextEntry(zipEntry)
+ }
+ if (buildAndExecuteLogFile != null) {
+ it.putNextEntry(Path.of(utgDir, buildAndExecuteLogDir, "buildAndExecuteLog").name, buildAndExecuteLogFile.inputStream)
+ }
+ }.toFile()
+
+ fun getProjectPayloadMetadata(): PayloadMetadata {
+ val files = mutableSetOf()
+ val traversedDirectories = mutableSetOf()
+ val stack = Stack()
+ var currentTotalFileSize = 0L
+ var currentTotalLines = 0L
+ val languageCounts = mutableMapOf()
+
+ moduleLoop@ for (module in project.modules) {
+ val changeListManager = ChangeListManager.getInstance(module.project)
+ if (module.guessModuleDir() != null) {
+ stack.push(module.guessModuleDir())
+ while (stack.isNotEmpty()) {
+ val current = stack.pop()
+
+ if (!current.isDirectory) {
+ if (current.isFile && !changeListManager.isIgnoredFile(current) &&
+ runBlocking { !featureDevSessionContext.ignoreFile(current) } &&
+ runReadAction { !fileIndex.isInLibrarySource(current) }
+ ) {
+ if (willExceedPayloadLimit(currentTotalFileSize, current.length)) {
+ fileTooLarge()
+ } else {
+ try {
+ val language = current.programmingLanguage()
+ if (language !is CodeWhispererUnknownLanguage) {
+ languageCounts[language] = (languageCounts[language] ?: 0) + 1
+ }
+ files.add(current.path)
+ currentTotalFileSize += current.length
+ currentTotalLines += countLinesInVirtualFile(current)
+ } catch (e: Exception) {
+ LOG.debug { "Error parsing the file: ${current.path} with error: ${e.message}" }
+ continue
+ }
+ }
+ }
+ } else {
+ // Directory case: only traverse if not ignored
+ if (!changeListManager.isIgnoredFile(current) &&
+ runBlocking { !featureDevSessionContext.ignoreFile(current) } &&
+ !traversedDirectories.contains(current) && current.isValid &&
+ runReadAction { !fileIndex.isInLibrarySource(current) }
+ ) {
+ for (child in current.children) {
+ stack.push(child)
+ }
+ }
+ traversedDirectories.add(current)
+ }
+ }
+ }
+ }
+
+ val maxCount = languageCounts.maxByOrNull { it.value }?.value ?: 0
+ val maxCountLanguage = languageCounts.filter { it.value == maxCount }.keys.firstOrNull()
+
+ if (maxCountLanguage == null) {
+ programmingLanguage = CodeWhispererUnknownLanguage.INSTANCE
+ cannotFindValidFile("Amazon Q: doesn't contain valid files to generate tests")
+ }
+ programmingLanguage = maxCountLanguage
+ return PayloadMetadata(files, currentTotalFileSize, currentTotalLines, maxCountLanguage.toTelemetryType())
+ }
+
+ fun getPath(root: String, relativePath: String = ""): Path? = try {
+ Path.of(root, relativePath).normalize()
+ } catch (e: Exception) {
+ LOG.debug { "Cannot find file at path $relativePath relative to the root $root" }
+ null
+ }
+
+ fun getRelativePath(): Path? = try {
+ selectedFile?.path?.let { Path.of(projectRoot.path).relativize(Path.of(it)).normalize() }
+ } catch (e: Exception) {
+ LOG.debug { "Cannot calculate relative path of $selectedFile with respect to $projectRoot" }
+ null
+ }
+
+ fun File.toVirtualFile() = LocalFileSystem.getInstance().findFileByIoFile(this)
+
+ companion object {
+ private val LOG = getLogger()
+ fun create(file: VirtualFile?, project: Project): CodeTestSessionConfig = CodeTestSessionConfig(file, project, null)
+ }
+}
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/credentials/CodeWhispererClientAdaptor.kt b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/credentials/CodeWhispererClientAdaptor.kt
index 27feadf917..cedd56e35a 100644
--- a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/credentials/CodeWhispererClientAdaptor.kt
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/credentials/CodeWhispererClientAdaptor.kt
@@ -24,11 +24,19 @@ import software.amazon.awssdk.services.codewhispererruntime.model.CreateUploadUr
import software.amazon.awssdk.services.codewhispererruntime.model.Dimension
import software.amazon.awssdk.services.codewhispererruntime.model.GenerateCompletionsRequest
import software.amazon.awssdk.services.codewhispererruntime.model.GenerateCompletionsResponse
+import software.amazon.awssdk.services.codewhispererruntime.model.GetCodeFixJobRequest
+import software.amazon.awssdk.services.codewhispererruntime.model.GetCodeFixJobResponse
+import software.amazon.awssdk.services.codewhispererruntime.model.GetTestGenerationResponse
+import software.amazon.awssdk.services.codewhispererruntime.model.IdeCategory
import software.amazon.awssdk.services.codewhispererruntime.model.InlineChatUserDecision
import software.amazon.awssdk.services.codewhispererruntime.model.ListAvailableCustomizationsRequest
import software.amazon.awssdk.services.codewhispererruntime.model.ListFeatureEvaluationsResponse
import software.amazon.awssdk.services.codewhispererruntime.model.SendTelemetryEventResponse
+import software.amazon.awssdk.services.codewhispererruntime.model.StartCodeFixJobRequest
+import software.amazon.awssdk.services.codewhispererruntime.model.StartCodeFixJobResponse
+import software.amazon.awssdk.services.codewhispererruntime.model.StartTestGenerationResponse
import software.amazon.awssdk.services.codewhispererruntime.model.SuggestionState
+import software.amazon.awssdk.services.codewhispererruntime.model.TargetCode
import software.amazon.awssdk.services.codewhispererruntime.model.UserIntent
import software.aws.toolkits.core.utils.debug
import software.aws.toolkits.core.utils.getLogger
@@ -87,8 +95,16 @@ interface CodeWhispererClientAdaptor : Disposable {
isSigv4: Boolean = shouldUseSigv4Client(project),
): ListCodeScanFindingsResponse
+ fun startCodeFixJob(request: StartCodeFixJobRequest): StartCodeFixJobResponse
+
+ fun getCodeFixJob(request: GetCodeFixJobRequest): GetCodeFixJobResponse
+
fun listAvailableCustomizations(): List
+ fun startTestGeneration(uploadId: String, targetCode: List, userInput: String): StartTestGenerationResponse
+
+ fun getTestGeneration(jobId: String, jobGroupName: String): GetTestGenerationResponse
+
fun sendUserTriggerDecisionTelemetry(
requestContext: RequestContext,
responseContext: ResponseContext,
@@ -135,6 +151,39 @@ interface CodeWhispererClientAdaptor : Disposable {
scope: CodeWhispererConstants.CodeAnalysisScope,
): SendTelemetryEventResponse
+ fun sendCodeScanSucceededTelemetry(
+ language: CodeWhispererProgrammingLanguage,
+ codeScanJobId: String?,
+ scope: CodeWhispererConstants.CodeAnalysisScope,
+ findings: Int,
+ ): SendTelemetryEventResponse
+
+ fun sendCodeScanFailedTelemetry(
+ language: CodeWhispererProgrammingLanguage,
+ codeScanJobId: String?,
+ scope: CodeWhispererConstants.CodeAnalysisScope,
+ ): SendTelemetryEventResponse
+
+ fun sendCodeFixGenerationTelemetry(
+ language: CodeWhispererProgrammingLanguage,
+ codeFixJobId: String?,
+ ruleId: String?,
+ detectorId: String?,
+ findingId: String?,
+ linesOfCodeGenerated: Int?,
+ charsOfCodeGenerated: Int?,
+ ): SendTelemetryEventResponse
+
+ fun sendCodeFixAcceptanceTelemetry(
+ language: CodeWhispererProgrammingLanguage,
+ codeFixJobId: String?,
+ ruleId: String?,
+ detectorId: String?,
+ findingId: String?,
+ linesOfCodeGenerated: Int?,
+ charsOfCodeGenerated: Int?,
+ ): SendTelemetryEventResponse
+
fun sendCodeScanRemediationTelemetry(
language: CodeWhispererProgrammingLanguage?,
codeScanRemediationEventType: String?,
@@ -146,6 +195,20 @@ interface CodeWhispererClientAdaptor : Disposable {
result: String?,
includesFix: Boolean?,
): SendTelemetryEventResponse
+
+ fun sendTestGenerationEvent(
+ jobId: String,
+ groupName: String,
+ language: CodeWhispererProgrammingLanguage?,
+ ideCategory: IdeCategory?,
+ numberOfUnitTestCasesGenerated: Int?,
+ numberOfUnitTestCasesAccepted: Int?,
+ linesOfCodeGenerated: Int?,
+ linesOfCodeAccepted: Int?,
+ charsOfCodeGenerated: Int?,
+ charsOfCodeAccepted: Int?,
+ ): SendTelemetryEventResponse
+
fun listFeatureEvaluations(): ListFeatureEvaluationsResponse
fun sendMetricDataTelemetry(eventName: String, metadata: Map): SendTelemetryEventResponse
@@ -210,6 +273,7 @@ interface CodeWhispererClientAdaptor : Disposable {
CodeWhispererExplorerActionManager.getInstance().checkActiveCodeWhispererConnectionType(project) == CodeWhispererLoginType.Accountless
const val INVALID_CODESCANJOBID = "Invalid_CodeScanJobID"
+ const val INVALID_CODEFIXJOBID = "Invalid_CodeFixJobID"
}
}
@@ -281,6 +345,10 @@ open class CodeWhispererClientAdaptorImpl(override val project: Project) : CodeW
bearerClient().listCodeAnalysisFindings(request.transform()).transform()
}
+ override fun startCodeFixJob(request: StartCodeFixJobRequest): StartCodeFixJobResponse = bearerClient().startCodeFixJob(request)
+
+ override fun getCodeFixJob(request: GetCodeFixJobRequest): GetCodeFixJobResponse = bearerClient().getCodeFixJob(request)
+
// DO NOT directly use this method to fetch customizations, use wrapper [CodeWhispererModelConfigurator.listCustomization()] instead
override fun listAvailableCustomizations(): List =
bearerClient().listAvailableCustomizationsPaginator(ListAvailableCustomizationsRequest.builder().build())
@@ -301,6 +369,20 @@ open class CodeWhispererClientAdaptorImpl(override val project: Project) : CodeW
}
}
+ override fun startTestGeneration(uploadId: String, targetCode: List, userInput: String): StartTestGenerationResponse =
+ bearerClient().startTestGeneration { builder ->
+ builder.uploadId(uploadId)
+ builder.targetCodeList(targetCode)
+ builder.userInput(userInput)
+ // TODO: client token
+ }
+
+ override fun getTestGeneration(jobId: String, jobGroupName: String): GetTestGenerationResponse =
+ bearerClient().getTestGeneration { builder ->
+ builder.testGenerationJobId(jobId)
+ builder.testGenerationJobGroupName(jobGroupName)
+ }
+
override fun sendUserTriggerDecisionTelemetry(
requestContext: RequestContext,
responseContext: ResponseContext,
@@ -459,6 +541,100 @@ open class CodeWhispererClientAdaptorImpl(override val project: Project) : CodeW
requestBuilder.optOutPreference(getTelemetryOptOutPreference())
requestBuilder.userContext(codeWhispererUserContext())
}
+
+ override fun sendCodeScanSucceededTelemetry(
+ language: CodeWhispererProgrammingLanguage,
+ codeScanJobId: String?,
+ scope: CodeWhispererConstants.CodeAnalysisScope,
+ findings: Int,
+ ): SendTelemetryEventResponse = bearerClient().sendTelemetryEvent { requestBuilder ->
+ requestBuilder.telemetryEvent { telemetryEventBuilder ->
+ telemetryEventBuilder.codeScanSucceededEvent {
+ it.programmingLanguage { languageBuilder ->
+ languageBuilder.languageName(language.toCodeWhispererRuntimeLanguage().languageId)
+ }
+ it.codeScanJobId(if (codeScanJobId.isNullOrEmpty()) CodeWhispererClientAdaptor.INVALID_CODESCANJOBID else codeScanJobId)
+ it.timestamp(Instant.now())
+ it.codeAnalysisScope(scope.value)
+ it.numberOfFindings(findings)
+ it.timestamp(Instant.now())
+ }
+ }
+ requestBuilder.optOutPreference(getTelemetryOptOutPreference())
+ requestBuilder.userContext(codeWhispererUserContext())
+ }
+
+ override fun sendCodeScanFailedTelemetry(
+ language: CodeWhispererProgrammingLanguage,
+ codeScanJobId: String?,
+ scope: CodeWhispererConstants.CodeAnalysisScope,
+ ): SendTelemetryEventResponse = bearerClient().sendTelemetryEvent { requestBuilder ->
+ requestBuilder.telemetryEvent { telemetryEventBuilder ->
+ telemetryEventBuilder.codeScanFailedEvent {
+ it.programmingLanguage { languageBuilder ->
+ languageBuilder.languageName(language.toCodeWhispererRuntimeLanguage().languageId)
+ }
+ it.codeScanJobId(if (codeScanJobId.isNullOrEmpty()) CodeWhispererClientAdaptor.INVALID_CODESCANJOBID else codeScanJobId)
+ it.timestamp(Instant.now())
+ it.codeAnalysisScope(scope.value)
+ }
+ }
+ requestBuilder.optOutPreference(getTelemetryOptOutPreference())
+ requestBuilder.userContext(codeWhispererUserContext())
+ }
+
+ override fun sendCodeFixGenerationTelemetry(
+ language: CodeWhispererProgrammingLanguage,
+ codeFixJobId: String?,
+ ruleId: String?,
+ detectorId: String?,
+ findingId: String?,
+ linesOfCodeGenerated: Int?,
+ charsOfCodeGenerated: Int?,
+ ): SendTelemetryEventResponse = bearerClient().sendTelemetryEvent { requestBuilder ->
+ requestBuilder.telemetryEvent { telemetryEventBuilder ->
+ telemetryEventBuilder.codeFixGenerationEvent {
+ it.programmingLanguage { languageBuilder ->
+ languageBuilder.languageName(language.toCodeWhispererRuntimeLanguage().languageId)
+ }
+ it.jobId(if (codeFixJobId.isNullOrEmpty()) CodeWhispererClientAdaptor.INVALID_CODEFIXJOBID else codeFixJobId)
+ it.ruleId(ruleId)
+ it.detectorId(detectorId)
+ it.findingId(findingId)
+ it.linesOfCodeGenerated(linesOfCodeGenerated)
+ it.charsOfCodeGenerated(charsOfCodeGenerated)
+ }
+ }
+ requestBuilder.optOutPreference(getTelemetryOptOutPreference())
+ requestBuilder.userContext(codeWhispererUserContext())
+ }
+
+ override fun sendCodeFixAcceptanceTelemetry(
+ language: CodeWhispererProgrammingLanguage,
+ codeFixJobId: String?,
+ ruleId: String?,
+ detectorId: String?,
+ findingId: String?,
+ linesOfCodeGenerated: Int?,
+ charsOfCodeGenerated: Int?,
+ ): SendTelemetryEventResponse = bearerClient().sendTelemetryEvent { requestBuilder ->
+ requestBuilder.telemetryEvent { telemetryEventBuilder ->
+ telemetryEventBuilder.codeFixAcceptanceEvent {
+ it.programmingLanguage { languageBuilder ->
+ languageBuilder.languageName(language.toCodeWhispererRuntimeLanguage().languageId)
+ }
+ it.jobId(if (codeFixJobId.isNullOrEmpty()) CodeWhispererClientAdaptor.INVALID_CODEFIXJOBID else codeFixJobId)
+ it.ruleId(ruleId)
+ it.detectorId(detectorId)
+ it.findingId(findingId)
+ it.linesOfCodeAccepted(linesOfCodeGenerated)
+ it.charsOfCodeAccepted(charsOfCodeGenerated)
+ }
+ }
+ requestBuilder.optOutPreference(getTelemetryOptOutPreference())
+ requestBuilder.userContext(codeWhispererUserContext())
+ }
+
override fun sendCodeScanRemediationTelemetry(
language: CodeWhispererProgrammingLanguage?,
codeScanRemediationEventType: String?,
@@ -490,6 +666,39 @@ open class CodeWhispererClientAdaptorImpl(override val project: Project) : CodeW
requestBuilder.userContext(codeWhispererUserContext())
}
+ override fun sendTestGenerationEvent(
+ jobId: String,
+ groupName: String,
+ language: CodeWhispererProgrammingLanguage?,
+ ideCategory: IdeCategory?,
+ numberOfUnitTestCasesGenerated: Int?,
+ numberOfUnitTestCasesAccepted: Int?,
+ linesOfCodeGenerated: Int?,
+ linesOfCodeAccepted: Int?,
+ charsOfCodeGenerated: Int?,
+ charsOfCodeAccepted: Int?,
+ ): SendTelemetryEventResponse = bearerClient().sendTelemetryEvent { requestBuilder ->
+ requestBuilder.telemetryEvent { telemetryEventBuilder ->
+ telemetryEventBuilder.testGenerationEvent {
+ it.programmingLanguage { languageBuilder ->
+ languageBuilder.languageName(language?.toCodeWhispererRuntimeLanguage()?.languageId)
+ }
+ it.jobId(jobId)
+ it.groupName(groupName)
+ it.ideCategory(ideCategory)
+ it.numberOfUnitTestCasesGenerated(numberOfUnitTestCasesGenerated)
+ it.numberOfUnitTestCasesAccepted(numberOfUnitTestCasesAccepted)
+ it.linesOfCodeGenerated(linesOfCodeGenerated)
+ it.linesOfCodeAccepted(linesOfCodeAccepted)
+ it.charsOfCodeGenerated(charsOfCodeGenerated)
+ it.charsOfCodeAccepted(charsOfCodeAccepted)
+ it.timestamp(Instant.now())
+ }
+ }
+ requestBuilder.optOutPreference(getTelemetryOptOutPreference())
+ requestBuilder.userContext(codeWhispererUserContext())
+ }
+
override fun listFeatureEvaluations(): ListFeatureEvaluationsResponse = bearerClient().listFeatureEvaluations {
it.userContext(codeWhispererUserContext())
}
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/customization/CodeWhispererModelConfigurator.kt b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/customization/CodeWhispererModelConfigurator.kt
index 3cacc48325..f6535a4bba 100644
--- a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/customization/CodeWhispererModelConfigurator.kt
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/customization/CodeWhispererModelConfigurator.kt
@@ -157,17 +157,24 @@ class DefaultCodeWhispererModelConfigurator : CodeWhispererModelConfigurator, Pe
return@calculateIfIamIdentityCenterConnection customizationUiItems
}
+ /**
+ * Gets the active customization for a user. If a user has manually selected a customization,
+ * respect that choice. If a user has not selected a customization, check if they have a customization
+ * assigned to them via an AB feature. If so, use that customization.
+ */
override fun activeCustomization(project: Project): CodeWhispererCustomization? {
- val result = calculateIfIamIdentityCenterConnection(project) { connectionIdToActiveCustomizationArn[it.id] }
-
- // A/B case
- val customizationFeature = CodeWhispererFeatureConfigService.getInstance().getCustomizationFeature()
- if (customizationFeature == null || customizationFeature.value.stringValue().isEmpty()) return result
- return CodeWhispererCustomization(
- arn = customizationFeature.value.stringValue(),
- name = customizationFeature.variation,
- description = result?.description
- )
+ val selectedCustomization = calculateIfIamIdentityCenterConnection(project) { connectionIdToActiveCustomizationArn[it.id] }
+
+ if (selectedCustomization != null) {
+ return selectedCustomization
+ } else {
+ val customizationOverride = CodeWhispererFeatureConfigService.getInstance().getCustomizationFeature()
+ if (customizationOverride == null || customizationOverride.value.stringValue().isEmpty()) return null
+ return CodeWhispererCustomization(
+ arn = customizationOverride.value.stringValue(),
+ name = customizationOverride.variation,
+ )
+ }
}
override fun switchCustomization(project: Project, newCustomization: CodeWhispererCustomization?) {
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/explorer/QStatusBarLoggedInActionGroup.kt b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/explorer/QStatusBarLoggedInActionGroup.kt
index e03b749726..4f7833a1fc 100644
--- a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/explorer/QStatusBarLoggedInActionGroup.kt
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/explorer/QStatusBarLoggedInActionGroup.kt
@@ -16,6 +16,7 @@ import software.aws.toolkits.jetbrains.services.codewhisperer.actions.CodeWhispe
import software.aws.toolkits.jetbrains.services.codewhisperer.actions.CodeWhispererLearnMoreAction
import software.aws.toolkits.jetbrains.services.codewhisperer.actions.CodeWhispererProvideFeedbackAction
import software.aws.toolkits.jetbrains.services.codewhisperer.actions.CodeWhispererShowSettingsAction
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.actions.CodeWhispererCodeScanRunAction
import software.aws.toolkits.jetbrains.services.codewhisperer.explorer.actions.ActionProvider
import software.aws.toolkits.jetbrains.services.codewhisperer.explorer.actions.Customize
import software.aws.toolkits.jetbrains.services.codewhisperer.explorer.actions.Learn
@@ -38,8 +39,7 @@ class QStatusBarLoggedInActionGroup : DefaultActionGroup() {
override val customize = Customize()
override val learn = Learn()
override val openChatPanel = ActionManager.getInstance().getAction("q.openchat")
- override val runScan = ActionManager.getInstance().getAction("codewhisperer.toolbar.security.scan")
- override val stopScan = ActionManager.getInstance().getAction("codewhisperer.toolbar.security.stopscan")
+ override val runScan = CodeWhispererCodeScanRunAction()
override val pauseAutoScans = PauseCodeScans()
override val resumeAutoScans = ResumeCodeScans()
override val sendFeedback = CodeWhispererProvideFeedbackAction()
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/explorer/actions/ActionFactory.kt b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/explorer/actions/ActionFactory.kt
index 3b4979572c..cf342acf4c 100644
--- a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/explorer/actions/ActionFactory.kt
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/explorer/actions/ActionFactory.kt
@@ -7,7 +7,6 @@ import com.intellij.openapi.project.Project
import software.aws.toolkits.jetbrains.core.credentials.ToolkitConnectionManager
import software.aws.toolkits.jetbrains.core.credentials.pinning.QConnection
import software.aws.toolkits.jetbrains.core.credentials.sono.isSono
-import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.CodeWhispererCodeScanManager
import software.aws.toolkits.jetbrains.services.codewhisperer.customization.CodeWhispererModelConfigurator
import software.aws.toolkits.jetbrains.services.codewhisperer.explorer.CodeWhispererExplorerActionManager
import software.aws.toolkits.jetbrains.services.codewhisperer.explorer.isUserBuilderId
@@ -23,7 +22,6 @@ interface ActionProvider {
val pauseAutoScans: T
val resumeAutoScans: T
val runScan: T
- val stopScan: T
val sendFeedback: T
val connectOnGithub: T
val documentation: T
@@ -48,7 +46,6 @@ fun buildActionListForInlineSuggestions(project: Project, actionProvider: Ac
fun buildActionListForCodeScan(project: Project, actionProvider: ActionProvider): List =
buildList {
- val codeScanManager = CodeWhispererCodeScanManager.getInstance(project)
val manager = CodeWhispererExplorerActionManager.getInstance()
if (!isUserBuilderId(project)) {
if (manager.isAutoEnabledForCodeScan()) {
@@ -57,11 +54,7 @@ fun buildActionListForCodeScan(project: Project, actionProvider: ActionProvi
add(actionProvider.resumeAutoScans)
}
}
- if (codeScanManager.isProjectScanInProgress()) {
- add(actionProvider.stopScan)
- } else {
- add(actionProvider.runScan)
- }
+ add(actionProvider.runScan)
}
fun buildActionListForOtherFeatures(project: Project, actionProvider: ActionProvider): List =
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/language/CodeWhispererProgrammingLanguage.kt b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/language/CodeWhispererProgrammingLanguage.kt
index 58167cb1ef..98e5486cd2 100644
--- a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/language/CodeWhispererProgrammingLanguage.kt
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/language/CodeWhispererProgrammingLanguage.kt
@@ -34,6 +34,12 @@ abstract class CodeWhispererProgrammingLanguage {
open fun toCodeWhispererRuntimeLanguage(): CodeWhispererProgrammingLanguage = this
+ open fun lineCommentPrefix(): String? = "//"
+
+ open fun blockCommentPrefix(): String? = "/*"
+
+ open fun blockCommentSuffix(): String? = "*/"
+
final override fun equals(other: Any?): Boolean {
if (other !is CodeWhispererProgrammingLanguage) return false
return this.languageId == other.languageId
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/language/languages/CodeWhispererJson.kt b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/language/languages/CodeWhispererJson.kt
index 5332d99145..aba531eb64 100644
--- a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/language/languages/CodeWhispererJson.kt
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/language/languages/CodeWhispererJson.kt
@@ -15,6 +15,12 @@ class CodeWhispererJson private constructor() : CodeWhispererProgrammingLanguage
override fun isAutoFileScanSupported(): Boolean = true
+ override fun lineCommentPrefix() = null
+
+ override fun blockCommentPrefix() = null
+
+ override fun blockCommentSuffix() = null
+
companion object {
const val ID = "json"
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/language/languages/CodeWhispererPlainText.kt b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/language/languages/CodeWhispererPlainText.kt
index 71f3d7b639..5fb15d737c 100644
--- a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/language/languages/CodeWhispererPlainText.kt
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/language/languages/CodeWhispererPlainText.kt
@@ -13,6 +13,12 @@ class CodeWhispererPlainText private constructor() : CodeWhispererProgrammingLan
override fun isAutoFileScanSupported(): Boolean = false
+ override fun lineCommentPrefix() = null
+
+ override fun blockCommentPrefix() = null
+
+ override fun blockCommentSuffix() = null
+
companion object {
const val ID = "plaintext"
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/language/languages/CodeWhispererPython.kt b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/language/languages/CodeWhispererPython.kt
index 4a9964c1a0..c2b4ce25f4 100644
--- a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/language/languages/CodeWhispererPython.kt
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/language/languages/CodeWhispererPython.kt
@@ -24,6 +24,12 @@ class CodeWhispererPython private constructor() : CodeWhispererProgrammingLangua
override fun isSupplementalContextSupported() = true
+ override fun lineCommentPrefix() = "#"
+
+ override fun blockCommentPrefix() = "\"\"\""
+
+ override fun blockCommentSuffix() = "\"\"\""
+
companion object {
const val ID = "python"
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/language/languages/CodeWhispererRuby.kt b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/language/languages/CodeWhispererRuby.kt
index 15be78d83f..aef64cac8f 100644
--- a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/language/languages/CodeWhispererRuby.kt
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/language/languages/CodeWhispererRuby.kt
@@ -15,6 +15,12 @@ class CodeWhispererRuby private constructor() : CodeWhispererProgrammingLanguage
override fun isAutoFileScanSupported(): Boolean = true
+ override fun lineCommentPrefix(): String = "#"
+
+ override fun blockCommentPrefix(): String = "=begin"
+
+ override fun blockCommentSuffix(): String = "=end"
+
companion object {
const val ID = "ruby"
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/language/languages/CodeWhispererShell.kt b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/language/languages/CodeWhispererShell.kt
index bd8fa3450f..a096ff1bb1 100644
--- a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/language/languages/CodeWhispererShell.kt
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/language/languages/CodeWhispererShell.kt
@@ -13,6 +13,12 @@ class CodeWhispererShell private constructor() : CodeWhispererProgrammingLanguag
override fun isCodeCompletionSupported(): Boolean = true
+ override fun lineCommentPrefix(): String = "#"
+
+ override fun blockCommentPrefix(): String = ": '"
+
+ override fun blockCommentSuffix(): String = "'"
+
companion object {
const val ID = "shell"
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/language/languages/CodeWhispererTf.kt b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/language/languages/CodeWhispererTf.kt
index 828c45cc25..0e5123b3cc 100644
--- a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/language/languages/CodeWhispererTf.kt
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/language/languages/CodeWhispererTf.kt
@@ -15,6 +15,8 @@ class CodeWhispererTf private constructor() : CodeWhispererProgrammingLanguage()
override fun isAutoFileScanSupported(): Boolean = true
+ override fun lineCommentPrefix(): String = "#"
+
companion object {
const val ID = "tf"
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/language/languages/CodeWhispererUnknownLanguage.kt b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/language/languages/CodeWhispererUnknownLanguage.kt
index dc9a2779ff..2723698f9a 100644
--- a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/language/languages/CodeWhispererUnknownLanguage.kt
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/language/languages/CodeWhispererUnknownLanguage.kt
@@ -13,6 +13,12 @@ class CodeWhispererUnknownLanguage private constructor() : CodeWhispererProgramm
override fun toCodeWhispererRuntimeLanguage(): CodeWhispererProgrammingLanguage = CodeWhispererPlainText.INSTANCE
+ override fun lineCommentPrefix() = null
+
+ override fun blockCommentPrefix() = null
+
+ override fun blockCommentSuffix() = null
+
companion object {
const val ID = "unknown"
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/language/languages/CodeWhispererYaml.kt b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/language/languages/CodeWhispererYaml.kt
index 986add8b8c..c110d797cb 100644
--- a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/language/languages/CodeWhispererYaml.kt
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/language/languages/CodeWhispererYaml.kt
@@ -15,6 +15,12 @@ class CodeWhispererYaml private constructor() : CodeWhispererProgrammingLanguage
override fun isAutoFileScanSupported(): Boolean = true
+ override fun lineCommentPrefix(): String = "#"
+
+ override fun blockCommentPrefix(): String? = null
+
+ override fun blockCommentSuffix(): String? = null
+
companion object {
const val ID = "yaml"
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/model/CodeWhispererModel.kt b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/model/CodeWhispererModel.kt
index 91da779ab1..efa3f7a0f5 100644
--- a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/model/CodeWhispererModel.kt
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/model/CodeWhispererModel.kt
@@ -254,16 +254,17 @@ data class CodeScanTelemetryEvent(
val totalProjectSizeInBytes: Double?,
val connection: ToolkitConnection?,
val codeAnalysisScope: CodeWhispererConstants.CodeAnalysisScope,
+ val initiatedByChat: Boolean = false,
)
-data class CodeScanServiceInvocationContext(
- val artifactsUploadDuration: Long,
- val serviceInvocationDuration: Long,
+data class CreateUploadUrlServiceInvocationContext(
+ val artifactsUploadDuration: Long = 0,
+ val serviceInvocationDuration: Long = 0,
)
data class CodeScanResponseContext(
val payloadContext: PayloadContext,
- val serviceInvocationContext: CodeScanServiceInvocationContext,
+ val serviceInvocationContext: CreateUploadUrlServiceInvocationContext,
val codeScanJobId: String? = null,
val codeScanTotalIssues: Int = 0,
val codeScanIssuesWithFixes: Int = 0,
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/settings/CodeWhispererConfigurable.kt b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/settings/CodeWhispererConfigurable.kt
index 8951d04aed..1c5f90a352 100644
--- a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/settings/CodeWhispererConfigurable.kt
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/settings/CodeWhispererConfigurable.kt
@@ -11,10 +11,13 @@ import com.intellij.openapi.options.SearchableConfigurable
import com.intellij.openapi.options.ex.Settings
import com.intellij.openapi.project.Project
import com.intellij.ui.components.ActionLink
+import com.intellij.ui.components.fields.ExpandableTextField
import com.intellij.ui.dsl.builder.bindIntText
import com.intellij.ui.dsl.builder.bindSelected
+import com.intellij.ui.dsl.builder.bindText
import com.intellij.ui.dsl.builder.panel
import com.intellij.util.concurrency.EdtExecutorService
+import com.intellij.util.execution.ParametersListUtil
import software.aws.toolkits.jetbrains.core.credentials.ToolkitConnection
import software.aws.toolkits.jetbrains.core.credentials.ToolkitConnectionManagerListener
import software.aws.toolkits.jetbrains.services.amazonq.CodeWhispererFeatureConfigService
@@ -202,6 +205,17 @@ class CodeWhispererConfigurable(private val project: Project) :
}
}
+ group(message("aws.settings.codewhisperer.code_review")) {
+ row {
+ ExpandableTextField(ParametersListUtil.COLON_LINE_PARSER, ParametersListUtil.COLON_LINE_JOINER).also {
+ cell(it)
+ .label(message("aws.settings.codewhisperer.code_review.title"))
+ .comment(message("aws.settings.codewhisperer.code_review.description"))
+ .bindText(codeWhispererSettings::getIgnoredCodeReviewIssues, codeWhispererSettings::setIgnoredCodeReviewIssues)
+ }
+ }
+ }
+
group(message("aws.settings.codewhisperer.group.data_sharing")) {
row {
checkBox(message("aws.settings.codewhisperer.configurable.opt_out.title")).apply {
@@ -232,6 +246,6 @@ class CodeWhispererConfigurable(private val project: Project) :
}
companion object {
- private const val Q_INLINE_KEYBINDING_SEARCH_TEXT = "inline suggestion"
+ private const val Q_INLINE_KEYBINDING_SEARCH_TEXT = "Amazon Q inline suggestion"
}
}
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/startup/CodeWhispererProjectStartupSettingsListener.kt b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/startup/CodeWhispererProjectStartupSettingsListener.kt
index 24dbd5bd7f..78403cc952 100644
--- a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/startup/CodeWhispererProjectStartupSettingsListener.kt
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/startup/CodeWhispererProjectStartupSettingsListener.kt
@@ -33,7 +33,7 @@ class CodeWhispererProjectStartupSettingsListener(private val project: Project)
CodeWhispererCodeReferenceManager.getInstance(project).toolWindow?.isAvailable = value
if (value) {
CodeWhispererSettings.getInstance().toggleIncludeCodeWithReference(true)
- CodeWhispererCodeScanManager.getInstance(project).addCodeScanUI()
+ CodeWhispererCodeScanManager.getInstance(project).buildCodeScanUI()
} else {
CodeWhispererCodeScanManager.getInstance(project).removeCodeScanUI()
}
@@ -43,7 +43,8 @@ class CodeWhispererProjectStartupSettingsListener(private val project: Project)
super.toolWindowShown(toolWindow)
if (toolWindow.id != ProblemsView.ID) return
if (!isCodeWhispererEnabled(project)) return
- CodeWhispererCodeScanManager.getInstance(project).addCodeScanUI()
+ CodeWhispererCodeScanManager.getInstance(project).buildCodeScanUI()
+ CodeWhispererCodeScanManager.getInstance(project).showCodeScanUI()
}
override fun activeConnectionChanged(newConnection: ToolkitConnection?) {
@@ -53,7 +54,7 @@ class CodeWhispererProjectStartupSettingsListener(private val project: Project)
CodeWhispererCodeReferenceManager.getInstance(project).toolWindow?.isAvailable = newConnection != null
}
if (newConnection != null) {
- CodeWhispererCodeScanManager.getInstance(project).addCodeScanUI()
+ CodeWhispererCodeScanManager.getInstance(project).buildCodeScanUI()
} else {
CodeWhispererCodeScanManager.getInstance(project).removeCodeScanUI()
}
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/telemetry/CodeWhispererTelemetryService.kt b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/telemetry/CodeWhispererTelemetryService.kt
index 77f62b65b9..59d82c3b44 100644
--- a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/telemetry/CodeWhispererTelemetryService.kt
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/telemetry/CodeWhispererTelemetryService.kt
@@ -38,6 +38,7 @@ import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhisperer
import software.aws.toolkits.jetbrains.services.codewhisperer.util.runIfIdcConnectionOrTelemetryEnabled
import software.aws.toolkits.jetbrains.settings.AwsSettings
import software.aws.toolkits.jetbrains.settings.CodeWhispererSettings
+import software.aws.toolkits.telemetry.CodeFixAction
import software.aws.toolkits.telemetry.CodewhispererCodeScanScope
import software.aws.toolkits.telemetry.CodewhispererCompletionType
import software.aws.toolkits.telemetry.CodewhispererGettingStartedTask
@@ -47,10 +48,14 @@ import software.aws.toolkits.telemetry.CodewhispererSuggestionState
import software.aws.toolkits.telemetry.CodewhispererTelemetry
import software.aws.toolkits.telemetry.CodewhispererTriggerType
import software.aws.toolkits.telemetry.Component
+import software.aws.toolkits.telemetry.CredentialSourceId
+import software.aws.toolkits.telemetry.MetricResult
import software.aws.toolkits.telemetry.Result
+import java.nio.file.Path
import java.time.Duration
import java.time.Instant
import java.util.Queue
+import kotlin.io.path.pathString
@Service
class CodeWhispererTelemetryService {
@@ -280,6 +285,18 @@ class CodeWhispererTelemetryService {
)
}
+ private fun mapToTelemetryScope(codeAnalysisScope: CodeWhispererConstants.CodeAnalysisScope, initiatedByChat: Boolean): CodewhispererCodeScanScope =
+ when (codeAnalysisScope) {
+ CodeWhispererConstants.CodeAnalysisScope.FILE -> {
+ if (initiatedByChat) {
+ CodewhispererCodeScanScope.FILEONDEMAND
+ } else {
+ CodewhispererCodeScanScope.FILEAUTO
+ }
+ }
+ CodeWhispererConstants.CodeAnalysisScope.PROJECT -> CodewhispererCodeScanScope.PROJECT
+ }
+
fun sendSecurityScanEvent(codeScanEvent: CodeScanTelemetryEvent, project: Project? = null) {
val payloadContext = codeScanEvent.codeScanResponseContext.payloadContext
val serviceInvocationContext = codeScanEvent.codeScanResponseContext.serviceInvocationContext
@@ -288,8 +305,9 @@ class CodeWhispererTelemetryService {
val issuesWithFixes = codeScanEvent.codeScanResponseContext.codeScanIssuesWithFixes
val reason = codeScanEvent.codeScanResponseContext.reason
val startUrl = getConnectionStartUrl(codeScanEvent.connection)
- val codeAnalysisScope = codeScanEvent.codeAnalysisScope
- val passive = codeAnalysisScope == CodeWhispererConstants.CodeAnalysisScope.FILE
+ val codeAnalysisScope = mapToTelemetryScope(codeScanEvent.codeAnalysisScope, codeScanEvent.initiatedByChat)
+ val passive = codeAnalysisScope == CodewhispererCodeScanScope.FILEAUTO
+ val source = if (codeScanEvent.initiatedByChat) "chat" else "menu"
LOG.debug {
"Recording code security scan event. \n" +
@@ -306,8 +324,9 @@ class CodeWhispererTelemetryService {
"Service invocation duration in milliseconds: ${serviceInvocationContext.serviceInvocationDuration}, \n" +
"Total number of lines scanned: ${payloadContext.totalLines}, \n" +
"Reason: $reason \n" +
- "Scope: ${codeAnalysisScope.value} \n" +
- "Passive: $passive \n"
+ "Scope: $codeAnalysisScope \n" +
+ "Passive: $passive \n" +
+ "Source: $source \n"
}
CodewhispererTelemetry.securityScan(
project = project,
@@ -327,8 +346,9 @@ class CodeWhispererTelemetryService {
reason = reason,
result = codeScanEvent.result,
credentialStartUrl = startUrl,
- codewhispererCodeScanScope = CodewhispererCodeScanScope.from(codeAnalysisScope.value),
- passive = passive
+ codewhispererCodeScanScope = codeAnalysisScope,
+ passive = passive,
+ source = source
)
}
@@ -342,7 +362,7 @@ class CodeWhispererTelemetryService {
)
}
- fun sendCodeScanIssueApplyFixEvent(issue: CodeWhispererCodeScanIssue, result: Result, reason: String? = null) {
+ fun sendCodeScanIssueApplyFixEvent(issue: CodeWhispererCodeScanIssue, result: Result, reason: String? = null, codeFixAction: CodeFixAction?) {
CodewhispererTelemetry.codeScanIssueApplyFix(
findingId = issue.findingId,
detectorId = issue.detectorId,
@@ -350,7 +370,48 @@ class CodeWhispererTelemetryService {
component = Component.Hover,
result = result,
reason = reason,
- credentialStartUrl = getCodeWhispererStartUrl(issue.project)
+ credentialStartUrl = getCodeWhispererStartUrl(issue.project),
+ codeFixAction = codeFixAction
+ )
+ }
+
+ fun sendCodeScanNewTabEvent(credentialSourceId: CredentialSourceId?) {
+ CodewhispererTelemetry.codeScanChatNewTab(
+ credentialSourceId = credentialSourceId
+ )
+ }
+
+ fun sendCodeScanIssueIgnore(
+ component: Component,
+ issue: CodeWhispererCodeScanIssue,
+ isIgnoreAll: Boolean,
+ ) {
+ CodewhispererTelemetry.codeScanIssueIgnore(
+ component = component,
+ credentialStartUrl = getCodeWhispererStartUrl(issue.project),
+ findingId = issue.findingId,
+ detectorId = issue.detectorId,
+ ruleId = issue.ruleId,
+ variant = if (isIgnoreAll) "all" else null
+ )
+ }
+
+ fun sendCodeScanIssueGenerateFix(
+ component: Component,
+ issue: CodeWhispererCodeScanIssue,
+ isRefresh: Boolean,
+ result: MetricResult,
+ reason: String? = null,
+ ) {
+ CodewhispererTelemetry.codeScanIssueGenerateFix(
+ component = component,
+ credentialStartUrl = getCodeWhispererStartUrl(issue.project),
+ findingId = issue.findingId,
+ detectorId = issue.detectorId,
+ ruleId = issue.ruleId,
+ variant = if (isRefresh) "refresh" else null,
+ result = result,
+ reason = reason
)
}
@@ -550,6 +611,14 @@ class CodeWhispererTelemetryService {
assert(ApplicationManager.getApplication().isUnitTestMode)
return this.previousUserTriggerDecisions
}
+
+ fun sendInvalidZipEvent(filePath: Path, projectRoot: Path, relativePath: String) {
+ CodewhispererTelemetry.invalidZip(
+ filePath = filePath.pathString,
+ workspaceRoot = projectRoot.pathString,
+ relativePath = relativePath
+ )
+ }
}
fun isTelemetryEnabled(): Boolean = AwsSettings.getInstance().isTelemetryEnabled
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/util/CodeWhispererConstants.kt b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/util/CodeWhispererConstants.kt
index eba9ce1641..d17e82aa40 100644
--- a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/util/CodeWhispererConstants.kt
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/util/CodeWhispererConstants.kt
@@ -3,12 +3,14 @@
package software.aws.toolkits.jetbrains.services.codewhisperer.util
+import com.intellij.openapi.actionSystem.DataKey
import com.intellij.openapi.editor.markup.EffectType
import com.intellij.openapi.editor.markup.TextAttributes
import com.intellij.ui.JBColor
import software.amazon.awssdk.regions.Region
import software.amazon.awssdk.services.codewhispererruntime.model.AccessDeniedException
import software.amazon.awssdk.services.codewhispererruntime.model.CodeWhispererRuntimeException
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.CodeScanResponse
import software.aws.toolkits.jetbrains.services.codewhisperer.language.languages.CodeWhispererJava
import software.aws.toolkits.telemetry.CodewhispererGettingStartedTask
import java.awt.Font
@@ -53,6 +55,10 @@ object CodeWhispererConstants {
// avoid ThrottlingException as much as possible.
const val INVOCATION_INTERVAL: Long = 2050
+ val runScanKey = DataKey.create("amazonq.codescan.run")
+ val scanResultsKey = DataKey.create("amazonq.codescan.result")
+ val scanScopeKey = DataKey.create("amazonq.codescan.scope")
+
const val Q_CUSTOM_LEARN_MORE_URI = "https://docs.aws.amazon.com/amazonq/latest/qdeveloper-ug/customizations.html"
const val Q_SUPPORTED_LANG_URI = "https://docs.aws.amazon.com/amazonq/latest/qdeveloper-ug/q-language-ide-support.html"
const val CODEWHISPERER_CODE_SCAN_LEARN_MORE_URI = "https://docs.aws.amazon.com/codewhisperer/latest/userguide/security-scans.html"
@@ -70,9 +76,12 @@ object CodeWhispererConstants {
const val FILE_SCAN_INITIAL_POLLING_INTERVAL_IN_SECONDS: Long = 10
const val PROJECT_SCAN_INITIAL_POLLING_INTERVAL_IN_SECONDS: Long = 30
const val CODE_SCAN_CREATE_PAYLOAD_TIMEOUT_IN_SECONDS: Long = 10
- const val FILE_SCAN_TIMEOUT_IN_SECONDS: Long = 60 // 60 seconds
+ const val FILE_SCAN_TIMEOUT_IN_SECONDS: Long = 60 * 10 // 10 minutes
const val FILE_SCAN_PAYLOAD_SIZE_LIMIT_IN_BYTES: Long = 1024 * 200 // 200KB
const val AUTO_SCAN_DEBOUNCE_DELAY_IN_SECONDS: Long = 30
+ const val CODE_FIX_CREATE_PAYLOAD_TIMEOUT_IN_SECONDS: Long = 10
+ const val CODE_FIX_POLLING_INTERVAL_IN_SECONDS: Long = 1
+ const val CODE_FIX_TIMEOUT_IN_SECONDS: Long = 60 // 60 seconds
const val TOTAL_BYTES_IN_KB = 1024
const val TOTAL_BYTES_IN_MB = 1024 * 1024
const val TOTAL_MILLIS_IN_SECOND = 1000
@@ -93,6 +102,7 @@ object CodeWhispererConstants {
const val PROJECT_SCANS_LIMIT_REACHED = "You have reached the monthly quota of project scans."
const val FILE_SCANS_THROTTLING_MESSAGE = "Maximum auto-scans count reached for this month"
const val PROJECT_SCANS_THROTTLING_MESSAGE = "Maximum project scan count reached for this month"
+ const val amazonqIgnoreNextLine = "amazonq-ignore-next-line"
// Date when Accountless is not supported
val EXPIRE_DATE = SimpleDateFormat("yyyy-MM-dd").parse("2023-01-31")
@@ -117,6 +127,24 @@ object CodeWhispererConstants {
PROJECT("PROJECT"),
}
+ enum class FeatureName(val value: String) {
+ TEST_GENERATION("TEST_GENERATION"),
+ CODE_REVIEW("CODE_REVIEW"),
+ }
+
+ enum class UploadTaskType(val value: String) {
+ SCAN_FILE("SCAN_FILE"),
+ SCAN_PROJECT("SCAN_PROJECT"),
+ UTG("UTG"),
+ CODE_FIX("CODE_FIX"),
+ }
+
+ enum class FixGenerationState(val value: String) {
+ GENERATING("GENERATING"),
+ COMPLETED("COMPLETED"),
+ FAILED("FAILED"),
+ }
+
object Config {
const val CODEWHISPERER_ENDPOINT = "https://codewhisperer.us-east-1.amazonaws.com/" // PROD
const val CODEWHISPERER_IDPOOL_ID = "us-east-1:70717e99-906f-4add-908c-bd9074a2f5b9"
@@ -149,6 +177,7 @@ object CodeWhispererConstants {
const val CHUNK_SIZE = 60
const val NUMBER_OF_LINE_IN_CHUNK = 50
const val NUMBER_OF_CHUNK_TO_FETCH = 3
+ const val MAX_TOTAL_LENGTH = 20480
}
object Utg {
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/util/CodeWhispererFileContextProvider.kt b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/util/CodeWhispererFileContextProvider.kt
index 0e68cda02b..fe06ccd64d 100644
--- a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/util/CodeWhispererFileContextProvider.kt
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/util/CodeWhispererFileContextProvider.kt
@@ -25,7 +25,6 @@ import software.aws.toolkits.core.utils.debug
import software.aws.toolkits.core.utils.getLogger
import software.aws.toolkits.core.utils.info
import software.aws.toolkits.core.utils.warn
-import software.aws.toolkits.jetbrains.services.amazonq.CodeWhispererFeatureConfigService
import software.aws.toolkits.jetbrains.services.amazonq.project.ProjectContextController
import software.aws.toolkits.jetbrains.services.codewhisperer.editor.CodeWhispererEditorUtil
import software.aws.toolkits.jetbrains.services.codewhisperer.language.CodeWhispererProgrammingLanguage
@@ -44,6 +43,7 @@ import java.io.DataInput
import java.io.DataOutput
import java.util.Collections
import kotlin.coroutines.coroutineContext
+import kotlin.time.measureTimedValue
private val contentRootPathProvider = CopyContentRootPathProvider()
@@ -147,12 +147,20 @@ class DefaultCodeWhispererFileContextProvider(private val project: Project) : Fi
val latency = System.currentTimeMillis() - startFetchingTimestamp
if (it.contents.isNotEmpty()) {
val logStr = buildString {
- append("Successfully fetched supplemental context with strategy ${it.strategy} with $latency ms")
+ append(
+ """Q inline completion supplemental context:
+ | Strategy: ${it.strategy},
+ | Latency: $latency ms,
+ | Contents: ${it.contents.size} chunks,
+ | ContentLength: ${it.contentLength} chars,
+ | TargetFile: ${it.targetFileName},
+ """.trimMargin()
+ )
it.contents.forEachIndexed { index, chunk ->
append(
"""
|
- | Chunk ${index + 1}:
+ | Chunk $index:
| path = ${chunk.path},
| score = ${chunk.score},
| contentLength = ${chunk.content.length}
@@ -219,55 +227,113 @@ class DefaultCodeWhispererFileContextProvider(private val project: Project) : Fi
val query = generateQuery(targetContext)
val contexts = withContext(coroutineContext) {
- val projectContextDeferred1 = if (CodeWhispererFeatureConfigService.getInstance().getInlineCompletion()) {
- async {
- val t0 = System.currentTimeMillis()
- val r = fetchProjectContext(query, psiFile, targetContext)
- val t1 = System.currentTimeMillis()
- LOG.debug {
- buildString {
- append("time elapse for fetching project context=${t1 - t0}ms; ")
- append("numberOfChunks=${r.contents.size}; ")
- append("totalLength=${r.contentLength}")
- }
+ val projectContextDeferred1 = async {
+ val timedCodemapContext = measureTimedValue { fetchProjectContext(query, psiFile, targetContext) }
+ val codemapContext = timedCodemapContext.value
+ LOG.debug {
+ buildString {
+ append("time elapse for fetching project context=${timedCodemapContext.duration.inWholeMilliseconds}ms; ")
+ append("numberOfChunks=${codemapContext.contents.size}; ")
+ append("totalLength=${codemapContext.contentLength}")
}
-
- r
}
- } else {
- null
+
+ codemapContext
}
val openTabsContextDeferred1 = async {
- val t0 = System.currentTimeMillis()
- val r = fetchOpenTabsContext(query, psiFile, targetContext)
- val t1 = System.currentTimeMillis()
+ val timedOpentabContext = measureTimedValue { fetchOpenTabsContext(query, psiFile, targetContext) }
+ val opentabContext = timedOpentabContext.value
LOG.debug {
buildString {
- append("time elapse for open tabs context=${t1 - t0}ms; ")
- append("numberOfChunks=${r.contents.size}; ")
- append("totalLength=${r.contentLength}")
+ append("time elapse for open tabs context=${timedOpentabContext.duration.inWholeMilliseconds}ms; ")
+ append("numberOfChunks=${opentabContext.contents.size}; ")
+ append("totalLength=${opentabContext.contentLength}")
}
}
- r
+ opentabContext
}
- if (projectContextDeferred1 != null) {
- awaitAll(projectContextDeferred1, openTabsContextDeferred1)
- } else {
- awaitAll(openTabsContextDeferred1)
- }
+ awaitAll(projectContextDeferred1, openTabsContextDeferred1)
}
- val projectContext = contexts.find { it.strategy == CrossFileStrategy.ProjectContext }
+ val projectContext = contexts.find { it.strategy == CrossFileStrategy.Codemap }
val openTabsContext = contexts.find { it.strategy == CrossFileStrategy.OpenTabsBM25 }
- return if (projectContext != null && projectContext.contents.isNotEmpty()) {
- projectContext
- } else {
- openTabsContext ?: SupplementalContextInfo.emptyCrossFileContextInfo(targetContext.filename)
+ /**
+ * We're using both codemap and opentabs context
+ * 1. If both are present, codemap should live in the first of supplemental context list, i.e [codemap, opentabs_0, opentabs_1...] with strategy name codemap
+ * 2. If only one is present, return the one present with corresponding strategy name, either codemap or opentabs
+ * 3. If none is present, return empty list with strategy name empty
+ *
+ * Service will throw 400 error when context length is greater than 20480, drop the last chunk until the total length fits in the cap
+ */
+ val contextBeforeTruncation = when {
+ projectContext == null && openTabsContext == null -> SupplementalContextInfo.emptyCrossFileContextInfo(targetContext.filename)
+
+ projectContext != null && openTabsContext != null -> {
+ val context1 = projectContext.contents
+ val context2 = openTabsContext.contents
+ val mergedContext = (context1 + context2).filter { it.content.isNotEmpty() }
+
+ val strategy = if (projectContext.contentLength != 0 && openTabsContext.contentLength != 0) {
+ CrossFileStrategy.Codemap
+ } else if (projectContext.contentLength != 0) {
+ CrossFileStrategy.Codemap
+ } else if (openTabsContext.contentLength != 0) {
+ CrossFileStrategy.OpenTabsBM25
+ } else {
+ CrossFileStrategy.Empty
+ }
+
+ SupplementalContextInfo(
+ isUtg = false,
+ contents = mergedContext,
+ targetFileName = targetContext.filename,
+ strategy = strategy
+ )
+ }
+
+ projectContext != null -> {
+ return if (projectContext.contentLength == 0) {
+ SupplementalContextInfo.emptyCrossFileContextInfo(targetContext.filename)
+ } else {
+ SupplementalContextInfo(
+ isUtg = false,
+ contents = projectContext.contents,
+ targetFileName = targetContext.filename,
+ strategy = CrossFileStrategy.Codemap
+ )
+ }
+ }
+
+ openTabsContext != null -> {
+ return if (openTabsContext.contentLength == 0) {
+ SupplementalContextInfo.emptyCrossFileContextInfo(targetContext.filename)
+ } else {
+ SupplementalContextInfo(
+ isUtg = false,
+ contents = openTabsContext.contents,
+ targetFileName = targetContext.filename,
+ strategy = CrossFileStrategy.OpenTabsBM25
+ )
+ }
+ }
+
+ else -> SupplementalContextInfo.emptyCrossFileContextInfo(targetContext.filename)
+ }
+
+ return truncateContext(contextBeforeTruncation)
+ }
+
+ fun truncateContext(context: SupplementalContextInfo): SupplementalContextInfo {
+ var c = context.contents
+ while (c.sumOf { it.content.length } >= CodeWhispererConstants.CrossFile.MAX_TOTAL_LENGTH) {
+ c = c.dropLast(1)
}
+
+ return context.copy(contents = c)
}
@VisibleForTesting
@@ -285,7 +351,7 @@ class DefaultCodeWhispererFileContextProvider(private val project: Project) : Fi
)
},
targetFileName = targetContext.filename,
- strategy = CrossFileStrategy.ProjectContext
+ strategy = CrossFileStrategy.Codemap
)
}
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/util/CodeWhispererUtil.kt b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/util/CodeWhispererUtil.kt
index a9c2a212c6..d90dc02b03 100644
--- a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/util/CodeWhispererUtil.kt
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/util/CodeWhispererUtil.kt
@@ -34,6 +34,10 @@ import software.aws.toolkits.jetbrains.core.credentials.pinning.CodeWhispererCon
import software.aws.toolkits.jetbrains.core.credentials.reauthConnectionIfNeeded
import software.aws.toolkits.jetbrains.core.credentials.sono.isSono
import software.aws.toolkits.jetbrains.core.credentials.sso.bearer.BearerTokenProvider
+import software.aws.toolkits.jetbrains.core.gettingstarted.editor.ActiveConnection
+import software.aws.toolkits.jetbrains.core.gettingstarted.editor.ActiveConnectionType
+import software.aws.toolkits.jetbrains.core.gettingstarted.editor.BearerTokenFeatureSet
+import software.aws.toolkits.jetbrains.core.gettingstarted.editor.checkBearerConnectionValidity
import software.aws.toolkits.jetbrains.services.codewhisperer.explorer.CodeWhispererExplorerActionManager
import software.aws.toolkits.jetbrains.services.codewhisperer.learn.LearnCodeWhispererManager.Companion.taskTypeToFilename
import software.aws.toolkits.jetbrains.services.codewhisperer.model.Chunk
@@ -50,6 +54,7 @@ import software.aws.toolkits.jetbrains.utils.pluginAwareExecuteOnPooledThread
import software.aws.toolkits.resources.message
import software.aws.toolkits.telemetry.CodewhispererCompletionType
import software.aws.toolkits.telemetry.CodewhispererGettingStartedTask
+import software.aws.toolkits.telemetry.CredentialSourceId
// Controls the condition to send telemetry event to CodeWhisperer service, currently:
// 1. It will be sent for Builder ID users, only if they have optin telemetry sharing.
@@ -104,6 +109,17 @@ suspend fun String.toCodeChunk(path: String): List {
}
}
+fun getAuthType(project: Project): CredentialSourceId? {
+ val connection = checkBearerConnectionValidity(project, BearerTokenFeatureSet.Q)
+ var authType: CredentialSourceId? = null
+ if (connection.connectionType == ActiveConnectionType.IAM_IDC && connection is ActiveConnection.ValidBearer) {
+ authType = CredentialSourceId.IamIdentityCenter
+ } else if (connection.connectionType == ActiveConnectionType.BUILDER_ID && connection is ActiveConnection.ValidBearer) {
+ authType = CredentialSourceId.AwsId
+ }
+ return authType
+}
+
// we refer 10 lines of code as "Code Chunk"
// [[L1, L2, ...L10], [L11, L12, ...L20]...]
// use VirtualFile.toCodeChunk
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/util/CodeWhispererZipUploadManager.kt b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/util/CodeWhispererZipUploadManager.kt
new file mode 100644
index 0000000000..b9ad79b12c
--- /dev/null
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/util/CodeWhispererZipUploadManager.kt
@@ -0,0 +1,189 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.services.codewhisperer.util
+
+import com.intellij.openapi.components.Service
+import com.intellij.openapi.components.service
+import com.intellij.openapi.project.Project
+import com.intellij.util.io.HttpRequests
+import org.apache.commons.codec.digest.DigestUtils
+import software.amazon.awssdk.services.codewhispererruntime.model.CodeAnalysisUploadContext
+import software.amazon.awssdk.services.codewhispererruntime.model.CodeFixUploadContext
+import software.amazon.awssdk.services.codewhispererruntime.model.CreateUploadUrlRequest
+import software.amazon.awssdk.services.codewhispererruntime.model.CreateUploadUrlResponse
+import software.amazon.awssdk.services.codewhispererruntime.model.UploadContext
+import software.amazon.awssdk.services.codewhispererruntime.model.UploadIntent
+import software.amazon.awssdk.utils.IoUtils
+import software.aws.toolkits.core.utils.debug
+import software.aws.toolkits.core.utils.getLogger
+import software.aws.toolkits.jetbrains.core.AwsClientManager
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.CodeWhispererCodeScanSession.Companion.APPLICATION_ZIP
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.CodeWhispererCodeScanSession.Companion.AWS_KMS
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.CodeWhispererCodeScanSession.Companion.CONTENT_MD5
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.CodeWhispererCodeScanSession.Companion.CONTENT_TYPE
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.CodeWhispererCodeScanSession.Companion.SERVER_SIDE_ENCRYPTION
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.CodeWhispererCodeScanSession.Companion.SERVER_SIDE_ENCRYPTION_AWS_KMS_KEY_ID
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.CodeWhispererCodeScanSession.Companion.SERVER_SIDE_ENCRYPTION_CONTEXT
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.codeScanServerException
+import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.invalidSourceZipError
+import software.aws.toolkits.jetbrains.services.codewhisperer.codetest.CodeTestException
+import software.aws.toolkits.jetbrains.services.codewhisperer.credentials.CodeWhispererClientAdaptor
+import software.aws.toolkits.resources.message
+import java.io.File
+import java.io.FileInputStream
+import java.io.IOException
+import java.net.HttpURLConnection
+import java.util.Base64
+import software.aws.toolkits.jetbrains.services.codewhisperer.codetest.invalidSourceZipError as testGenerationInvalidSourceZipError
+
+@Service
+class CodeWhispererZipUploadManager(private val project: Project) {
+
+ fun createUploadUrlAndUpload(
+ zipFile: File,
+ artifactType: String,
+ taskType: CodeWhispererConstants.UploadTaskType,
+ taskName: String,
+ featureUseCase: CodeWhispererConstants.FeatureName,
+ ): CreateUploadUrlResponse {
+ // Throw error if zipFile is invalid.
+ if (!zipFile.exists()) {
+ when (featureUseCase) {
+ CodeWhispererConstants.FeatureName.CODE_REVIEW -> invalidSourceZipError()
+ CodeWhispererConstants.FeatureName.TEST_GENERATION -> testGenerationInvalidSourceZipError()
+ else -> throw IllegalArgumentException("Unsupported feature case: $featureUseCase") // Adding else for safety check
+ }
+ }
+ val fileMd5: String = Base64.getEncoder().encodeToString(DigestUtils.md5(FileInputStream(zipFile)))
+ val createUploadUrlResponse = createUploadUrl(fileMd5, artifactType, taskType, taskName, featureUseCase)
+ val url = createUploadUrlResponse.uploadUrl()
+ LOG.debug { "$featureUseCase: Uploading $artifactType using the presigned URL." }
+
+ uploadArtifactToS3(
+ url,
+ createUploadUrlResponse.uploadId(),
+ zipFile,
+ fileMd5,
+ createUploadUrlResponse.kmsKeyArn(),
+ createUploadUrlResponse.requestHeaders(),
+ featureUseCase
+ )
+ return createUploadUrlResponse
+ }
+
+ @Throws(IOException::class)
+ fun uploadArtifactToS3(
+ url: String,
+ uploadId: String,
+ fileToUpload: File,
+ md5: String,
+ kmsArn: String?,
+ requestHeaders: Map?,
+ featureUseCase: CodeWhispererConstants.FeatureName,
+ ) {
+ try {
+ val uploadIdJson = """{"uploadId":"$uploadId"}"""
+ HttpRequests.put(url, "application/zip").userAgent(AwsClientManager.getUserAgent()).tuner {
+ if (requestHeaders.isNullOrEmpty()) {
+ it.setRequestProperty(CONTENT_MD5, md5)
+ it.setRequestProperty(CONTENT_TYPE, APPLICATION_ZIP)
+ it.setRequestProperty(SERVER_SIDE_ENCRYPTION, AWS_KMS)
+ if (kmsArn?.isNotEmpty() == true) {
+ it.setRequestProperty(SERVER_SIDE_ENCRYPTION_AWS_KMS_KEY_ID, kmsArn)
+ }
+ it.setRequestProperty(SERVER_SIDE_ENCRYPTION_CONTEXT, Base64.getEncoder().encodeToString(uploadIdJson.toByteArray()))
+ } else {
+ requestHeaders.forEach { entry ->
+ it.setRequestProperty(entry.key, entry.value)
+ }
+ }
+ }.connect {
+ val connection = it.connection as HttpURLConnection
+ connection.setFixedLengthStreamingMode(fileToUpload.length())
+ IoUtils.copy(fileToUpload.inputStream(), connection.outputStream)
+ }
+ } catch (e: Exception) {
+ LOG.debug { "$featureUseCase: Artifact failed to upload in the S3 bucket: ${e.message}" }
+ val errorMessage = getTelemetryErrorMessage(e, featureUseCase)
+ when (featureUseCase) {
+ CodeWhispererConstants.FeatureName.CODE_REVIEW -> codeScanServerException("CreateUploadUrlException: $errorMessage")
+ CodeWhispererConstants.FeatureName.TEST_GENERATION -> throw CodeTestException(
+ "UploadTestArtifactToS3Error: $errorMessage",
+ "UploadTestArtifactToS3Error",
+ message("testgen.error.generic_technical_error_message")
+ )
+ else -> throw RuntimeException(errorMessage) // Adding else for safety check
+ }
+ }
+ }
+
+ fun createUploadUrl(
+ md5Content: String,
+ artifactType: String,
+ uploadTaskType: CodeWhispererConstants.UploadTaskType,
+ taskName: String,
+ featureUseCase: CodeWhispererConstants.FeatureName,
+ ): CreateUploadUrlResponse = try {
+ CodeWhispererClientAdaptor.getInstance(project).createUploadUrl(
+ CreateUploadUrlRequest.builder()
+ .contentMd5(md5Content)
+ .artifactType(artifactType)
+ .uploadIntent(getUploadIntent(uploadTaskType))
+ .uploadContext(
+ // For UTG we don't need uploadContext but sending else case as UploadContext
+ if (uploadTaskType == CodeWhispererConstants.UploadTaskType.CODE_FIX) {
+ UploadContext.fromCodeFixUploadContext(CodeFixUploadContext.builder().codeFixName(taskName).build())
+ } else {
+ UploadContext.fromCodeAnalysisUploadContext(CodeAnalysisUploadContext.builder().codeScanName(taskName).build())
+ }
+ )
+ .build()
+ )
+ } catch (e: Exception) {
+ LOG.debug { "$featureUseCase: Create Upload URL failed: ${e.message}" }
+ val errorMessage = getTelemetryErrorMessage(e, featureUseCase)
+ when (featureUseCase) {
+ CodeWhispererConstants.FeatureName.CODE_REVIEW -> codeScanServerException("CreateUploadUrlException: $errorMessage")
+ CodeWhispererConstants.FeatureName.TEST_GENERATION -> throw CodeTestException(
+ "CreateUploadUrlError: $errorMessage",
+ "CreateUploadUrlError",
+ message("testgen.error.generic_technical_error_message")
+ )
+ else -> throw RuntimeException(errorMessage) // Adding else for safety check
+ }
+ }
+
+ private fun getUploadIntent(uploadTaskType: CodeWhispererConstants.UploadTaskType): UploadIntent = when (uploadTaskType) {
+ CodeWhispererConstants.UploadTaskType.SCAN_FILE -> UploadIntent.AUTOMATIC_FILE_SECURITY_SCAN
+ CodeWhispererConstants.UploadTaskType.SCAN_PROJECT -> UploadIntent.FULL_PROJECT_SECURITY_SCAN
+ CodeWhispererConstants.UploadTaskType.UTG -> UploadIntent.UNIT_TESTS_GENERATION
+ CodeWhispererConstants.UploadTaskType.CODE_FIX -> UploadIntent.CODE_FIX_GENERATION
+ }
+
+ companion object {
+ fun getInstance(project: Project) = project.service()
+ private val LOG = getLogger()
+ }
+}
+
+fun getTelemetryErrorMessage(e: Exception, featureUseCase: CodeWhispererConstants.FeatureName): String = when {
+ e.message?.contains("Resource not found.") == true -> "Resource not found."
+ e.message?.contains("Maximum com.amazon.aws.codewhisperer.StartCodeAnalysis reached for this month.") == true -> message(
+ "testgen.error.maximum_generations_reach"
+ )
+ e.message?.contains("Maximum com.amazon.aws.codewhisperer.runtime.StartTestGeneration reached for this month.") == true
+ -> "Maximum com.amazon.aws.codewhisperer.runtime.StartTestGeneration reached for this month."
+ e.message?.contains("Service returned HTTP status code 407") == true -> "Service returned HTTP status code 407"
+ e.message?.contains("Improperly formed request") == true -> "Improperly formed request"
+ e.message?.contains("Service returned HTTP status code 403") == true -> "Service returned HTTP status code 403"
+ e.message?.contains("Service returned HTTP status code 503") == true -> "Service returned HTTP status code 503"
+ e.message?.contains("invalid_grant: Invalid token provided") == true -> "invalid_grant: Invalid token provided"
+ e.message?.contains("Connect timed out") == true -> "Unable to execute HTTP request: Connect timed out" // Error: Connect to host failed
+ e.message?.contains("Encountered an unexpected error when processing the request, please try again.") == true ->
+ "Encountered an unexpected error when processing the request, please try again."
+ else -> e.message ?: when (featureUseCase) {
+ CodeWhispererConstants.FeatureName.CODE_REVIEW -> message("codewhisperer.codescan.run_scan_error_telemetry")
+ else -> message("testgen.message.failed")
+ }
+}
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/util/SupplementalContextStrategy.kt b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/util/SupplementalContextStrategy.kt
index 6e139a896e..17ab38d428 100644
--- a/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/util/SupplementalContextStrategy.kt
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/src/software/aws/toolkits/jetbrains/services/codewhisperer/util/SupplementalContextStrategy.kt
@@ -12,9 +12,9 @@ enum class UtgStrategy : SupplementalContextStrategy {
;
override fun toString() = when (this) {
- ByName -> "ByName"
- ByContent -> "ByContent"
- Empty -> "Empty"
+ ByName -> "byName"
+ ByContent -> "byContent"
+ Empty -> "empty"
}
}
@@ -22,11 +22,13 @@ enum class CrossFileStrategy : SupplementalContextStrategy {
OpenTabsBM25,
Empty,
ProjectContext,
+ Codemap,
;
override fun toString() = when (this) {
- OpenTabsBM25 -> "OpenTabs_BM25"
- Empty -> "Empty"
- ProjectContext -> "ProjectContext"
+ OpenTabsBM25 -> "opentabs"
+ Empty -> "empty"
+ ProjectContext -> "projectContext"
+ Codemap -> "codemap"
}
}
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codewhisperer/CodeWhispererConfigurableTest.kt b/plugins/amazonq/codewhisperer/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codewhisperer/CodeWhispererConfigurableTest.kt
index 25a5cbb18f..5224284b42 100644
--- a/plugins/amazonq/codewhisperer/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codewhisperer/CodeWhispererConfigurableTest.kt
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codewhisperer/CodeWhispererConfigurableTest.kt
@@ -23,7 +23,8 @@ class CodeWhispererConfigurableTest : CodeWhispererTestBase() {
@Test
fun `test CodeWhisperer configurable`() {
val codeScanManagerSpy = Mockito.spy(CodeWhispererCodeScanManager.getInstance(projectRule.project))
- doNothing().`when`(codeScanManagerSpy).addCodeScanUI()
+ doNothing().`when`(codeScanManagerSpy).buildCodeScanUI()
+ doNothing().`when`(codeScanManagerSpy).showCodeScanUI()
doNothing().`when`(codeScanManagerSpy).removeCodeScanUI()
projectRule.project.replaceService(CodeWhispererCodeScanManager::class.java, codeScanManagerSpy, disposableRule.disposable)
val configurable = CodeWhispererConfigurable(projectRule.project)
@@ -48,7 +49,7 @@ class CodeWhispererConfigurableTest : CodeWhispererTestBase() {
)
val comments = panel.components.filterIsInstance()
- assertThat(comments.size).isEqualTo(7)
+ assertThat(comments.size).isEqualTo(8)
mockCodeWhispererEnabledStatus(false)
ApplicationManager.getApplication().messageBus.syncPublisher(ToolkitConnectionManagerListener.TOPIC)
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codewhisperer/CodeWhispererFeatureConfigServiceTest.kt b/plugins/amazonq/codewhisperer/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codewhisperer/CodeWhispererFeatureConfigServiceTest.kt
index c79905fd97..56f6f66b51 100644
--- a/plugins/amazonq/codewhisperer/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codewhisperer/CodeWhispererFeatureConfigServiceTest.kt
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codewhisperer/CodeWhispererFeatureConfigServiceTest.kt
@@ -26,6 +26,8 @@ import software.amazon.awssdk.services.codewhispererruntime.model.ListAvailableC
import software.amazon.awssdk.services.codewhispererruntime.model.ListFeatureEvaluationsRequest
import software.amazon.awssdk.services.codewhispererruntime.model.ListFeatureEvaluationsResponse
import software.amazon.awssdk.services.codewhispererruntime.paginators.ListAvailableCustomizationsIterable
+import software.aws.toolkits.core.TokenConnectionSettings
+import software.aws.toolkits.core.region.AwsRegion
import software.aws.toolkits.jetbrains.core.MockClientManagerRule
import software.aws.toolkits.jetbrains.core.credentials.LegacyManagedBearerSsoConnection
import software.aws.toolkits.jetbrains.core.credentials.ToolkitConnectionManager
@@ -58,6 +60,44 @@ class CodeWhispererFeatureConfigServiceTest {
assertThat(CodeWhispererFeatureConfigService.FEATURE_DEFINITIONS).containsKeys("testFeature")
}
+ @Test
+ fun `test highlightCommand returns non-empty`() {
+ mockClientManagerRule.create().stub {
+ on { listFeatureEvaluations(any()) } doReturn ListFeatureEvaluationsResponse.builder().featureEvaluations(
+ listOf(
+ FeatureEvaluation.builder()
+ .feature("highlightCommand")
+ .variation("a new command")
+ .value(FeatureValue.fromStringValue("@highlight"))
+ .build()
+ )
+ ).build()
+ }
+
+ val mockTokenSettings = mock {
+ on { providerId } doReturn "mock"
+ on { region } doReturn AwsRegion.GLOBAL
+ }
+
+ val mockSsoConnection = mock {
+ on { startUrl } doReturn "fake sso url"
+ on { getConnectionSettings() } doReturn mockTokenSettings
+ }
+
+ projectRule.project.replaceService(
+ ToolkitConnectionManager::class.java,
+ mock { on { activeConnectionForFeature(eq(QConnection.getInstance())) } doReturn mockSsoConnection },
+ disposableRule.disposable
+ )
+
+ runBlocking {
+ CodeWhispererFeatureConfigService.getInstance().fetchFeatureConfigs(projectRule.project)
+ }
+
+ assertThat(CodeWhispererFeatureConfigService.getInstance().getHighlightCommandFeature()?.value?.stringValue()).isEqualTo("@highlight")
+ assertThat(CodeWhispererFeatureConfigService.getInstance().getHighlightCommandFeature()?.variation).isEqualTo("a new command")
+ }
+
@Test
fun `test customizationArnOverride returns empty for BID users`() {
testCustomizationArnOverrideABHelper(isIdc = false, isInListAvailableCustomizations = false)
@@ -80,7 +120,7 @@ class CodeWhispererFeatureConfigServiceTest {
on { listFeatureEvaluations(any()) } doReturn ListFeatureEvaluationsResponse.builder().featureEvaluations(
listOf(
FeatureEvaluation.builder()
- .feature(CodeWhispererFeatureConfigService.CUSTOMIZATION_ARN_OVERRIDE_NAME)
+ .feature("customizationArnOverride")
.variation("customization-name")
.value(FeatureValue.fromStringValue("test arn"))
.build()
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codewhisperer/CodeWhispererFileContextProviderTest.kt b/plugins/amazonq/codewhisperer/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codewhisperer/CodeWhispererFileContextProviderTest.kt
index 81d52f35bb..8507196582 100644
--- a/plugins/amazonq/codewhisperer/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codewhisperer/CodeWhispererFileContextProviderTest.kt
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codewhisperer/CodeWhispererFileContextProviderTest.kt
@@ -49,7 +49,9 @@ import software.aws.toolkits.jetbrains.services.codewhisperer.language.languages
import software.aws.toolkits.jetbrains.services.codewhisperer.language.languages.CodeWhispererTsx
import software.aws.toolkits.jetbrains.services.codewhisperer.language.languages.CodeWhispererTypeScript
import software.aws.toolkits.jetbrains.services.codewhisperer.model.CaretContext
+import software.aws.toolkits.jetbrains.services.codewhisperer.model.Chunk
import software.aws.toolkits.jetbrains.services.codewhisperer.model.FileContextInfo
+import software.aws.toolkits.jetbrains.services.codewhisperer.model.SupplementalContextInfo
import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererConstants
import software.aws.toolkits.jetbrains.services.codewhisperer.util.CrossFileStrategy
import software.aws.toolkits.jetbrains.services.codewhisperer.util.DefaultCodeWhispererFileContextProvider
@@ -137,7 +139,7 @@ class CodeWhispererFileContextProviderTest {
@Test
fun `extractSupplementalFileContext should timeout 50ms`() = runTest {
- featureConfigService.stub { on { getInlineCompletion() } doReturn false }
+ mockProjectContext.stub { onBlocking { queryInline(any(), any()) }.doReturn(emptyList()) }
sut = spy(sut)
val files = NaiveSampleCase.setupFixture(fixture)
@@ -159,7 +161,26 @@ class CodeWhispererFileContextProviderTest {
}
@Test
- fun `should only call and use openTabsContext if projectContext is disabled`() = runTest {
+ fun `should return empty if both project context and opentabs context return empty`() = runTest {
+ sut = spy(sut)
+
+ mockProjectContext.stub { onBlocking { queryInline(any(), any()) }.doReturn(emptyList()) }
+ val queryPsi = projectRule.fixture.addFileToProject("Foo.java", "public Foo {}")
+ val mockFileContext = aFileContextInfo(CodeWhispererJava.INSTANCE)
+
+ val result = sut.extractSupplementalFileContextForSrc(queryPsi, mockFileContext)
+
+ verify(sut, times(1)).fetchProjectContext(any(), any(), any())
+ verify(sut, times(1)).fetchOpenTabsContext(any(), any(), any())
+
+ assertThat(result.isUtg).isFalse
+ assertThat(result.strategy).isEqualTo(CrossFileStrategy.Empty)
+ assertThat(result.contents).isEmpty()
+ }
+
+ @Test
+ fun `should only use openTabsContext if projectContext is empty`() = runTest {
+ mockProjectContext.stub { onBlocking { queryInline(any(), any()) }.doReturn(emptyList()) }
featureConfigService.stub { on { getInlineCompletion() } doReturn false }
sut = spy(sut)
@@ -169,7 +190,7 @@ class CodeWhispererFileContextProviderTest {
val result = sut.extractSupplementalFileContextForSrc(queryPsi, mockFileContext)
- verify(sut, times(0)).fetchProjectContext(any(), any(), any())
+ verify(sut, times(1)).fetchProjectContext(any(), any(), any())
verify(sut, times(1)).fetchOpenTabsContext(any(), any(), any())
assertThat(result.isUtg).isFalse
@@ -208,7 +229,7 @@ class CodeWhispererFileContextProviderTest {
assertThat(providerContext.constructed()).hasSize(1)
assertThat(serverContext.constructed()).hasSize(1)
- whenever(providerContext.constructed()[0].queryInline(any(), any())).thenThrow(RuntimeException("mock exception"))
+ whenever(providerContext.constructed()[0].queryInline(any(), any(), any())).thenThrow(RuntimeException("mock exception"))
val result = controller.queryInline("query", "filePath")
assertThat(result).isEmpty()
@@ -217,19 +238,16 @@ class CodeWhispererFileContextProviderTest {
}
@Test
- fun `should use project context if it is present`() = runTest {
+ fun `should use both project context and open tabs if both are present`() = runTest {
mockProjectContext.stub {
runBlocking {
doReturn(
listOf(
InlineBm25Chunk("project_context1", "path1", 0.0),
- InlineBm25Chunk("project_context2", "path2", 0.0),
- InlineBm25Chunk("project_context3", "path3", 0.0),
)
).whenever(it).queryInline(any(), any())
}
}
- featureConfigService.stub { on { getInlineCompletion() } doReturn true }
sut = spy(sut)
val files = NaiveSampleCase.setupFixture(fixture)
val queryPsi = files[0]
@@ -238,8 +256,8 @@ class CodeWhispererFileContextProviderTest {
val result = sut.extractSupplementalFileContextForSrc(queryPsi, mockFileContext)
assertThat(result.isUtg).isFalse
- assertThat(result.strategy).isEqualTo(CrossFileStrategy.ProjectContext)
- assertThat(result.contents).hasSize(3)
+ assertThat(result.strategy).isEqualTo(CrossFileStrategy.Codemap)
+ assertThat(result.contents).hasSize(4)
}
@Test
@@ -419,7 +437,8 @@ class CodeWhispererFileContextProviderTest {
}
@Test
- fun `extractSupplementalFileContext from src file should extract src`() = runTest {
+ fun `extractSupplementalFileContext should return opentabs context if project context is empty`() = runTest {
+ mockProjectContext.stub { onBlocking { queryInline(any(), any()) }.doReturn(emptyList()) }
val files = NaiveSampleCase.setupFixture(fixture)
val queryPsi = files[0]
@@ -436,6 +455,10 @@ class CodeWhispererFileContextProviderTest {
assertThat(supplementalContext?.contents)
.isNotNull
.isNotEmpty
+
+ assertThat(supplementalContext?.strategy)
+ .isNotNull
+ .isEqualTo(CrossFileStrategy.OpenTabsBM25)
verify(sut).extractSupplementalFileContextForSrc(any(), any())
verify(sut, times(0)).extractSupplementalFileContextForTst(any(), any())
}
@@ -484,6 +507,28 @@ class CodeWhispererFileContextProviderTest {
verify(sut).extractSupplementalFileContextForTst(any(), any())
}
+ @Test
+ fun `truncate context should make context length fit in 20480 cap`() {
+ val supplementalContext = SupplementalContextInfo(
+ isUtg = false,
+ contents = listOf(
+ Chunk(content = "a".repeat(10000), path = "a.java"),
+ Chunk(content = "b".repeat(10000), path = "b.java"),
+ Chunk(content = "c".repeat(10000), path = "c.java"),
+ Chunk(content = "d".repeat(10000), path = "d.java"),
+ Chunk(content = "e".repeat(10000), path = "e.java"),
+ ),
+ targetFileName = "foo",
+ strategy = CrossFileStrategy.Codemap
+ )
+
+ val r = sut.truncateContext(supplementalContext)
+ assertThat(r.contents).hasSize(2)
+ assertThat(r.contentLength).isEqualTo(20000)
+ assertThat(r.strategy).isEqualTo(CrossFileStrategy.Codemap)
+ assertThat(r.targetFileName).isEqualTo("foo")
+ }
+
private fun setupFixture(fixture: JavaCodeInsightTestFixture): List {
val psiFile1 = fixture.addFileToProject("Main.java", JAVA_MAIN)
val psiFile2 = fixture.addFileToProject("UtilClass.java", JAVA_UTILCLASS)
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codewhisperer/CodeWhispererModelConfiguratorTest.kt b/plugins/amazonq/codewhisperer/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codewhisperer/CodeWhispererModelConfiguratorTest.kt
index 169473a97e..d960870240 100644
--- a/plugins/amazonq/codewhisperer/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codewhisperer/CodeWhispererModelConfiguratorTest.kt
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codewhisperer/CodeWhispererModelConfiguratorTest.kt
@@ -113,17 +113,37 @@ class CodeWhispererModelConfiguratorTest {
}
@Test
- fun `should override customization arn if there is one under AB test`() {
+ fun `should not override customization arn if there is one under AB test and manual selection has been made`() {
val ssoConn = spy(LegacyManagedBearerSsoConnection(region = "us-east-1", startUrl = "url 1", scopes = Q_SCOPES))
ToolkitConnectionManager.getInstance(projectRule.project).switchConnection(ssoConn)
- sut.switchCustomization(projectRule.project, CodeWhispererCustomization("foo", "customization_1", "description_1"))
- assertThat(sut.activeCustomization(projectRule.project)).isEqualTo(CodeWhispererCustomization("foo", "customization_1", "description_1"))
+ sut.switchCustomization(projectRule.project, CodeWhispererCustomization("selectedCustomizationArn", "customization_1", "description_1"))
+ assertThat(sut.activeCustomization(projectRule.project))
+ .isEqualTo(CodeWhispererCustomization("selectedCustomizationArn", "customization_1", "description_1"))
abManager.stub {
on { getCustomizationFeature() }.thenReturn(FeatureContext("customizationArnOverride", "foo", FeatureValue.builder().stringValue("bar").build()))
}
- assertThat(sut.activeCustomization(projectRule.project)).isEqualTo(CodeWhispererCustomization("bar", "foo", "description_1"))
+ assertThat(sut.activeCustomization(projectRule.project))
+ .isEqualTo(CodeWhispererCustomization("selectedCustomizationArn", "customization_1", "description_1"))
+ }
+
+ @Test
+ fun `should override customization arn if there is one under AB test and manual selection has not been made`() {
+ val ssoConn = spy(LegacyManagedBearerSsoConnection(region = "us-east-1", startUrl = "url 1", scopes = Q_SCOPES))
+ ToolkitConnectionManager.getInstance(projectRule.project).switchConnection(ssoConn)
+
+ sut.switchCustomization(projectRule.project, CodeWhispererCustomization("selectedCustomizationArn", "customization_1", "description_1"))
+ assertThat(sut.activeCustomization(projectRule.project))
+ .isEqualTo(CodeWhispererCustomization("selectedCustomizationArn", "customization_1", "description_1"))
+ sut.invalidateCustomization("selectedCustomizationArn")
+
+ abManager.stub {
+ on { getCustomizationFeature() }.thenReturn(
+ FeatureContext("customizationArnOverride", "foo", FeatureValue.builder().stringValue("overrideArn").build())
+ )
+ }
+ assertThat(sut.activeCustomization(projectRule.project)).isEqualTo(CodeWhispererCustomization("overrideArn", "foo", null))
}
@Test
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/CodeWhispererCodeFileScanTest.kt b/plugins/amazonq/codewhisperer/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/CodeWhispererCodeFileScanTest.kt
index aec774efbc..a39164933e 100644
--- a/plugins/amazonq/codewhisperer/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/CodeWhispererCodeFileScanTest.kt
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/CodeWhispererCodeFileScanTest.kt
@@ -16,10 +16,8 @@ import org.mockito.ArgumentMatchers.anyString
import org.mockito.Mockito.mock
import org.mockito.kotlin.any
import org.mockito.kotlin.argumentCaptor
-import org.mockito.kotlin.doNothing
import org.mockito.kotlin.eq
import org.mockito.kotlin.inOrder
-import org.mockito.kotlin.isNull
import org.mockito.kotlin.spy
import org.mockito.kotlin.stub
import org.mockito.kotlin.times
@@ -103,7 +101,8 @@ class CodeWhispererCodeFileScanTest : CodeWhispererCodeScanTestBase(PythonCodeIn
CodeScanSessionConfig.create(
pyPsiFile.virtualFile,
project,
- CodeWhispererConstants.CodeAnalysisScope.FILE
+ CodeWhispererConstants.CodeAnalysisScope.FILE,
+ false
)
)
setupResponse(pyPsiFile.virtualFile.toNioPath().relativeTo(pySession.projectRoot.toNioPath()))
@@ -115,7 +114,6 @@ class CodeWhispererCodeFileScanTest : CodeWhispererCodeScanTestBase(PythonCodeIn
// Mock CodeWhispererClient needs to be setup before initializing CodeWhispererCodeScanSession
val pySessionContext = CodeScanSessionContext(project, pySession, CodeWhispererConstants.CodeAnalysisScope.FILE)
codeScanSessionSpy = spy(CodeWhispererCodeScanSession(pySessionContext))
- doNothing().whenever(codeScanSessionSpy).uploadArtifactToS3(any(), any(), any(), any(), isNull(), any())
mockClient.stub {
// setupResponse dynamically modifies these fake responses so this is very hard to follow and makes me question if we even need this
@@ -136,13 +134,13 @@ class CodeWhispererCodeFileScanTest : CodeWhispererCodeScanTestBase(PythonCodeIn
CodeScanSessionConfig.create(
psiFile.virtualFile,
project,
- CodeWhispererConstants.CodeAnalysisScope.FILE
+ CodeWhispererConstants.CodeAnalysisScope.FILE,
+ false
)
)
setupResponse(psiFile.virtualFile.toNioPath().relativeTo(sessionConfig.projectRoot.toNioPath()))
val sessionContext = CodeScanSessionContext(project, sessionConfig, CodeWhispererConstants.CodeAnalysisScope.FILE)
val session = spy(CodeWhispererCodeScanSession(sessionContext))
- doNothing().whenever(session).uploadArtifactToS3(any(), any(), any(), any(), isNull(), any())
// Set up CPU and Memory monitoring
val runtime = Runtime.getRuntime()
@@ -186,13 +184,13 @@ class CodeWhispererCodeFileScanTest : CodeWhispererCodeScanTestBase(PythonCodeIn
CodeScanSessionConfig.create(
psiFile.virtualFile,
project,
- CodeWhispererConstants.CodeAnalysisScope.FILE
+ CodeWhispererConstants.CodeAnalysisScope.FILE,
+ false
)
)
setupResponse(psiFile.virtualFile.toNioPath().relativeTo(sessionConfig.projectRoot.toNioPath()))
val sessionContext = CodeScanSessionContext(project, sessionConfig, CodeWhispererConstants.CodeAnalysisScope.FILE)
val session = spy(CodeWhispererCodeScanSession(sessionContext))
- doNothing().whenever(session).uploadArtifactToS3(any(), any(), any(), any(), isNull(), any())
// Set up CPU and Memory monitoring
val runtime = Runtime.getRuntime()
@@ -230,28 +228,47 @@ class CodeWhispererCodeFileScanTest : CodeWhispererCodeScanTestBase(PythonCodeIn
fun `test createUploadUrlAndUpload()`() {
val file = pyPsiFile.virtualFile.toNioPath().toFile()
val fileMd5: String = Base64.getEncoder().encodeToString(DigestUtils.md5(FileInputStream(file)))
- codeScanSessionSpy.stub {
- onGeneric { codeScanSessionSpy.createUploadUrl(any(), any(), any()) }
+ zipUploadManagerSpy.stub {
+ onGeneric { zipUploadManagerSpy.createUploadUrl(any(), any(), any(), any(), any()) }
.thenReturn(fakeCreateUploadUrlResponse)
}
- codeScanSessionSpy.createUploadUrlAndUpload(file, "artifactType", codeScanName)
+ zipUploadManagerSpy.createUploadUrlAndUpload(
+ file,
+ "artifactType",
+ CodeWhispererConstants.UploadTaskType.SCAN_FILE,
+ codeScanName,
+ CodeWhispererConstants.FeatureName.CODE_REVIEW
+ )
- val inOrder = inOrder(codeScanSessionSpy)
- inOrder.verify(codeScanSessionSpy).createUploadUrl(eq(fileMd5), eq("artifactType"), any())
- inOrder.verify(codeScanSessionSpy).uploadArtifactToS3(
+ val inOrder = inOrder(zipUploadManagerSpy)
+ inOrder.verify(zipUploadManagerSpy).createUploadUrl(
+ eq(fileMd5),
+ eq("artifactType"),
+ eq(CodeWhispererConstants.UploadTaskType.SCAN_FILE),
+ any(),
+ any()
+ )
+ inOrder.verify(zipUploadManagerSpy).uploadArtifactToS3(
eq(fakeCreateUploadUrlResponse.uploadUrl()),
eq(fakeCreateUploadUrlResponse.uploadId()),
eq(file),
eq(fileMd5),
eq(null),
+ any(),
any()
)
}
@Test
fun `test createUploadUrl()`() {
- val response = codeScanSessionSpy.createUploadUrl("md5", "type", codeScanName)
+ val response = zipUploadManagerSpy.createUploadUrl(
+ "md5",
+ "type",
+ CodeWhispererConstants.UploadTaskType.SCAN_FILE,
+ codeScanName,
+ CodeWhispererConstants.FeatureName.CODE_REVIEW
+ )
argumentCaptor().apply {
verify(mockClient).createUploadUrl(capture())
@@ -268,7 +285,7 @@ class CodeWhispererCodeFileScanTest : CodeWhispererCodeScanTestBase(PythonCodeIn
fakeListCodeScanFindingsResponse.codeScanFindings(),
getFakeRecommendationsOnNonExistentFile()
)
- val res = codeScanSessionSpy.mapToCodeScanIssues(recommendations)
+ val res = codeScanSessionSpy.mapToCodeScanIssues(recommendations, project)
assertThat(res).hasSize(2)
}
@@ -284,8 +301,8 @@ class CodeWhispererCodeFileScanTest : CodeWhispererCodeScanTestBase(PythonCodeIn
assertThat(it.responseContext.codeScanJobId).isEqualTo("jobId")
}
+ verify(zipUploadManagerSpy, times(1)).createUploadUrlAndUpload(eq(file), eq("SourceCode"), any(), anyString(), any())
val inOrder = inOrder(codeScanSessionSpy)
- inOrder.verify(codeScanSessionSpy, times(1)).createUploadUrlAndUpload(eq(file), eq("SourceCode"), anyString())
inOrder.verify(codeScanSessionSpy, times(1)).createCodeScan(eq(CodewhispererLanguage.Python.toString()), anyString())
inOrder.verify(codeScanSessionSpy, times(1)).getCodeScan(any())
inOrder.verify(codeScanSessionSpy, times(1)).listCodeScanFindings(eq("jobId"), eq(null))
@@ -311,7 +328,8 @@ class CodeWhispererCodeFileScanTest : CodeWhispererCodeScanTestBase(PythonCodeIn
CodeScanSessionConfig.create(
externalFile.virtualFile,
project,
- CodeWhispererConstants.CodeAnalysisScope.FILE
+ CodeWhispererConstants.CodeAnalysisScope.FILE,
+ false
)
)
@@ -337,8 +355,7 @@ class CodeWhispererCodeFileScanTest : CodeWhispererCodeScanTestBase(PythonCodeIn
scanManagerSpy.runCodeScan(CodeWhispererConstants.CodeAnalysisScope.FILE)
// verify that function was run but none of the results/error handling methods were called.
- verify(scanManagerSpy, times(0)).updateFileIssues(any(), any())
- verify(scanManagerSpy, times(0)).handleError(any(), any(), any())
+ verify(scanManagerSpy, times(0)).handleError(any(), any())
verify(scanManagerSpy, times(0)).handleException(any(), any(), any())
}
@@ -347,7 +364,7 @@ class CodeWhispererCodeFileScanTest : CodeWhispererCodeScanTestBase(PythonCodeIn
assertNotNull(pySession)
mockClient.stub {
- onGeneric { codeScanSessionSpy.createUploadUrlAndUpload(any(), any(), any()) }.thenThrow(
+ onGeneric { zipUploadManagerSpy.createUploadUrlAndUpload(any(), any(), any(), any(), any()) }.thenThrow(
CodeWhispererException.builder()
.message("File Scan Monthly Exceeded")
.requestId("abc123")
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/CodeWhispererCodeScanTest.kt b/plugins/amazonq/codewhisperer/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/CodeWhispererCodeScanTest.kt
index 6b0112e8c9..de232778da 100644
--- a/plugins/amazonq/codewhisperer/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/CodeWhispererCodeScanTest.kt
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/CodeWhispererCodeScanTest.kt
@@ -14,12 +14,11 @@ import org.mockito.ArgumentMatchers.anyString
import org.mockito.internal.verification.Times
import org.mockito.kotlin.any
import org.mockito.kotlin.argumentCaptor
-import org.mockito.kotlin.doNothing
import org.mockito.kotlin.eq
import org.mockito.kotlin.inOrder
-import org.mockito.kotlin.isNull
import org.mockito.kotlin.spy
import org.mockito.kotlin.stub
+import org.mockito.kotlin.times
import org.mockito.kotlin.verify
import software.amazon.awssdk.awscore.exception.AwsErrorDetails
import software.amazon.awssdk.services.codewhisperer.model.CodeWhispererException
@@ -30,6 +29,7 @@ import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.sessionco
import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.sessionconfig.PayloadContext
import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererConstants
import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererConstants.TOTAL_MILLIS_IN_SECOND
+import software.aws.toolkits.jetbrains.services.codewhisperer.util.getTelemetryErrorMessage
import software.aws.toolkits.jetbrains.utils.isInstanceOf
import software.aws.toolkits.jetbrains.utils.isInstanceOfSatisfying
import software.aws.toolkits.jetbrains.utils.rules.PythonCodeInsightTestFixtureRule
@@ -69,7 +69,8 @@ class CodeWhispererCodeScanTest : CodeWhispererCodeScanTestBase(PythonCodeInsigh
CodeScanSessionConfig.create(
psifile.virtualFile,
project,
- CodeWhispererConstants.CodeAnalysisScope.PROJECT
+ CodeWhispererConstants.CodeAnalysisScope.PROJECT,
+ false
)
)
setupResponse(psifile.virtualFile.toNioPath().relativeTo(sessionConfigSpy.projectRoot.toNioPath()))
@@ -81,7 +82,6 @@ class CodeWhispererCodeScanTest : CodeWhispererCodeScanTestBase(PythonCodeInsigh
// Mock CodeWhispererClient needs to be setup before initializing CodeWhispererCodeScanSession
codeScanSessionContext = CodeScanSessionContext(project, sessionConfigSpy, CodeWhispererConstants.CodeAnalysisScope.PROJECT)
codeScanSessionSpy = spy(CodeWhispererCodeScanSession(codeScanSessionContext))
- doNothing().`when`(codeScanSessionSpy).uploadArtifactToS3(any(), any(), any(), any(), isNull(), any())
mockClient.stub {
onGeneric { createUploadUrl(any()) }.thenReturn(fakeCreateUploadUrlResponse)
@@ -94,21 +94,28 @@ class CodeWhispererCodeScanTest : CodeWhispererCodeScanTestBase(PythonCodeInsigh
@Test
fun `test createUploadUrlAndUpload()`() {
val fileMd5: String = Base64.getEncoder().encodeToString(DigestUtils.md5(FileInputStream(file)))
- codeScanSessionSpy.stub {
- onGeneric { codeScanSessionSpy.createUploadUrl(any(), any(), any()) }
+ zipUploadManagerSpy.stub {
+ onGeneric { zipUploadManagerSpy.createUploadUrl(any(), any(), any(), any(), any()) }
.thenReturn(fakeCreateUploadUrlResponse)
}
- codeScanSessionSpy.createUploadUrlAndUpload(file, "artifactType", codeScanName)
+ zipUploadManagerSpy.createUploadUrlAndUpload(
+ file,
+ "artifactType",
+ CodeWhispererConstants.UploadTaskType.SCAN_FILE,
+ codeScanName,
+ CodeWhispererConstants.FeatureName.CODE_REVIEW
+ )
- val inOrder = inOrder(codeScanSessionSpy)
- inOrder.verify(codeScanSessionSpy).createUploadUrl(eq(fileMd5), eq("artifactType"), any())
- inOrder.verify(codeScanSessionSpy).uploadArtifactToS3(
+ val inOrder = inOrder(zipUploadManagerSpy)
+ inOrder.verify(zipUploadManagerSpy).createUploadUrl(eq(fileMd5), eq("artifactType"), any(), any(), any())
+ inOrder.verify(zipUploadManagerSpy).uploadArtifactToS3(
eq(fakeCreateUploadUrlResponse.uploadUrl()),
eq(fakeCreateUploadUrlResponse.uploadId()),
eq(file),
eq(fileMd5),
eq(null),
+ any(),
any()
)
}
@@ -118,13 +125,25 @@ class CodeWhispererCodeScanTest : CodeWhispererCodeScanTestBase(PythonCodeInsigh
val invalidZipFile = File("/path/file.zip")
assertThrows {
- codeScanSessionSpy.createUploadUrlAndUpload(invalidZipFile, "artifactType", codeScanName)
+ zipUploadManagerSpy.createUploadUrlAndUpload(
+ invalidZipFile,
+ "artifactType",
+ CodeWhispererConstants.UploadTaskType.SCAN_FILE,
+ codeScanName,
+ CodeWhispererConstants.FeatureName.CODE_REVIEW
+ )
}
}
@Test
fun `test createUploadUrl()`() {
- val response = codeScanSessionSpy.createUploadUrl("md5", "type", codeScanName)
+ val response = zipUploadManagerSpy.createUploadUrl(
+ "md5",
+ "type",
+ CodeWhispererConstants.UploadTaskType.SCAN_FILE,
+ codeScanName,
+ featureUseCase = CodeWhispererConstants.FeatureName.CODE_REVIEW
+ )
argumentCaptor().apply {
verify(mockClient).createUploadUrl(capture())
@@ -141,7 +160,7 @@ class CodeWhispererCodeScanTest : CodeWhispererCodeScanTestBase(PythonCodeInsigh
fakeListCodeScanFindingsResponse.codeScanFindings(),
getFakeRecommendationsOnNonExistentFile()
)
- val res = codeScanSessionSpy.mapToCodeScanIssues(recommendations)
+ val res = codeScanSessionSpy.mapToCodeScanIssues(recommendations, project)
assertThat(res).hasSize(2)
}
@@ -150,7 +169,7 @@ class CodeWhispererCodeScanTest : CodeWhispererCodeScanTestBase(PythonCodeInsigh
val recommendations = listOf(
fakeListCodeScanFindingsOutOfBoundsIndexResponse.codeScanFindings(),
)
- val res = codeScanSessionSpy.mapToCodeScanIssues(recommendations)
+ val res = codeScanSessionSpy.mapToCodeScanIssues(recommendations, project)
assertThat(res).hasSize(1)
}
@@ -179,7 +198,7 @@ class CodeWhispererCodeScanTest : CodeWhispererCodeScanTestBase(PythonCodeInsigh
)
exceptions.forEachIndexed { index, exception ->
- val actualMessage = codeScanSessionSpy.getTelemetryErrorMessage(exception)
+ val actualMessage = getTelemetryErrorMessage(exception, featureUseCase = CodeWhispererConstants.FeatureName.CODE_REVIEW)
assertThat(expectedMessages[index]).isEqualTo(actualMessage)
}
}
@@ -194,8 +213,8 @@ class CodeWhispererCodeScanTest : CodeWhispererCodeScanTestBase(PythonCodeInsigh
assertThat(it.responseContext.codeScanJobId).isEqualTo("jobId")
}
+ verify(zipUploadManagerSpy, times(1)).createUploadUrlAndUpload(eq(file), eq("SourceCode"), any(), anyString(), any())
val inOrder = inOrder(codeScanSessionSpy)
- inOrder.verify(codeScanSessionSpy, Times(1)).createUploadUrlAndUpload(eq(file), eq("SourceCode"), anyString())
inOrder.verify(codeScanSessionSpy, Times(1)).createCodeScan(eq(CodewhispererLanguage.Python.toString()), anyString())
inOrder.verify(codeScanSessionSpy, Times(1)).getCodeScan(any())
inOrder.verify(codeScanSessionSpy, Times(1)).listCodeScanFindings(eq("jobId"), eq(null))
@@ -206,9 +225,9 @@ class CodeWhispererCodeScanTest : CodeWhispererCodeScanTestBase(PythonCodeInsigh
assertNotNull(sessionConfigSpy)
mockClient.stub {
- onGeneric { codeScanSessionSpy.createUploadUrlAndUpload(any(), any(), any()) }.thenThrow(
+ onGeneric { zipUploadManagerSpy.createUploadUrlAndUpload(any(), any(), any(), any(), any()) }.thenThrow(
CodeWhispererException.builder()
- .message("Project Scan Monthly Exceeded")
+ .message("Project Review Monthly Exceeded")
.requestId("abc123")
.statusCode(400)
.cause(RuntimeException("Something went wrong"))
@@ -228,7 +247,7 @@ class CodeWhispererCodeScanTest : CodeWhispererCodeScanTestBase(PythonCodeInsigh
assertThat(codeScanResponse).isInstanceOf()
if (codeScanResponse is CodeScanResponse.Failure) {
assertThat(codeScanResponse.failureReason).isInstanceOf()
- assertThat(codeScanResponse.failureReason.toString()).contains("Project Scan Monthly Exceeded")
+ assertThat(codeScanResponse.failureReason.toString()).contains("Project Review Monthly Exceeded")
assertThat(codeScanResponse.failureReason.cause.toString()).contains("java.lang.RuntimeException: Something went wrong")
}
}
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/CodeWhispererCodeScanTestBase.kt b/plugins/amazonq/codewhisperer/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/CodeWhispererCodeScanTestBase.kt
index ac85aecdd4..ba486c97d5 100644
--- a/plugins/amazonq/codewhisperer/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/CodeWhispererCodeScanTestBase.kt
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/CodeWhispererCodeScanTestBase.kt
@@ -8,6 +8,7 @@ import com.github.tomakehurst.wiremock.junit.WireMockRule
import com.intellij.analysis.problemsView.toolWindow.ProblemsView
import com.intellij.openapi.application.ApplicationManager
import com.intellij.openapi.project.Project
+import com.intellij.openapi.vfs.VirtualFile
import com.intellij.openapi.wm.RegisterToolWindowTask
import com.intellij.openapi.wm.ToolWindowManager
import com.intellij.testFramework.ApplicationRule
@@ -33,7 +34,12 @@ import software.amazon.awssdk.services.codewhisperer.model.CodeScanStatus
import software.amazon.awssdk.services.codewhisperer.model.CreateCodeScanResponse
import software.amazon.awssdk.services.codewhisperer.model.GetCodeScanResponse
import software.amazon.awssdk.services.codewhisperer.model.ListCodeScanFindingsResponse
+import software.amazon.awssdk.services.codewhispererruntime.model.CodeFixJobStatus
import software.amazon.awssdk.services.codewhispererruntime.model.CreateUploadUrlResponse
+import software.amazon.awssdk.services.codewhispererruntime.model.GetCodeFixJobResponse
+import software.amazon.awssdk.services.codewhispererruntime.model.Reference
+import software.amazon.awssdk.services.codewhispererruntime.model.Span
+import software.amazon.awssdk.services.codewhispererruntime.model.StartCodeFixJobResponse
import software.aws.toolkits.jetbrains.core.MockClientManagerRule
import software.aws.toolkits.jetbrains.services.codewhisperer.CodeWhispererTestUtil
import software.aws.toolkits.jetbrains.services.codewhisperer.codescan.sessionconfig.CodeScanSessionConfig
@@ -41,6 +47,7 @@ import software.aws.toolkits.jetbrains.services.codewhisperer.credentials.CodeWh
import software.aws.toolkits.jetbrains.services.codewhisperer.credentials.CodeWhispererLoginType
import software.aws.toolkits.jetbrains.services.codewhisperer.explorer.CodeWhispererExplorerActionManager
import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererConstants
+import software.aws.toolkits.jetbrains.services.codewhisperer.util.CodeWhispererZipUploadManager
import software.aws.toolkits.jetbrains.utils.rules.CodeInsightTestFixtureRule
import software.aws.toolkits.telemetry.CodewhispererLanguage
import java.nio.file.Path
@@ -81,12 +88,15 @@ open class CodeWhispererCodeScanTestBase(projectRule: CodeInsightTestFixtureRule
internal lateinit var fakeGetCodeScanResponse: GetCodeScanResponse
internal lateinit var fakeGetCodeScanResponsePending: GetCodeScanResponse
internal lateinit var fakeGetCodeScanResponseFailed: GetCodeScanResponse
+ internal lateinit var fakeGetCodeFixJobResponse: GetCodeFixJobResponse
+ internal lateinit var fakeStartCodeFixJobResponse: StartCodeFixJobResponse
internal val metadata: DefaultAwsResponseMetadata = DefaultAwsResponseMetadata.create(
mapOf(AwsHeader.AWS_REQUEST_ID to CodeWhispererTestUtil.testRequestId)
)
internal lateinit var scanManagerSpy: CodeWhispererCodeScanManager
+ internal lateinit var zipUploadManagerSpy: CodeWhispererZipUploadManager
internal lateinit var project: Project
@Before
@@ -95,7 +105,12 @@ open class CodeWhispererCodeScanTestBase(projectRule: CodeInsightTestFixtureRule
s3endpoint = "http://127.0.0.1:${wireMock.port()}"
scanManagerSpy = spy(CodeWhispererCodeScanManager.getInstance(project))
- doNothing().whenever(scanManagerSpy).addCodeScanUI(any())
+ doNothing().whenever(scanManagerSpy).buildCodeScanUI()
+ doNothing().whenever(scanManagerSpy).showCodeScanUI()
+
+ zipUploadManagerSpy = spy(CodeWhispererZipUploadManager.getInstance(project))
+ doNothing().whenever(zipUploadManagerSpy).uploadArtifactToS3(any(), any(), any(), any(), isNull(), any(), any())
+ projectRule.project.replaceService(CodeWhispererZipUploadManager::class.java, zipUploadManagerSpy, disposableRule.disposable)
mockClient = mock().also {
project.replaceService(CodeWhispererClientAdaptor::class.java, it, disposableRule.disposable)
@@ -142,7 +157,7 @@ open class CodeWhispererCodeScanTestBase(projectRule: CodeInsightTestFixtureRule
"codeSnippet": [
$codeSnippetJson
],
- "severity": "severity",
+ "severity": "Low",
"remediation": {
"recommendation": {
"text": "recommendationText",
@@ -212,6 +227,36 @@ open class CodeWhispererCodeScanTestBase(projectRule: CodeInsightTestFixtureRule
.responseMetadata(metadata)
.build() as CreateUploadUrlResponse
+ fakeGetCodeFixJobResponse = GetCodeFixJobResponse.builder()
+ .jobStatus(CodeFixJobStatus.SUCCEEDED)
+ .suggestedFix(
+ software.amazon.awssdk.services.codewhispererruntime.model.SuggestedFix.builder()
+ .codeDiff("diff")
+ .description("description")
+ .references(
+ Reference.builder()
+ .url(s3endpoint)
+ .licenseName("license")
+ .repository("repo")
+ .recommendationContentSpan(
+ Span.builder()
+ .start(6)
+ .end(8)
+ .build()
+ )
+ .build()
+ )
+ .build()
+ )
+ .responseMetadata(metadata)
+ .build() as GetCodeFixJobResponse
+
+ fakeStartCodeFixJobResponse = StartCodeFixJobResponse.builder()
+ .jobId(JOB_ID)
+ .status(CodeFixJobStatus.IN_PROGRESS)
+ .responseMetadata(metadata)
+ .build() as StartCodeFixJobResponse
+
fakeCreateCodeScanResponse = CreateCodeScanResponse.builder()
.status(CodeScanStatus.COMPLETED)
.jobId(JOB_ID)
@@ -286,7 +331,7 @@ open class CodeWhispererCodeScanTestBase(projectRule: CodeInsightTestFixtureRule
"content": "codeBlock2"
}
],
- "severity": "severity",
+ "severity": "Low",
"remediation": {
"recommendation": {
"text": "recommendationText",
@@ -336,7 +381,6 @@ open class CodeWhispererCodeScanTestBase(projectRule: CodeInsightTestFixtureRule
) {
val codeScanContext = CodeScanSessionContext(project, sessionConfigSpy, CodeWhispererConstants.CodeAnalysisScope.PROJECT)
val sessionMock = spy(CodeWhispererCodeScanSession(codeScanContext))
- doNothing().`when`(sessionMock).uploadArtifactToS3(any(), any(), any(), any(), isNull(), any())
ToolWindowManager.getInstance(project).registerToolWindow(
RegisterToolWindowTask(
@@ -364,8 +408,58 @@ open class CodeWhispererCodeScanTestBase(projectRule: CodeInsightTestFixtureRule
assertThat(treeModel.getTotalIssuesCount()).isEqualTo(expectedTotalIssues)
}
+ fun createCodeScanIssue(project: Project, virtualFile: VirtualFile): CodeWhispererCodeScanIssue =
+ CodeWhispererCodeScanIssue(
+ project = project,
+ file = virtualFile,
+ startLine = 10,
+ startCol = 5,
+ endLine = 15,
+ endCol = 20,
+ title = "Potential Security Vulnerability",
+ description = Description(
+ text = "A detailed description of the security issue found in the code",
+ markdown = "# Security Issue\n\nA detailed description of the security issue found in the code"
+ ),
+ detectorId = "AWS-DETECTOR-001",
+ detectorName = "SecurityScanner",
+ findingId = "FINDING-123",
+ ruleId = "RULE-456",
+ relatedVulnerabilities = listOf(
+ "CVE-2023-12345",
+ "CVE-2023-67890"
+ ),
+ severity = "HIGH",
+ recommendation = Recommendation(
+ text = "Consider implementing secure coding practices",
+ url = "https://docs.aws.amazon.com/security-best-practices"
+ ),
+ suggestedFixes = emptyList(), // Empty list as requested
+ codeSnippet = listOf(
+ CodeLine(
+ number = 10,
+ content = "val unsecureCode = performOperation()"
+ ),
+ CodeLine(
+ number = 11,
+ content = "processData(unsecureCode)"
+ )
+ )
+ )
+
+// You might need these data classes depending on your implementation
+ data class Description(val message: String)
+ data class Recommendation(val text: String)
+ data class SuggestedFix(val description: String, val code: String)
+ data class CodeLine(val lineNumber: Int, val content: String)
+
companion object {
const val UPLOAD_ID = "uploadId"
const val JOB_ID = "jobId"
+ const val KMS_KEY_ARN = "kmsKeyArn"
+ val REQUEST_HEADERS = mapOf(
+ "Content-Type" to "application/zip",
+ "test" to "aws:test",
+ )
}
}
diff --git a/plugins/amazonq/codewhisperer/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/CodeWhispererProjectCodeScanTest.kt b/plugins/amazonq/codewhisperer/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/CodeWhispererProjectCodeScanTest.kt
index 45b06857e5..47d2c5ff6e 100644
--- a/plugins/amazonq/codewhisperer/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/CodeWhispererProjectCodeScanTest.kt
+++ b/plugins/amazonq/codewhisperer/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/codewhisperer/codescan/CodeWhispererProjectCodeScanTest.kt
@@ -45,10 +45,10 @@ class CodeWhispererProjectCodeScanTest : CodeWhispererCodeScanTestBase(PythonCod
override fun setup() {
super.setup()
setupCsharpProject()
- sessionConfigSpy = spy(CodeScanSessionConfig.create(testCs, project, CodeWhispererConstants.CodeAnalysisScope.PROJECT))
+ sessionConfigSpy = spy(CodeScanSessionConfig.create(testCs, project, CodeWhispererConstants.CodeAnalysisScope.PROJECT, true))
setupResponse(testCs.toNioPath().relativeTo(sessionConfigSpy.projectRoot.toNioPath()))
- sessionConfigSpy2 = spy(CodeScanSessionConfig.create(testCs, project, CodeWhispererConstants.CodeAnalysisScope.FILE))
+ sessionConfigSpy2 = spy(CodeScanSessionConfig.create(testCs, project, CodeWhispererConstants.CodeAnalysisScope.FILE, true))
setupResponse(testCs.toNioPath().relativeTo(sessionConfigSpy2.projectRoot.toNioPath()))
mockClient.stub {
@@ -372,6 +372,6 @@ class CodeWhispererProjectCodeScanTest : CodeWhispererCodeScanTestBase(PythonCod
// Adding gitignore file and gitignore file member for testing.
// The tests include the markdown file but not these two files.
projectRule.fixture.addFileToProject("/.gitignore", "node_modules\n.idea\n.vscode\n.DS_Store").virtualFile
- projectRule.fixture.addFileToProject("test.idea", "ref: refs/heads/main")
+ projectRule.fixture.addFileToProject("/.idea/ref", "ref: refs/heads/main")
}
}
diff --git a/plugins/amazonq/mynah-ui/package-lock.json b/plugins/amazonq/mynah-ui/package-lock.json
index 37e7bd7efc..650fc7194c 100644
--- a/plugins/amazonq/mynah-ui/package-lock.json
+++ b/plugins/amazonq/mynah-ui/package-lock.json
@@ -9,7 +9,7 @@
"version": "1.0.0",
"license": "ISC",
"dependencies": {
- "@aws/mynah-ui-chat": "npm:@aws/mynah-ui@4.18.1",
+ "@aws/mynah-ui-chat": "npm:@aws/mynah-ui@4.21.2",
"@types/node": "^14.18.5",
"fs-extra": "^10.0.1",
"sanitize-html": "^2.12.1",
@@ -57,9 +57,9 @@
},
"node_modules/@aws/mynah-ui-chat": {
"name": "@aws/mynah-ui",
- "version": "4.18.1",
- "resolved": "https://registry.npmjs.org/@aws/mynah-ui/-/mynah-ui-4.18.1.tgz",
- "integrity": "sha512-531FL5509O081eWIk0P0reQhGTm/ZaXhRu6FLNqMvAKySPtJyyxee0ieeGAR8h5CVI75learQbXJEGJC6XibAA==",
+ "version": "4.21.2",
+ "resolved": "https://registry.npmjs.org/@aws/mynah-ui/-/mynah-ui-4.21.2.tgz",
+ "integrity": "sha512-zC/ck/m5nYXRwTs3EoiGNYR0jTfbrnovRloqlD07fmvTt9OpbWLhagg14Jr/+mqoYX3YWpqbLs9U56mqCLwHHQ==",
"hasInstallScript": true,
"license": "Apache License 2.0",
"dependencies": {
@@ -2368,9 +2368,9 @@
"dev": true
},
"node_modules/nanoid": {
- "version": "3.3.7",
- "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.7.tgz",
- "integrity": "sha512-eSRppjcPIatRIMC1U6UngP8XFcz8MQWGQdt1MTBQ7NaAmvXDfvNxbvWV3x2y6CdEUciCSsDHDQZbhYaB8QEo2g==",
+ "version": "3.3.8",
+ "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.8.tgz",
+ "integrity": "sha512-WNLf5Sd8oZxOm+TzppcYk8gVOgP+l58xNy58D0nbUnOxOWRWvlcCV4kUF7ltmI6PsrLl/BgKEyS4mqsGChFN0w==",
"funding": [
{
"type": "github",
diff --git a/plugins/amazonq/mynah-ui/package.json b/plugins/amazonq/mynah-ui/package.json
index edb7db6624..f2e32a78a1 100644
--- a/plugins/amazonq/mynah-ui/package.json
+++ b/plugins/amazonq/mynah-ui/package.json
@@ -12,7 +12,7 @@
"lintfix": "eslint -c .eslintrc.js --fix --ext .ts ."
},
"dependencies": {
- "@aws/mynah-ui-chat": "npm:@aws/mynah-ui@4.18.1",
+ "@aws/mynah-ui-chat": "npm:@aws/mynah-ui@4.21.2",
"@types/node": "^14.18.5",
"fs-extra": "^10.0.1",
"sanitize-html": "^2.12.1",
diff --git a/plugins/amazonq/mynah-ui/src/mynah-ui/ui/apps/amazonqCommonsConnector.ts b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/apps/amazonqCommonsConnector.ts
index 01af350411..9fb0d54c9a 100644
--- a/plugins/amazonq/mynah-ui/src/mynah-ui/ui/apps/amazonqCommonsConnector.ts
+++ b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/apps/amazonqCommonsConnector.ts
@@ -3,15 +3,19 @@
* SPDX-License-Identifier: Apache-2.0
*/
-import { ChatItemAction } from '@aws/mynah-ui-chat'
+import { ChatItemAction, ChatPrompt } from '@aws/mynah-ui-chat'
import { AuthFollowUpType } from '../followUps/generator'
import { ExtensionMessage } from '../commands'
+import {getTabCommandFromTabType, isTabType, TabType } from '../storages/tabsStorage'
+import {codeScanUserGuide, codeTestUserGuide, codeTransformUserGuide, docUserGuide, featureDevUserGuide} from "../texts/constants";
+import {createClickTelemetry, createOpenAgentTelemetry, Trigger} from "../telemetry/actions";
export type WelcomeFollowupType = 'continue-to-chat'
export interface ConnectorProps {
sendMessageToExtension: (message: ExtensionMessage) => void
onWelcomeFollowUpClicked: (tabID: string, welcomeFollowUpType: WelcomeFollowupType) => void
+ handleCommand: (chatPrompt: ChatPrompt, tabId: string) => void
}
export interface CodeReference {
licenseName?: string
@@ -26,10 +30,12 @@ export interface CodeReference {
export class Connector {
private readonly sendMessageToExtension
private readonly onWelcomeFollowUpClicked
+ private readonly handleCommand
constructor(props: ConnectorProps) {
this.sendMessageToExtension = props.sendMessageToExtension
this.onWelcomeFollowUpClicked = props.onWelcomeFollowUpClicked
+ this.handleCommand = props.handleCommand
}
followUpClicked = (tabID: string, followUp: ChatItemAction): void => {
@@ -46,4 +52,63 @@ export class Connector {
tabType,
})
}
+
+ onCustomFormAction(
+ tabId: string,
+ action: {
+ id: string
+ text?: string | undefined
+ formItemValues?: Record | undefined
+ }
+ ) {
+ const tabType = action.id.split('-')[2]
+ if (!isTabType(tabType)) {
+ return
+ }
+
+ if (action.id.startsWith('user-guide-')) {
+ this.processUserGuideLink(tabType, action.id)
+ return
+ }
+
+ if (action.id.startsWith('quick-start-')) {
+ this.handleCommand(
+ {
+ command: getTabCommandFromTabType(tabType),
+ },
+ tabId
+ )
+
+ this.sendMessageToExtension(createOpenAgentTelemetry(tabType, 'quick-start'))
+ }
+ }
+
+ private processUserGuideLink(tabType: TabType, actionId: string) {
+ let userGuideLink = ''
+ switch (tabType) {
+ case 'codescan':
+ userGuideLink = codeScanUserGuide
+ break
+ case 'codetest':
+ userGuideLink = codeTestUserGuide
+ break
+ case 'codetransform':
+ userGuideLink = codeTransformUserGuide
+ break
+ case 'doc':
+ userGuideLink = docUserGuide
+ break
+ case 'featuredev':
+ userGuideLink = featureDevUserGuide
+ break
+ }
+
+ // e.g. amazonq-explore-user-guide-featuredev
+ this.sendMessageToExtension(createClickTelemetry(`amazonq-explore-${actionId}`))
+
+ this.sendMessageToExtension({
+ command: 'open-user-guide',
+ userGuideLink,
+ })
+ }
}
diff --git a/plugins/amazonq/mynah-ui/src/mynah-ui/ui/apps/codeScanChatConnector.ts b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/apps/codeScanChatConnector.ts
new file mode 100644
index 0000000000..60ffe3222f
--- /dev/null
+++ b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/apps/codeScanChatConnector.ts
@@ -0,0 +1,183 @@
+/*!
+ * Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+import {ExtensionMessage} from "../commands";
+import {ChatItem, ChatItemType, ProgressField} from "@aws/mynah-ui-chat";
+import {FormButtonIds} from "../forms/constants";
+
+export interface ICodeScanChatConnectorProps {
+ sendMessageToExtension: (message: ExtensionMessage) => void
+ onCodeScanMessageReceived: (tabID: string, message: ChatItem, isLoading: boolean, clearPreviousItemButtons?: boolean, runReview?: boolean) => void
+ onUpdatePlaceholder: (tabID: string, newPlaceholder: string) => void,
+ onUpdatePromptProgress: (tabID: string, progressField: ProgressField | null | undefined) => void
+ onChatInputEnabled: (tabID: string, enabled: boolean) => void
+}
+
+export class CodeScanChatConnector {
+ private readonly sendMessageToExtension
+ private readonly onCodeScanMessageReceived
+ private readonly updatePlaceholder
+ private readonly updatePromptProgress
+ private readonly chatInputEnabled
+
+ constructor(props: ICodeScanChatConnectorProps) {
+ this.sendMessageToExtension = props.sendMessageToExtension
+ this.onCodeScanMessageReceived = props.onCodeScanMessageReceived
+ this.updatePlaceholder = props.onUpdatePlaceholder
+ this.updatePromptProgress = props.onUpdatePromptProgress
+ this.chatInputEnabled = props.onChatInputEnabled
+ }
+
+ private processChatMessage = (messageData: any): void => {
+ const runReview = messageData.command === "review"
+ if (this.onCodeScanMessageReceived === undefined) {
+ return
+ }
+
+ const tabID = messageData.tabID
+ const isAddingNewItem: boolean = messageData.isAddingNewItem
+ const isLoading: boolean = messageData.isLoading
+ const clearPreviousItemButtons: boolean = messageData.clearPreviousItemButtons
+ const type = messageData.messageType
+
+ if (isAddingNewItem && type === ChatItemType.ANSWER_PART) {
+ this.onCodeScanMessageReceived(tabID, {
+ type: ChatItemType.ANSWER_STREAM,
+ }, isLoading)
+ }
+
+ const chatItem: ChatItem = {
+ type: type,
+ body: messageData.message ?? undefined,
+ messageId: messageData.messageId ?? messageData.triggerID ?? '',
+ relatedContent: undefined,
+ canBeVoted: messageData.canBeVoted ?? true,
+ formItems: messageData.formItems,
+ buttons:
+ messageData.buttons !== undefined && messageData.buttons.length > 0 ? messageData.buttons : undefined,
+ followUp:
+ messageData.followUps !== undefined && messageData.followUps.length > 0
+ ? {
+ text: '',
+ options: messageData.followUps,
+ }
+ : undefined
+ }
+ this.onCodeScanMessageReceived(tabID, chatItem, isLoading, clearPreviousItemButtons, runReview)
+ }
+
+ handleMessageReceive = async (messageData: any): Promise => {
+ if (messageData.type === 'chatMessage') {
+ this.processChatMessage(messageData)
+ return
+ }
+ if (messageData.type === 'updatePlaceholderMessage') {
+ this.updatePlaceholder(messageData.tabID, messageData.newPlaceholder)
+ return
+ }
+
+ if(messageData.type === 'updatePromptProgress') {
+ this.updatePromptProgress(messageData.tabID, messageData.progressField)
+ }
+
+ if(messageData.type === 'chatInputEnabledMessage') {
+ this.chatInputEnabled(messageData.tabID, messageData.enabled)
+ }
+ }
+
+ onFormButtonClick = (
+ tabID: string,
+ action: {
+ id: string
+ text?: string
+ formItemValues?: Record
+ }
+ ) => {
+ if (action.id === FormButtonIds.CodeScanStartProjectScan) {
+ this.sendMessageToExtension({
+ command: 'codescan_start_project_scan',
+ tabID,
+ tabType: 'codescan',
+ })
+ } else if (action.id === FormButtonIds.CodeScanStartFileScan) {
+ this.sendMessageToExtension({
+ command: 'codescan_start_file_scan',
+ tabID,
+ tabType: 'codescan'
+ })
+ } else if (action.id === FormButtonIds.CodeScanStopFileScan) {
+ this.sendMessageToExtension({
+ command: 'codescan_stop_file_scan',
+ tabID,
+ tabType: 'codescan'
+ })
+ } else if (action.id === FormButtonIds.CodeScanStopProjectScan) {
+ this.sendMessageToExtension({
+ command: 'codescan_stop_project_scan',
+ tabID,
+ tabType: 'codescan'
+ })
+ } else if (action.id === FormButtonIds.CodeScanOpenIssues) {
+ this.sendMessageToExtension({
+ command: 'codescan_open_issues',
+ tabID,
+ tabType: 'codescan'
+ })
+ }
+ }
+ onResponseBodyLinkClick(tabID: string, messageId: string, link: string) {
+ this.sendMessageToExtension({
+ command: 'response-body-link-click',
+ tabID,
+ messageId,
+ link,
+ tabType: 'codescan',
+ })
+ }
+
+ clearChat = (tabID: string): void => {
+ this.sendMessageToExtension({
+ tabID: tabID,
+ command: 'clear',
+ chatMessage: '',
+ tabType: 'codescan',
+ })
+ }
+
+ help = (tabID: string): void => {
+ console.log("reached here")
+ this.sendMessageToExtension({
+ tabID: tabID,
+ command: 'help',
+ chatMessage: '',
+ tabType: 'codescan',
+ })
+ }
+
+ onTabOpen = (tabID: string) => {
+ this.sendMessageToExtension({
+ tabID,
+ command: 'new-tab-was-created',
+ tabType: 'codescan'
+ })
+ }
+
+ onTabRemove = (tabID: string) => {
+ this.sendMessageToExtension({
+ tabID,
+ command: 'tab-was-removed',
+ tabType: 'codescan'
+ })
+ }
+
+ scan = (tabID: string): void => {
+ this.sendMessageToExtension({
+ tabID: tabID,
+ command: 'scan',
+ chatMessage: 'scan',
+ tabType: 'codescan'
+ })
+ }
+}
diff --git a/plugins/amazonq/mynah-ui/src/mynah-ui/ui/apps/codeTestChatConnector.ts b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/apps/codeTestChatConnector.ts
new file mode 100644
index 0000000000..5a6e174497
--- /dev/null
+++ b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/apps/codeTestChatConnector.ts
@@ -0,0 +1,584 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+import {ExtensionMessage} from "../commands";
+import {ChatPayload, ConnectorProps} from "../connector";
+import {FormButtonIds} from "../forms/constants";
+import {ChatItem, ChatItemAction, ChatItemType, MynahIcons, MynahUIDataModel} from '@aws/mynah-ui-chat'
+import {CodeReference} from "./amazonqCommonsConnector";
+import {Status} from "@aws/mynah-ui-chat/dist/static";
+import {EmptyMynahUIDataModel} from "@aws/mynah-ui-chat/dist/helper/store";
+import {doesNotMatch} from "node:assert";
+
+export interface ICodeTestChatConnectorProps {
+ sendMessageToExtension: (message: ExtensionMessage) => void
+ onChatAnswerReceived?: (tabID: string, message: ChatItem) => void
+ onUpdateAuthentication: (
+ featureDevEnabled: boolean,
+ codeTransformEnabled: boolean,
+ docEnabled: boolean,
+ codeScanEnabled: boolean,
+ codeTestEnabled: boolean,
+ authenticatingTabIDs: string[]
+ ) => void
+ onRunTestMessageReceived?: (tabID: string, showRunTestMessage: boolean) => void
+ onChatAnswerUpdated?: (tabID: string, message: ChatItem) => void
+ onChatInputEnabled: (tabID: string, enabled: boolean) => void
+ onUpdatePlaceholder: (tabID: string, newPlaceholder: string) => void
+ onError: (tabID: string, message: string, title: string) => void
+}
+
+interface IntroductionCardContentType {
+ title: string
+ description: string
+ icon: MynahIcons
+ content: {
+ body: string
+ }
+}
+
+interface MessageData {
+ message?: string
+ messageType: ChatItemType
+ messageId?: string
+ triggerID?: string
+ informationCard?: boolean
+ canBeVoted: boolean
+ filePath?: string
+ tabID: string
+}
+
+function getIntroductionCardContent(): IntroductionCardContentType {
+ const introductionBody = [
+ "I can generate unit tests for your active file. ",
+ "\n\n",
+ "After you select the functions or methods I should focus on, I will:\n",
+ "1. Generate unit tests\n",
+ "2. Place them into relevant test file\n",
+ "\n\n",
+ "To learn more, check out our [user guide](https://aws.amazon.com/q/developer/)."
+ ].join("");
+
+ return {
+ title: "/test",
+ description: "Included in your Q Developer subscription",
+ icon: MynahIcons.CHECK_LIST,
+ content: {
+ body: introductionBody
+ }
+ }
+}
+
+export class CodeTestChatConnector {
+ private readonly sendMessageToExtension
+ private readonly onChatAnswerReceived
+ private readonly onChatAnswerUpdated
+ private readonly onMessageReceived
+ private readonly onUpdateAuthentication
+ private readonly chatInputEnabled
+ private readonly updatePlaceholder
+ private readonly updatePromptProgress
+ private readonly onError
+ private readonly runTestMessageReceived
+
+ constructor(props: ConnectorProps) {
+ this.sendMessageToExtension = props.sendMessageToExtension
+ this.onChatAnswerReceived = props.onChatAnswerReceived
+ this.onChatAnswerUpdated = props.onChatAnswerUpdated
+ this.runTestMessageReceived = props.onRunTestMessageReceived
+ this.onMessageReceived = props.onMessageReceived
+ this.onUpdateAuthentication = props.onUpdateAuthentication
+ this.chatInputEnabled = props.onChatInputEnabled
+ this.updatePlaceholder = props.onUpdatePlaceholder
+ this.updatePromptProgress = props.onUpdatePromptProgress
+ this.onError = props.onError
+ }
+
+ private addAnswer = (messageData: any): void => {
+ console.log("message data in addAnswer:")
+ console.log(messageData)
+ if (this.onChatAnswerReceived === undefined) {
+ return
+ }
+ if (messageData.command === 'test' && this.runTestMessageReceived) {
+ this.runTestMessageReceived(messageData.tabID, true)
+ return
+ }
+ const answer: ChatItem = {
+ type: messageData.messageType,
+ messageId: messageData.messageId ?? messageData.triggerID,
+ body: messageData.message,
+ relatedContent: undefined,
+ snapToTop: messageData.snapToTop,
+ canBeVoted: messageData.canBeVoted ?? false,
+ followUp: messageData.followUps ? {
+ text: '',
+ options: messageData.followUps,
+ } : undefined,
+ buttons: messageData.buttons ?? undefined,
+ fileList: messageData.fileList ? {
+ rootFolderTitle: messageData.projectRootName,
+ fileTreeTitle: 'READY FOR REVIEW',
+ filePaths: messageData.fileList,
+ details: {
+ [messageData.filePaths]: {
+ icon: MynahIcons.FILE,
+ },
+ },
+ } : undefined,
+ codeBlockActions: {
+ 'insert-to-cursor': undefined
+ },
+ codeReference: messageData.codeReference?.length ? messageData.codeReference : undefined
+ }
+
+ this.onChatAnswerReceived(messageData.tabID, answer)
+ }
+
+ private updateAnswer = (messageData: any): void => {
+ console.log("message data in updateAnswer:")
+ console.log(messageData)
+ if (this.onChatAnswerUpdated == undefined) {
+ return
+ }
+ const answer: ChatItem = {
+ type: messageData.messageType,
+ messageId: messageData.messageId ?? messageData.triggerID,
+ body: messageData.message,
+ buttons: messageData.buttons ?? undefined,
+ followUp: messageData.followUps ? {
+ text: '',
+ options: messageData.followUps,
+ } : undefined,
+ canBeVoted: true,
+ fileList: messageData.fileList ? {
+ rootFolderTitle: messageData.projectRootName,
+ fileTreeTitle: 'READY FOR REVIEW',
+ filePaths: messageData.fileList,
+ details: {
+ [messageData.fileList]: {
+ icon: MynahIcons.FILE,
+ },
+ },
+ } : undefined,
+ footer: messageData.footer ? {
+ fileList: {
+ rootFolderTitle: undefined,
+ fileTreeTitle: '',
+ filePaths: messageData.footer,
+ details: {
+ [messageData.footer]: {
+ icon: MynahIcons.FILE,
+ },
+ },
+ },
+ } : undefined,
+ codeReference: messageData.codeReference?.length ? messageData.codeReference : undefined
+ }
+ this.onChatAnswerUpdated(messageData.tabID, answer)
+ }
+
+ private updateUI = (messageData: any): void => {
+ if (!this.onMessageReceived) {
+ return
+ }
+
+ const settings: MynahUIDataModel = {
+ ...(messageData.loadingChat !== undefined ? { loadingChat: messageData.loadingChat } : {}),
+ ...(messageData.cancelButtonWhenLoading !== undefined ? { cancelButtonWhenLoading: messageData.cancelButtonWhenLoading } : {}),
+ ...(messageData.promptInputPlaceholder !== undefined ? { promptInputPlaceholder: messageData.promptInputPlaceholder } : {}),
+ ...(messageData.promptInputProgress !== undefined ? { promptInputProgress: messageData.promptInputProgress } : {}),
+ // ...(messageData.promptInputDisabledState !== undefined ? { promptInputDisabledState: messageData.promptInputDisabledState } : {}),
+ }
+
+ console.log("UI settings to be updated")
+ console.log(settings)
+ this.onMessageReceived(messageData.tabID, settings, false)
+ this.chatInputEnabled(messageData.tabID, !messageData.promptInputDisabledState)
+ }
+
+ private processChatMessage = async (messageData: any): Promise => {
+ if (!this.onChatAnswerReceived) {
+ return
+ }
+ if (messageData.message === undefined && !messageData.informationCard) {
+ return
+ }
+ const answer: ChatItem = {
+ type: messageData.messageType,
+ messageId: messageData.messageId || messageData.triggerID,
+ body: messageData.informationCard ? "" : messageData.message,
+ canBeVoted: messageData.canBeVoted,
+ informationCard: messageData.informationCard ? getIntroductionCardContent() : undefined,
+ footer: messageData.filePath
+ ? {
+ fileList: {
+ rootFolderTitle: undefined,
+ fileTreeTitle: '',
+ filePaths: [messageData.filePaths],
+ },
+ }
+ : undefined,
+ }
+ this.onChatAnswerReceived(messageData.tabID, answer)
+ }
+
+ private processAuthNeededException = async (messageData: any): Promise => {
+ if (this.onChatAnswerReceived === undefined) {
+ return
+ }
+ this.onChatAnswerReceived(
+ messageData.tabID,
+ {
+ type: ChatItemType.SYSTEM_PROMPT,
+ body: messageData.message,
+ }
+ )
+ }
+
+ private processCodeResultMessage = async (messageData: any): Promise => {
+ if (this.onChatAnswerReceived !== undefined) {
+ const answer: ChatItem = {
+ type: ChatItemType.ANSWER,
+ canBeVoted: true,
+ messageId: messageData.uploadId,
+ followUp: {
+ text: '',
+ options: messageData.followUps,
+ },
+ fileList: {
+ fileTreeTitle: 'READY FOR REVIEW',
+ rootFolderTitle: messageData.projectName,
+ filePaths: messageData.filePaths,
+ },
+ body: messageData.message,
+ }
+ this.onChatAnswerReceived(messageData.tabID, answer)
+ }
+ }
+
+
+ private processChatAIPromptMessage = async (messageData: any): Promise => {
+ if (this.onChatAnswerReceived === undefined) {
+ return
+ }
+
+ if (messageData.message !== undefined) {
+ const answer: ChatItem = {
+ type: messageData.messageType,
+ messageId: messageData.messageId ?? messageData.triggerID,
+ body: messageData.message,
+ relatedContent: undefined,
+ snapToTop: messageData.snapToTop,
+ canBeVoted: false,
+ }
+
+ this.onChatAnswerReceived(messageData.tabID, answer)
+ }
+ }
+
+ private processChatSummaryMessage = async (messageData: any): Promise => {
+ if (this.onChatAnswerUpdated === undefined) {
+ return
+ }
+ if (messageData.message !== undefined) {
+ const answer: ChatItem = {
+ type: messageData.messageType,
+ messageId: messageData.messageId ?? messageData.triggerID,
+ body: messageData.message,
+ buttons: messageData.buttons ?? [],
+ canBeVoted: true,
+ footer: {
+ fileList: {
+ rootFolderTitle: undefined,
+ fileTreeTitle: '',
+ filePaths: [messageData.filePath],
+ },
+ },
+ }
+ this.onChatAnswerUpdated(messageData.tabID, answer)
+ }
+ }
+
+ handleMessageReceive = async (messageData: any): Promise => {
+ // TODO: Implement the logic to handle received messages for Unit Test generator chat
+ switch(messageData.type){
+ case 'authNeededException':
+ await this.processAuthNeededException(messageData)
+ break
+ case 'authenticationUpdateMessage':
+ this.onUpdateAuthentication(
+ messageData.featureDevEnabled,
+ messageData.codeTransformEnabled,
+ messageData.docEnabled,
+ messageData.codeScanEnabled,
+ messageData.codeTestEnabled,
+ messageData.authenticatingTabIDs
+ )
+ break
+ case 'chatInputEnabledMessage':
+ this.chatInputEnabled(messageData.tabID, messageData.enabled)
+ break
+ case 'updatePromptProgress':
+ this.updatePromptProgress(messageData.tabID, messageData.progressField)
+ break
+ case 'chatMessage':
+ await this.processChatMessage(messageData)
+ break
+ case 'addAnswer':
+ this.addAnswer(messageData)
+ break
+ case 'updateAnswer':
+ this.updateAnswer(messageData)
+ break
+ case 'chatAIPromptMessage':
+ await this.processChatAIPromptMessage(messageData)
+ break
+ case 'chatSummaryMessage':
+ await this.processChatSummaryMessage(messageData)
+ break
+ case 'updatePlaceholderMessage':
+ this.updatePlaceholder(messageData.tabID, messageData.newPlaceholder)
+ break
+ case 'codeResultMessage':
+ await this.processCodeResultMessage(messageData)
+ break
+ case 'errorMessage':
+ this.onError(messageData.tabID, messageData.message, messageData.title)
+ break
+ case 'updateUI':
+ this.updateUI(messageData)
+ break
+ }
+ }
+
+ onFormButtonClick = (
+ tabID: string,
+ messageId: string,
+ action: {
+ id: string
+ text?: string
+ formItemValues?: Record
+ }
+ ) => {
+ if (action === undefined) {
+ return
+ }
+
+ this.sendMessageToExtension({
+ command: 'button-click',
+ actionID: action.id,
+ formSelectedValues: action.formItemValues,
+ tabType: 'codetest',
+ tabID: tabID,
+ })
+
+ if (this.onChatAnswerUpdated === undefined) {
+ return
+ }
+
+ const answer: ChatItem = {
+ type: ChatItemType.ANSWER,
+ messageId: messageId,
+ buttons: []
+ };
+
+ switch (action.id) {
+ case FormButtonIds.CodeTestViewDiff:
+ // does nothing
+ break
+ case FormButtonIds.CodeTestAccept:
+ answer.buttons = [
+ {
+ keepCardAfterClick: true,
+ text: 'Accepted',
+ id: 'utg_accepted',
+ status: 'success',
+ position: 'outside',
+ disabled: true
+ }
+ ];
+ break;
+ case FormButtonIds.CodeTestReject:
+ answer.buttons = [
+ {
+ keepCardAfterClick: true,
+ text: 'Rejected',
+ id: 'utg_rejected',
+ status: 'error',
+ position: 'outside',
+ disabled: true
+ }
+ ];
+ break;
+ case FormButtonIds.CodeTestBuildAndExecute:
+ answer.buttons = [
+ {
+ keepCardAfterClick: true,
+ text: 'Build and execute',
+ id: 'utg_build_and_execute',
+ status: 'primary',
+ position: 'outside',
+ disabled: true
+ }
+ ]
+ break;
+ case FormButtonIds.CodeTestSkipAndFinish:
+ answer.buttons = [
+ {
+ keepCardAfterClick: true,
+ text: 'Skip and finish',
+ id: 'utg_skip_and_finish',
+ status: 'primary',
+ position: 'outside',
+ disabled: true
+ }
+ ]
+ break;
+ /*
+ //TODO: generate button
+ case FormButtonIds.CodeTestRegenerate:
+ answer.buttons = [
+ {
+ keepCardAfterClick: true,
+ text: 'Regenerate',
+ id: 'utg_regenerate',
+ status: 'info',
+ position: 'outside',
+ disabled: true
+ }
+ ]
+ break;
+ */
+ case FormButtonIds.CodeTestRejectAndRevert:
+ // TODO: what behavior should this be?
+ break;
+ case FormButtonIds.CodeTestProceed:
+ answer.buttons = [
+ {
+ keepCardAfterClick: true,
+ text: 'Proceeded',
+ id: 'utg_proceeded',
+ status: 'primary',
+ position: 'outside',
+ disabled: true
+ }
+ ]
+ break;
+ case FormButtonIds.CodeTestModifyCommand:
+ answer.buttons = [
+ {
+ keepCardAfterClick: true,
+ text: 'Modify command',
+ id: 'utg_modify_command',
+ status: 'primary',
+ position: 'outside',
+ disabled: true
+ }
+ ]
+ break;
+ default:
+ console.warn(`Unhandled action ID: ${action.id}`);
+ break;
+ }
+
+ this.onChatAnswerUpdated(tabID, answer);
+ }
+
+ clearChat = (tabID: string): void => {
+ this.sendMessageToExtension({
+ tabID: tabID,
+ command: 'clear',
+ chatMessage: '',
+ tabType: 'codetest',
+ })
+ }
+
+ help = (tabID: string): void => {
+ this.sendMessageToExtension({
+ tabID: tabID,
+ command: 'help',
+ chatMessage: '',
+ tabType: 'codetest',
+ })
+ }
+
+ onTabOpen = (tabID: string) => {
+ this.sendMessageToExtension({
+ tabID,
+ command: 'new-tab-was-created',
+ tabType: 'codetest'
+ })
+ }
+
+ /**
+ * Ignore for this pr, this request Answer function is used to in the future to receive users' input
+ */
+ requestAnswer = (tabID: string, payload: ChatPayload) => {
+ this.sendMessageToExtension({
+ tabID,
+ command: 'chat-prompt',
+ chatMessage: payload.chatMessage,
+ tabType: 'codetest'
+ })
+ }
+
+ onTabRemove = (tabID: string) => {
+ this.sendMessageToExtension({
+ tabID,
+ command: 'tab-was-removed',
+ tabType: 'codetest'
+ })
+ }
+
+ startTestGen = (tabID: string, prompt: string): void => {
+ console.log("calling generate-test here")
+ this.sendMessageToExtension({
+ tabID: tabID,
+ command: 'start-test-gen',
+ prompt,
+ tabType: 'codetest'
+ })
+ }
+
+ onCodeInsertToCursorPosition = (tabID: string, code?: string, type?: 'selection' | 'block', codeReference?: CodeReference[]): void => {
+ this.sendMessageToExtension({
+ tabID: tabID,
+ code,
+ command: 'insert_code_at_cursor_position',
+ codeReference,
+ tabType: 'codetest'
+ })
+ }
+
+ onFileClick = (tabID: string, filePath: string, deleted: boolean, messageId?: string): void => {
+ this.sendMessageToExtension({
+ command: 'open-diff',
+ tabID,
+ filePath,
+ deleted,
+ messageId,
+ tabType: 'codetest',
+ })
+ }
+
+ followUpClicked = (tabID: string, followUp: ChatItemAction): void => {
+ this.sendMessageToExtension({
+ command: 'follow-up-was-clicked',
+ followUp,
+ tabID,
+ tabType: 'codetest',
+ })
+ }
+
+ onResponseBodyLinkClick = (tabID: string, messageId: string, link: string): void => {
+ this.sendMessageToExtension({
+ command: 'response-body-link-click',
+ tabID,
+ messageId,
+ link,
+ tabType: 'codetest',
+ })
+ }
+
+}
diff --git a/plugins/amazonq/mynah-ui/src/mynah-ui/ui/apps/codeTransformChatConnector.ts b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/apps/codeTransformChatConnector.ts
index b335a3d6fb..4a9dc9a141 100644
--- a/plugins/amazonq/mynah-ui/src/mynah-ui/ui/apps/codeTransformChatConnector.ts
+++ b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/apps/codeTransformChatConnector.ts
@@ -20,7 +20,14 @@ export interface ICodeTransformChatConnectorProps {
onUpdatePlaceholder: (tabID: string, newPlaceholder: string) => void
onNotification: (props: {content: string; title?: string; type: NotificationType}) => void
onStartNewTransform: (tabID: string) => void
- onUpdateAuthentication: (featureDevEnabled: boolean, codeTransformEnabled: boolean, authenticatingTabIDs: string[]) => void
+ onUpdateAuthentication: (
+ featureDevEnabled: boolean,
+ codeTransformEnabled: boolean,
+ docEnabled: boolean,
+ codeScanEnabled: boolean,
+ codeTestEnabled: boolean,
+ authenticatingTabIDs: string[]
+ ) => void
tabsStorage: TabsStorage
onNewTab: (tabType: TabType) => void
}
@@ -201,7 +208,14 @@ export class CodeTransformChatConnector {
}
if (messageData.type === 'authenticationUpdateMessage') {
- this.onUpdateAuthentication(messageData.featureDevEnabled, messageData.codeTransformEnabled, messageData.authenticatingTabIDs)
+ this.onUpdateAuthentication(
+ messageData.featureDevEnabled,
+ messageData.codeTransformEnabled,
+ messageData.docEnabled,
+ messageData.codeScanEnabled,
+ messageData.codeTestEnabled,
+ messageData.authenticatingTabIDs
+ )
return
}
diff --git a/plugins/amazonq/mynah-ui/src/mynah-ui/ui/apps/docChatConnector.ts b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/apps/docChatConnector.ts
new file mode 100644
index 0000000000..e92421213d
--- /dev/null
+++ b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/apps/docChatConnector.ts
@@ -0,0 +1,395 @@
+/*!
+ * Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+import { ChatItem, ChatItemAction, ChatItemType, FeedbackPayload, MynahIcons, ProgressField } from '@aws/mynah-ui-chat'
+import { ExtensionMessage } from '../commands'
+import { TabType, TabsStorage } from '../storages/tabsStorage'
+import { CodeReference } from './amazonqCommonsConnector'
+import { FollowUpGenerator } from '../followUps/generator'
+import { getActions } from '../diffTree/actions'
+import { DiffTreeFileInfo } from '../diffTree/types'
+
+interface ChatPayload {
+ chatMessage: string
+}
+
+export interface ConnectorProps {
+ sendMessageToExtension: (message: ExtensionMessage) => void
+ onMessageReceived?: (tabID: string, messageData: any, needToShowAPIDocsTab: boolean) => void
+ onUpdatePromptProgress: (tabID: string, progressField: ProgressField) => void
+ onAsyncEventProgress: (tabID: string, inProgress: boolean, message: string, cancelButtonWhenLoading?: boolean) => void
+ onChatAnswerReceived?: (tabID: string, message: ChatItem) => void
+ sendFeedback?: (tabId: string, feedbackPayload: FeedbackPayload) => void | undefined
+ onError: (tabID: string, message: string, title: string) => void
+ onWarning: (tabID: string, message: string, title: string) => void
+ onUpdatePlaceholder: (tabID: string, newPlaceholder: string) => void
+ onChatInputEnabled: (tabID: string, enabled: boolean) => void
+ onUpdateAuthentication: (
+ featureDevEnabled: boolean,
+ codeTransformEnabled: boolean,
+ docEnabled: boolean,
+ codeScanEnabled: boolean,
+ codeTestEnabled: boolean,
+ authenticatingTabIDs: string[]
+ ) => void
+ onNewTab: (tabType: TabType) => void
+ tabsStorage: TabsStorage
+ onFileComponentUpdate: (
+ tabID: string,
+ filePaths: DiffTreeFileInfo[],
+ deletedFiles: DiffTreeFileInfo[],
+ messageId: string
+ ) => void
+}
+
+export class Connector {
+ private readonly sendMessageToExtension
+ private readonly onError
+ private readonly onWarning
+ private readonly onChatAnswerReceived
+ private readonly onUpdatePromptProgress
+ private readonly onAsyncEventProgress
+ private readonly updatePlaceholder
+ private readonly chatInputEnabled
+ private readonly onUpdateAuthentication
+ private readonly followUpGenerator: FollowUpGenerator
+ private readonly onNewTab
+ private readonly onFileComponentUpdate
+
+ constructor(props: ConnectorProps) {
+ this.sendMessageToExtension = props.sendMessageToExtension
+ this.onChatAnswerReceived = props.onChatAnswerReceived
+ this.onWarning = props.onWarning
+ this.onError = props.onError
+ this.onUpdatePromptProgress = props.onUpdatePromptProgress
+ this.onAsyncEventProgress = props.onAsyncEventProgress
+ this.updatePlaceholder = props.onUpdatePlaceholder
+ this.chatInputEnabled = props.onChatInputEnabled
+ this.onUpdateAuthentication = props.onUpdateAuthentication
+ this.followUpGenerator = new FollowUpGenerator()
+ this.onNewTab = props.onNewTab
+ this.onFileComponentUpdate = props.onFileComponentUpdate
+ }
+
+ onCodeInsertToCursorPosition = (
+ tabID: string,
+ code?: string,
+ type?: 'selection' | 'block',
+ codeReference?: CodeReference[]
+ ): void => {
+ this.sendMessageToExtension({
+ tabID: tabID,
+ code,
+ command: 'insert_code_at_cursor_position',
+ codeReference,
+ tabType: 'doc',
+ })
+ }
+
+ onCopyCodeToClipboard = (
+ tabID: string,
+ code?: string,
+ type?: 'selection' | 'block',
+ codeReference?: CodeReference[]
+ ): void => {
+ this.sendMessageToExtension({
+ tabID: tabID,
+ code,
+ command: 'code_was_copied_to_clipboard',
+ codeReference,
+ tabType: 'doc',
+ })
+ }
+
+ onOpenDiff = (tabID: string, filePath: string, deleted: boolean): void => {
+ this.sendMessageToExtension({
+ command: 'open-diff',
+ tabID,
+ filePath,
+ deleted,
+ tabType: 'doc',
+ })
+ }
+
+ followUpClicked = (tabID: string, followUp: ChatItemAction): void => {
+ this.sendMessageToExtension({
+ command: 'follow-up-was-clicked',
+ followUp,
+ tabID,
+ tabType: 'doc',
+ })
+ }
+
+ requestGenerativeAIAnswer = (tabID: string, payload: ChatPayload): Promise =>
+ new Promise((resolve, reject) => {
+ const message: ExtensionMessage = {
+ tabID: tabID,
+ command: 'chat-prompt',
+ chatMessage: payload.chatMessage,
+ tabType: 'doc',
+ }
+ this.sendMessageToExtension(message)
+ })
+
+ onFileActionClick = (tabID: string, messageId: string, filePath: string, actionName: string): void => {
+ this.sendMessageToExtension({
+ command: 'file-click',
+ tabID,
+ messageId,
+ filePath,
+ actionName,
+ tabType: 'doc',
+ })
+ }
+
+ private processFolderConfirmationMessage = async (messageData: any, folderPath: string): Promise => {
+ if (this.onChatAnswerReceived !== undefined) {
+ const answer: ChatItem = {
+ type: ChatItemType.ANSWER,
+ body: messageData.message ?? undefined,
+ messageId: messageData.messageID ?? messageData.triggerID ?? '',
+ fileList: {
+ rootFolderTitle: undefined,
+ fileTreeTitle: '',
+ filePaths: [folderPath],
+ details: {
+ [folderPath]: {
+ icon: MynahIcons.FOLDER,
+ clickable: false,
+ },
+ },
+ },
+ followUp: {
+ text: '',
+ options: messageData.followUps,
+ },
+ }
+ this.onChatAnswerReceived(messageData.tabID, answer)
+ }
+ }
+
+ private processChatMessage = async (messageData: any): Promise => {
+ if (this.onChatAnswerReceived !== undefined) {
+ const answer: ChatItem = {
+ type: messageData.messageType,
+ body: messageData.message ?? undefined,
+ messageId: messageData.messageID ?? messageData.triggerID ?? '',
+ relatedContent: undefined,
+ canBeVoted: messageData.canBeVoted,
+ snapToTop: messageData.snapToTop,
+ followUp:
+ messageData.followUps !== undefined && messageData.followUps.length > 0
+ ? {
+ text:
+ messageData.messageType === ChatItemType.SYSTEM_PROMPT
+ ? ''
+ : 'Please follow up with one of these',
+ options: messageData.followUps,
+ }
+ : undefined,
+ }
+ this.onChatAnswerReceived(messageData.tabID, answer)
+ }
+ }
+
+ private processCodeResultMessage = async (messageData: any): Promise => {
+ if (this.onChatAnswerReceived !== undefined) {
+ const answer: ChatItem = {
+ type: ChatItemType.ANSWER,
+ relatedContent: undefined,
+ followUp: undefined,
+ canBeVoted: false,
+ codeReference: messageData.references,
+ // TODO get the backend to store a message id in addition to conversationID
+ messageId: messageData.messageID ?? messageData.triggerID ?? messageData.conversationID,
+ fileList: {
+ fileTreeTitle: 'Documents ready',
+ rootFolderTitle: 'Generated documentation',
+ filePaths: (messageData.filePaths as DiffTreeFileInfo[]).map(path => path.zipFilePath),
+ deletedFiles: (messageData.deletedFiles as DiffTreeFileInfo[]).map(path => path.zipFilePath)
+ },
+ body: '',
+ }
+ this.onChatAnswerReceived(messageData.tabID, answer)
+ }
+ }
+
+ private processAuthNeededException = async (messageData: any): Promise => {
+ if (this.onChatAnswerReceived === undefined) {
+ return
+ }
+
+ this.onChatAnswerReceived(messageData.tabID, {
+ type: ChatItemType.ANSWER,
+ body: messageData.message,
+ followUp: undefined,
+ canBeVoted: false,
+ })
+
+ this.onChatAnswerReceived(messageData.tabID, {
+ type: ChatItemType.SYSTEM_PROMPT,
+ body: undefined,
+ followUp: this.followUpGenerator.generateAuthFollowUps('doc', messageData.authType),
+ canBeVoted: false,
+ })
+
+ return
+ }
+
+ handleMessageReceive = async (messageData: any): Promise => {
+ if (messageData.type === 'updateFileComponent') {
+ this.onFileComponentUpdate(
+ messageData.tabID,
+ messageData.filePaths,
+ messageData.deletedFiles,
+ messageData.messageId
+ )
+ return
+ }
+ if (messageData.type === 'errorMessage') {
+ this.onError(messageData.tabID, messageData.message, messageData.title)
+ return
+ }
+
+ if (messageData.type === 'showInvalidTokenNotification') {
+ this.onWarning(messageData.tabID, messageData.message, messageData.title)
+ return
+ }
+
+ if (messageData.type === 'folderConfirmationMessage') {
+ await this.processFolderConfirmationMessage(messageData, messageData.folderPath)
+ return
+ }
+
+ if (messageData.type === 'chatMessage') {
+ await this.processChatMessage(messageData)
+ return
+ }
+
+ if (messageData.type === 'codeResultMessage') {
+ await this.processCodeResultMessage(messageData)
+ return
+ }
+
+ if (messageData.type === 'asyncEventProgressMessage') {
+ this.onAsyncEventProgress(messageData.tabID, messageData.inProgress, messageData.message ?? undefined)
+ return
+ }
+
+ if (messageData.type === 'updatePlaceholderMessage') {
+ this.updatePlaceholder(messageData.tabID, messageData.newPlaceholder)
+ return
+ }
+
+ if (messageData.type === 'chatInputEnabledMessage') {
+ this.chatInputEnabled(messageData.tabID, messageData.enabled)
+ return
+ }
+
+ if (messageData.type === 'authenticationUpdateMessage') {
+ this.onUpdateAuthentication(
+ messageData.featureDevEnabled,
+ messageData.codeTransformEnabled,
+ messageData.docEnabled,
+ messageData.codeScanEnabled,
+ messageData.codeTestEnabled,
+ messageData.authenticatingTabIDs
+ )
+ return
+ }
+
+ if (messageData.type === 'authNeededException') {
+ this.processAuthNeededException(messageData)
+ return
+ }
+
+ if (messageData.type === 'openNewTabMessage') {
+ this.onNewTab('doc')
+ return
+ }
+
+ if (messageData.type === 'updatePromptProgress') {
+ this.onUpdatePromptProgress(messageData.tabId, messageData.progressField)
+ }
+ }
+
+ onStopChatResponse = (tabID: string): void => {
+ this.sendMessageToExtension({
+ tabID: tabID,
+ command: 'stop-response',
+ })
+ }
+
+ onTabOpen = (tabID: string): void => {
+ this.sendMessageToExtension({
+ tabID,
+ command: 'new-tab-was-created',
+ tabType: 'doc',
+ })
+ }
+
+ onTabRemove = (tabID: string): void => {
+ this.sendMessageToExtension({
+ tabID: tabID,
+ command: 'tab-was-removed',
+ tabType: 'doc',
+ })
+ }
+
+ sendFeedback = (tabId: string, feedbackPayload: FeedbackPayload): void | undefined => {
+ this.sendMessageToExtension({
+ command: 'chat-item-feedback',
+ ...feedbackPayload,
+ tabType: 'doc',
+ tabID: tabId,
+ })
+ }
+
+ onChatItemVoted = (tabId: string, messageId: string, vote: string): void | undefined => {
+ this.sendMessageToExtension({
+ tabID: tabId,
+ messageId: messageId,
+ vote: vote,
+ command: 'chat-item-voted',
+ tabType: 'doc',
+ })
+ }
+
+ onResponseBodyLinkClick = (tabID: string, messageId: string, link: string): void => {
+ this.sendMessageToExtension({
+ command: 'response-body-link-click',
+ tabID,
+ messageId,
+ link,
+ tabType: 'doc',
+ })
+ }
+
+ sendFolderConfirmationMessage = (tabID: string, messageId: string): void => {
+ this.sendMessageToExtension({
+ command: 'folderConfirmationMessage',
+ tabID,
+ messageId,
+ tabType: 'doc',
+ })
+ }
+
+ onFormButtonClick = (
+ tabID: string,
+ action: {
+ id: string
+ text?: string
+ formItemValues?: Record
+ }
+ ) => {
+ if (action.id === "doc_stop_generate") {
+ this.sendMessageToExtension({
+ command: 'doc_stop_generate',
+ tabID,
+ tabType: 'doc',
+ })
+ }
+ }
+}
diff --git a/plugins/amazonq/mynah-ui/src/mynah-ui/ui/apps/featureDevChatConnector.ts b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/apps/featureDevChatConnector.ts
index bfef0844e0..3b1e44c5bf 100644
--- a/plugins/amazonq/mynah-ui/src/mynah-ui/ui/apps/featureDevChatConnector.ts
+++ b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/apps/featureDevChatConnector.ts
@@ -18,7 +18,7 @@ interface ChatPayload {
export interface ConnectorProps {
sendMessageToExtension: (message: ExtensionMessage) => void
onMessageReceived?: (tabID: string, messageData: any, needToShowAPIDocsTab: boolean) => void
- onAsyncEventProgress: (tabID: string, inProgress: boolean, message: string) => void
+ onAsyncEventProgress: (tabID: string, inProgress: boolean, message: string, cancelButtonWhenLoading?: boolean) => void
onChatAnswerReceived?: (tabID: string, message: ChatItem) => void
onChatAnswerUpdated?: (tabID: string, message: ChatItem) => void
sendFeedback?: (tabId: string, feedbackPayload: FeedbackPayload) => void | undefined
@@ -26,7 +26,14 @@ export interface ConnectorProps {
onWarning: (tabID: string, message: string, title: string) => void
onUpdatePlaceholder: (tabID: string, newPlaceholder: string) => void
onChatInputEnabled: (tabID: string, enabled: boolean) => void
- onUpdateAuthentication: (featureDevEnabled: boolean, codeTransformEnabled: boolean, authenticatingTabIDs: string[]) => void
+ onUpdateAuthentication: (
+ featureDevEnabled: boolean,
+ codeTransformEnabled: boolean,
+ docEnabled: boolean,
+ codeScanEnabled: boolean,
+ codeTestEnabled: boolean,
+ authenticatingTabIDs: string[]
+ ) => void
onNewTab: (tabType: TabType) => void
tabsStorage: TabsStorage
onFileComponentUpdate: (tabID: string, filePaths: DiffTreeFileInfo[], deletedFiles: DiffTreeFileInfo[], messageId: string, disableFileActions: boolean) => void
@@ -196,6 +203,7 @@ export class Connector {
relatedContent: undefined,
canBeVoted: messageData.canBeVoted ?? undefined,
snapToTop: messageData.snapToTop ?? undefined,
+ informationCard: messageData.informationCard ?? undefined,
followUp:
messageData.followUps !== undefined && Array.isArray(messageData.followUps)
? {
@@ -241,7 +249,7 @@ export class Connector {
}
if (messageData.type === 'asyncEventProgressMessage') {
- this.onAsyncEventProgress(messageData.tabID, messageData.inProgress, messageData.message ?? undefined)
+ this.onAsyncEventProgress(messageData.tabID, messageData.inProgress, messageData.message ?? undefined, true)
return
}
@@ -256,7 +264,14 @@ export class Connector {
}
if (messageData.type === 'authenticationUpdateMessage') {
- this.onUpdateAuthentication(messageData.featureDevEnabled, messageData.codeTransformEnabled, messageData.authenticatingTabIDs)
+ this.onUpdateAuthentication(
+ messageData.featureDevEnabled,
+ messageData.codeTransformEnabled,
+ messageData.docEnabled,
+ messageData.codeScanEnabled,
+ messageData.codeTestEnabled,
+ messageData.authenticatingTabIDs
+ )
return
}
diff --git a/plugins/amazonq/mynah-ui/src/mynah-ui/ui/commands.ts b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/commands.ts
index f55f1b66d5..5df19981b3 100644
--- a/plugins/amazonq/mynah-ui/src/mynah-ui/ui/commands.ts
+++ b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/commands.ts
@@ -12,6 +12,7 @@ type MessageCommand =
| 'tab-was-removed'
| 'tab-was-changed'
| 'ui-is-ready'
+ | 'disclaimer-acknowledged'
| 'ui-focus'
| 'follow-up-was-clicked'
| 'auth-follow-up-was-clicked'
@@ -48,7 +49,21 @@ type MessageCommand =
| 'codetransform-pom-file-open-click'
| 'file-click'
| 'open-settings'
+ | 'button-click'
| 'store-code-result-message-id'
+ | 'folderConfirmationMessage'
+ | 'scan'
+ | 'codescan_start_project_scan'
+ | 'codescan_start_file_scan'
+ | 'codescan_stop_project_scan'
+ | 'codescan_stop_file_scan'
+ | 'codescan_open_issues'
+ | 'generate-test'
+ | 'start-test-gen'
+ | 'open-user-guide'
+ | 'send-telemetry'
+ | 'doc_stop_generate'
+ | 'updatePromptProgress'
export type ExtensionMessage = Record & { command: MessageCommand }
diff --git a/plugins/amazonq/mynah-ui/src/mynah-ui/ui/connector.ts b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/connector.ts
index e51291b0c8..e3734b449a 100644
--- a/plugins/amazonq/mynah-ui/src/mynah-ui/ui/connector.ts
+++ b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/connector.ts
@@ -3,17 +3,28 @@
* SPDX-License-Identifier: Apache-2.0
*/
-import { ChatItem, ChatItemAction, FeedbackPayload, Engagement, NotificationType } from '@aws/mynah-ui-chat'
+import {
+ ChatItem,
+ ChatItemAction,
+ FeedbackPayload,
+ Engagement,
+ NotificationType,
+ ProgressField,
+ ChatPrompt,
+} from '@aws/mynah-ui-chat'
import { Connector as CWChatConnector } from './apps/cwChatConnector'
import { Connector as FeatureDevChatConnector } from './apps/featureDevChatConnector'
+import { Connector as DocChatConnector } from './apps/docChatConnector'
import { Connector as AmazonQCommonsConnector } from './apps/amazonqCommonsConnector'
import { ExtensionMessage } from './commands'
import { TabType, TabsStorage } from './storages/tabsStorage'
import { WelcomeFollowupType } from './apps/amazonqCommonsConnector'
import { AuthFollowUpType } from './followUps/generator'
import { CodeTransformChatConnector } from './apps/codeTransformChatConnector'
-import { isFormButtonCodeTransform } from './forms/constants'
+import { isFormButtonCodeTest, isFormButtonCodeScan, isFormButtonCodeTransform } from './forms/constants'
import { DiffTreeFileInfo } from './diffTree/types'
+import { CodeScanChatConnector } from "./apps/codeScanChatConnector";
+import { CodeTestChatConnector } from './apps/codeTestChatConnector'
export interface CodeReference {
licenseName?: string
@@ -39,6 +50,7 @@ export interface ConnectorProps {
sendMessageToExtension: (message: ExtensionMessage) => void
onMessageReceived?: (tabID: string, messageData: any, needToShowAPIDocsTab: boolean) => void
onChatAnswerReceived?: (tabID: string, message: ChatItem) => void
+ onChatAnswerUpdated?: (tabID: string, message:ChatItem) => void
onCodeTransformChatDisabled: (tabID: string) => void
onCodeTransformMessageReceived: (
tabID: string,
@@ -47,8 +59,9 @@ export interface ConnectorProps {
clearPreviousItemButtons?: boolean
) => void
onCodeTransformMessageUpdate: (tabID: string, messageId: string, chatItem: Partial) => void
+ onRunTestMessageReceived?: (tabID: string, showRunTestMessage: boolean) => void
onWelcomeFollowUpClicked: (tabID: string, welcomeFollowUpType: WelcomeFollowupType) => void
- onAsyncEventProgress: (tabID: string, inProgress: boolean, message: string | undefined) => void
+ onAsyncEventProgress: (tabID: string, inProgress: boolean, message: string | undefined, cancelButtonWhenLoading?: boolean) => void
onCWCContextCommandMessage: (message: ChatItem, command?: string) => string | undefined
onCWCOnboardingPageInteractionMessage: (message: ChatItem) => string | undefined
onOpenSettingsMessage: (tabID: string) => void
@@ -58,14 +71,17 @@ export interface ConnectorProps {
tabID: string,
filePaths: DiffTreeFileInfo[],
deletedFiles: DiffTreeFileInfo[],
- messageId: string,
- disableFileActions: boolean
+ messageId: string
) => void
onUpdatePlaceholder: (tabID: string, newPlaceholder: string) => void
+ onUpdatePromptProgress: (tabID: string, progressField: ProgressField | null | undefined) => void
onChatInputEnabled: (tabID: string, enabled: boolean) => void
onUpdateAuthentication: (
featureDevEnabled: boolean,
codeTransformEnabled: boolean,
+ docEnabled: boolean,
+ codeScanEnabled: boolean,
+ codeTestEnabled: boolean,
authenticatingTabIDs: string[]
) => void
onNewTab: (tabType: TabType) => void
@@ -73,7 +89,9 @@ export interface ConnectorProps {
onCodeTransformCommandMessageReceived: (message: ChatItem, command?: string) => void
onNotification: (props: { content: string; title?: string; type: NotificationType }) => void
onFileActionClick: (tabID: string, messageId: string, filePath: string, actionName: string) => void
+ handleCommand: (chatPrompt: ChatPrompt, tabId: string) => void
tabsStorage: TabsStorage
+ onCodeScanMessageReceived: (tabID: string, message: ChatItem, isLoading: boolean, clearPreviousItemButtons?: boolean) => void
}
export class Connector {
@@ -82,6 +100,9 @@ export class Connector {
private readonly cwChatConnector
private readonly featureDevChatConnector
private readonly codeTransformChatConnector: CodeTransformChatConnector
+ private readonly docChatConnector
+ private readonly codeScanChatConnector: CodeScanChatConnector
+ private readonly codeTestChatConnector: CodeTestChatConnector
private readonly tabsStorage
private readonly amazonqCommonsConnector: AmazonQCommonsConnector
@@ -93,9 +114,13 @@ export class Connector {
this.cwChatConnector = new CWChatConnector(props as ConnectorProps)
this.featureDevChatConnector = new FeatureDevChatConnector(props)
this.codeTransformChatConnector = new CodeTransformChatConnector(props)
+ this.docChatConnector = new DocChatConnector(props)
+ this.codeScanChatConnector = new CodeScanChatConnector(props)
+ this.codeTestChatConnector = new CodeTestChatConnector(props)
this.amazonqCommonsConnector = new AmazonQCommonsConnector({
sendMessageToExtension: this.sendMessageToExtension,
onWelcomeFollowUpClicked: props.onWelcomeFollowUpClicked,
+ handleCommand: props.handleCommand,
})
this.tabsStorage = props.tabsStorage
}
@@ -119,6 +144,15 @@ export class Connector {
case 'codetransform':
this.codeTransformChatConnector.onResponseBodyLinkClick(tabID, messageId, link)
break
+ case 'codescan':
+ this.codeScanChatConnector.onResponseBodyLinkClick(tabID, messageId, link)
+ break
+ case 'codetest':
+ this.codeTestChatConnector.onResponseBodyLinkClick(tabID, messageId, link)
+ break
+ case 'doc':
+ this.docChatConnector.onResponseBodyLinkClick(tabID, messageId, link)
+ break
}
}
@@ -134,6 +168,8 @@ export class Connector {
switch (this.tabsStorage.getTab(tabID)?.type) {
case 'codetransform':
return this.codeTransformChatConnector.requestAnswer(tabID, payload)
+ case 'codetest':
+ return this.codeTestChatConnector.requestAnswer(tabID, payload)
}
}
@@ -144,6 +180,9 @@ export class Connector {
case 'featuredev':
this.featureDevChatConnector.requestGenerativeAIAnswer(tabID, payload)
break
+ case 'doc':
+ this.docChatConnector.requestGenerativeAIAnswer(tabID, payload)
+ break
default:
this.cwChatConnector.requestGenerativeAIAnswer(tabID, payload)
break
@@ -156,11 +195,18 @@ export class Connector {
}
})
+ //TODO: Create a common connector to share this options across the features
clearChat = (tabID: string): void => {
switch (this.tabsStorage.getTab(tabID)?.type) {
case 'cwc':
this.cwChatConnector.clearChat(tabID)
break
+ case 'codetest':
+ this.codeTestChatConnector.clearChat(tabID)
+ break
+ case 'codescan':
+ this.codeScanChatConnector.clearChat(tabID)
+ break
}
}
@@ -169,6 +215,12 @@ export class Connector {
case 'cwc':
this.cwChatConnector.help(tabID)
break
+ case 'codetest':
+ this.codeTestChatConnector.help(tabID)
+ break
+ case 'codescan':
+ this.codeScanChatConnector.help(tabID)
+ break
}
}
@@ -180,6 +232,18 @@ export class Connector {
}
}
+ scan = (tabID: string): void => {
+ switch (this.tabsStorage.getTab(tabID)?.type) {
+ default:
+ this.codeScanChatConnector.scan(tabID)
+ break
+ }
+ }
+
+ startTestGen = (tabID: string, prompt: string): void => {
+ this.codeTestChatConnector.startTestGen(tabID, prompt)
+ }
+
handleMessageReceive = async (message: MessageEvent): Promise => {
if (message.data === undefined) {
return
@@ -192,12 +256,27 @@ export class Connector {
return
}
- if (messageData.sender === 'CWChat') {
- void this.cwChatConnector.handleMessageReceive(messageData)
- } else if (messageData.sender === 'featureDevChat') {
- void this.featureDevChatConnector.handleMessageReceive(messageData)
- } else if (messageData.sender === 'codetransform') {
- void this.codeTransformChatConnector.handleMessageReceive(messageData)
+ switch (messageData.sender) {
+ case 'CWChat':
+ void this.cwChatConnector.handleMessageReceive(messageData)
+ break
+ case 'featureDevChat':
+ void this.featureDevChatConnector.handleMessageReceive(messageData)
+ break
+ case 'codetransform':
+ void this.codeTransformChatConnector.handleMessageReceive(messageData)
+ break
+ case 'docChat':
+ void this.docChatConnector.handleMessageReceive(messageData)
+ break
+ case 'codescan':
+ void this.codeScanChatConnector.handleMessageReceive(messageData)
+ break
+ case 'codetest':
+ void this.codeTestChatConnector.handleMessageReceive(messageData)
+ break
+ default:
+ break
}
}
@@ -227,6 +306,15 @@ export class Connector {
case 'codetransform':
this.codeTransformChatConnector.onTabOpen(tabID)
break
+ case 'doc':
+ this.docChatConnector.onTabOpen(tabID)
+ break
+ case 'codescan':
+ this.codeScanChatConnector.onTabOpen(tabID)
+ break
+ case 'codetest':
+ this.codeTestChatConnector.onTabOpen(tabID)
+ break
}
}
@@ -265,6 +353,8 @@ export class Connector {
case 'featuredev':
this.featureDevChatConnector.onCodeInsertToCursorPosition(tabID, code, type, codeReference)
break
+ case 'codetest':
+ this.codeTestChatConnector.onCodeInsertToCursorPosition(tabID, code, type, codeReference)
}
}
@@ -314,6 +404,15 @@ export class Connector {
case 'codetransform':
this.codeTransformChatConnector.onTabRemove(tabID)
break
+ case 'doc':
+ this.docChatConnector.onTabRemove(tabID)
+ break
+ case 'codescan':
+ this.codeScanChatConnector.onTabRemove(tabID)
+ break
+ case 'codetest':
+ this.codeTestChatConnector.onTabRemove(tabID)
+ break
}
}
@@ -363,8 +462,11 @@ export class Connector {
switch (tabType) {
case 'codetransform':
case 'cwc':
+ case 'doc':
case 'featuredev':
+ case 'codetest':
this.amazonqCommonsConnector.authFollowUpClicked(tabID, tabType, authType)
+ break
}
}
@@ -382,6 +484,12 @@ export class Connector {
case 'codetransform':
this.codeTransformChatConnector.followUpClicked(tabID, followUp)
break
+ case 'doc':
+ this.docChatConnector.followUpClicked(tabID, followUp)
+ break
+ case 'codetest':
+ this.codeTestChatConnector.followUpClicked(tabID, followUp)
+ break
default:
this.cwChatConnector.followUpClicked(tabID, messageId, followUp)
break
@@ -393,6 +501,24 @@ export class Connector {
case 'featuredev':
this.featureDevChatConnector.onFileActionClick(tabID, messageId, filePath, actionName)
break
+ case 'doc':
+ this.docChatConnector.onFileActionClick(tabID, messageId, filePath, actionName)
+ break
+ }
+ }
+
+ onFileClick = (tabID: string, filePath: string, deleted: boolean, messageId?: string): void => {
+ switch (this.tabsStorage.getTab(tabID)?.type) {
+ case 'featuredev':
+ this.featureDevChatConnector.onOpenDiff(tabID, filePath, deleted)
+ break
+ /*
+ TODO: This is for temporary solution to show correct viewdiff panel by clicking the filename
+ Would re-factor it later for the next task
+ */
+ case 'codetest':
+ this.codeTestChatConnector.onFormButtonClick(tabID, messageId ?? '', {id: "utg_view_diff"})
+ break
}
}
@@ -401,6 +527,35 @@ export class Connector {
case 'featuredev':
this.featureDevChatConnector.onOpenDiff(tabID, filePath, deleted)
break
+ case 'doc':
+ this.docChatConnector.onOpenDiff(tabID, filePath, deleted)
+ break
+ }
+ }
+
+ onCustomFormAction = (
+ tabId: string,
+ messageId: string | undefined,
+ action: any,
+ eventId: string | undefined = undefined
+ ): void | undefined => {
+ switch (this.tabsStorage.getTab(tabId)?.type) {
+ case 'codescan':
+ this.codeScanChatConnector.onFormButtonClick(tabId, action)
+ break
+ case 'codetest':
+ this.codeTestChatConnector.onFormButtonClick(tabId, messageId ?? '', action)
+ break
+ case 'codetransform':
+ this.codeTransformChatConnector.onFormButtonClick(tabId, action)
+ break
+ case 'doc':
+ this.docChatConnector.onFormButtonClick(tabId, action)
+ break
+ case 'agentWalkthrough': {
+ this.amazonqCommonsConnector.onCustomFormAction(tabId, action)
+ break
+ }
}
}
@@ -448,6 +603,12 @@ export class Connector {
) => {
if (isFormButtonCodeTransform(action.id)) {
this.codeTransformChatConnector.onFormButtonClick(tabId, action)
+ } else if (isFormButtonCodeScan(action.id)) {
+ this.codeScanChatConnector.onFormButtonClick(tabId, action)
+ } else if (isFormButtonCodeTest(action.id)) {
+ this.codeTestChatConnector.onFormButtonClick(tabId, messageId, action)
+ } else if (action.id === 'doc') {
+ this.docChatConnector.onFormButtonClick(tabId, action)
}
switch (this.tabsStorage.getTab(tabId)?.type) {
case 'cwc':
diff --git a/plugins/amazonq/mynah-ui/src/mynah-ui/ui/followUps/generator.ts b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/followUps/generator.ts
index 286c6ec98f..6505e02391 100644
--- a/plugins/amazonq/mynah-ui/src/mynah-ui/ui/followUps/generator.ts
+++ b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/followUps/generator.ts
@@ -52,11 +52,37 @@ export class FollowUpGenerator {
},
],
}
+ case 'doc':
+ return {
+ text: 'Select one of the following...',
+ options: [
+ {
+ pillText: 'Create a README',
+ prompt: 'Create a README',
+ type: 'CreateDocumentation',
+ },
+ {
+ pillText: 'Update an existing README',
+ prompt: 'Update an existing README',
+ type: 'UpdateDocumentation',
+ },
+ ],
+ }
case 'codetransform':
return {
text: '',
options: [],
}
+ case 'codescan':
+ return {
+ text: '',
+ options: []
+ }
+ case 'codetest':
+ return {
+ text: '',
+ options: [],
+ }
default:
return {
text: 'Try Examples:',
diff --git a/plugins/amazonq/mynah-ui/src/mynah-ui/ui/followUps/handler.ts b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/followUps/handler.ts
index 8234672976..55e28eaacc 100644
--- a/plugins/amazonq/mynah-ui/src/mynah-ui/ui/followUps/handler.ts
+++ b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/followUps/handler.ts
@@ -73,6 +73,7 @@ export class FollowUpInteractionHandler {
return
}
}
+
this.connector.onFollowUpClicked(tabID, messageId, followUp)
}
diff --git a/plugins/amazonq/mynah-ui/src/mynah-ui/ui/forms/constants.ts b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/forms/constants.ts
index 2ad5730375..f727647729 100644
--- a/plugins/amazonq/mynah-ui/src/mynah-ui/ui/forms/constants.ts
+++ b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/forms/constants.ts
@@ -19,6 +19,23 @@ export const enum FormButtonIds {
ConfirmHilSelection = 'confirm_hil_selection',
RejectHilSelection = 'reject_hil_selection',
OpenDependencyErrorPom = "open_dependency_error_pom",
+ CodeScanStartProjectScan = "codescan_start_project_scan",
+ CodeScanStartFileScan = "codescan_start_file_scan",
+ CodeScanStopProjectScan = "codescan_stop_project_scan",
+ CodeScanStopFileScan = "codescan_stop_file_scan",
+ CodeScanOpenIssues = "codescan_open_issues",
+ CodeTestStartGeneration = "code_test_start_generation",
+ CodeTestViewDiff = "utg_view_diff",
+ CodeTestAccept = "utg_accept",
+ CodeTestRegenerate = "utg_regenerate",
+ CodeTestReject = "utg_reject",
+ CodeTestBuildAndExecute = "utg_build_and_execute",
+ CodeTestModifyCommand = "utg_modify_command",
+ CodeTestSkipAndFinish = "utg_skip_and_finish",
+ CodeTestInstallAndContinue = "utg_install_and_continue",
+ CodeTestRejectAndRevert = "utg_reject_and_revert",
+ CodeTestProceed = "utg_proceed",
+
}
export const isFormButtonCodeTransform = (id: string): boolean => {
@@ -40,3 +57,19 @@ export const isFormButtonCodeTransform = (id: string): boolean => {
id === FormButtonIds.OpenDependencyErrorPom
)
}
+
+export const isFormButtonCodeTest = (id: string): boolean => {
+ return (
+ id === FormButtonIds.CodeTestStartGeneration || id.startsWith("utg")
+ )
+}
+
+export const isFormButtonCodeScan = (id: string): boolean => {
+ return (
+ id === FormButtonIds.CodeScanStartProjectScan ||
+ id === FormButtonIds.CodeScanStartFileScan ||
+ id === FormButtonIds.CodeScanStopProjectScan ||
+ id === FormButtonIds.CodeScanStopFileScan ||
+ id === FormButtonIds.CodeScanOpenIssues
+ )
+}
diff --git a/plugins/amazonq/mynah-ui/src/mynah-ui/ui/main.ts b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/main.ts
index f7d1cae719..685c496d1a 100644
--- a/plugins/amazonq/mynah-ui/src/mynah-ui/ui/main.ts
+++ b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/main.ts
@@ -3,7 +3,16 @@
* SPDX-License-Identifier: Apache-2.0
*/
import { Connector, CWCChatItem } from './connector'
-import {ChatItem, ChatItemType, MynahIcons, MynahUI, MynahUIDataModel, NotificationType, ReferenceTrackerInformation} from '@aws/mynah-ui-chat'
+import {
+ ChatItem,
+ ChatItemType,
+ MynahIcons,
+ MynahUI,
+ MynahUIDataModel,
+ NotificationType,
+ ProgressField, QuickActionCommand,
+ ReferenceTrackerInformation
+} from '@aws/mynah-ui-chat'
import './styles/dark.scss'
import { TabsStorage, TabType } from './storages/tabsStorage'
import { WelcomeFollowupType } from './apps/amazonqCommonsConnector'
@@ -17,9 +26,25 @@ import { MessageController } from './messages/controller'
import { getActions, getDetails } from './diffTree/actions'
import { DiffTreeFileInfo } from './diffTree/types'
import './styles.css'
-import {CodeSelectionType} from "@aws/mynah-ui-chat/dist/static";
+import { ChatPrompt, CodeSelectionType} from "@aws/mynah-ui-chat/dist/static";
+import {welcomeScreenTabData} from "./walkthrough/welcome";
+import { agentWalkthroughDataModel } from './walkthrough/agent'
+import {createClickTelemetry, createOpenAgentTelemetry} from "./telemetry/actions";
+import {disclaimerAcknowledgeButtonId, disclaimerCard} from "./texts/disclaimer";
+
+export const createMynahUI = (
+ ideApi: any,
+ showWelcomePage: boolean,
+ disclaimerAcknowledged: boolean,
+ featureDevInitEnabled: boolean,
+ codeTransformInitEnabled: boolean,
+ docInitEnabled: boolean,
+ codeScanEnabled: boolean,
+ codeTestEnabled: boolean,
+ highlightCommand?: QuickActionCommand,
+) => {
+ let disclaimerCardActive = !disclaimerAcknowledged
-export const createMynahUI = (ideApi: any, featureDevInitEnabled: boolean, codeTransformInitEnabled: boolean) => {
// eslint-disable-next-line prefer-const
let mynahUI: MynahUI
// eslint-disable-next-line prefer-const
@@ -42,7 +67,7 @@ export const createMynahUI = (ideApi: any, featureDevInitEnabled: boolean, codeT
tabsStorage.addTab({
id: 'tab-1',
status: 'free',
- type: 'cwc',
+ type: showWelcomePage ? 'welcome' : 'cwc',
isSelected: true,
})
@@ -51,9 +76,19 @@ export const createMynahUI = (ideApi: any, featureDevInitEnabled: boolean, codeT
let isCodeTransformEnabled = codeTransformInitEnabled
+ let isDocEnabled = docInitEnabled
+
+ let isCodeScanEnabled = codeScanEnabled
+
+ let isCodeTestEnabled = codeTestEnabled
+
const tabDataGenerator = new TabDataGenerator({
isFeatureDevEnabled,
isCodeTransformEnabled,
+ isDocEnabled,
+ isCodeScanEnabled,
+ isCodeTestEnabled,
+ highlightCommand
})
// eslint-disable-next-line prefer-const
@@ -68,18 +103,36 @@ export const createMynahUI = (ideApi: any, featureDevInitEnabled: boolean, codeT
// eslint-disable-next-line prefer-const
connector = new Connector({
tabsStorage,
+ /**
+ * Proxy for allowing underlying common connectors to call quick action handlers
+ */
+ handleCommand: (chatPrompt: ChatPrompt, tabId: string) => {
+ quickActionHandler.handleCommand(chatPrompt, tabId)
+ },
onUpdateAuthentication: (
featureDevEnabled: boolean,
codeTransformEnabled: boolean,
+ docEnabled: boolean,
+ codeScanEnabled: boolean,
+ codeTestEnabled: boolean,
authenticatingTabIDs: string[]
): void => {
isFeatureDevEnabled = featureDevEnabled
isCodeTransformEnabled = codeTransformEnabled
+ isDocEnabled = docEnabled
+ isCodeScanEnabled = codeScanEnabled
+ isCodeTestEnabled = codeTestEnabled
quickActionHandler.isFeatureDevEnabled = isFeatureDevEnabled
quickActionHandler.isCodeTransformEnabled = isCodeTransformEnabled
+ quickActionHandler.isDocEnabled = isDocEnabled
+ quickActionHandler.isCodeTestEnabled = isCodeTestEnabled
+ quickActionHandler.isCodeScanEnabled = isCodeScanEnabled
tabDataGenerator.quickActionsGenerator.isFeatureDevEnabled = isFeatureDevEnabled
tabDataGenerator.quickActionsGenerator.isCodeTransformEnabled = isCodeTransformEnabled
+ tabDataGenerator.quickActionsGenerator.isDocEnabled = isDocEnabled
+ tabDataGenerator.quickActionsGenerator.isCodeScanEnabled = isCodeScanEnabled
+ tabDataGenerator.quickActionsGenerator.isCodeTestEnabled = isCodeTestEnabled
// Set the new defaults for the quick action commands in all tabs now that isFeatureDevEnabled and isCodeTransformEnabled were enabled/disabled
for (const tab of tabsStorage.getTabs()) {
@@ -93,7 +146,10 @@ export const createMynahUI = (ideApi: any, featureDevInitEnabled: boolean, codeT
const tabType = tabsStorage.getTab(tabID)?.type
if (
(tabType === 'featuredev' && featureDevEnabled) ||
- (tabType === 'codetransform' && codeTransformEnabled)
+ (tabType === 'codetransform' && codeTransformEnabled) ||
+ (tabType === 'doc' && docEnabled) ||
+ (tabType === 'codetransform' && codeTransformEnabled) ||
+ (tabType === 'codetest' && codeTestEnabled)
) {
mynahUI.addChatItem(tabID, {
type: ChatItemType.ANSWER,
@@ -114,7 +170,12 @@ export const createMynahUI = (ideApi: any, featureDevInitEnabled: boolean, codeT
if (command === 'aws.amazonq.sendToPrompt') {
return messageController.sendSelectedCodeToTab(message)
} else {
- return messageController.sendMessageToTab(message, 'cwc')
+ const tabID = messageController.sendMessageToTab(message, 'cwc')
+ if (tabID && command) {
+ ideApi.postMessage(createOpenAgentTelemetry('cwc', 'right-click'))
+ }
+
+ return tabID
}
},
onWelcomeFollowUpClicked: (tabID: string, welcomeFollowUpType: WelcomeFollowupType) => {
@@ -125,12 +186,12 @@ export const createMynahUI = (ideApi: any, featureDevInitEnabled: boolean, codeT
promptInputDisabledState: tabsStorage.isTabDead(tabID) || !enabled,
})
},
- onAsyncEventProgress: (tabID: string, inProgress: boolean, message: string | undefined) => {
+ onAsyncEventProgress: (tabID: string, inProgress: boolean, message: string | undefined, cancelButtonWhenLoading: boolean = false) => {
if (inProgress) {
mynahUI.updateStore(tabID, {
loadingChat: true,
promptInputDisabledState: true,
- cancelButtonWhenLoading: true,
+ cancelButtonWhenLoading,
})
if (message) {
mynahUI.updateLastChatAnswer(tabID, {
@@ -235,7 +296,7 @@ export const createMynahUI = (ideApi: any, featureDevInitEnabled: boolean, codeT
mynahUI.selectTab(codeTransformTab.id, eventId)
} else {
// Click to open a new code transform tab
- quickActionHandler.handle({ command: '/transform' }, '', eventId)
+ quickActionHandler.handleCommand({ command: '/transform' }, '', eventId)
}
},
})
@@ -244,6 +305,27 @@ export const createMynahUI = (ideApi: any, featureDevInitEnabled: boolean, codeT
sendMessageToExtension: message => {
ideApi.postMessage(message)
},
+ onChatAnswerUpdated: (tabID: string, item) => {
+ if (item.messageId !== undefined) {
+ mynahUI.updateChatAnswerWithMessageId(tabID, item.messageId, {
+ ...(item.body !== undefined ? { body: item.body } : {}),
+ ...(item.buttons !== undefined ? { buttons: item.buttons } : {}),
+ ...(item.fileList !== undefined ? { fileList: item.fileList } : {}),
+ ...(item.footer !== undefined ? { footer: item.footer } : {}),
+ ...(item.canBeVoted !== undefined ? { canBeVoted: item.canBeVoted } : {}),
+ ...(item.codeReference !== undefined ? { codeReference: item.codeReference } : {}),
+ })
+ } else {
+ mynahUI.updateLastChatAnswer(tabID, {
+ ...(item.body !== undefined ? { body: item.body } : {}),
+ ...(item.buttons !== undefined ? { buttons: item.buttons } : {}),
+ ...(item.fileList !== undefined ? { fileList: item.fileList } : {}),
+ ...(item.footer !== undefined ? { footer: item.footer } : {}),
+ ...(item.canBeVoted !== undefined ? { canBeVoted: item.canBeVoted } : {}),
+ ...(item.codeReference !== undefined ? { codeReference: item.codeReference } : {}),
+ } as ChatItem)
+ }
+ },
onChatAnswerReceived: (tabID: string, item: CWCChatItem) => {
if (item.type === ChatItemType.ANSWER_PART || item.type === ChatItemType.CODE_RESULT) {
mynahUI.updateLastChatAnswer(tabID, {
@@ -255,6 +337,7 @@ export const createMynahUI = (ideApi: any, featureDevInitEnabled: boolean, codeT
...(item.type === ChatItemType.CODE_RESULT
? { type: ChatItemType.CODE_RESULT, fileList: item.fileList }
: {}),
+ ...(item.codeReference !== undefined ? { codeReference: item.codeReference } : {}),
})
if (item.messageId !== undefined && item.userIntent !== undefined && item.codeBlockLanguage !== undefined) {
responseMetadata.set(item.messageId, [item.userIntent, item.codeBlockLanguage])
@@ -289,6 +372,11 @@ export const createMynahUI = (ideApi: any, featureDevInitEnabled: boolean, codeT
tabsStorage.updateTabStatus(tabID, 'free')
}
},
+ onRunTestMessageReceived: (tabID: string, shouldRunTestMessage: boolean) => {
+ if (shouldRunTestMessage) {
+ quickActionHandler.handleCommand({ command: '/test' }, tabID)
+ }
+ },
onMessageReceived: (tabID: string, messageData: MynahUIDataModel) => {
mynahUI.updateStore(tabID, messageData)
},
@@ -369,6 +457,12 @@ export const createMynahUI = (ideApi: any, featureDevInitEnabled: boolean, codeT
promptInputPlaceholder: newPlaceholder,
})
},
+ onUpdatePromptProgress(tabID: string, progressField: ProgressField | null | undefined) {
+ mynahUI.updateStore(tabID, {
+ // eslint-disable-next-line no-null/no-null
+ promptInputProgress: progressField ? progressField : null,
+ })
+ },
onNewTab(tabType: TabType) {
const newTabID = mynahUI.updateStore('', {})
if (!newTabID) {
@@ -406,15 +500,69 @@ export const createMynahUI = (ideApi: any, featureDevInitEnabled: boolean, codeT
})
return
},
+ onCodeScanMessageReceived(tabID: string, chatItem: ChatItem, isLoading: boolean, clearPreviousItemButtons?: boolean, runReview?: boolean) {
+ if (runReview) {
+ quickActionHandler.handleCommand({ command: "/review" }, "")
+ return
+ }
+ if (chatItem.type === ChatItemType.ANSWER_PART) {
+ mynahUI.updateLastChatAnswer(tabID, {
+ ...(chatItem.messageId !== undefined ? { messageId: chatItem.messageId } : {}),
+ ...(chatItem.canBeVoted !== undefined ? { canBeVoted: chatItem.canBeVoted } : {}),
+ ...(chatItem.codeReference !== undefined ? { codeReference: chatItem.codeReference } : {}),
+ ...(chatItem.body !== undefined ? { body: chatItem.body } : {}),
+ ...(chatItem.relatedContent !== undefined ? { relatedContent: chatItem.relatedContent } : {}),
+ ...(chatItem.formItems !== undefined ? { formItems: chatItem.formItems } : {}),
+ ...(chatItem.buttons !== undefined ? { buttons: chatItem.buttons } : { buttons: [] }),
+ // For loading animation to work, do not update the chat item type
+ ...(chatItem.followUp !== undefined ? { followUp: chatItem.followUp } : {}),
+ })
+
+ if (!isLoading) {
+ mynahUI.updateStore(tabID, {
+ loadingChat: false,
+ })
+ } else {
+ mynahUI.updateStore(tabID, {
+ cancelButtonWhenLoading: false
+ })
+ }
+ }
+
+ if (
+ chatItem.type === ChatItemType.PROMPT ||
+ chatItem.type === ChatItemType.ANSWER_STREAM ||
+ chatItem.type === ChatItemType.ANSWER
+ ) {
+ if (chatItem.followUp === undefined && clearPreviousItemButtons === true) {
+ mynahUI.updateLastChatAnswer(tabID, {
+ buttons: [],
+ followUp: { options: [] },
+ })
+ }
+
+ mynahUI.addChatItem(tabID, chatItem)
+ mynahUI.updateStore(tabID, {
+ loadingChat: chatItem.type !== ChatItemType.ANSWER
+ })
+
+ if (chatItem.type === ChatItemType.PROMPT) {
+ tabsStorage.updateTabStatus(tabID, 'busy')
+ } else if (chatItem.type === ChatItemType.ANSWER) {
+ tabsStorage.updateTabStatus(tabID, 'free')
+ }
+ }
+ }
})
mynahUI = new MynahUI({
onReady: connector.uiReady,
onTabAdd: (tabID: string) => {
- // If featureDev or gumby has changed availability inbetween the default store settings and now
+ // If featureDev or gumby has changed availability in between the default store settings and now
// make sure to show/hide it accordingly
mynahUI.updateStore(tabID, {
quickActionCommands: tabDataGenerator.quickActionsGenerator.generateForTab('unknown'),
+ ...(disclaimerCardActive ? { promptInputStickyCard: disclaimerCard } : {}),
})
connector.onTabAdd(tabID)
},
@@ -441,10 +589,41 @@ export const createMynahUI = (ideApi: any, featureDevInitEnabled: boolean, codeT
chatMessage: prompt.prompt ?? ''
})
return
+ } else if (tabsStorage.getTab(tabID)?.type === 'codetest') {
+ if(prompt.command !== undefined && prompt.command.trim() !== '' && prompt.command !== '/test') {
+ quickActionHandler.handleCommand(prompt, tabID, eventId)
+ return
+ } else {
+ connector.requestAnswer(tabID, {
+ chatMessage: prompt.prompt ?? ''
+ })
+ return
+ }
+ } else if (tabsStorage.getTab(tabID)?.type === 'codescan') {
+ if(prompt.command !== undefined && prompt.command.trim() !== '') {
+ quickActionHandler.handleCommand(prompt, tabID, eventId)
+ return
+ }
+ }
+
+ if (tabsStorage.getTab(tabID)?.type === 'welcome') {
+ mynahUI.updateStore(tabID, {
+ tabHeaderDetails: void 0,
+ compactMode: false,
+ tabBackground: false,
+ promptInputText: '',
+ promptInputLabel: void 0,
+ chatItems: [],
+ })
}
if (prompt.command !== undefined && prompt.command.trim() !== '') {
- quickActionHandler.handle(prompt, tabID, eventId)
+ quickActionHandler.handleCommand(prompt, tabID, eventId)
+
+ const newTabType = tabsStorage.getSelectedTab()?.type
+ if (newTabType) {
+ ideApi.postMessage(createOpenAgentTelemetry(newTabType, 'quick-action'))
+ }
return
}
@@ -511,15 +690,76 @@ export const createMynahUI = (ideApi: any, featureDevInitEnabled: boolean, codeT
onFileActionClick: async (tabID: string, messageId: string, filePath: string, actionName: string) => {
connector.onFileActionClick(tabID, messageId, filePath, actionName)
},
- onOpenDiff: connector.onOpenDiff,
+ onFileClick: connector.onFileClick,
+ onChatPromptProgressActionButtonClicked: (tabID, action) => {
+ connector.onCustomFormAction(tabID, undefined, action)
+ },
tabs: {
'tab-1': {
isSelected: true,
- store: tabDataGenerator.getTabData('cwc', true),
+ store: {
+ ...(showWelcomePage
+ ? welcomeScreenTabData(tabDataGenerator).store
+ : tabDataGenerator.getTabData('cwc', true)),
+ ...(disclaimerCardActive ? { promptInputStickyCard: disclaimerCard } : {}),
+ },
},
},
- onInBodyButtonClicked: (tabId, messageId, action) => {
- connector.onFormButtonClick(tabId, messageId, action)
+ onInBodyButtonClicked: (tabId, messageId, action, eventId) => {
+ if (action.id === disclaimerAcknowledgeButtonId) {
+ disclaimerCardActive = false
+ // post message to tell IDE that disclaimer is acknowledged
+ ideApi.postMessage({
+ command: 'disclaimer-acknowledged',
+ })
+
+ // create telemetry
+ ideApi.postMessage(createClickTelemetry('amazonq-disclaimer-acknowledge-button'))
+
+ // remove all disclaimer cards from all tabs
+ Object.keys(mynahUI.getAllTabs()).forEach((storeTabKey) => {
+ // eslint-disable-next-line no-null/no-null
+ mynahUI.updateStore(storeTabKey, { promptInputStickyCard: null })
+ })
+ }
+
+ if (action.id === 'quick-start') {
+ /**
+ * quick start is the action on the welcome page. When its
+ * clicked it collapses the view and puts it into regular
+ * "chat" which is cwc
+ */
+ tabsStorage.updateTabTypeFromUnknown(tabId, 'cwc')
+
+ // show quick start in the current tab instead of a new one
+ mynahUI.updateStore(tabId, {
+ tabHeaderDetails: undefined,
+ compactMode: false,
+ tabBackground: false,
+ promptInputText: '/',
+ promptInputLabel: undefined,
+ chatItems: [],
+ })
+
+ ideApi.postMessage(createClickTelemetry('amazonq-welcome-quick-start-button'))
+ return
+ }
+
+ if (action.id === 'explore') {
+ const newTabId = mynahUI.updateStore('', agentWalkthroughDataModel)
+ if (newTabId === undefined) {
+ mynahUI.notify({
+ content: uiComponentsTexts.noMoreTabsTooltip,
+ type: NotificationType.WARNING,
+ })
+ return
+ }
+ tabsStorage.updateTabTypeFromUnknown(newTabId, 'agentWalkthrough')
+ ideApi.postMessage(createClickTelemetry('amazonq-welcome-explore-button'))
+ return
+ }
+
+ connector.onCustomFormAction(tabId, messageId, action, eventId)
},
defaults: {
store: tabDataGenerator.getTabData('cwc', true),
@@ -542,6 +782,9 @@ export const createMynahUI = (ideApi: any, featureDevInitEnabled: boolean, codeT
tabsStorage,
isFeatureDevEnabled,
isCodeTransformEnabled,
+ isDocEnabled,
+ isCodeScanEnabled,
+ isCodeTestEnabled,
})
textMessageHandler = new TextMessageHandler({
mynahUI,
@@ -554,5 +797,8 @@ export const createMynahUI = (ideApi: any, featureDevInitEnabled: boolean, codeT
tabsStorage,
isFeatureDevEnabled,
isCodeTransformEnabled,
+ isDocEnabled,
+ isCodeScanEnabled,
+ isCodeTestEnabled,
})
}
diff --git a/plugins/amazonq/mynah-ui/src/mynah-ui/ui/messages/controller.ts b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/messages/controller.ts
index 142f4d4093..72635f0775 100644
--- a/plugins/amazonq/mynah-ui/src/mynah-ui/ui/messages/controller.ts
+++ b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/messages/controller.ts
@@ -15,6 +15,9 @@ export interface MessageControllerProps {
tabsStorage: TabsStorage
isFeatureDevEnabled: boolean
isCodeTransformEnabled: boolean
+ isDocEnabled: boolean
+ isCodeScanEnabled: boolean
+ isCodeTestEnabled: boolean
}
export class MessageController {
@@ -30,6 +33,9 @@ export class MessageController {
this.tabDataGenerator = new TabDataGenerator({
isFeatureDevEnabled: props.isFeatureDevEnabled,
isCodeTransformEnabled: props.isCodeTransformEnabled,
+ isDocEnabled: props.isDocEnabled,
+ isCodeScanEnabled: props.isCodeScanEnabled,
+ isCodeTestEnabled: props.isCodeTestEnabled,
})
}
diff --git a/plugins/amazonq/mynah-ui/src/mynah-ui/ui/quickActions/generator.ts b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/quickActions/generator.ts
index 1f62a45ce0..1033f29a8e 100644
--- a/plugins/amazonq/mynah-ui/src/mynah-ui/ui/quickActions/generator.ts
+++ b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/quickActions/generator.ts
@@ -5,60 +5,114 @@
import { QuickActionCommand, QuickActionCommandGroup } from '@aws/mynah-ui-chat/dist/static'
import { TabType } from '../storages/tabsStorage'
+import {MynahIcons} from "@aws/mynah-ui-chat";
export interface QuickActionGeneratorProps {
isFeatureDevEnabled: boolean
isCodeTransformEnabled: boolean
+ isDocEnabled: boolean
+ isCodeScanEnabled: boolean
+ isCodeTestEnabled: boolean
}
export class QuickActionGenerator {
public isFeatureDevEnabled: boolean
public isCodeTransformEnabled: boolean
+ public isDocEnabled: boolean
+ public isCodeScanEnabled: boolean
+ public isCodeTestEnabled: boolean
constructor(props: QuickActionGeneratorProps) {
this.isFeatureDevEnabled = props.isFeatureDevEnabled
this.isCodeTransformEnabled = props.isCodeTransformEnabled
+ this.isDocEnabled = props.isDocEnabled
+ this.isCodeScanEnabled = props.isCodeScanEnabled
+ this.isCodeTestEnabled = props.isCodeTestEnabled
}
public generateForTab(tabType: TabType): QuickActionCommandGroup[] {
+ // agentWalkthrough is static and doesn't have any quick actions
+ if (tabType === 'agentWalkthrough') {
+ return []
+ }
+
const quickActionCommands = [
{
+ groupName: `Q Developer Agent for Software Development `,
commands: [
...(this.isFeatureDevEnabled
? [
{
command: '/dev',
+ icon: MynahIcons.CODE_BLOCK,
placeholder: 'Describe your task or issue in as much detail as possible',
description: 'Generate code to make a change in your project',
},
]
: []),
+ ...(this.isDocEnabled
+ ? [
+ {
+ command: '/doc',
+ icon: MynahIcons.FILE,
+ description: 'Generate documentation for your code',
+ },
+ ]
+ : []),
+ ...(this.isCodeScanEnabled
+ ? [
+ {
+ command: '/review',
+ icon: MynahIcons.BUG,
+ description: 'Identify and fix code issues before committing'
+ }
+ ]
+ : []),
+ ...(this.isCodeTestEnabled
+ ? [
+ {
+ command: '/test',
+ icon: MynahIcons.CHECK_LIST,
+ placeholder: 'Specify a function(s) in the current file(optional)',
+ description: 'Generate unit tests',
+ },
+ ]
+ : []),
+ ],
+ },
+ {
+ groupName: `Q Developer Agent for Code Transformation `,
+ commands:[
...(this.isCodeTransformEnabled
? [
- {
- command: '/transform',
- description: 'Transform your Java project',
- },
- ]
+ {
+ command: '/transform',
+ icon: MynahIcons.TRANSFORM,
+ description: 'Transform your Java project',
+ },
+ ]
: []),
],
},
{
+ groupName: 'Quick Actions',
commands: [
{
command: '/help',
+ icon: MynahIcons.HELP,
description: 'Learn more about Amazon Q',
},
{
command: '/clear',
+ icon: MynahIcons.TRASH,
description: 'Clear this session',
},
],
},
- ]
+ ].filter((section) => section.commands.length > 0)
const commandUnavailability: Record<
- TabType,
+ Exclude,
{
description: string
unavailableItems: string[]
@@ -70,11 +124,27 @@ export class QuickActionGenerator {
},
featuredev: {
description: "This command isn't available in /dev",
- unavailableItems: ['/dev', '/transform', '/help', '/clear'],
+ unavailableItems: ['/dev', '/transform', '/doc', '/help', '/clear', '/review', '/test'],
},
codetransform: {
description: "This command isn't available in /transform",
- unavailableItems: ['/dev', '/transform'],
+ unavailableItems: ['/help', '/clear'],
+ },
+ codescan: {
+ description: "This command isn't available in /review",
+ unavailableItems: ['/help', '/clear'],
+ },
+ codetest: {
+ description: "This command isn't available in /test",
+ unavailableItems: ['/help', '/clear'],
+ },
+ doc: {
+ description: "This command isn't available in /doc",
+ unavailableItems: ['/help', '/clear'],
+ },
+ welcome: {
+ description: '',
+ unavailableItems: ['/clear'],
},
unknown: {
description: '',
@@ -84,6 +154,7 @@ export class QuickActionGenerator {
return quickActionCommands.map((commandGroup: QuickActionCommandGroup) => {
return {
+ groupName: commandGroup.groupName,
commands: commandGroup.commands.map((commandItem: QuickActionCommand) => {
const commandNotAvailable = commandUnavailability[tabType].unavailableItems.includes(
commandItem.command
diff --git a/plugins/amazonq/mynah-ui/src/mynah-ui/ui/quickActions/handler.ts b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/quickActions/handler.ts
index 426dc0a325..e9de0369f2 100644
--- a/plugins/amazonq/mynah-ui/src/mynah-ui/ui/quickActions/handler.ts
+++ b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/quickActions/handler.ts
@@ -3,7 +3,7 @@
* SPDX-License-Identifier: Apache-2.0
*/
-import { ChatItemType, ChatPrompt, MynahUI, NotificationType } from '@aws/mynah-ui-chat'
+import { ChatItemType, ChatPrompt, MynahIcons, MynahUI, NotificationType } from '@aws/mynah-ui-chat'
import { TabDataGenerator } from '../tabs/generator'
import { Connector } from '../connector'
import { Tab, TabsStorage } from '../storages/tabsStorage'
@@ -15,6 +15,9 @@ export interface QuickActionsHandlerProps {
tabsStorage: TabsStorage
isFeatureDevEnabled: boolean
isCodeTransformEnabled: boolean
+ isDocEnabled: boolean
+ isCodeScanEnabled: boolean
+ isCodeTestEnabled: boolean
}
export class QuickActionHandler {
@@ -24,6 +27,9 @@ export class QuickActionHandler {
private tabDataGenerator: TabDataGenerator
public isFeatureDevEnabled: boolean
public isCodeTransformEnabled: boolean
+ public isDocEnabled: boolean
+ public isCodeScanEnabled: boolean
+ public isCodeTestEnabled: boolean
constructor(props: QuickActionsHandlerProps) {
this.mynahUI = props.mynahUI
@@ -32,12 +38,19 @@ export class QuickActionHandler {
this.tabDataGenerator = new TabDataGenerator({
isFeatureDevEnabled: props.isFeatureDevEnabled,
isCodeTransformEnabled: props.isCodeTransformEnabled,
+ isDocEnabled: props.isDocEnabled,
+ isCodeScanEnabled: props.isCodeScanEnabled,
+ isCodeTestEnabled: props.isCodeTestEnabled,
})
this.isFeatureDevEnabled = props.isFeatureDevEnabled
this.isCodeTransformEnabled = props.isCodeTransformEnabled
+ this.isDocEnabled = props.isDocEnabled
+ this.isCodeScanEnabled = props.isCodeScanEnabled
+ this.isCodeTestEnabled = props.isCodeTestEnabled
}
- public handle(chatPrompt: ChatPrompt, tabID: string, eventId?: string) {
+ // Entry point for `/xxx` commands
+ public handleCommand(chatPrompt: ChatPrompt, tabID: string, eventId?: string) {
this.tabsStorage.resetTabTimer(tabID)
switch (chatPrompt.command) {
case '/dev':
@@ -49,6 +62,15 @@ export class QuickActionHandler {
case '/transform':
this.handleCodeTransformCommand(tabID, eventId)
break
+ case '/doc':
+ this.handleDocCommand(chatPrompt, tabID, 'Q - Doc')
+ break
+ case '/review':
+ this.handleCodeScanCommand(tabID, eventId)
+ break
+ case '/test':
+ this.handleCodeTestCommand(chatPrompt, tabID, eventId)
+ break
case '/clear':
this.handleClearCommand(tabID)
break
@@ -143,15 +165,38 @@ export class QuickActionHandler {
this.mynahUI.updateStore(affectedTabId, { chatItems: [] })
this.mynahUI.updateStore(
affectedTabId,
- this.tabDataGenerator.getTabData('featuredev', realPromptText === '', taskName)
+ this.tabDataGenerator.getTabData('featuredev', false, taskName)
)
+ const addInformationCard = (tabId: string) => {
+ this.mynahUI.addChatItem(tabId, {
+ type: ChatItemType.ANSWER,
+ informationCard: {
+ title: "Feature development",
+ description: "Amazon Q Developer Agent for Software Development",
+ icon: MynahIcons.BUG,
+ content: {
+ body: [
+ "I can generate code to accomplish a task or resolve an issue.",
+ "After you provide a task, I will:",
+ "1. Generate code based on your description and the code in your workspace",
+ "2. Provide a list of suggestions for you to review and add to your workspace",
+ "3. If needed, iterate based on your feedback",
+ "To learn more, visit the [user guide](https://docs.aws.amazon.com/amazonq/latest/qdeveloper-ug/software-dev.html).",
+ ].join("\n")
+ },
+ },
+ })
+ };
+
if (realPromptText !== '') {
this.mynahUI.addChatItem(affectedTabId, {
type: ChatItemType.PROMPT,
body: realPromptText,
})
+ addInformationCard(affectedTabId)
+
this.mynahUI.addChatItem(affectedTabId, {
type: ChatItemType.ANSWER_STREAM,
body: '',
@@ -165,7 +210,174 @@ export class QuickActionHandler {
this.connector.requestGenerativeAIAnswer(affectedTabId, {
chatMessage: realPromptText,
})
+ } else {
+ addInformationCard(affectedTabId)
+ }
+ }
+ }
+
+private handleDocCommand(chatPrompt: ChatPrompt, tabID: string, taskName: string) {
+ if (!this.isDocEnabled) {
+ return
+ }
+
+ let affectedTabId: string | undefined = tabID
+ const realPromptText = chatPrompt.escapedPrompt?.trim() ?? ''
+
+ if (this.tabsStorage.getTab(affectedTabId)?.type !== 'unknown') {
+ affectedTabId = this.mynahUI.updateStore('', {})
+ }
+
+ if (affectedTabId === undefined) {
+ this.mynahUI.notify({
+ content: uiComponentsTexts.noMoreTabsTooltip,
+ type: NotificationType.WARNING,
+ })
+ return
+ } else {
+ this.tabsStorage.updateTabTypeFromUnknown(affectedTabId, 'doc')
+ this.connector.onKnownTabOpen(affectedTabId)
+ this.connector.onUpdateTabType(affectedTabId)
+
+ this.mynahUI.updateStore(affectedTabId, { chatItems: [] })
+
+ this.mynahUI.updateStore(
+ affectedTabId, {
+ ...this.tabDataGenerator.getTabData('doc', realPromptText === '', taskName),
+ promptInputDisabledState: true
+ }
+ )
+
+ if (realPromptText !== '') {
+ this.mynahUI.addChatItem(affectedTabId, {
+ type: ChatItemType.PROMPT,
+ body: realPromptText,
+ })
+
+ this.mynahUI.updateStore(affectedTabId, {
+ loadingChat: true,
+ promptInputDisabledState: true,
+ })
+
+ void this.connector.requestGenerativeAIAnswer(affectedTabId, {
+ chatMessage: realPromptText,
+ })
}
}
}
+
+ private showScanInTab( tabId: string) {
+ this.mynahUI.addChatItem(tabId, {
+ type: ChatItemType.PROMPT,
+ body: "Run a code review",
+ })
+ this.mynahUI.addChatItem(tabId, {
+ type: ChatItemType.ANSWER,
+ informationCard: {
+ title: "/review",
+ description: "Included in your Q Developer subscription",
+ icon: MynahIcons.BUG,
+ content: {
+ body: "Automated code review allowing developers to identify and resolve code quality issues, " +
+ "security vulnerabilities, misconfigurations, and deviations from coding best practices.\n\n" +
+ "For this workflow, Q will:\n1. Review the project or a particular file you select and identify issues before code commit\n" +
+ "2. Provide a list of findings from where you can follow up with Q to find solutions\n3. Generate on-demand code fixes inline\n\n" +
+ "To learn more, check out our [user guide](https://aws.amazon.com/q/developer/)."
+ },
+ },
+ })
+ this.connector.scan(tabId)
+ }
+
+ private handleCodeScanCommand(tabID: string, eventId?: string) {
+ if (!this.isCodeScanEnabled) {
+ return
+ }
+
+ // Check for existing opened code scan tab
+ const existingCodeScanTab = this.tabsStorage.getTabs().find(tab => tab.type === 'codescan')
+ if (existingCodeScanTab !== undefined ) {
+ this.mynahUI.selectTab(existingCodeScanTab.id, eventId || "")
+ this.connector.onTabChange(existingCodeScanTab.id)
+
+ this.mynahUI.notify({
+ title: "Q - Review",
+ content: "Switched to the opened code review tab"
+ });
+ this.showScanInTab(existingCodeScanTab.id)
+ return
+ }
+
+ // Add new tab
+ let affectedTabId: string | undefined = tabID
+ if (this.tabsStorage.getTab(affectedTabId)?.type !== 'unknown') {
+ affectedTabId = this.mynahUI.updateStore('', {})
+ }
+ if (affectedTabId === undefined) {
+ this.mynahUI.notify({
+ content: uiComponentsTexts.noMoreTabsTooltip,
+ type: NotificationType.WARNING
+ })
+ return
+ } else {
+ this.tabsStorage.updateTabTypeFromUnknown(affectedTabId, 'codescan')
+ this.connector.onKnownTabOpen(affectedTabId)
+ // Clear unknown tab type's welcome message
+ this.mynahUI.updateStore(affectedTabId, {chatItems: []})
+ this.mynahUI.updateStore(affectedTabId, this.tabDataGenerator.getTabData('codescan', true))
+ this.mynahUI.updateStore(affectedTabId, {
+ promptInputDisabledState: true,
+ promptInputPlaceholder: 'Waiting on your inputs...',
+ loadingChat: true,
+ })
+
+ this.connector.onTabAdd(affectedTabId)
+ }
+ this.showScanInTab(affectedTabId)
+ }
+
+ private handleCodeTestCommand(chatPrompt: ChatPrompt, tabID: string, eventId: string | undefined) {
+ if (!this.isCodeTestEnabled) {
+ return
+ }
+ const testTabId = this.tabsStorage.getTabs().find((tab) => tab.type === 'codetest')?.id
+ const realPromptText = chatPrompt.escapedPrompt?.trim() ?? ''
+ if (testTabId !== undefined) {
+ this.mynahUI.selectTab(testTabId, eventId || '')
+ this.connector.onTabChange(testTabId)
+ this.connector.startTestGen(testTabId, realPromptText)
+ return
+ }
+ let affectedTabId: string | undefined = tabID
+
+ // if there is no test tab, open a new one
+ if (this.tabsStorage.getTab(affectedTabId)?.type !== 'unknown') {
+ affectedTabId = this.mynahUI.updateStore('', {
+ loadingChat: true,
+ })
+ }
+ if (affectedTabId === undefined) {
+ this.mynahUI.notify({
+ content: uiComponentsTexts.noMoreTabsTooltip,
+ type: NotificationType.WARNING,
+ })
+ return
+ } else {
+ this.tabsStorage.updateTabTypeFromUnknown(affectedTabId, 'codetest')
+ this.connector.onKnownTabOpen(affectedTabId)
+ this.connector.onUpdateTabType(affectedTabId)
+ // reset chat history
+ this.mynahUI.updateStore(affectedTabId, {
+ chatItems: [],
+ })
+
+ // creating a new tab and printing some title
+ this.mynahUI.updateStore(
+ affectedTabId,
+ this.tabDataGenerator.getTabData('codetest', realPromptText === '', 'Q - Test')
+ )
+
+ this.connector.startTestGen(affectedTabId, realPromptText)
+ }
+ }
}
diff --git a/plugins/amazonq/mynah-ui/src/mynah-ui/ui/storages/tabsStorage.ts b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/storages/tabsStorage.ts
index 85b9b1fb6c..6b36eae684 100644
--- a/plugins/amazonq/mynah-ui/src/mynah-ui/ui/storages/tabsStorage.ts
+++ b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/storages/tabsStorage.ts
@@ -4,7 +4,38 @@
*/
export type TabStatus = 'free' | 'busy' | 'dead'
-export type TabType = 'cwc' | 'featuredev' | 'codetransform' | 'unknown'
+const TabTypes = [
+ 'cwc',
+ 'featuredev',
+ 'codetransform',
+ 'doc',
+ 'codescan',
+ 'codetest',
+ 'agentWalkthrough',
+ 'welcome',
+ 'unknown',
+] as const
+export type TabType = (typeof TabTypes)[number]
+export function isTabType(value: string): value is TabType {
+ return (TabTypes as readonly string[]).includes(value)
+}
+
+export function getTabCommandFromTabType(tabType: TabType): string {
+ switch (tabType) {
+ case 'featuredev':
+ return '/dev'
+ case 'codetransform':
+ return '/transform'
+ case 'doc':
+ return '/doc'
+ case 'codescan':
+ return '/review'
+ case 'codetest':
+ return '/test'
+ default:
+ return ''
+ }
+}
export type TabOpenType = 'click' | 'contextMenu' | 'hotkeys'
const TabTimeoutDuration = 172_800_000 // 48hrs
@@ -73,7 +104,10 @@ export class TabsStorage {
public updateTabTypeFromUnknown(tabID: string, tabType: TabType) {
const currentTabValue = this.tabs.get(tabID)
- if (currentTabValue === undefined || currentTabValue.type !== 'unknown') {
+ if (
+ currentTabValue === undefined ||
+ (currentTabValue.type !== 'unknown' && currentTabValue.type !== 'welcome')
+ ) {
return
}
diff --git a/plugins/amazonq/mynah-ui/src/mynah-ui/ui/tabs/generator.ts b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/tabs/generator.ts
index 42711a25f0..b5fb3d047f 100644
--- a/plugins/amazonq/mynah-ui/src/mynah-ui/ui/tabs/generator.ts
+++ b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/tabs/generator.ts
@@ -3,7 +3,7 @@
* SPDX-License-Identifier: Apache-2.0
*/
-import { ChatItemType, MynahUIDataModel, QuickActionCommandGroup } from '@aws/mynah-ui-chat'
+import { ChatItemType, MynahUIDataModel, QuickActionCommandGroup, QuickActionCommand } from '@aws/mynah-ui-chat'
import { TabType } from '../storages/tabsStorage'
import { FollowUpGenerator } from '../followUps/generator'
import { QuickActionGenerator } from '../quickActions/generator'
@@ -12,23 +12,34 @@ import { workspaceCommand } from '../commands'
export interface TabDataGeneratorProps {
isFeatureDevEnabled: boolean
isCodeTransformEnabled: boolean
+ isDocEnabled: boolean
+ isCodeScanEnabled: boolean
+ isCodeTestEnabled: boolean
+ highlightCommand?: QuickActionCommand
}
export class TabDataGenerator {
private followUpsGenerator: FollowUpGenerator
public quickActionsGenerator: QuickActionGenerator
+ private highlightCommand?: QuickActionCommand
private tabTitle: Map = new Map([
['unknown', 'Chat'],
['cwc', 'Chat'],
['featuredev', 'Q - Dev'],
['codetransform', 'Q - Transform'],
+ ['doc', 'Q - Documentation'],
+ ['codescan', 'Q - Review'],
+ ['codetest', 'Q - Test'],
])
private tabInputPlaceholder: Map = new Map([
['unknown', 'Ask a question or enter "/" for quick commands'],
['cwc', 'Ask a question or enter "/" for quick commands'],
['featuredev', 'Describe your task or issue in detail'],
+ ['doc', 'Ask Amazon Q to generate documentation for your project'],
+ ['codescan', 'Waiting for your inputs...'],
+ ['codetest', 'Specify a function(s) in the current file(optional)'],
])
private tabWelcomeMessage: Map = new Map([
@@ -56,6 +67,14 @@ What would you like to work on?`,
'codetransform',
`Welcome to Code Transformation!`,
],
+ [
+ 'doc',
+ `Welcome to doc generation!\n\nI can help generate documentation for your code. To get started, choose what type of doc update you'd like to make.`,
+ ],
+ [
+ 'codetest',
+ `Welcome to Amazon Q Unit Test Generation. I can help you generate unit tests for your active file.`,
+ ]
])
private tabContextCommand: Map = new Map([
@@ -67,7 +86,11 @@ What would you like to work on?`,
this.quickActionsGenerator = new QuickActionGenerator({
isFeatureDevEnabled: props.isFeatureDevEnabled,
isCodeTransformEnabled: props.isCodeTransformEnabled,
+ isDocEnabled: props.isDocEnabled,
+ isCodeScanEnabled: props.isCodeScanEnabled,
+ isCodeTestEnabled: props.isCodeTestEnabled,
})
+ this.highlightCommand = props.highlightCommand
}
public getTabData(tabType: TabType, needWelcomeMessages: boolean, taskName?: string): MynahUIDataModel {
@@ -77,7 +100,7 @@ What would you like to work on?`,
'Amazon Q Developer uses generative AI. You may need to verify responses. See the [AWS Responsible AI Policy](https://aws.amazon.com/machine-learning/responsible-ai/policy/).',
quickActionCommands: this.quickActionsGenerator.generateForTab(tabType),
promptInputPlaceholder: this.tabInputPlaceholder.get(tabType),
- contextCommands: this.tabContextCommand.get(tabType),
+ contextCommands: this.getContextCommands(tabType),
chatItems: needWelcomeMessages
? [
{
@@ -92,4 +115,23 @@ What would you like to work on?`,
: [],
}
}
+
+ private getContextCommands(tabType: TabType): QuickActionCommandGroup[] | undefined {
+ const contextCommands = this.tabContextCommand.get(tabType)
+
+ if (this.highlightCommand) {
+ const commandHighlight: QuickActionCommandGroup = {
+ groupName: 'Additional Commands',
+ commands: [this.highlightCommand],
+ }
+
+ if (contextCommands !== undefined) {
+ return [...contextCommands, commandHighlight]
+ }
+
+ return [commandHighlight]
+ }
+
+ return contextCommands
+ }
}
diff --git a/plugins/amazonq/mynah-ui/src/mynah-ui/ui/telemetry/actions.ts b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/telemetry/actions.ts
new file mode 100644
index 0000000000..ffd65684ff
--- /dev/null
+++ b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/telemetry/actions.ts
@@ -0,0 +1,38 @@
+/*!
+ * Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+import { ExtensionMessage } from '../commands'
+import { TabType } from '../storages/tabsStorage'
+
+export function createClickTelemetry(source: string): ExtensionMessage {
+ return {
+ command: 'send-telemetry',
+ source,
+ }
+}
+export function isClickTelemetry(message: ExtensionMessage): boolean {
+ return (
+ message.command === 'send-telemetry' && typeof message.source === 'string' && Object.keys(message).length === 2
+ )
+}
+
+export function createOpenAgentTelemetry(module: TabType, trigger: Trigger): ExtensionMessage {
+ return {
+ command: 'send-telemetry',
+ module,
+ trigger,
+ }
+}
+
+export type Trigger = 'right-click' | 'quick-action' | 'quick-start'
+
+export function isOpenAgentTelemetry(message: ExtensionMessage): boolean {
+ return (
+ message.command === 'send-telemetry' &&
+ typeof message.module === 'string' &&
+ typeof message.trigger === 'string' &&
+ Object.keys(message).length === 3
+ )
+}
diff --git a/plugins/amazonq/mynah-ui/src/mynah-ui/ui/texts/constants.ts b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/texts/constants.ts
index 577f6d3f3d..9c6c6dbc9a 100644
--- a/plugins/amazonq/mynah-ui/src/mynah-ui/ui/texts/constants.ts
+++ b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/texts/constants.ts
@@ -26,3 +26,9 @@ export const uiComponentsTexts = {
spinnerText: 'Generating your answer...',
pleaseSelect: 'Please select',
}
+
+export const docUserGuide = 'https://docs.aws.amazon.com/amazonq/latest/qdeveloper-ug/doc-generation.html'
+export const featureDevUserGuide = 'https://docs.aws.amazon.com/amazonq/latest/qdeveloper-ug/software-dev.html'
+export const codeTransformUserGuide = 'https://docs.aws.amazon.com/amazonq/latest/aws-builder-use-ug/code-transformation.html'
+export const codeTestUserGuide = 'https://docs.aws.amazon.com/amazonq/latest/qdeveloper-ug/test-generation.html'
+export const codeScanUserGuide = 'https://docs.aws.amazon.com/amazonq/latest/qdeveloper-ug/code-reviews.html'
diff --git a/plugins/amazonq/mynah-ui/src/mynah-ui/ui/texts/disclaimer.ts b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/texts/disclaimer.ts
new file mode 100644
index 0000000000..4bf62164a8
--- /dev/null
+++ b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/texts/disclaimer.ts
@@ -0,0 +1,20 @@
+/*!
+ * Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+import { ChatItem, MynahIcons } from '@aws/mynah-ui-chat'
+
+export const disclaimerAcknowledgeButtonId = 'amazonq-disclaimer-acknowledge-button-id'
+export const disclaimerCard: Partial = {
+ messageId: 'amazonq-disclaimer-card',
+ body: 'Amazon Q Developer uses generative AI. You may need to verify responses. See the [AWS Responsible AI Policy](https://aws.amazon.com/machine-learning/responsible-ai/policy/). Amazon Q Developer processes data across all US Regions. See [here](https://docs.aws.amazon.com/amazonq/latest/qdeveloper-ug/cross-region-inference.html) for more info. Amazon Q may retain chats to provide and maintain the service.',
+ buttons: [
+ {
+ text: 'Acknowledge',
+ id: disclaimerAcknowledgeButtonId,
+ status: 'info',
+ icon: MynahIcons.OK,
+ },
+ ],
+}
diff --git a/plugins/amazonq/mynah-ui/src/mynah-ui/ui/walkthrough/agent.ts b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/walkthrough/agent.ts
new file mode 100644
index 0000000000..02b2818a2f
--- /dev/null
+++ b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/walkthrough/agent.ts
@@ -0,0 +1,201 @@
+/*!
+ * Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+import { ChatItemContent, ChatItemType, MynahIcons, MynahUIDataModel } from '@aws/mynah-ui-chat'
+
+function createdTabbedData(examples: string[], agent: string): ChatItemContent['tabbedContent'] {
+ const exampleText = examples.map((example) => `- ${example}`).join('\n')
+ return [
+ {
+ label: 'Examples',
+ value: 'examples',
+ content: {
+ body: `**Example use cases:**\n${exampleText}\n\nEnter ${agent} in Q Chat to get started`,
+ },
+ },
+ ]
+}
+
+export const agentWalkthroughDataModel: MynahUIDataModel = {
+ tabBackground: false,
+ compactMode: false,
+ tabTitle: 'Explore',
+ promptInputVisible: false,
+ tabHeaderDetails: {
+ icon: MynahIcons.ASTERISK,
+ title: 'Amazon Q Developer agents capabilities',
+ description: '',
+ },
+ chatItems: [
+ {
+ type: ChatItemType.ANSWER,
+ snapToTop: true,
+ hoverEffect: true,
+ body: `### Feature development
+Implement features or make changes across your workspace, all from a single prompt.
+`,
+ icon: MynahIcons.CODE_BLOCK,
+ footer: {
+ tabbedContent: createdTabbedData(
+ [
+ '/dev update app.py to add a new api',
+ '/dev fix the error',
+ '/dev add a new button to sort by ',
+ ],
+ '/dev'
+ ),
+ },
+ buttons: [
+ {
+ status: 'clear',
+ id: 'user-guide-featuredev',
+ disabled: false,
+ text: 'Read user guide',
+ },
+ {
+ status: 'main',
+ disabled: false,
+ flash: 'once',
+ fillState: 'hover',
+ icon: MynahIcons.RIGHT_OPEN,
+ id: 'quick-start-featuredev',
+ text: `Quick start with **/dev**`,
+ },
+ ],
+ },
+ {
+ type: ChatItemType.ANSWER,
+ hoverEffect: true,
+ body: `### Unit test generation
+Automatically generate unit tests for your active file.
+`,
+ icon: MynahIcons.BUG,
+ footer: {
+ tabbedContent: createdTabbedData(
+ ['Generate tests for specific functions', 'Generate tests for null and empty inputs'],
+ '/test'
+ ),
+ },
+ buttons: [
+ {
+ status: 'clear',
+ id: 'user-guide-codetest',
+ disabled: false,
+ text: 'Read user guide',
+ },
+ {
+ status: 'main',
+ disabled: false,
+ flash: 'once',
+ fillState: 'hover',
+ icon: MynahIcons.RIGHT_OPEN,
+ id: 'quick-start-codetest',
+ text: `Quick start with **/test**`,
+ },
+ ],
+ },
+ {
+ type: ChatItemType.ANSWER,
+ hoverEffect: true,
+ body: `### Documentation generation
+Create and update READMEs for better documented code.
+`,
+ icon: MynahIcons.CHECK_LIST,
+ footer: {
+ tabbedContent: createdTabbedData(
+ [
+ 'Generate new READMEs for your project',
+ 'Update existing READMEs with recent code changes',
+ 'Request specific changes to a README',
+ ],
+ '/doc'
+ ),
+ },
+ buttons: [
+ {
+ status: 'clear',
+ id: 'user-guide-doc',
+ disabled: false,
+ text: 'Read user guide',
+ },
+ {
+ status: 'main',
+ disabled: false,
+ flash: 'once',
+ fillState: 'hover',
+ icon: MynahIcons.RIGHT_OPEN,
+ id: 'quick-start-doc',
+ text: `Quick start with **/doc**`,
+ },
+ ],
+ },
+ {
+ type: ChatItemType.ANSWER,
+ hoverEffect: true,
+ body: `### Code reviews
+Review code for issues, then get suggestions to fix your code instantaneously.
+`,
+ icon: MynahIcons.TRANSFORM,
+ footer: {
+ tabbedContent: createdTabbedData(
+ [
+ 'Review code for security vulnerabilities and code quality issues',
+ 'Get detailed explanations about code issues',
+ 'Apply automatic code fixes to your files',
+ ],
+ '/review'
+ ),
+ },
+ buttons: [
+ {
+ status: 'clear',
+ id: 'user-guide-codescan',
+ disabled: false,
+ text: 'Read user guide',
+ },
+ {
+ status: 'main',
+ disabled: false,
+ flash: 'once',
+ fillState: 'hover',
+ icon: MynahIcons.RIGHT_OPEN,
+ id: 'quick-start-codescan',
+ text: `Quick start with **/review**`,
+ },
+ ],
+ },
+ {
+ type: ChatItemType.ANSWER,
+ hoverEffect: true,
+ body: `### Transformation
+Upgrade library and language versions in your codebase.
+`,
+ icon: MynahIcons.TRANSFORM,
+ footer: {
+ tabbedContent: createdTabbedData(
+ ['Upgrade Java language and dependency versions', 'Convert embedded SQL code in Java apps'],
+ '/transform'
+ ),
+ },
+ buttons: [
+ {
+ status: 'clear',
+ id: 'user-guide-codetransform',
+ disabled: false,
+ text: 'Read user guide',
+ },
+ {
+ status: 'main',
+ disabled: false,
+ flash: 'once',
+ fillState: 'hover',
+ icon: MynahIcons.RIGHT_OPEN,
+ id: 'quick-start-codetransform',
+ text: `Quick start with **/transform**`,
+ },
+ ],
+ },
+ ],
+}
diff --git a/plugins/amazonq/mynah-ui/src/mynah-ui/ui/walkthrough/welcome.ts b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/walkthrough/welcome.ts
new file mode 100644
index 0000000000..e5fe67fdd7
--- /dev/null
+++ b/plugins/amazonq/mynah-ui/src/mynah-ui/ui/walkthrough/welcome.ts
@@ -0,0 +1,47 @@
+/*!
+ * Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+import { ChatItemType, MynahIcons, MynahUITabStoreTab } from '@aws/mynah-ui-chat'
+import { TabDataGenerator } from '../tabs/generator'
+
+export const welcomeScreenTabData = (tabs: TabDataGenerator): MynahUITabStoreTab => ({
+ isSelected: true,
+ store: {
+ quickActionCommands: tabs.quickActionsGenerator.generateForTab('welcome'),
+ contextCommands: tabs.getTabData('cwc', false).contextCommands,
+ tabTitle: 'Welcome to Q',
+ tabBackground: true,
+ chatItems: [
+ {
+ type: ChatItemType.ANSWER,
+ icon: MynahIcons.ASTERISK,
+ messageId: 'new-welcome-card',
+ body: `#### Work on a task using agentic capabilities
+_Generate code, scan for issues, and more._`,
+ buttons: [
+ {
+ id: 'explore',
+ disabled: false,
+ text: 'Explore',
+ },
+ {
+ id: 'quick-start',
+ text: 'Quick start',
+ disabled: false,
+ status: 'main',
+ },
+ ],
+ },
+ ],
+ promptInputLabel: 'Or, start a chat',
+ promptInputPlaceholder: 'Type your question',
+ compactMode: true,
+ tabHeaderDetails: {
+ title: "Hi, I'm Amazon Q.",
+ description: 'Where would you like to start?',
+ icon: MynahIcons.Q,
+ },
+ },
+})
diff --git a/plugins/amazonq/shared/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/CodeWhispererFeatureConfigService.kt b/plugins/amazonq/shared/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/CodeWhispererFeatureConfigService.kt
index de57180cba..757bdd1036 100644
--- a/plugins/amazonq/shared/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/CodeWhispererFeatureConfigService.kt
+++ b/plugins/amazonq/shared/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/CodeWhispererFeatureConfigService.kt
@@ -111,6 +111,8 @@ class CodeWhispererFeatureConfigService {
fun getCustomizationFeature(): FeatureContext? = getFeature(CUSTOMIZATION_ARN_OVERRIDE_NAME)
+ fun getHighlightCommandFeature(): FeatureContext? = getFeature(HIGHLIGHT_COMMAND_NAME)
+
fun getNewAutoTriggerUX(): Boolean = getFeatureValueForKey(NEW_AUTO_TRIGGER_UX).stringValue() == "TREATMENT"
fun getInlineCompletion(): Boolean = getFeatureValueForKey(INLINE_COMPLETION).stringValue() == "TREATMENT"
@@ -131,7 +133,8 @@ class CodeWhispererFeatureConfigService {
fun getInstance(): CodeWhispererFeatureConfigService = service()
private const val TEST_FEATURE_NAME = "testFeature"
private const val INLINE_COMPLETION = "ProjectContextV2"
- const val CUSTOMIZATION_ARN_OVERRIDE_NAME = "customizationArnOverride"
+ private const val CUSTOMIZATION_ARN_OVERRIDE_NAME = "customizationArnOverride"
+ private const val HIGHLIGHT_COMMAND_NAME = "highlightCommand"
private const val NEW_AUTO_TRIGGER_UX = "newAutoTriggerUX"
private val LOG = getLogger()
diff --git a/plugins/amazonq/shared/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/project/LspMessage.kt b/plugins/amazonq/shared/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/project/LspMessage.kt
index cd4f743028..e95e3f35f3 100644
--- a/plugins/amazonq/shared/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/project/LspMessage.kt
+++ b/plugins/amazonq/shared/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/project/LspMessage.kt
@@ -52,6 +52,7 @@ data class QueryChatRequest(
data class QueryInlineCompletionRequest(
val query: String,
val filePath: String,
+ val target: String,
) : LspRequest
data class LspResponse(
@@ -70,6 +71,14 @@ enum class IndexOption(val command: String) {
DEFAULT("default"),
}
+enum class InlineContextTarget(private val v: String) {
+ CODEMAP("codemap"),
+ BM25("bm25"),
+ DEFAULT("default"), ;
+
+ override fun toString(): String = this.v
+}
+
// TODO: unify with [software.aws.toolkits.jetbrains.services.codewhisperer.model.Chunk]
data class InlineBm25Chunk(
val content: String,
diff --git a/plugins/amazonq/shared/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/project/ProjectContextController.kt b/plugins/amazonq/shared/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/project/ProjectContextController.kt
index 7e98c61b74..9864acd5e2 100644
--- a/plugins/amazonq/shared/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/project/ProjectContextController.kt
+++ b/plugins/amazonq/shared/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/project/ProjectContextController.kt
@@ -47,7 +47,7 @@ class ProjectContextController(private val project: Project, private val cs: Cor
fun getProjectContextIndexComplete() = projectContextProvider.isIndexComplete.get()
- suspend fun query(prompt: String, timeout: Long?): List {
+ suspend fun queryChat(prompt: String, timeout: Long?): List {
try {
return projectContextProvider.query(prompt, timeout)
} catch (e: Exception) {
@@ -58,7 +58,7 @@ class ProjectContextController(private val project: Project, private val cs: Cor
suspend fun queryInline(query: String, filePath: String): List =
try {
- projectContextProvider.queryInline(query, filePath)
+ projectContextProvider.queryInline(query, filePath, InlineContextTarget.CODEMAP)
} catch (e: Exception) {
var logStr = "error while querying inline for project context $e.message"
if (e is TimeoutCancellationException || e is TimeoutException) {
diff --git a/plugins/amazonq/shared/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/project/ProjectContextProvider.kt b/plugins/amazonq/shared/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/project/ProjectContextProvider.kt
index 1f657837c4..4a516bcfdb 100644
--- a/plugins/amazonq/shared/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/project/ProjectContextProvider.kt
+++ b/plugins/amazonq/shared/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/project/ProjectContextProvider.kt
@@ -171,9 +171,9 @@ class ProjectContextProvider(val project: Project, private val encoderServer: En
}.await()
}
- suspend fun queryInline(query: String, filePath: String): List = withTimeout(SUPPLEMENTAL_CONTEXT_TIMEOUT) {
+ suspend fun queryInline(query: String, filePath: String, target: InlineContextTarget): List = withTimeout(SUPPLEMENTAL_CONTEXT_TIMEOUT) {
cs.async {
- val encrypted = encryptRequest(QueryInlineCompletionRequest(query, filePath))
+ val encrypted = encryptRequest(QueryInlineCompletionRequest(query, filePath, target.toString()))
val r = sendMsgToLsp(LspMessage.QueryInlineCompletion, encrypted)
return@async mapper.readValue>(r.responseBody)
}.await()
diff --git a/plugins/amazonq/shared/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/project/manifest/ManifestManager.kt b/plugins/amazonq/shared/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/project/manifest/ManifestManager.kt
index ba6e274e8b..4693db1004 100644
--- a/plugins/amazonq/shared/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/project/manifest/ManifestManager.kt
+++ b/plugins/amazonq/shared/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/project/manifest/ManifestManager.kt
@@ -15,7 +15,7 @@ import software.aws.toolkits.jetbrains.core.getTextFromUrl
class ManifestManager {
private val cloudFrontUrl = "https://aws-toolkit-language-servers.amazonaws.com/q-context/manifest.json"
- val currentVersion = "0.1.27"
+ val currentVersion = "0.1.32"
val currentOs = getOs()
private val arch = CpuArch.CURRENT
private val mapper = jacksonObjectMapper()
diff --git a/plugins/amazonq/shared/jetbrains-community/src/software/aws/toolkits/jetbrains/settings/CodeWhispererSettings.kt b/plugins/amazonq/shared/jetbrains-community/src/software/aws/toolkits/jetbrains/settings/CodeWhispererSettings.kt
index 4916d241f5..208e86c8e4 100644
--- a/plugins/amazonq/shared/jetbrains-community/src/software/aws/toolkits/jetbrains/settings/CodeWhispererSettings.kt
+++ b/plugins/amazonq/shared/jetbrains-community/src/software/aws/toolkits/jetbrains/settings/CodeWhispererSettings.kt
@@ -97,6 +97,15 @@ class CodeWhispererSettings : PersistentStateComponent()
val intValue by map()
val projectAutoBuildConfigurationMap by map()
+ val stringValue by map()
}
enum class CodeWhispererConfigurationType {
@@ -133,6 +146,10 @@ enum class CodeWhispererConfigurationType {
HasEnabledProjectContextOnce,
}
+enum class CodeWhispererStringConfigurationType {
+ IgnoredCodeReviewIssues,
+}
+
enum class CodeWhispererIntConfigurationType {
ProjectContextIndexThreadCount,
ProjectContextIndexMaxSize,
diff --git a/plugins/amazonq/shared/jetbrains-community/src/software/aws/toolkits/jetbrains/settings/MeetQSettings.kt b/plugins/amazonq/shared/jetbrains-community/src/software/aws/toolkits/jetbrains/settings/MeetQSettings.kt
index e23dda3070..77a30aea0a 100644
--- a/plugins/amazonq/shared/jetbrains-community/src/software/aws/toolkits/jetbrains/settings/MeetQSettings.kt
+++ b/plugins/amazonq/shared/jetbrains-community/src/software/aws/toolkits/jetbrains/settings/MeetQSettings.kt
@@ -26,10 +26,24 @@ class MeetQSettings : PersistentStateComponent {
state.shouldDisplayPage = value
}
+ var reinvent2024OnboardingCount: Int
+ get() = state.reinvent2024OnboardingCount
+ set(value) {
+ state.reinvent2024OnboardingCount = value
+ }
+
+ var disclaimerAcknowledged: Boolean
+ get() = state.disclaimerAcknowledged
+ set(value) {
+ state.disclaimerAcknowledged = value
+ }
+
companion object {
fun getInstance(): MeetQSettings = service()
}
}
data class MeetQSettingsConfiguration(
var shouldDisplayPage: Boolean = true,
+ var reinvent2024OnboardingCount: Int = 0,
+ var disclaimerAcknowledged: Boolean = false,
)
diff --git a/plugins/amazonq/src/main/resources/META-INF/plugin.xml b/plugins/amazonq/src/main/resources/META-INF/plugin.xml
index f4ce6dfac2..3ffcb9ab7a 100644
--- a/plugins/amazonq/src/main/resources/META-INF/plugin.xml
+++ b/plugins/amazonq/src/main/resources/META-INF/plugin.xml
@@ -5,38 +5,47 @@
amazon.q
Amazon Q
Amazon Q is your generative AI-powered assistant across the software development lifecycle.
- Inline code suggestions
- Code faster with inline code suggestions as you type.
-
-15+ languages supported including Java, Python, TypeScript, Rust, Terraform, AWS Cloudformation, and more
- Chat
- Generate code, refactor existing code, explain code, and get answers to questions about software development.
-
+ The most capable generative AI-powered assistant for building, operating, and transforming software, with advanced capabilities for managing data and AI
+
+ Agent capabilities
+ Implement new features
+ /dev
to task Amazon Q with generating new code across your entire project and implement features.
+
+ Generate documentation
+ /doc
to task Amazon Q with writing API, technical design, and onboarding documentation.
+
+ Automate code reviews
+ /review
to ask Amazon Q to perform code reviews, flagging suspicious code patterns and assessing deployment risk.
+
+ Generate unit tests
+ /test
to ask Amazon Q to generate unit tests and add them to your project, helping you improve code quality, fast.
- Security scans
- Analyze and fix security vulnerabilities in your project.
-10 languages supported including Java, Python, Javascript, Golang, and more
+ Transform workloads
+ /transform
to upgrade your Java applications in minutes, not weeks.
- Agent for software development
- Let Amazon Q plan and implement new functionality across multiple files in your workspace. Type “/” in chat to open the quick actions menu and choose the “/dev” action.
+ Core features
- Agent for code transformation
- Upgrade your Java applications in minutes, not weeks. Type “/” in chat to open the quick actions menu and choose the “/transform” action.
- Currently supports upgrading Java 8 or 11 Maven projects to Java 17
-
+ Inline chat
+ Seamlessly initial chat within the inline coding experience. Select a section of code that you need assistance with and initiate chat within the editor to request actions such as "Optimize this code", "Add comments", or "Write tests".
+
+ Chat
+ Generate code, explain code, and get answers about software development.
+
+ Inline suggestions
+
+ 15+ languages supported including Python, TypeScript, Rust, Terraform, AWS Cloudformation, and more
Code reference log
Attribute code from Amazon Q that is similar to training data. When code suggestions similar to training data are accepted, they will be added to the code reference log.
- Getting Started
- Free Tier - create or log in with an AWS Builder ID (no AWS account needed!).
- Pro Tier - if your organization is on the Amazon Q Developer Pro tier, log in with single sign-on.
-
+ Getting Started
+ Free Tier - create or log in with an AWS Builder ID (a personal profile from AWS).
+ Pro Tier - if your organization is on the Amazon Q Developer Pro tier, log in with single sign-on.
+
- Troubleshooting & feedback
- File a bug or submit a feature request on our Github repository.
+ Troubleshooting & feedback
+ File a bug or submit a feature request on our Github repository.
]]>
1.0
diff --git a/plugins/core/core/build.gradle.kts b/plugins/core/core/build.gradle.kts
index 32464f6cb6..00bc6beedc 100644
--- a/plugins/core/core/build.gradle.kts
+++ b/plugins/core/core/build.gradle.kts
@@ -24,7 +24,6 @@ dependencies {
implementation(libs.commonmark)
testImplementation(libs.junit4)
- testRuntimeOnly(libs.junit5.jupiterVintage)
testRuntimeOnly(project(":plugin-core:resources"))
testRuntimeOnly(project(":plugin-core:sdk-codegen"))
}
diff --git a/plugins/core/core/src/software/aws/toolkits/core/utils/RemoteResourceResolver.kt b/plugins/core/core/src/software/aws/toolkits/core/utils/RemoteResourceResolver.kt
index ce0603bf0f..a8107c5b3e 100644
--- a/plugins/core/core/src/software/aws/toolkits/core/utils/RemoteResourceResolver.kt
+++ b/plugins/core/core/src/software/aws/toolkits/core/utils/RemoteResourceResolver.kt
@@ -13,22 +13,56 @@ import java.time.Instant
import java.util.UUID
import java.util.concurrent.Callable
import java.util.concurrent.CompletionStage
+import java.util.concurrent.atomic.AtomicBoolean
interface RemoteResourceResolver {
fun resolve(resource: RemoteResource): CompletionStage
+ fun checkForUpdates(endpoint: String, eTagProvider: ETagProvider): UpdateCheckResult = UpdateCheckResult.NoUpdates
+ fun getLocalResourcePath(filename: String): Path? = null
}
interface RemoteResolveParser {
fun canBeParsed(data: InputStream): Boolean
}
+interface ETagProvider {
+ var etag: String?
+ fun updateETag(newETag: String?)
+}
+
+sealed class UpdateCheckResult {
+ data object HasUpdates : UpdateCheckResult()
+ data object NoUpdates : UpdateCheckResult()
+ data object FirstPollCheck : UpdateCheckResult()
+}
+
class DefaultRemoteResourceResolver(
private val urlFetcher: UrlFetcher,
private val cacheBasePath: Path,
private val executor: (Callable) -> CompletionStage,
) : RemoteResourceResolver {
+ private val isFirstPoll = AtomicBoolean(true)
override fun resolve(resource: RemoteResource): CompletionStage = executor(Callable { internalResolve(resource) })
+ override fun getLocalResourcePath(filename: String): Path? {
+ val expectedLocation = cacheBasePath.resolve(filename)
+ return expectedLocation.existsOrNull()
+ }
+
+ override fun checkForUpdates(endpoint: String, eTagProvider: ETagProvider): UpdateCheckResult {
+ val hasETagUpdate = updateETags(eTagProvider, endpoint)
+ // for when we need to notify on first poll even when there's no new ETag
+ if (isFirstPoll.compareAndSet(true, false) && !hasETagUpdate) {
+ return UpdateCheckResult.FirstPollCheck
+ }
+
+ return if (hasETagUpdate) {
+ UpdateCheckResult.HasUpdates
+ } else {
+ UpdateCheckResult.NoUpdates
+ }
+ }
+
private fun internalResolve(resource: RemoteResource): Path {
val expectedLocation = cacheBasePath.resolve(resource.name)
val current = expectedLocation.existsOrNull()
@@ -82,6 +116,21 @@ class DefaultRemoteResourceResolver(
return expectedLocation
}
+ private fun updateETags(eTagProvider: ETagProvider, endpoint: String): Boolean {
+ val currentEtag = eTagProvider.etag
+ val remoteEtag = getEndpointETag(endpoint)
+ eTagProvider.etag = remoteEtag
+ return currentEtag != remoteEtag
+ }
+
+ private fun getEndpointETag(endpoint: String): String =
+ try {
+ urlFetcher.getETag(endpoint)
+ } catch (e: Exception) {
+ LOG.warn { "Failed to fetch ETag: $e.message" }
+ throw e
+ }
+
private companion object {
val LOG = getLogger()
fun Path.existsOrNull() = if (this.exists()) {
@@ -103,6 +152,7 @@ class DefaultRemoteResourceResolver(
interface UrlFetcher {
fun fetch(url: String, file: Path)
+ fun getETag(url: String): String
}
/**
diff --git a/plugins/core/jetbrains-community/build.gradle.kts b/plugins/core/jetbrains-community/build.gradle.kts
index f286818006..bc21062a1e 100644
--- a/plugins/core/jetbrains-community/build.gradle.kts
+++ b/plugins/core/jetbrains-community/build.gradle.kts
@@ -73,6 +73,8 @@ dependencies {
testFixturesApi(libs.wiremock) {
// conflicts with transitive inclusion from docker plugin
exclude(group = "org.apache.httpcomponents.client5")
+ // provided by IDE
+ exclude(group = "commons-io")
}
testImplementation(project(":plugin-core:core"))
diff --git a/plugins/core/jetbrains-community/resources/META-INF/aws.toolkit.core.xml b/plugins/core/jetbrains-community/resources/META-INF/aws.toolkit.core.xml
index 2ae2fdc75a..88b37350a6 100644
--- a/plugins/core/jetbrains-community/resources/META-INF/aws.toolkit.core.xml
+++ b/plugins/core/jetbrains-community/resources/META-INF/aws.toolkit.core.xml
@@ -66,6 +66,7 @@
+
@@ -77,6 +78,9 @@
restartRequired="true"/>
+
+
diff --git a/plugins/core/jetbrains-community/resources/icons/resources/codewhisperer/severity-initial-critical.svg b/plugins/core/jetbrains-community/resources/icons/resources/codewhisperer/severity-initial-critical.svg
new file mode 100644
index 0000000000..7733994d24
--- /dev/null
+++ b/plugins/core/jetbrains-community/resources/icons/resources/codewhisperer/severity-initial-critical.svg
@@ -0,0 +1,4 @@
+
+
+
+
diff --git a/plugins/core/jetbrains-community/resources/icons/resources/codewhisperer/severity-initial-high.svg b/plugins/core/jetbrains-community/resources/icons/resources/codewhisperer/severity-initial-high.svg
new file mode 100644
index 0000000000..ff92aebc81
--- /dev/null
+++ b/plugins/core/jetbrains-community/resources/icons/resources/codewhisperer/severity-initial-high.svg
@@ -0,0 +1,4 @@
+
+
+
+
diff --git a/plugins/core/jetbrains-community/resources/icons/resources/codewhisperer/severity-initial-info.svg b/plugins/core/jetbrains-community/resources/icons/resources/codewhisperer/severity-initial-info.svg
new file mode 100644
index 0000000000..dbf7860917
--- /dev/null
+++ b/plugins/core/jetbrains-community/resources/icons/resources/codewhisperer/severity-initial-info.svg
@@ -0,0 +1,4 @@
+
+
+
+
diff --git a/plugins/core/jetbrains-community/resources/icons/resources/codewhisperer/severity-initial-low.svg b/plugins/core/jetbrains-community/resources/icons/resources/codewhisperer/severity-initial-low.svg
new file mode 100644
index 0000000000..4ca6d96961
--- /dev/null
+++ b/plugins/core/jetbrains-community/resources/icons/resources/codewhisperer/severity-initial-low.svg
@@ -0,0 +1,4 @@
+
+
+
+
diff --git a/plugins/core/jetbrains-community/resources/icons/resources/codewhisperer/severity-initial-medium.svg b/plugins/core/jetbrains-community/resources/icons/resources/codewhisperer/severity-initial-medium.svg
new file mode 100644
index 0000000000..a906d9b487
--- /dev/null
+++ b/plugins/core/jetbrains-community/resources/icons/resources/codewhisperer/severity-initial-medium.svg
@@ -0,0 +1,4 @@
+
+
+
+
diff --git a/plugins/core/jetbrains-community/resources/telemetryOverride.json b/plugins/core/jetbrains-community/resources/telemetryOverride.json
index d0fb37e5ce..39e9d4ba31 100644
--- a/plugins/core/jetbrains-community/resources/telemetryOverride.json
+++ b/plugins/core/jetbrains-community/resources/telemetryOverride.json
@@ -160,6 +160,11 @@
],
"description": "Identifies the specific interaction that opens the chat panel"
},
+ {
+ "name": "executedCount",
+ "type": "int",
+ "description": "The number of executed operations"
+ },
{
"name": "reAuth",
"type": "boolean",
@@ -172,6 +177,21 @@
"reloaded"
],
"description": "Toolkit run state."
+ },
+ {
+ "name": "filePath",
+ "type": "string",
+ "description": "The path of the file"
+ },
+ {
+ "name": "workspaceRoot",
+ "type": "string",
+ "description": "The path of the project root"
+ },
+ {
+ "name": "relativePath",
+ "type": "string",
+ "description": "The relative path of the file"
}
],
"metrics": [
@@ -525,6 +545,24 @@
}
],
"passive": true
+ },
+ {
+ "name": "codewhisperer_invalidZip",
+ "description": "Invalid zip file",
+ "metadata": [
+ {
+ "type": "filePath",
+ "required": true
+ },
+ {
+ "type": "workspaceRoot",
+ "required": true
+ },
+ {
+ "type": "relativePath",
+ "required": true
+ }
+ ]
}
]
}
diff --git a/plugins/core/jetbrains-community/src/icons/AwsIcons.kt b/plugins/core/jetbrains-community/src/icons/AwsIcons.kt
index 024c57f80c..3ed96be9f5 100644
--- a/plugins/core/jetbrains-community/src/icons/AwsIcons.kt
+++ b/plugins/core/jetbrains-community/src/icons/AwsIcons.kt
@@ -115,6 +115,8 @@ object AwsIcons {
object CodeWhisperer {
@JvmField val CUSTOM = load("icons/resources/CodewhispererCustom.svg") // 16 * 16
+ // Icons with full severity string
+
@JvmField val SEVERITY_INFO = load("/icons/resources/codewhisperer/severity-info.svg")
@JvmField val SEVERITY_LOW = load("/icons/resources/codewhisperer/severity-low.svg")
@@ -124,6 +126,18 @@ object AwsIcons {
@JvmField val SEVERITY_HIGH = load("/icons/resources/codewhisperer/severity-high.svg")
@JvmField val SEVERITY_CRITICAL = load("/icons/resources/codewhisperer/severity-critical.svg")
+
+ // Icons with severity initials
+
+ @JvmField val SEVERITY_INITIAL_INFO = load("/icons/resources/codewhisperer/severity-initial-info.svg")
+
+ @JvmField val SEVERITY_INITIAL_LOW = load("/icons/resources/codewhisperer/severity-initial-low.svg")
+
+ @JvmField val SEVERITY_INITIAL_MEDIUM = load("/icons/resources/codewhisperer/severity-initial-medium.svg")
+
+ @JvmField val SEVERITY_INITIAL_HIGH = load("/icons/resources/codewhisperer/severity-initial-high.svg")
+
+ @JvmField val SEVERITY_INITIAL_CRITICAL = load("/icons/resources/codewhisperer/severity-initial-critical.svg")
}
}
diff --git a/plugins/core/jetbrains-community/src/migration/software/aws/toolkits/jetbrains/settings/AwsSettings.kt b/plugins/core/jetbrains-community/src/migration/software/aws/toolkits/jetbrains/settings/AwsSettings.kt
index 6e7852c736..929810fa29 100644
--- a/plugins/core/jetbrains-community/src/migration/software/aws/toolkits/jetbrains/settings/AwsSettings.kt
+++ b/plugins/core/jetbrains-community/src/migration/software/aws/toolkits/jetbrains/settings/AwsSettings.kt
@@ -6,7 +6,7 @@ package migration.software.aws.toolkits.jetbrains.settings
import com.intellij.openapi.components.service
import software.aws.toolkits.jetbrains.settings.ProfilesNotification
import software.aws.toolkits.jetbrains.settings.UseAwsCredentialRegion
-import java.util.*
+import java.util.UUID
interface AwsSettings {
var isTelemetryEnabled: Boolean
diff --git a/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/AwsToolkit.kt b/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/AwsToolkit.kt
index 05d1e21e37..883c64a905 100644
--- a/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/AwsToolkit.kt
+++ b/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/AwsToolkit.kt
@@ -28,6 +28,7 @@ object AwsToolkit {
const val GITHUB_URL = "https://github.com/aws/aws-toolkit-jetbrains"
const val AWS_DOCS_URL = "https://docs.aws.amazon.com/console/toolkit-for-jetbrains"
+ const val GITHUB_CHANGELOG = "https://github.com/aws/aws-toolkit-jetbrains/blob/main/CHANGELOG.md"
}
data class PluginInfo(val id: String, val name: String) {
diff --git a/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/DefaultRemoteResourceResolverProvider.kt b/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/DefaultRemoteResourceResolverProvider.kt
index 3e570126b6..af45361ac3 100644
--- a/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/DefaultRemoteResourceResolverProvider.kt
+++ b/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/DefaultRemoteResourceResolverProvider.kt
@@ -4,6 +4,7 @@
package software.aws.toolkits.jetbrains.core
import com.intellij.openapi.application.PathManager
+import com.intellij.util.io.HttpRequests
import com.intellij.util.io.createDirectories
import software.aws.toolkits.core.utils.DefaultRemoteResourceResolver
import software.aws.toolkits.core.utils.UrlFetcher
@@ -38,6 +39,13 @@ class DefaultRemoteResourceResolverProvider : RemoteResourceResolverProvider {
override fun fetch(url: String, file: Path) {
saveFileFromUrl(url, file)
}
+
+ override fun getETag(url: String): String =
+ HttpRequests.head(url)
+ .userAgent("AWS Toolkit for JetBrains")
+ .connect { request ->
+ request.connection.headerFields["ETag"]?.firstOrNull().orEmpty()
+ }
}
}
}
diff --git a/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/credentials/ToolkitConnectionImpls.kt b/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/credentials/ToolkitConnectionImpls.kt
index fb89054b5f..f7c85b05e6 100644
--- a/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/credentials/ToolkitConnectionImpls.kt
+++ b/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/credentials/ToolkitConnectionImpls.kt
@@ -3,6 +3,7 @@
package software.aws.toolkits.jetbrains.core.credentials
+import com.fasterxml.jackson.annotation.JsonIgnore
import com.intellij.openapi.Disposable
import com.intellij.openapi.util.Disposer
import software.aws.toolkits.core.TokenConnectionSettings
@@ -73,6 +74,7 @@ sealed class ManagedBearerSsoConnection(
region
)
+ @JsonIgnore
override fun getConnectionSettings(): TokenConnectionSettings = provider
override fun dispose() {
@@ -99,6 +101,7 @@ class DetectedDiskSsoSessionConnection(
region
)
+ @JsonIgnore
override fun getConnectionSettings(): TokenConnectionSettings = provider
override fun dispose() {
diff --git a/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/credentials/actions/SsoLogoutAction.kt b/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/credentials/actions/SsoLogoutAction.kt
index 49eb918a14..1c03c39b87 100644
--- a/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/credentials/actions/SsoLogoutAction.kt
+++ b/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/credentials/actions/SsoLogoutAction.kt
@@ -10,7 +10,6 @@ import com.intellij.openapi.ui.MessageDialogBuilder
import software.aws.toolkits.jetbrains.core.credentials.AwsBearerTokenConnection
import software.aws.toolkits.jetbrains.core.credentials.ProfileSsoManagedBearerSsoConnection
import software.aws.toolkits.jetbrains.core.credentials.ToolkitConnectionManagerListener
-import software.aws.toolkits.jetbrains.core.credentials.deleteSsoConnection
import software.aws.toolkits.jetbrains.core.credentials.logoutFromSsoConnection
import software.aws.toolkits.resources.AwsCoreBundle
import software.aws.toolkits.telemetry.UiTelemetry
@@ -23,8 +22,8 @@ class SsoLogoutAction(private val value: AwsBearerTokenConnection) : DumbAwareAc
AwsCoreBundle.message("gettingstarted.auth.idc.sign.out.confirmation.title"),
AwsCoreBundle.message("gettingstarted.auth.idc.sign.out.confirmation")
).yesText(AwsCoreBundle.message("general.confirm")).ask(e.project)
- if (confirmDeletion) {
- deleteSsoConnection(value)
+ if (!confirmDeletion) {
+ return
}
}
logoutFromSsoConnection(e.project, value)
diff --git a/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/credentials/sso/bearer/BearerTokenProvider.kt b/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/credentials/sso/bearer/BearerTokenProvider.kt
index 8cdecbc64a..420613c2d6 100644
--- a/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/credentials/sso/bearer/BearerTokenProvider.kt
+++ b/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/credentials/sso/bearer/BearerTokenProvider.kt
@@ -48,6 +48,9 @@ interface BearerTokenProvider : SdkTokenProvider, SdkAutoCloseable, ToolkitBeare
*/
fun currentToken(): AccessToken?
+ /**
+ * Not meant to be invoked outside the implementation
+ */
fun refresh(): AccessToken
/**
@@ -58,13 +61,13 @@ interface BearerTokenProvider : SdkTokenProvider, SdkAutoCloseable, ToolkitBeare
/**
* Request provider to interactively request user input to obtain a new [AccessToken]
*/
- open fun reauthenticate() {
+ fun reauthenticate() {
throw UnsupportedOperationException("Provider is not interactive and cannot reauthenticate")
}
- open fun supportsLogout() = this is BearerTokenLogoutSupport
+ fun supportsLogout() = this is BearerTokenLogoutSupport
- open fun invalidate() {
+ fun invalidate() {
throw UnsupportedOperationException("Provider is not interactive and cannot be invalidated")
}
@@ -90,10 +93,9 @@ class InteractiveBearerTokenProvider(
val startUrl: String,
val region: String,
val scopes: List,
- id: String,
+ override val id: String,
cache: DiskCache = diskCache,
) : BearerTokenProvider, BearerTokenLogoutSupport, Disposable {
- override val id = id
override val displayName = ToolkitBearerTokenProvider.ssoDisplayName(startUrl)
private val ssoOidcClient: SsoOidcClient = buildUnmanagedSsoOidcClient(region)
@@ -107,7 +109,7 @@ class InteractiveBearerTokenProvider(
)
private val supplier = CachedSupplier.builder { refreshToken() }.prefetchStrategy(NonBlocking("AWS SSO bearer token refresher")).build()
- private val lastToken = AtomicReference()
+ internal val lastToken = AtomicReference()
val pendingAuthorization: PendingAuthorization?
get() = accessTokenProvider.authorization
@@ -134,6 +136,7 @@ class InteractiveBearerTokenProvider(
)
}
+ // we need to seed CachedSupplier with an initial value, then subsequent calls need to hit the network
private fun refreshToken(): RefreshResult {
val lastToken = lastToken.get() ?: throw NoTokenInitializedException("Token refresh started before session initialized")
val token = if (Duration.between(Instant.now(), lastToken.expiresAt) > Duration.ofMinutes(30)) {
@@ -148,6 +151,7 @@ class InteractiveBearerTokenProvider(
.build()
}
+ // how we expect consumers to obtain a token
override fun resolveToken() = supplier.get()
override fun close() {
@@ -159,6 +163,7 @@ class InteractiveBearerTokenProvider(
close()
}
+ // internal nonsense so we can query the token without triggering a refresh
override fun currentToken() = lastToken.get()
/**
@@ -189,7 +194,7 @@ class InteractiveBearerTokenProvider(
class NoTokenInitializedException(message: String) : Exception(message)
-public enum class BearerTokenAuthState {
+enum class BearerTokenAuthState {
AUTHORIZED,
NEEDS_REFRESH,
NOT_AUTHENTICATED,
diff --git a/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/notifications/CustomizeNotificationsUi.kt b/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/notifications/CustomizeNotificationsUi.kt
new file mode 100644
index 0000000000..ba18c46d60
--- /dev/null
+++ b/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/notifications/CustomizeNotificationsUi.kt
@@ -0,0 +1,130 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.core.notifications
+
+import com.intellij.icons.AllIcons
+import com.intellij.ide.BrowserUtil
+import com.intellij.openapi.actionSystem.AnAction
+import com.intellij.openapi.actionSystem.AnActionEvent
+import com.intellij.openapi.application.runInEdt
+import com.intellij.openapi.progress.ProgressIndicator
+import com.intellij.openapi.progress.ProgressManager
+import com.intellij.openapi.progress.Task
+import com.intellij.openapi.project.Project
+import com.intellij.openapi.ui.Messages
+import com.intellij.ui.EditorNotificationPanel
+import software.aws.toolkits.jetbrains.AwsPlugin
+import software.aws.toolkits.jetbrains.AwsToolkit
+import software.aws.toolkits.jetbrains.core.plugin.PluginUpdateManager
+import software.aws.toolkits.resources.AwsCoreBundle
+
+fun checkSeverity(notificationSeverity: String): NotificationSeverity = when (notificationSeverity) {
+ "Critical" -> NotificationSeverity.CRITICAL
+ "Warning" -> NotificationSeverity.WARNING
+ "Info" -> NotificationSeverity.INFO
+ else -> NotificationSeverity.INFO
+}
+
+object NotificationManager {
+ fun createActions(
+ project: Project,
+ followupActions: List?,
+ message: String,
+ title: String,
+
+ ): List = buildList {
+ var url: String? = null
+ followupActions?.forEach { action ->
+ if (action.type == "ShowUrl") {
+ url = action.content.locale.url
+ }
+
+ if (action.type == "UpdateExtension") {
+ add(
+ NotificationActionList(AwsCoreBundle.message("notification.update")) {
+ updatePlugins()
+ }
+ )
+ }
+
+ if (action.type == "OpenChangelog") {
+ add(
+ NotificationActionList(AwsCoreBundle.message("notification.changelog")) {
+ BrowserUtil.browse(AwsToolkit.GITHUB_CHANGELOG)
+ }
+ )
+ }
+ }
+ add(
+ NotificationActionList(AwsCoreBundle.message("general.more_dialog")) {
+ if (url == null) {
+ Messages.showMessageDialog(
+ project,
+ message,
+ title,
+ AllIcons.General.Error
+ )
+ } else {
+ val link = url ?: AwsToolkit.GITHUB_URL
+ val openLink = Messages.showYesNoDialog(
+ project,
+ message,
+ title,
+ AwsCoreBundle.message(AwsCoreBundle.message("notification.learn_more")),
+ AwsCoreBundle.message("general.cancel"),
+ AllIcons.General.Error
+ )
+ if (openLink == 0) {
+ BrowserUtil.browse(link)
+ }
+ }
+ }
+ )
+ }
+
+ fun buildNotificationActions(actions: List): List = actions.map { (title, block) ->
+ object : AnAction(title) {
+ override fun actionPerformed(e: AnActionEvent) {
+ block()
+ }
+ }
+ }
+
+ fun buildBannerPanel(panel: EditorNotificationPanel, actions: List): EditorNotificationPanel {
+ actions.forEach { (actionTitle, block) ->
+ panel.createActionLabel(actionTitle) {
+ block()
+ }
+ }
+
+ return panel
+ }
+ private fun updatePlugins() {
+ val pluginUpdateManager = PluginUpdateManager()
+ runInEdt {
+ ProgressManager.getInstance().run(object : Task.Backgroundable(
+ null,
+ AwsCoreBundle.message("aws.settings.auto_update.progress.message")
+ ) {
+ override fun run(indicator: ProgressIndicator) {
+ pluginUpdateManager.checkForUpdates(indicator, AwsPlugin.CORE)
+ pluginUpdateManager.checkForUpdates(indicator, AwsPlugin.TOOLKIT)
+ pluginUpdateManager.checkForUpdates(indicator, AwsPlugin.Q)
+ }
+ })
+ }
+ }
+}
+
+data class NotificationActionList(
+ val title: String,
+ val blockToExecute: () -> Unit,
+)
+
+data class BannerContent(
+ val title: String,
+ val message: String,
+ val actions: List,
+ val id: String,
+)
diff --git a/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/notifications/DisplayToastNotifications.kt b/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/notifications/DisplayToastNotifications.kt
new file mode 100644
index 0000000000..870672f78a
--- /dev/null
+++ b/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/notifications/DisplayToastNotifications.kt
@@ -0,0 +1,6 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.core.notifications
+
+object DisplayToastNotifications
diff --git a/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/notifications/NotificationCustomDeserializers.kt b/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/notifications/NotificationCustomDeserializers.kt
new file mode 100644
index 0000000000..ef0316b318
--- /dev/null
+++ b/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/notifications/NotificationCustomDeserializers.kt
@@ -0,0 +1,127 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.core.notifications
+
+import com.fasterxml.jackson.core.JsonParser
+import com.fasterxml.jackson.core.JsonToken
+import com.fasterxml.jackson.databind.DeserializationContext
+import com.fasterxml.jackson.databind.JsonDeserializer
+import com.fasterxml.jackson.databind.JsonMappingException
+import com.fasterxml.jackson.databind.JsonNode
+
+class OperationConditionDeserializer : JsonDeserializer() {
+ override fun deserialize(parser: JsonParser, ctxt: DeserializationContext): NotificationExpression.OperationCondition = when (parser.currentToken) {
+ JsonToken.VALUE_STRING -> {
+ // Handle direct string value
+ NotificationExpression.OperationCondition(parser.valueAsString)
+ }
+ else -> throw JsonMappingException(parser, "Cannot deserialize OperatingCondition")
+ }
+}
+
+class ComparisonConditionDeserializer : JsonDeserializer() {
+ override fun deserialize(parser: JsonParser, ctxt: DeserializationContext): NotificationExpression.ComparisonCondition {
+ val op = OperationConditionDeserializer().deserialize(parser, ctxt)
+ return NotificationExpression.ComparisonCondition(op.value)
+ }
+}
+
+class NotEqualsConditionDeserializer : JsonDeserializer() {
+ override fun deserialize(parser: JsonParser, ctxt: DeserializationContext): NotificationExpression.NotEqualsCondition {
+ val op = OperationConditionDeserializer().deserialize(parser, ctxt)
+ return NotificationExpression.NotEqualsCondition(op.value)
+ }
+}
+class GreaterThanConditionDeserializer : JsonDeserializer() {
+ override fun deserialize(parser: JsonParser, ctxt: DeserializationContext): NotificationExpression.GreaterThanCondition {
+ val op = OperationConditionDeserializer().deserialize(parser, ctxt)
+ return NotificationExpression.GreaterThanCondition(op.value)
+ }
+}
+class GreaterThanOrEqualsConditionDeserializer : JsonDeserializer() {
+ override fun deserialize(parser: JsonParser, ctxt: DeserializationContext): NotificationExpression.GreaterThanOrEqualsCondition {
+ val op = OperationConditionDeserializer().deserialize(parser, ctxt)
+ return NotificationExpression.GreaterThanOrEqualsCondition(op.value)
+ }
+}
+class LessThanConditionDeserializer : JsonDeserializer() {
+ override fun deserialize(parser: JsonParser, ctxt: DeserializationContext): NotificationExpression.LessThanCondition {
+ val op = OperationConditionDeserializer().deserialize(parser, ctxt)
+ return NotificationExpression.LessThanCondition(op.value)
+ }
+}
+class LessThanOrEqualsConditionDeserializer : JsonDeserializer() {
+ override fun deserialize(parser: JsonParser, ctxt: DeserializationContext): NotificationExpression.LessThanOrEqualsCondition {
+ val op = OperationConditionDeserializer().deserialize(parser, ctxt)
+ return NotificationExpression.LessThanOrEqualsCondition(op.value)
+ }
+}
+class ComplexOperationConditionDeserializer : JsonDeserializer() {
+ override fun deserialize(parser: JsonParser, ctxt: DeserializationContext): NotificationExpression.ComplexOperationCondition {
+ val node = parser.codec.readTree(parser)
+ if (!node.isArray) {
+ throw JsonMappingException(parser, "anyOf/noneOf must contain an array of values")
+ }
+ val values = node.map { it.asText() }
+ return NotificationExpression.ComplexOperationCondition(values)
+ }
+}
+class AnyOfConditionDeserializer : JsonDeserializer() {
+ override fun deserialize(parser: JsonParser, ctxt: DeserializationContext): NotificationExpression.AnyOfCondition {
+ val op = ComplexOperationConditionDeserializer().deserialize(parser, ctxt)
+ return NotificationExpression.AnyOfCondition(op.value)
+ }
+}
+
+class NoneOfConditionDeserializer : JsonDeserializer() {
+ override fun deserialize(parser: JsonParser, ctxt: DeserializationContext): NotificationExpression.NoneOfCondition {
+ val op = ComplexOperationConditionDeserializer().deserialize(parser, ctxt)
+ return NotificationExpression.NoneOfCondition(op.value)
+ }
+}
+
+class ComplexConditionDeserializer : JsonDeserializer() {
+ override fun deserialize(parser: JsonParser, ctxt: DeserializationContext): NotificationExpression.ComplexCondition {
+ val node = parser.codec.readTree(parser)
+ if (!node.isArray) {
+ throw JsonMappingException(parser, "or/and must contain an array of values")
+ }
+ return NotificationExpression.ComplexCondition(node.toNotificationExpressions(parser))
+ }
+}
+class OrConditionDeserializer : JsonDeserializer() {
+ override fun deserialize(parser: JsonParser, ctxt: DeserializationContext): NotificationExpression.OrCondition {
+ val op = ComplexConditionDeserializer().deserialize(parser, ctxt)
+ return NotificationExpression.OrCondition(op.expectedValueList)
+ }
+}
+
+class AndConditionDeserializer : JsonDeserializer() {
+ override fun deserialize(parser: JsonParser, ctxt: DeserializationContext): NotificationExpression.AndCondition {
+ val op = ComplexConditionDeserializer().deserialize(parser, ctxt)
+ return NotificationExpression.AndCondition(op.expectedValueList)
+ }
+}
+
+class NotConditionDeserializer : JsonDeserializer() {
+ override fun deserialize(p: JsonParser, ctxt: DeserializationContext): NotificationExpression.NotCondition {
+ val node = p.codec.readTree(p)
+ val parser = node.traverse(p.codec)
+ parser.nextToken()
+
+ return NotificationExpression.NotCondition(parser.readValueAs(NotificationExpression::class.java))
+ }
+}
+
+// Create a custom deserializer if needed
+class NotificationTypeDeserializer : JsonDeserializer() {
+ override fun deserialize(p: JsonParser, ctxt: DeserializationContext): NotificationScheduleType =
+ NotificationScheduleType.fromString(p.valueAsString)
+}
+
+private fun JsonNode.toNotificationExpressions(p: JsonParser): List = this.map { element ->
+ val parser = element.traverse(p.codec)
+ parser.nextToken()
+ parser.readValueAs(NotificationExpression::class.java)
+}
diff --git a/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/notifications/NotificationFormatUtils.kt b/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/notifications/NotificationFormatUtils.kt
new file mode 100644
index 0000000000..f582ce7d32
--- /dev/null
+++ b/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/notifications/NotificationFormatUtils.kt
@@ -0,0 +1,200 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.core.notifications
+
+import com.fasterxml.jackson.annotation.JsonProperty
+import com.fasterxml.jackson.annotation.JsonSubTypes
+import com.fasterxml.jackson.annotation.JsonTypeInfo
+import com.fasterxml.jackson.databind.annotation.JsonDeserialize
+
+data class NotificationsList(
+ val schema: Schema,
+ val notifications: List?,
+)
+
+data class Schema(
+ val version: String,
+)
+
+data class NotificationData(
+ val id: String,
+ val schedule: NotificationSchedule,
+ val severity: String,
+ val condition: NotificationDisplayCondition?,
+ val content: NotificationContentDescriptionLocale,
+ val actions: List? = emptyList(),
+)
+
+data class NotificationSchedule(
+ @JsonDeserialize(using = NotificationTypeDeserializer::class)
+ val type: NotificationScheduleType,
+) {
+ constructor(type: String) : this(NotificationScheduleType.fromString(type))
+}
+
+enum class NotificationSeverity {
+ INFO,
+ WARNING,
+ CRITICAL,
+}
+
+enum class NotificationScheduleType {
+ STARTUP,
+ EMERGENCY,
+ ;
+
+ companion object {
+ fun fromString(value: String): NotificationScheduleType =
+ when (value.lowercase()) {
+ "startup" -> STARTUP
+ else -> EMERGENCY
+ }
+ }
+}
+
+data class NotificationContentDescriptionLocale(
+ @JsonProperty("en-US")
+ val locale: NotificationContentDescription,
+)
+
+data class NotificationContentDescription(
+ val title: String,
+ val description: String,
+)
+
+data class NotificationFollowupActions(
+ val type: String,
+ val content: NotificationFollowupActionsContent,
+)
+
+data class NotificationFollowupActionsContent(
+ @JsonProperty("en-US")
+ val locale: NotificationActionDescription,
+)
+
+data class NotificationActionDescription(
+ val title: String,
+ val url: String?,
+)
+
+data class NotificationDisplayCondition(
+ val compute: ComputeType?,
+ val os: SystemType?,
+ val ide: SystemType?,
+ val extension: List?,
+ val authx: List?,
+)
+
+data class ComputeType(
+ val type: NotificationExpression?,
+ val architecture: NotificationExpression?,
+)
+
+data class SystemType(
+ val type: NotificationExpression?,
+ val version: NotificationExpression?,
+)
+
+data class ExtensionType(
+ val id: String?,
+ val version: NotificationExpression?,
+)
+
+@JsonTypeInfo(
+ use = JsonTypeInfo.Id.NAME,
+ include = JsonTypeInfo.As.WRAPPER_OBJECT
+)
+@JsonSubTypes(
+ JsonSubTypes.Type(value = NotificationExpression.ComparisonCondition::class, name = "=="),
+ JsonSubTypes.Type(value = NotificationExpression.NotEqualsCondition::class, name = "!="),
+ JsonSubTypes.Type(value = NotificationExpression.GreaterThanCondition::class, name = ">"),
+ JsonSubTypes.Type(value = NotificationExpression.GreaterThanOrEqualsCondition::class, name = ">="),
+ JsonSubTypes.Type(value = NotificationExpression.LessThanCondition::class, name = "<"),
+ JsonSubTypes.Type(value = NotificationExpression.LessThanOrEqualsCondition::class, name = "<="),
+ JsonSubTypes.Type(value = NotificationExpression.AnyOfCondition::class, name = "anyOf"),
+ JsonSubTypes.Type(value = NotificationExpression.NotCondition::class, name = "not"),
+ JsonSubTypes.Type(value = NotificationExpression.OrCondition::class, name = "or"),
+ JsonSubTypes.Type(value = NotificationExpression.AndCondition::class, name = "and"),
+ JsonSubTypes.Type(value = NotificationExpression.NoneOfCondition::class, name = "noneOf")
+)
+sealed interface NotificationExpression {
+ @JsonDeserialize(using = NotConditionDeserializer::class)
+ data class NotCondition(
+ val expectedValue: NotificationExpression,
+ ) : NotificationExpression
+
+ @JsonDeserialize(using = OrConditionDeserializer::class)
+ data class OrCondition(
+ val expectedValueList: List,
+ ) : NotificationExpression
+
+ @JsonDeserialize(using = AndConditionDeserializer::class)
+ data class AndCondition(
+ val expectedValueList: List,
+ ) : NotificationExpression
+
+ @JsonDeserialize(using = ComplexConditionDeserializer::class)
+ data class ComplexCondition(
+ val expectedValueList: List,
+ ) : NotificationExpression
+
+ // General class for comparison operators
+ @JsonDeserialize(using = OperationConditionDeserializer::class)
+ data class OperationCondition(
+ val value: String,
+ ) : NotificationExpression
+
+ @JsonDeserialize(using = ComplexOperationConditionDeserializer::class)
+ data class ComplexOperationCondition(
+ val value: List,
+ ) : NotificationExpression
+
+ @JsonDeserialize(using = ComparisonConditionDeserializer::class)
+ data class ComparisonCondition(
+ val value: String,
+ ) : NotificationExpression
+
+ @JsonDeserialize(using = NotEqualsConditionDeserializer::class)
+ data class NotEqualsCondition(
+ val value: String,
+ ) : NotificationExpression
+
+ @JsonDeserialize(using = GreaterThanConditionDeserializer::class)
+ data class GreaterThanCondition(
+ val value: String,
+ ) : NotificationExpression
+
+ @JsonDeserialize(using = GreaterThanOrEqualsConditionDeserializer::class)
+ data class GreaterThanOrEqualsCondition(
+ val value: String,
+ ) : NotificationExpression
+
+ @JsonDeserialize(using = LessThanConditionDeserializer::class)
+ data class LessThanCondition(
+ val value: String,
+ ) : NotificationExpression
+
+ @JsonDeserialize(using = LessThanOrEqualsConditionDeserializer::class)
+ data class LessThanOrEqualsCondition(
+ val value: String,
+ ) : NotificationExpression
+
+ @JsonDeserialize(using = AnyOfConditionDeserializer::class)
+ data class AnyOfCondition(
+ val value: List,
+ ) : NotificationExpression
+
+ @JsonDeserialize(using = NoneOfConditionDeserializer::class)
+ data class NoneOfCondition(
+ val value: List,
+ ) : NotificationExpression
+}
+
+data class AuthxType(
+ val feature: String,
+ val type: NotificationExpression?,
+ val region: NotificationExpression?,
+ val connectionState: NotificationExpression?,
+ val ssoScopes: NotificationExpression?,
+)
diff --git a/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/notifications/NotificationPanel.kt b/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/notifications/NotificationPanel.kt
new file mode 100644
index 0000000000..e9b6564adc
--- /dev/null
+++ b/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/notifications/NotificationPanel.kt
@@ -0,0 +1,40 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.core.notifications
+
+import com.intellij.icons.AllIcons
+import com.intellij.openapi.application.runInEdt
+import com.intellij.ui.EditorNotificationPanel
+import com.intellij.ui.components.panels.Wrapper
+import com.intellij.util.ui.components.BorderLayoutPanel
+import software.aws.toolkits.resources.AwsCoreBundle
+
+class NotificationPanel : BorderLayoutPanel() {
+ private val wrapper = Wrapper()
+ init {
+ isOpaque = false
+ addToCenter(wrapper)
+ BannerNotificationService.getInstance().getNotifications().forEach { (_, content) ->
+ updateNotificationPanel(content)
+ }
+ }
+
+ private fun removeNotificationPanel(notificationId: String) = runInEdt {
+ BannerNotificationService.getInstance().removeNotification(notificationId)
+ NotificationDismissalState.getInstance().dismissNotification(notificationId)
+ wrapper.removeAll()
+ }
+
+ fun updateNotificationPanel(bannerContent: BannerContent) {
+ val panel = EditorNotificationPanel()
+ panel.text = bannerContent.title
+ panel.icon(AllIcons.General.Error)
+ val panelWithActions = NotificationManager.buildBannerPanel(panel, bannerContent.actions)
+ panelWithActions.createActionLabel(AwsCoreBundle.message("general.dismiss")) {
+ removeNotificationPanel(bannerContent.id)
+ }
+
+ wrapper.setContent(panelWithActions)
+ }
+}
diff --git a/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/notifications/NotificationPollingService.kt b/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/notifications/NotificationPollingService.kt
new file mode 100644
index 0000000000..2709aa180a
--- /dev/null
+++ b/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/notifications/NotificationPollingService.kt
@@ -0,0 +1,143 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.core.notifications
+
+import com.fasterxml.jackson.module.kotlin.readValue
+import com.intellij.openapi.Disposable
+import com.intellij.openapi.application.ApplicationManager
+import com.intellij.openapi.components.Service
+import com.intellij.openapi.util.registry.Registry
+import com.intellij.util.Alarm
+import com.intellij.util.AlarmFactory
+import kotlinx.coroutines.delay
+import kotlinx.coroutines.runBlocking
+import software.aws.toolkits.core.utils.RemoteResolveParser
+import software.aws.toolkits.core.utils.RemoteResource
+import software.aws.toolkits.core.utils.UpdateCheckResult
+import software.aws.toolkits.core.utils.getLogger
+import software.aws.toolkits.core.utils.info
+import software.aws.toolkits.core.utils.warn
+import software.aws.toolkits.jetbrains.core.DefaultRemoteResourceResolverProvider
+import software.aws.toolkits.jetbrains.core.RemoteResourceResolverProvider
+import software.aws.toolkits.telemetry.Component
+import software.aws.toolkits.telemetry.ToolkitTelemetry
+import java.io.InputStream
+import java.time.Duration
+
+private const val MAX_RETRIES = 3
+private const val RETRY_DELAY_MS = 1000L
+internal const val FILENAME = "notifications.json"
+
+object NotificationFileValidator : RemoteResolveParser {
+ override fun canBeParsed(data: InputStream): Boolean =
+ try {
+ NotificationMapperUtil.mapper.readValue(data)
+ true
+ } catch (e: Exception) {
+ false
+ }
+}
+
+object NotificationEndpoint {
+ fun getEndpoint(): String =
+ Registry.get("aws.toolkit.notification.endpoint").asString()
+}
+
+@Service(Service.Level.APP)
+internal final class NotificationPollingService : Disposable {
+ private val observers = mutableListOf<() -> Unit>()
+ private val alarm = AlarmFactory.getInstance().create(Alarm.ThreadToUse.POOLED_THREAD, this)
+ private val pollingIntervalMs = Duration.ofMinutes(10).toMillis()
+ private val resourceResolver: RemoteResourceResolverProvider = DefaultRemoteResourceResolverProvider()
+ private val notificationsResource = object : RemoteResource {
+ override val name: String = FILENAME
+ override val urls: List = listOf(NotificationEndpoint.getEndpoint())
+ override val remoteResolveParser: RemoteResolveParser = NotificationFileValidator
+ override val ttl: Duration = Duration.ofMillis(1)
+ // ttl forces resolver to fetch from endpoint every time
+ }
+
+ fun startPolling() {
+ val newNotifications = runBlocking { pollForNotifications() }
+ if (newNotifications) {
+ notifyObservers()
+ }
+ alarm.addRequest(
+ { startPolling() },
+ pollingIntervalMs
+ )
+ }
+
+ private suspend fun pollForNotifications(): Boolean {
+ var retryCount = 0
+ var lastException: Exception? = null
+ while (retryCount < MAX_RETRIES) {
+ LOG.info { "Polling for notifications" }
+ try {
+ when (
+ resourceResolver.get().checkForUpdates(
+ NotificationEndpoint.getEndpoint(),
+ NotificationEtagState.getInstance()
+ )
+ ) {
+ is UpdateCheckResult.HasUpdates -> {
+ resourceResolver.get()
+ .resolve(notificationsResource)
+ .toCompletableFuture()
+ .get()
+ LOG.info { "New notifications fetched" }
+ return true
+ }
+ is UpdateCheckResult.FirstPollCheck -> {
+ LOG.info { "No new notifications, checking cached notifications on first poll" }
+ return true
+ }
+ is UpdateCheckResult.NoUpdates -> {
+ LOG.info { "No new notifications to fetch" }
+ return false
+ }
+ }
+ } catch (e: Exception) {
+ lastException = e
+ LOG.warn { "Failed to poll for notifications (attempt ${retryCount + 1}/$MAX_RETRIES)" }
+ retryCount++
+ if (retryCount < MAX_RETRIES) {
+ val backoffDelay = RETRY_DELAY_MS * (1L shl (retryCount - 1))
+ delay(backoffDelay)
+ }
+ }
+ }
+ emitFailureMetric(lastException)
+ return false
+ }
+
+ private fun emitFailureMetric(e: Exception?) {
+ ToolkitTelemetry.showNotification(
+ project = null,
+ component = Component.Filesystem,
+ id = "",
+ reason = "Failed to poll for notifications",
+ success = false,
+ reasonDesc = "${e?.javaClass?.simpleName ?: "Unknown"}: ${e?.message ?: "No message"}",
+ )
+ }
+
+ fun addObserver(observer: () -> Unit) = observers.add(observer)
+
+ private fun notifyObservers() {
+ observers.forEach { observer ->
+ observer()
+ }
+ }
+
+ override fun dispose() {
+ alarm.dispose()
+ }
+
+ companion object {
+ private val LOG = getLogger()
+ fun getInstance(): NotificationPollingService =
+ ApplicationManager.getApplication().getService(NotificationPollingService::class.java)
+ }
+}
diff --git a/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/notifications/NotificationServiceInitializer.kt b/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/notifications/NotificationServiceInitializer.kt
new file mode 100644
index 0000000000..150e256379
--- /dev/null
+++ b/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/notifications/NotificationServiceInitializer.kt
@@ -0,0 +1,23 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.core.notifications
+
+import com.intellij.openapi.application.ApplicationManager
+import com.intellij.openapi.project.Project
+import com.intellij.openapi.startup.ProjectActivity
+import java.util.concurrent.atomic.AtomicBoolean
+
+internal class NotificationServiceInitializer : ProjectActivity {
+
+ private val initialized = AtomicBoolean(false)
+
+ override suspend fun execute(project: Project) {
+ ProcessNotificationsBase.getInstance(project)
+ if (ApplicationManager.getApplication().isUnitTestMode) return
+ if (initialized.compareAndSet(false, true)) {
+ val service = NotificationPollingService.getInstance()
+ service.startPolling()
+ }
+ }
+}
diff --git a/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/notifications/NotificationStateUtils.kt b/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/notifications/NotificationStateUtils.kt
new file mode 100644
index 0000000000..7c86de0420
--- /dev/null
+++ b/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/notifications/NotificationStateUtils.kt
@@ -0,0 +1,109 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.core.notifications
+
+import com.intellij.openapi.components.PersistentStateComponent
+import com.intellij.openapi.components.Service
+import com.intellij.openapi.components.State
+import com.intellij.openapi.components.Storage
+import com.intellij.openapi.components.service
+import software.aws.toolkits.core.utils.ETagProvider
+import java.time.Duration
+import java.time.Instant
+
+data class DismissedNotification(
+ var id: String = "",
+ var dismissedAt: String = Instant.now().toEpochMilli().toString(),
+)
+
+data class NotificationDismissalConfiguration(
+ var dismissedNotifications: MutableSet = mutableSetOf(),
+)
+
+@Service
+@State(name = "notificationDismissals", storages = [Storage("aws.xml")])
+class NotificationDismissalState : PersistentStateComponent {
+ private var state = NotificationDismissalConfiguration()
+ private val retentionPeriod = Duration.ofDays(60) // 2 months
+
+ override fun getState(): NotificationDismissalConfiguration = state
+
+ override fun loadState(state: NotificationDismissalConfiguration) {
+ this.state = state
+ cleanExpiredNotifications()
+ }
+
+ fun isDismissed(notificationId: String): Boolean =
+ state.dismissedNotifications.any { it.id == notificationId }
+
+ fun dismissNotification(notificationId: String) {
+ state.dismissedNotifications.add(
+ DismissedNotification(
+ id = notificationId
+ )
+ )
+ }
+
+ private fun cleanExpiredNotifications() {
+ val now = Instant.now()
+ state.dismissedNotifications.removeAll { notification ->
+ Duration.between(Instant.ofEpochMilli(notification.dismissedAt.toLong()), now) > retentionPeriod
+ }
+ }
+
+ companion object {
+ fun getInstance(): NotificationDismissalState = service()
+ }
+}
+
+@Service
+@State(name = "notificationEtag", storages = [Storage("aws.xml")])
+class NotificationEtagState : PersistentStateComponent, ETagProvider {
+ private val state = NotificationEtagConfiguration()
+
+ override fun updateETag(newETag: String?) {
+ etag = newETag
+ }
+
+ override fun getState(): NotificationEtagConfiguration = state
+
+ override fun loadState(state: NotificationEtagConfiguration) {
+ this.state.etag = state.etag
+ }
+
+ override var etag: String?
+ get() = state.etag
+ set(value) {
+ state.etag = value
+ }
+
+ companion object {
+ fun getInstance(): NotificationEtagState =
+ service()
+ }
+}
+
+data class NotificationEtagConfiguration(
+ var etag: String? = null,
+)
+
+@Service
+class BannerNotificationService {
+ private val notifications = mutableMapOf()
+
+ fun addNotification(id: String, content: BannerContent) {
+ notifications[id] = content
+ }
+
+ fun getNotifications(): Map = notifications
+
+ fun removeNotification(id: String) {
+ notifications.remove(id)
+ }
+
+ companion object {
+ fun getInstance(): BannerNotificationService =
+ service()
+ }
+}
diff --git a/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/notifications/ProcessNotificationsBase.kt b/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/notifications/ProcessNotificationsBase.kt
new file mode 100644
index 0000000000..1313f2b687
--- /dev/null
+++ b/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/notifications/ProcessNotificationsBase.kt
@@ -0,0 +1,139 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.core.notifications
+
+import com.fasterxml.jackson.databind.DeserializationFeature
+import com.fasterxml.jackson.databind.ObjectMapper
+import com.fasterxml.jackson.module.kotlin.jacksonObjectMapper
+import com.fasterxml.jackson.module.kotlin.readValue
+import com.intellij.notification.NotificationType
+import com.intellij.openapi.actionSystem.AnAction
+import com.intellij.openapi.components.Service
+import com.intellij.openapi.components.service
+import com.intellij.openapi.project.Project
+import software.aws.toolkits.core.utils.getLogger
+import software.aws.toolkits.core.utils.info
+import software.aws.toolkits.core.utils.inputStream
+import software.aws.toolkits.core.utils.warn
+import software.aws.toolkits.jetbrains.core.RemoteResourceResolverProvider
+import software.aws.toolkits.jetbrains.utils.notifyStickyWithData
+import software.aws.toolkits.telemetry.Component
+import software.aws.toolkits.telemetry.Result
+import software.aws.toolkits.telemetry.ToolkitTelemetry
+import java.util.concurrent.atomic.AtomicBoolean
+
+object NotificationMapperUtil {
+ val mapper: ObjectMapper = jacksonObjectMapper().configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false)
+}
+private var isStartup: AtomicBoolean = AtomicBoolean(true)
+
+@Service(Service.Level.PROJECT)
+class ProcessNotificationsBase(
+ private val project: Project,
+) {
+ private val notifListener = mutableListOf()
+ init {
+ LOG.info { "Initializing ProcessNotificationsBase" }
+ NotificationPollingService.getInstance().addObserver {
+ retrieveStartupAndEmergencyNotifications()
+ }
+ }
+
+ private fun getNotificationsFromFile(): NotificationsList? {
+ try {
+ val path = RemoteResourceResolverProvider
+ .getInstance()
+ .get()
+ .getLocalResourcePath(FILENAME)
+ if (path == null) {
+ LOG.warn { "Notifications file not found" }
+ return null
+ }
+ val content = path.inputStream().bufferedReader().use { it.readText() }
+ if (content.isEmpty()) {
+ return null
+ }
+ return NotificationMapperUtil.mapper.readValue(content)
+ } catch (e: Exception) {
+ LOG.warn { "Error reading notifications file: $e" }
+ return null
+ }
+ }
+
+ fun retrieveStartupAndEmergencyNotifications() {
+ val isStartupPoll = isStartup.compareAndSet(true, false)
+ LOG.info { "Retrieving notifications for processing. StartUp notifications included: $isStartupPoll" }
+ val notifications = getNotificationsFromFile()
+ notifications?.let { notificationsList ->
+ val activeNotifications = notificationsList.notifications
+ ?.filter { notification ->
+ // Keep notification if:
+ // - it's not a startup notification, OR
+ // - it is a startup notification AND this is the first poll
+ notification.schedule.type != NotificationScheduleType.STARTUP || isStartupPoll
+ }
+ ?.filter { notification ->
+ !NotificationDismissalState.getInstance().isDismissed(notification.id)
+ }
+ .orEmpty()
+
+ activeNotifications.forEach { processNotification(project, it) }
+ }
+ LOG.info { "Finished processing notifications" }
+ }
+
+ fun processNotification(project: Project, notificationData: NotificationData) {
+ val shouldShow = RulesEngine.displayNotification(project, notificationData)
+ if (shouldShow) {
+ LOG.info { "Showing notification with id: ${notificationData.id}" }
+ val notificationContent = notificationData.content.locale
+ val severity = notificationData.severity
+ val followupActions = NotificationManager.createActions(
+ project,
+ notificationData.actions,
+ notificationContent.description,
+ notificationContent.title
+ )
+ showToast(
+ notificationContent.title,
+ notificationContent.description,
+ NotificationManager.buildNotificationActions(followupActions),
+ checkSeverity(severity),
+ notificationData.id
+ )
+ if (severity == "Critical") {
+ val bannerContent = BannerContent(notificationContent.title, notificationContent.description, followupActions, notificationData.id)
+ BannerNotificationService.getInstance().addNotification(notificationData.id, bannerContent)
+ notifyListenerForNotification(bannerContent)
+ }
+ ToolkitTelemetry.showNotification(
+ id = notificationData.id,
+ result = Result.Succeeded,
+ component = Component.Infobar
+ )
+ }
+ }
+
+ private fun showToast(title: String, message: String, action: List, notificationType: NotificationSeverity, notificationId: String) {
+ val notifyType = when (notificationType) {
+ NotificationSeverity.CRITICAL -> NotificationType.ERROR
+ NotificationSeverity.WARNING -> NotificationType.WARNING
+ NotificationSeverity.INFO -> NotificationType.INFORMATION
+ }
+ notifyStickyWithData(notifyType, title, message, null, action, notificationId)
+ }
+
+ fun notifyListenerForNotification(bannerContent: BannerContent) =
+ notifListener.forEach { it(bannerContent) }
+
+ fun addListenerForNotification(newNotifListener: NotifListener) =
+ notifListener.add(newNotifListener)
+
+ companion object {
+ private val LOG = getLogger()
+ fun getInstance(project: Project): ProcessNotificationsBase = project.service()
+ }
+}
+
+typealias NotifListener = (bannerContent: BannerContent) -> Unit
diff --git a/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/notifications/RulesEngine.kt b/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/notifications/RulesEngine.kt
new file mode 100644
index 0000000000..4f5c174409
--- /dev/null
+++ b/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/notifications/RulesEngine.kt
@@ -0,0 +1,210 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.core.notifications
+
+import com.intellij.ide.plugins.PluginManagerCore
+import com.intellij.openapi.application.ApplicationInfo
+import com.intellij.openapi.extensions.PluginId
+import com.intellij.openapi.project.Project
+import com.intellij.openapi.util.SystemInfo
+import software.aws.toolkits.jetbrains.core.gettingstarted.editor.ActiveConnection
+import software.aws.toolkits.jetbrains.core.gettingstarted.editor.ActiveConnectionType
+import software.aws.toolkits.jetbrains.core.gettingstarted.editor.BearerTokenFeatureSet
+import software.aws.toolkits.jetbrains.core.gettingstarted.editor.checkBearerConnectionValidity
+import software.aws.toolkits.jetbrains.core.gettingstarted.editor.checkIamProfileByCredentialType
+import software.aws.toolkits.jetbrains.utils.isRunningOnRemoteBackend
+
+object RulesEngine {
+
+ fun displayNotification(project: Project, notification: NotificationData): Boolean {
+ // If no conditions provided, show display the notification to everyone
+ val shouldShow = notification.condition?.let { matchesAllRules(it, project) } ?: true
+ return shouldShow
+ }
+
+ fun matchesAllRules(notificationConditions: NotificationDisplayCondition, project: Project): Boolean {
+ val sysDetails = getCurrentSystemAndConnectionDetails()
+ // If any of these conditions are null, we assume the condition to be true
+ val compute = notificationConditions.compute?.let { matchesCompute(it, sysDetails.computeType, sysDetails.computeArchitecture) } ?: true
+ val os = notificationConditions.os?.let { matchesOs(it, sysDetails.osType, sysDetails.osVersion) } ?: true
+ val ide = notificationConditions.ide?.let { matchesIde(it, sysDetails.ideType, sysDetails.ideVersion) } ?: true
+ val extension = matchesExtension(notificationConditions.extension, sysDetails.pluginVersions)
+ val authx = matchesAuth(notificationConditions.authx, project)
+ return compute && os && ide && extension && authx
+ }
+
+ private fun matchesCompute(notificationCompute: ComputeType, actualCompute: String, actualArchitecture: String): Boolean {
+ val type = notificationCompute.type?.let { evaluateNotificationExpression(it, actualCompute) } ?: true
+ val architecture = notificationCompute.architecture?.let { evaluateNotificationExpression(it, actualArchitecture) } ?: true
+ return type && architecture
+ }
+
+ private fun matchesOs(notificationOs: SystemType, actualOs: String, actualOsVersion: String): Boolean {
+ val os = notificationOs.type?.let { evaluateNotificationExpression(it, actualOs) } ?: true
+ val osVersion = notificationOs.version?.let { evaluateNotificationExpression(it, actualOsVersion) } ?: true
+ return os && osVersion
+ }
+
+ private fun matchesIde(notificationIde: SystemType, actualIde: String, actualIdeVersion: String): Boolean {
+ val ide = notificationIde.type?.let { evaluateNotificationExpression(it, actualIde) } ?: true
+ val ideVersion = notificationIde.version?.let { evaluateNotificationExpression(it, actualIdeVersion) } ?: true
+ return ide && ideVersion
+ }
+
+ private fun matchesExtension(notificationExtension: List?, actualPluginVersions: Map): Boolean {
+ if (notificationExtension.isNullOrEmpty()) return true
+ val extensionsToBeChecked = notificationExtension.map { it.id }
+ val pluginVersions = actualPluginVersions.filterKeys { extensionsToBeChecked.contains(it) }
+ if (pluginVersions.isEmpty()) return false
+ return notificationExtension.all { extension ->
+ val actualVersion = pluginVersions[extension.id]
+ if (actualVersion == null) {
+ true
+ } else {
+ extension.version?.let { evaluateNotificationExpression(it, actualVersion) } ?: true
+ }
+ }
+ }
+
+ private fun matchesAuth(notificationAuth: List?, project: Project): Boolean {
+ if (notificationAuth.isNullOrEmpty()) return true
+ return notificationAuth.all { feature ->
+ val actualConnection = when (feature.feature) {
+ "q" -> getConnectionDetailsForFeature(project, BearerTokenFeatureSet.Q)
+ "codeCatalyst" -> getConnectionDetailsForFeature(project, BearerTokenFeatureSet.CODECATALYST)
+ "toolkit" -> getConnectionDetailsForToolkit(project)
+ else -> return true
+ }
+
+ if (actualConnection == null) {
+ false
+ } else {
+ val authType = feature.type?.let { evaluateNotificationExpression(it, actualConnection.connectionType) } ?: true
+ val authRegion = feature.region?.let { evaluateNotificationExpression(it, actualConnection.region) } ?: true
+ val connectionState = feature.connectionState?.let { evaluateNotificationExpression(it, actualConnection.connectionState) } ?: true
+ // TODO: Add condition for matching scopes
+ authType && authRegion && connectionState
+ }
+ }
+ }
+
+ private fun evaluateNotificationExpression(notificationExpression: NotificationExpression, value: String): Boolean = when (notificationExpression) {
+ is NotificationExpression.NotCondition -> performNotOp(notificationExpression, value)
+ is NotificationExpression.OrCondition -> performOrOp(notificationExpression, value)
+ is NotificationExpression.AndCondition -> performAndOp(notificationExpression, value)
+ is NotificationExpression.ComparisonCondition -> notificationExpression.value == value
+ is NotificationExpression.NotEqualsCondition -> notificationExpression.value != value
+ is NotificationExpression.GreaterThanCondition -> value > notificationExpression.value
+ is NotificationExpression.LessThanCondition -> value < notificationExpression.value
+ is NotificationExpression.GreaterThanOrEqualsCondition -> value >= notificationExpression.value
+ is NotificationExpression.LessThanOrEqualsCondition -> value <= notificationExpression.value
+ is NotificationExpression.AnyOfCondition -> notificationExpression.value.contains(value)
+ is NotificationExpression.NoneOfCondition -> !notificationExpression.value.contains(value)
+ else -> true
+ }
+
+ private fun performNotOp(notificationOperation: NotificationExpression.NotCondition, actualValue: String): Boolean =
+ !evaluateNotificationExpression(notificationOperation.expectedValue, actualValue)
+
+ private fun performOrOp(notificationOperation: NotificationExpression.OrCondition, actualValue: String): Boolean =
+ notificationOperation.expectedValueList.any { evaluateNotificationExpression(it, actualValue) }
+
+ private fun performAndOp(notificationOperation: NotificationExpression.AndCondition, actualValue: String): Boolean =
+ notificationOperation.expectedValueList.all { evaluateNotificationExpression(it, actualValue) }
+}
+
+fun getCurrentSystemAndConnectionDetails(): SystemDetails {
+ val computeType: String = if (isRunningOnRemoteBackend()) "Remote" else "Local"
+ val computeArchitecture: String = SystemInfo.OS_ARCH
+
+ val osType: String = SystemInfo.OS_NAME
+ val osVersion: String = SystemInfo.OS_VERSION
+
+ val ideInfo = ApplicationInfo.getInstance()
+ val ideType: String = ideInfo.build.productCode
+ val ideVersion = ideInfo.shortVersion
+
+ val pluginVersionMap = createPluginVersionMap()
+
+ return SystemDetails(computeType, computeArchitecture, osType, osVersion, ideType, ideVersion, pluginVersionMap)
+}
+
+data class FeatureAuthDetails(
+ val connectionType: String,
+ val region: String,
+ val connectionState: String,
+)
+
+data class SystemDetails(
+ val computeType: String,
+ val computeArchitecture: String,
+ val osType: String,
+ val osVersion: String,
+ val ideType: String,
+ val ideVersion: String,
+ val pluginVersions: Map,
+)
+
+fun createPluginVersionMap(): Map {
+ val pluginVersionMap = mutableMapOf()
+ val pluginIds = listOf(
+ "amazon.q",
+ "aws.toolkit.core",
+ "aws.toolkit"
+ )
+ pluginIds.forEach { pluginId ->
+ val plugin = PluginManagerCore.getPlugin(PluginId.getId(pluginId))
+ val pluginVersion = plugin?.version
+ if (pluginVersion != null) {
+ pluginVersionMap[pluginId] = pluginVersion
+ }
+ }
+ return pluginVersionMap
+}
+
+private fun getConnectionDetailsForToolkit(project: Project): FeatureAuthDetails? {
+ val connection = checkIamProfileByCredentialType(project)
+ if (connection.activeConnectionIam == null) return null
+ val authType = when (connection.connectionType) {
+ ActiveConnectionType.IAM_IDC -> "Idc"
+ ActiveConnectionType.IAM -> "Iam"
+ else -> "Unknown"
+ }
+ val authRegion = connection.activeConnectionIam?.defaultRegionId ?: "Unknown"
+
+ val connectionState = when (connection) {
+ is ActiveConnection.NotConnected -> "NotConnected"
+ is ActiveConnection.ValidIam -> "Connected"
+ is ActiveConnection.ExpiredIam -> "Expired"
+ else -> "Unknown"
+ }
+ return FeatureAuthDetails(
+ authType,
+ authRegion,
+ connectionState
+ )
+}
+
+fun getConnectionDetailsForFeature(project: Project, featureId: BearerTokenFeatureSet): FeatureAuthDetails? {
+ val connection = checkBearerConnectionValidity(project, featureId)
+ if (connection.activeConnectionBearer == null) return null
+ val authType = when (connection.connectionType) {
+ ActiveConnectionType.BUILDER_ID -> "BuilderId"
+ ActiveConnectionType.IAM_IDC -> "Idc"
+ else -> "Unknown"
+ }
+ val authRegion = connection.activeConnectionBearer?.region ?: "Unknown"
+
+ val connectionState = when (connection) {
+ is ActiveConnection.NotConnected -> "NotConnected"
+ is ActiveConnection.ValidBearer -> "Connected"
+ is ActiveConnection.ExpiredBearer -> "Expired"
+ else -> "Unknown"
+ }
+ return FeatureAuthDetails(
+ authType,
+ authRegion,
+ connectionState
+ )
+}
diff --git a/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/plugin/PluginUpdateManager.kt b/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/plugin/PluginUpdateManager.kt
index 80dea49e17..384c38bf1e 100644
--- a/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/plugin/PluginUpdateManager.kt
+++ b/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/plugin/PluginUpdateManager.kt
@@ -200,7 +200,7 @@ class PluginUpdateManager : Disposable {
// TODO: Optimize this to only search the result for AWS plugins
fun getUpdateInfo(): Collection = UpdateChecker.getPluginUpdates() ?: emptyList()
- internal fun updatePlugin(pluginDescriptor: IdeaPluginDescriptor, progressIndicator: ProgressIndicator): Boolean {
+ fun updatePlugin(pluginDescriptor: IdeaPluginDescriptor, progressIndicator: ProgressIndicator): Boolean {
val pluginName = pluginDescriptor.name
// wasUpdatedWithRestart means that, it was an update and it needs to restart to apply
diff --git a/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/webview/LoginBrowser.kt b/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/webview/LoginBrowser.kt
index dc308c1d51..e3c6e81fba 100644
--- a/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/webview/LoginBrowser.kt
+++ b/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/core/webview/LoginBrowser.kt
@@ -130,6 +130,7 @@ abstract class LoginBrowser(
protected val onPendingToken: (InteractiveBearerTokenProvider) -> Unit = { provider ->
startBrowserOpenTimer(provider.startUrl, provider.region, provider.scopes)
+
projectCoroutineScope(project).launch {
val authorization = pollForAuthorization(provider)
if (authorization != null) {
diff --git a/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/FeatureDevSessionContext.kt b/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/FeatureDevSessionContext.kt
index ee3a10bf70..23fb7a3e28 100644
--- a/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/FeatureDevSessionContext.kt
+++ b/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/services/amazonq/FeatureDevSessionContext.kt
@@ -16,9 +16,7 @@ import kotlinx.coroutines.launch
import kotlinx.coroutines.runBlocking
import kotlinx.coroutines.withContext
import org.apache.commons.codec.digest.DigestUtils
-import software.aws.toolkits.core.utils.outputStream
-import software.aws.toolkits.core.utils.putNextEntry
-import software.aws.toolkits.jetbrains.core.coroutines.EDT
+import org.apache.commons.io.FileUtils
import software.aws.toolkits.jetbrains.core.coroutines.getCoroutineBgContext
import software.aws.toolkits.jetbrains.services.telemetry.ALLOWED_CODE_EXTENSIONS
import software.aws.toolkits.jetbrains.utils.isDevFile
@@ -26,12 +24,19 @@ import software.aws.toolkits.resources.AwsCoreBundle
import software.aws.toolkits.telemetry.AmazonqTelemetry
import java.io.File
import java.io.FileInputStream
+import java.net.URI
+import java.nio.file.FileSystem
+import java.nio.file.FileSystems
import java.nio.file.Files
import java.nio.file.Path
+import java.nio.file.Paths
+import java.nio.file.StandardCopyOption
import java.util.Base64
-import java.util.zip.ZipOutputStream
+import java.util.UUID
import kotlin.coroutines.coroutineContext
import kotlin.io.path.Path
+import kotlin.io.path.createParentDirectories
+import kotlin.io.path.getPosixFilePermissions
import kotlin.io.path.relativeTo
interface RepoSizeError {
@@ -41,35 +46,46 @@ class RepoSizeLimitError(override val message: String) : RuntimeException(), Rep
class FeatureDevSessionContext(val project: Project, val maxProjectSizeBytes: Long? = null) {
// TODO: Need to correct this class location in the modules going further to support both amazonq and codescan.
+ private val requiredBinaryFilesForExecution = setOf("gradle/wrapper/gradle-wrapper.jar")
- private val ignorePatterns = setOf(
- "\\.aws-sam",
- "\\.svn",
- "\\.hg/?",
- "\\.rvm",
- "\\.git/?",
- "\\.project",
- "\\.gem",
- "/\\.idea/?",
- "\\.zip$",
- "\\.bin$",
- "\\.png$",
- "\\.jpg$",
- "\\.svg$",
- "\\.pyc$",
- "/license\\.txt$",
- "/License\\.txt$",
- "/LICENSE\\.txt$",
- "/license\\.md$",
- "/License\\.md$",
- "/LICENSE\\.md$",
- "node_modules/?",
- "build/?",
- "dist/?"
- ).map { Regex(it) }
+ private val additionalGitIgnoreRules = setOf(
+ ".aws-sam",
+ ".gem",
+ ".git",
+ ".gradle",
+ ".hg",
+ ".idea",
+ ".project",
+ ".rvm",
+ ".svn",
+ "*.zip",
+ "*.bin",
+ "*.png",
+ "*.jpg",
+ "*.svg",
+ "*.pyc",
+ "license.txt",
+ "License.txt",
+ "LICENSE.txt",
+ "license.md",
+ "License.md",
+ "LICENSE.md",
+ "node_modules",
+ "build",
+ "dist"
+ )
+
+ // well known source files that do not have extensions
+ private val wellKnownSourceFiles = setOf(
+ "Dockerfile",
+ "Dockerfile.build",
+ "gradlew",
+ "mvnw"
+ )
// projectRoot: is the directory where the project is located when selected to open a project.
val projectRoot = project.guessProjectDir() ?: error("Cannot guess base directory for project ${project.name}")
+ private val projectRootPath = Paths.get(projectRoot.path) ?: error("Can not find project root path")
// selectedSourceFolder: is the directory selected in replacement of the root, this happens when the project is too big to bundle for uploading.
private var _selectedSourceFolder = projectRoot
@@ -77,7 +93,16 @@ class FeatureDevSessionContext(val project: Project, val maxProjectSizeBytes: Lo
private val gitIgnoreFile = File(selectedSourceFolder.path, ".gitignore")
init {
- ignorePatternsWithGitIgnore = (ignorePatterns + parseGitIgnore().map { Regex(it) }).toList()
+ ignorePatternsWithGitIgnore = try {
+ buildList {
+ addAll(additionalGitIgnoreRules.map { convertGitIgnorePatternToRegex(it) })
+ addAll(parseGitIgnore())
+ }.mapNotNull { pattern ->
+ runCatching { Regex(pattern) }.getOrNull()
+ }
+ } catch (e: Exception) {
+ emptyList()
+ }
}
// This function checks for existence of `devfile.yaml` in customer's repository, currently only `devfile.yaml` is supported for this feature.
@@ -101,7 +126,6 @@ class FeatureDevSessionContext(val project: Project, val maxProjectSizeBytes: Lo
fun isFileExtensionAllowed(file: VirtualFile): Boolean {
// if it is a directory, it is allowed
if (file.isDirectory) return true
-
val extension = file.extension ?: return false
return ALLOWED_CODE_EXTENSIONS.contains(extension)
}
@@ -109,13 +133,24 @@ class FeatureDevSessionContext(val project: Project, val maxProjectSizeBytes: Lo
private fun ignoreFileByExtension(file: VirtualFile) =
!isFileExtensionAllowed(file)
- suspend fun ignoreFile(file: VirtualFile): Boolean = ignoreFile(file.path)
+ suspend fun ignoreFile(file: VirtualFile): Boolean = ignoreFile(file.presentableUrl)
suspend fun ignoreFile(path: String): Boolean {
+ // explicitly allow the Gradle wrapper JAR file
+ if (requiredBinaryFilesForExecution.any { path.endsWith(it) }) {
+ return false
+ }
+
// this method reads like something a JS dev would write and doesn't do what the author thinks
val deferredResults = ignorePatternsWithGitIgnore.map { pattern ->
withContext(coroutineContext) {
- async { pattern.containsMatchIn(path) }
+ // avoid partial match (pattern.containsMatchIn) since it causes us matching files
+ // against folder patterns. (e.g. settings.gradle ignored by .gradle rule!)
+ // we convert the glob rules to regex, add a trailing /* to all rules and then match
+ // entries against them by adding a trailing /.
+ // TODO: Add unit tests for gitignore matching
+ val relative = if (path.startsWith(projectRootPath.toString())) Paths.get(path).relativeTo(projectRootPath) else path
+ async { pattern.matches("$relative/") }
}
}
@@ -125,6 +160,8 @@ class FeatureDevSessionContext(val project: Project, val maxProjectSizeBytes: Lo
return deferredResults.any { it.await() }
}
+ fun wellKnown(file: VirtualFile): Boolean = wellKnownSourceFiles.contains(file.name)
+
suspend fun zipFiles(projectRoot: VirtualFile, isAutoBuildFeatureEnabled: Boolean?): File = withContext(getCoroutineBgContext()) {
val files = mutableListOf()
val ignoredExtensionMap = mutableMapOf().withDefault { 0L }
@@ -134,10 +171,11 @@ class FeatureDevSessionContext(val project: Project, val maxProjectSizeBytes: Lo
projectRoot,
object : VirtualFileVisitor() {
override fun visitFile(file: VirtualFile): Boolean {
+ val isWellKnown = runBlocking { wellKnown(file) }
val isFileIgnoredByExtension = runBlocking { ignoreFileByExtension(file) }
// if `isAutoBuildFeatureEnabled` is false, then filter devfile
val isFilterDevFile = if (isAutoBuildFeatureEnabled == true) false else isDevFile(file)
- if (isFileIgnoredByExtension) {
+ if (!isWellKnown && isFileIgnoredByExtension) {
val extension = file.extension.orEmpty()
ignoredExtensionMap[extension] = (ignoredExtensionMap[extension] ?: 0) + 1
return false
@@ -184,18 +222,43 @@ class FeatureDevSessionContext(val project: Project, val maxProjectSizeBytes: Lo
}
}
- createTemporaryZipFileAsync { zipOutput ->
+ val zipFilePath = createTemporaryZipFileAsync { zipfs ->
+ val posixFileAttributeSubstr = "posix"
+ val isPosix = FileSystems.getDefault().supportedFileAttributeViews().contains(posixFileAttributeSubstr)
filesToIncludeFlow.collect { file ->
- val relativePath = Path(file.path).relativeTo(projectRoot.toNioPath())
- zipOutput.putNextEntry(relativePath.toString(), Path(file.path))
+
+ if (!file.isDirectory) {
+ val externalFilePath = Path(file.path)
+ val relativePath = Path(file.path).relativeTo(projectRootPath)
+ val zipfsPath = zipfs.getPath("/$relativePath")
+ withContext(getCoroutineBgContext()) {
+ zipfsPath.createParentDirectories()
+ try {
+ Files.copy(externalFilePath, zipfsPath, StandardCopyOption.REPLACE_EXISTING)
+ if (isPosix) {
+ val zipPermissionAttributeName = "zip:permissions"
+ Files.setAttribute(zipfsPath, zipPermissionAttributeName, externalFilePath.getPosixFilePermissions())
+ }
+ } catch (e: NoSuchFileException) {
+ // Noop: Skip if file was deleted
+ }
+ }
+ }
}
}
+ zipFilePath
}.toFile()
- private suspend fun createTemporaryZipFileAsync(block: suspend (ZipOutputStream) -> Unit): Path = withContext(EDT) {
- val file = Files.createTempFile(null, ".zip")
- ZipOutputStream(file.outputStream()).use { zipOutput -> block(zipOutput) }
- file
+ private suspend fun createTemporaryZipFileAsync(block: suspend (FileSystem) -> Unit): Path = withContext(getCoroutineBgContext()) {
+ // Don't use Files.createTempFile since the file must not be created for ZipFS to work
+ val tempFilePath: Path = Paths.get(FileUtils.getTempDirectory().absolutePath, "${UUID.randomUUID()}.zip")
+ val uri = URI.create("jar:${tempFilePath.toUri()}")
+ val env = hashMapOf("create" to "true")
+ val zipfs = FileSystems.newFileSystem(uri, env)
+ zipfs.use {
+ block(zipfs)
+ }
+ tempFilePath
}
private fun parseGitIgnore(): Set {
@@ -213,7 +276,7 @@ class FeatureDevSessionContext(val project: Project, val maxProjectSizeBytes: Lo
private fun convertGitIgnorePatternToRegex(pattern: String): String = pattern
.replace(".", "\\.")
.replace("*", ".*")
- .let { if (it.endsWith("/")) "$it?" else it } // Handle directory-specific patterns by optionally matching trailing slash
+ .let { if (it.endsWith("/")) "$it.*" else "$it/.*" } // Add a trailing /* to all patterns. (we add a trailing / to all files when matching)
var selectedSourceFolder: VirtualFile
set(newRoot) {
diff --git a/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/services/telemetry/PluginResolver.kt b/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/services/telemetry/PluginResolver.kt
index f450145ae6..58154e319f 100644
--- a/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/services/telemetry/PluginResolver.kt
+++ b/plugins/core/jetbrains-community/src/software/aws/toolkits/jetbrains/services/telemetry/PluginResolver.kt
@@ -45,7 +45,7 @@ class PluginResolver private constructor(callerStackTrace: Array,
+ id: String,
+) {
+ val notification = Notification(GROUP_DISPLAY_ID_STICKY, title, content, type)
+ notificationActions.forEach {
+ notification.addAction(it)
+ }
+
+ notification.addAction(
+ createNotificationExpiringAction(
+ object : AnAction("Dismiss") {
+ override fun actionPerformed(e: AnActionEvent) {
+ BannerNotificationService.getInstance().removeNotification(id)
+ NotificationDismissalState.getInstance().dismissNotification(id)
+ }
+ }
+ )
+
+ )
+
+ notify(notification, project)
+}
+
private fun notifySticky(type: NotificationType, title: String, content: String = "", project: Project? = null, notificationActions: Collection) {
val notification = Notification(GROUP_DISPLAY_ID_STICKY, title, content, type)
notificationActions.forEach {
diff --git a/plugins/core/jetbrains-community/tst-resources/exampleNotification2.json b/plugins/core/jetbrains-community/tst-resources/exampleNotification2.json
new file mode 100644
index 0000000000..344f45191c
--- /dev/null
+++ b/plugins/core/jetbrains-community/tst-resources/exampleNotification2.json
@@ -0,0 +1,124 @@
+{
+ "schema": {
+ "version": "2.0"
+},
+ "notifications": [
+ {
+
+ "id": "example_id_12344",
+ "schedule": {
+ "type": "StartUp"
+ },
+ "severity": "Critical",
+ "condition": {
+ "compute": {
+ "type": {
+ "or": [
+ {
+ "==": "ec2"
+ },
+ {
+ "==": "desktop"
+ }
+ ]
+ },
+ "architecture": {
+ "!=": "x64"
+ }
+ },
+ "os": {
+ "type": {
+ "anyOf": [
+ "Darwin",
+ "Linux"
+ ]
+ },
+ "version": {
+ "<=": "23.0.1.0"
+ }
+
+ },
+ "ide": {
+ "type": {
+ "noneOf": [
+ "PyCharm",
+ "IDEA"
+ ]
+ },
+ "version": {
+ "and": [
+ {
+ ">=": "1.0"
+ },
+ {
+ "<": "2.0"
+ }
+ ]
+ }
+ },
+ "extensions": [
+ {
+ "id": "aws.toolkit",
+ "version": {
+ "!=": "1.3334"
+ }
+ },
+ {
+ "id": "amazon.q",
+ "version": {
+ "!=": "3.37.0"
+ }
+ }
+ ]
+
+ ,
+ "authx": [{
+ "feature" : "q",
+ "type": {
+ "anyOf": [
+ "IamIdentityCenter",
+ "AwsBuilderId"
+ ]
+ },
+ "region": {
+ "==": "us-east-1"
+ },
+ "connectionState": {
+ "!=": "Connected"
+ },
+ "ssoScopes": {
+ "noneOf": [
+ "codewhisperer:scope1",
+ "sso:account:access"
+ ]
+ }
+ } ]
+ },
+ "content": {
+ "en-US": {
+ "title": "Look at this!",
+ "description": "Some bug is there"
+ }
+ },
+ "actions": [
+ {
+ "type": "ShowMarketplace",
+ "content": {
+ "en-US": {
+ "title": "Go to market"
+ }
+ }
+ },
+ {
+ "type": "ShowUrl",
+ "content": {
+ "en-US": {
+ "title": "Click me!",
+ "url": "http://nowhere"
+ }
+ }
+ }
+ ]
+ }
+ ]
+}
diff --git a/plugins/core/jetbrains-community/tst-resources/olderNotification.json b/plugins/core/jetbrains-community/tst-resources/olderNotification.json
new file mode 100644
index 0000000000..9ceee415d3
--- /dev/null
+++ b/plugins/core/jetbrains-community/tst-resources/olderNotification.json
@@ -0,0 +1,115 @@
+{
+ "schema": {
+ "version": "2.0"
+ },
+ "notifications": [
+ {
+
+ "id": "example_id_12344",
+ "schedule": {
+ "type": "StartUp"
+ },
+ "severity": "Critical",
+ "condition": {
+ "compute": {
+ "type": {
+ "or": [
+ {
+ "==": "ec2"
+ },
+ {
+ "==": "desktop"
+ }
+ ]
+ },
+ "architecture": {
+ "!=": "x64"
+ }
+ },
+ "os": {
+ "type": {
+ "anyOf": [
+ "Darwin",
+ "Linux"
+ ]
+ },
+ "version": {
+ "<=": "23.0.1.0"
+ }
+
+ },
+ "ide": {
+ "type": {
+ "noneOf": [
+ "PyCharm",
+ "IDEA"
+ ]
+ },
+ "version": {
+ "and": [
+ {
+ ">=": "1.0"
+ },
+ {
+ "<": "2.0"
+ }
+ ]
+ }
+ },
+ "extension": {
+ "type": {
+ "==": "AWS Toolkit for JetBrains"
+ },
+ "version": {
+ "<": "1.47.0.0"
+ }
+ },
+ "authx": {
+ "type": {
+ "anyOf": [
+ "IamIdentityCenter",
+ "AwsBuilderId"
+ ]
+ },
+ "region": {
+ "==": "us-east-1"
+ },
+ "connectionState": {
+ "!=": "Connected"
+ },
+ "ssoScopes": {
+ "noneOf": [
+ "codewhisperer:scope1",
+ "sso:account:access"
+ ]
+ }
+ }
+ },
+ "content": {
+ "en-US": {
+ "title": "Look at this!",
+ "description": "Some bug is there"
+ }
+ },
+ "actions": [
+ {
+ "type": "ShowMarketplace",
+ "content": {
+ "en-US": {
+ "title": "Go to market"
+ }
+ }
+ },
+ {
+ "type": "ShowUrl",
+ "content": {
+ "en-US": {
+ "title": "Click me!",
+ "url": "http://nowhere"
+ }
+ }
+ }
+ ]
+ }
+ ]
+}
diff --git a/plugins/core/jetbrains-community/tst/software/aws/toolkits/jetbrains/core/credentials/DefaultToolkitAuthManagerTest.kt b/plugins/core/jetbrains-community/tst/software/aws/toolkits/jetbrains/core/credentials/DefaultToolkitAuthManagerTest.kt
index 3f5d496243..f63be201d7 100644
--- a/plugins/core/jetbrains-community/tst/software/aws/toolkits/jetbrains/core/credentials/DefaultToolkitAuthManagerTest.kt
+++ b/plugins/core/jetbrains-community/tst/software/aws/toolkits/jetbrains/core/credentials/DefaultToolkitAuthManagerTest.kt
@@ -3,6 +3,7 @@
package software.aws.toolkits.jetbrains.core.credentials
+import com.fasterxml.jackson.module.kotlin.jacksonObjectMapper
import com.intellij.openapi.Disposable
import com.intellij.openapi.application.ApplicationManager
import com.intellij.testFramework.ProjectExtension
@@ -453,6 +454,22 @@ class DefaultToolkitAuthManagerTest {
}
}
+ @Test
+ fun `serializing LegacyManagedBearerSsoConnection does not include connectionSettings`() {
+ val profile = ManagedSsoProfile("us-east-1", "startUrl000", listOf("scopes"))
+ val connection = sut.createConnection(profile) as LegacyManagedBearerSsoConnection
+
+ assertThat(jacksonObjectMapper().writeValueAsString(connection)).doesNotContain("connectionSettings")
+ }
+
+ @Test
+ fun `serializing ProfileSsoManagedBearerSsoConnection does not include connectionSettings`() {
+ val profile = UserConfigSsoSessionProfile("sessionName", "us-east-1", "startUrl000", listOf("scopes"))
+ val connection = sut.createConnection(profile) as ProfileSsoManagedBearerSsoConnection
+
+ assertThat(jacksonObjectMapper().writeValueAsString(connection)).doesNotContain("connectionSettings")
+ }
+
private companion object {
@ExtendWith(ProjectExtension::class)
val projectRule = ProjectRule()
diff --git a/plugins/core/jetbrains-community/tst/software/aws/toolkits/jetbrains/core/notifications/NotificationDismissalStateTest.kt b/plugins/core/jetbrains-community/tst/software/aws/toolkits/jetbrains/core/notifications/NotificationDismissalStateTest.kt
new file mode 100644
index 0000000000..eca7e1e6ca
--- /dev/null
+++ b/plugins/core/jetbrains-community/tst/software/aws/toolkits/jetbrains/core/notifications/NotificationDismissalStateTest.kt
@@ -0,0 +1,83 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.core.notifications
+
+import org.junit.jupiter.api.Assertions.assertEquals
+import org.junit.jupiter.api.Assertions.assertFalse
+import org.junit.jupiter.api.Assertions.assertTrue
+import org.junit.jupiter.api.BeforeEach
+import org.junit.jupiter.api.Test
+import java.time.Instant
+import java.time.temporal.ChronoUnit
+
+class NotificationDismissalStateTest {
+ private lateinit var state: NotificationDismissalState
+
+ @BeforeEach
+ fun setUp() {
+ state = NotificationDismissalState()
+ }
+
+ @Test
+ fun `notifications less than 2 months old are not removed`() {
+ val recentNotification = DismissedNotification(
+ id = "recent-notification",
+ dismissedAt = Instant.now().minus(30, ChronoUnit.DAYS).toEpochMilli().toString()
+ )
+
+ state.loadState(NotificationDismissalConfiguration(mutableSetOf(recentNotification)))
+
+ val persistedState = state.getState()
+
+ assertEquals(1, persistedState.dismissedNotifications.size)
+ assertTrue(persistedState.dismissedNotifications.any { it.id == "recent-notification" })
+ assertTrue(state.isDismissed("recent-notification"))
+ }
+
+ @Test
+ fun `notifications older than 2 months are removed`() {
+ val oldNotification = DismissedNotification(
+ id = "old-notification",
+ dismissedAt = Instant.now().minus(61, ChronoUnit.DAYS).toEpochMilli().toString()
+ )
+
+ state.loadState(NotificationDismissalConfiguration(mutableSetOf(oldNotification)))
+
+ val persistedState = state.getState()
+
+ assertEquals(0, persistedState.dismissedNotifications.size)
+ assertFalse(state.isDismissed("old-notification"))
+ }
+
+ @Test
+ fun `mixed age notifications are handled correctly`() {
+ val recentNotification = DismissedNotification(
+ id = "recent-notification",
+ dismissedAt = Instant.now().minus(30, ChronoUnit.DAYS).toEpochMilli().toString()
+ )
+ val oldNotification = DismissedNotification(
+ id = "old-notification",
+ dismissedAt = Instant.now().minus(61, ChronoUnit.DAYS).toEpochMilli().toString()
+ )
+
+ state.loadState(
+ NotificationDismissalConfiguration(
+ mutableSetOf(recentNotification, oldNotification)
+ )
+ )
+
+ val persistedState = state.getState()
+
+ assertEquals(1, persistedState.dismissedNotifications.size)
+ assertTrue(state.isDismissed("recent-notification"))
+ assertFalse(state.isDismissed("old-notification"))
+ }
+
+ @Test
+ fun `dismissing new notification retains it`() {
+ state.dismissNotification("new-notification")
+
+ assertTrue(state.isDismissed("new-notification"))
+ }
+}
diff --git a/plugins/core/jetbrains-community/tst/software/aws/toolkits/jetbrains/core/notifications/NotificationFormatUtilsTest.kt b/plugins/core/jetbrains-community/tst/software/aws/toolkits/jetbrains/core/notifications/NotificationFormatUtilsTest.kt
new file mode 100644
index 0000000000..74a7da0371
--- /dev/null
+++ b/plugins/core/jetbrains-community/tst/software/aws/toolkits/jetbrains/core/notifications/NotificationFormatUtilsTest.kt
@@ -0,0 +1,156 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.core.notifications
+
+import com.fasterxml.jackson.databind.DeserializationFeature
+import com.fasterxml.jackson.module.kotlin.jacksonObjectMapper
+import com.fasterxml.jackson.module.kotlin.readValue
+import com.intellij.testFramework.ApplicationExtension
+import com.intellij.testFramework.ProjectRule
+import io.mockk.every
+import io.mockk.mockkStatic
+import io.mockk.unmockkAll
+import org.assertj.core.api.Assertions.assertThat
+import org.junit.Rule
+import org.junit.jupiter.api.AfterEach
+import org.junit.jupiter.api.BeforeEach
+import org.junit.jupiter.api.Test
+import org.junit.jupiter.api.assertDoesNotThrow
+import org.junit.jupiter.api.assertThrows
+import org.junit.jupiter.api.extension.ExtendWith
+import org.junit.jupiter.params.ParameterizedTest
+import org.junit.jupiter.params.provider.Arguments
+import org.junit.jupiter.params.provider.MethodSource
+import software.aws.toolkits.core.utils.exists
+import software.aws.toolkits.core.utils.inputStream
+import software.aws.toolkits.jetbrains.core.gettingstarted.editor.BearerTokenFeatureSet
+import java.io.InputStream
+import java.nio.file.Paths
+import java.util.stream.Stream
+
+@ExtendWith(ApplicationExtension::class)
+class NotificationFormatUtilsTest {
+ @Rule
+ @JvmField
+ val projectRule = ProjectRule()
+
+ private lateinit var mockSystemDetails: SystemDetails
+ private lateinit var mockSystemDetailsWithNoPlugin: SystemDetails
+ private lateinit var exampleNotification: InputStream
+
+ @BeforeEach
+ fun setUp() {
+ mockSystemDetails = SystemDetails(
+ computeType = "Local",
+ computeArchitecture = "x86_64",
+ osType = "Linux",
+ osVersion = "5.4.0",
+ ideType = "IC",
+ ideVersion = "2023.1",
+ pluginVersions = mapOf(
+ "aws.toolkit" to "1.0",
+ "amazon.q" to "2.0"
+ )
+ )
+
+ mockSystemDetailsWithNoPlugin = SystemDetails(
+ computeType = "Local",
+ computeArchitecture = "x86_64",
+ osType = "Linux",
+ osVersion = "5.4.0",
+ ideType = "IC",
+ ideVersion = "2023.1",
+ pluginVersions = mapOf(
+ "aws.toolkit" to "1.0",
+ )
+ )
+
+ exampleNotification = javaClass.getResource("/exampleNotification2.json")?.let {
+ Paths.get(it.toURI()).takeIf { f -> f.exists() }
+ }?.inputStream() ?: throw RuntimeException("Test not found")
+
+ mockkStatic("software.aws.toolkits.jetbrains.core.notifications.RulesEngineKt")
+ every { getCurrentSystemAndConnectionDetails() } returns mockSystemDetails
+ every { getConnectionDetailsForFeature(projectRule.project, BearerTokenFeatureSet.Q) } returns FeatureAuthDetails(
+ "Idc",
+ "us-west-2",
+ "Connected"
+ )
+ }
+
+ @AfterEach
+ fun tearDown() {
+ unmockkAll()
+ }
+
+ @Test
+ fun `test System Details`() {
+ val result = getCurrentSystemAndConnectionDetails()
+ assertThat(mockSystemDetails).isEqualTo(result)
+ }
+
+ @Test
+ fun `check Json Validity which has all the required fields`() {
+ assertDoesNotThrow {
+ mapper.readValue(exampleNotification)
+ }
+ }
+
+ @Test
+ fun `No schema version associated with the notification file throws an exception`() {
+ assertThrows {
+ mapper.readValue(exampleNotificationWithoutSchema)
+ }
+ }
+
+ @Test
+ fun `No notifications present with the version file does not throw an exception`() {
+ assertDoesNotThrow {
+ mapper.readValue(exampleNotificationWithoutNotification)
+ }
+ }
+
+ @Test
+ fun `If plugin is not present, notification is not shown`() {
+ every { getCurrentSystemAndConnectionDetails() } returns mockSystemDetailsWithNoPlugin
+ val shouldShow = RulesEngine.displayNotification(projectRule.project, pluginNotPresentData)
+ assertThat(shouldShow).isFalse
+ }
+
+ @ParameterizedTest
+ @MethodSource("validNotifications")
+ fun `The notification is shown`(notification: String, expectedData: NotificationData) {
+ val notificationData = mapper.readValue(notification)
+ assertThat(notificationData).isEqualTo(expectedData)
+ val shouldShow = RulesEngine.displayNotification(projectRule.project, notificationData)
+ assertThat(shouldShow).isTrue
+ }
+
+ @ParameterizedTest
+ @MethodSource("invalidNotifications")
+ fun `The notification is not shown`(notification: String, expectedData: NotificationData) {
+ val notificationData = mapper.readValue(notification)
+ assertThat(notificationData).isEqualTo(expectedData)
+ val shouldShow = RulesEngine.displayNotification(projectRule.project, notificationData)
+ assertThat(shouldShow).isFalse
+ }
+
+ companion object {
+ @JvmStatic
+ fun validNotifications(): Stream = Stream.of(
+ Arguments.of(notificationWithConditionsOrActions, notificationWithConditionsOrActionsData),
+ Arguments.of(notificationWithoutConditionsOrActions, notificationsWithoutConditionsOrActionsData),
+ Arguments.of(notificationWithValidConnection, notificationWithValidConnectionData)
+ )
+
+ @JvmStatic
+ fun invalidNotifications(): Stream = Stream.of(
+ Arguments.of(validComputeInvalidOs, validOsInvalidComputeData),
+ Arguments.of(invalidExtensionVersion, invalidExtensionVersionData),
+ Arguments.of(invalidIdeTypeAndVersion, invalidIdeTypeAndVersionData)
+ )
+
+ private val mapper = jacksonObjectMapper().configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false)
+ }
+}
diff --git a/plugins/core/jetbrains-community/tst/software/aws/toolkits/jetbrains/core/notifications/NotificationFormatUtilsTestCases.kt b/plugins/core/jetbrains-community/tst/software/aws/toolkits/jetbrains/core/notifications/NotificationFormatUtilsTestCases.kt
new file mode 100644
index 0000000000..f033474f85
--- /dev/null
+++ b/plugins/core/jetbrains-community/tst/software/aws/toolkits/jetbrains/core/notifications/NotificationFormatUtilsTestCases.kt
@@ -0,0 +1,388 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.core.notifications
+
+val validComputeInvalidOs = """{
+ "id": "example_id_12344",
+ "schedule": {
+ "type": "StartUp"
+},
+ "severity": "Critical",
+ "condition": {
+ "compute": {
+ "type": {"==": "Local"}
+},
+"os": {
+ "type": {"==": "Windows"}
+}
+},
+ "actions": [
+ {
+ "type": "ShowMarketplace",
+ "content": {
+ "en-US": {
+ "title": "Go to market"
+ }
+ }
+ }
+ ],
+ "content": {
+ "en-US": {
+ "title": "Look at this!",
+ "description": "Some bug is there"
+}
+}
+}
+""".trimIndent()
+
+val validOsInvalidComputeData = NotificationData(
+ id = "example_id_12344",
+ schedule = NotificationSchedule(type = "StartUp"),
+ severity = "Critical",
+ condition = NotificationDisplayCondition(
+ compute = ComputeType(type = NotificationExpression.ComparisonCondition("Local"), architecture = null),
+ os = SystemType(type = NotificationExpression.ComparisonCondition("Windows"), version = null),
+ ide = null,
+ extension = null,
+ authx = null
+ ),
+ actions = listOf(
+ NotificationFollowupActions(
+ type = "ShowMarketplace",
+ content = NotificationFollowupActionsContent(
+ NotificationActionDescription(
+ title = "Go to market",
+ url = null
+ )
+ )
+ )
+ ),
+ content = NotificationContentDescriptionLocale(
+ NotificationContentDescription(
+ title = "Look at this!",
+ description = "Some bug is there"
+ )
+ )
+)
+
+val invalidExtensionVersion = """{
+ "id": "example_id_12344",
+ "schedule": {
+ "type": "StartUp"
+},
+ "severity": "Critical",
+ "condition": {
+ "extension": [
+ {
+ "id": "aws.toolkit",
+ "version": {
+ "!=": "1.3334"
+ }
+ },
+ {
+ "id": "amazon.q",
+ "version": {
+ ">": "3.37.0"
+ }
+ }
+ ]
+},
+ "content": {
+ "en-US": {
+ "title": "Look at this!",
+ "description": "Some bug is there"
+}
+}
+}
+""".trimIndent()
+
+val invalidExtensionVersionData = NotificationData(
+ id = "example_id_12344",
+ schedule = NotificationSchedule(type = "StartUp"),
+ severity = "Critical",
+ condition = NotificationDisplayCondition(
+ compute = null,
+ os = null,
+ ide = null,
+ extension = listOf(
+ ExtensionType(
+ id = "aws.toolkit",
+ version = NotificationExpression.NotEqualsCondition("1.3334")
+ ),
+ ExtensionType(
+ id = "amazon.q",
+ version = NotificationExpression.GreaterThanCondition("3.37.0")
+ )
+ ),
+ authx = null
+
+ ),
+ actions = emptyList(),
+ content = NotificationContentDescriptionLocale(
+ NotificationContentDescription(
+ title = "Look at this!",
+ description = "Some bug is there"
+ )
+ )
+)
+
+val exampleNotificationWithoutSchema = """
+ {
+ "notifications": [
+ {
+ "id": "notification-001",
+ "title": "Test Notification",
+ "description": "This is a test notification",
+ "type": "INFO",
+
+ "rules": {
+ "computeType": "Local",
+ "osType": "Linux",
+ "ideType": "IC",
+ "pluginVersion": {
+ "aws.toolkit": "1.0"
+ }
+ }
+ }
+ ]
+ }
+""".trimIndent()
+
+val exampleNotificationWithoutNotification = """
+ {
+ "schema": {
+ "version": "2.0"
+}
+
+ }
+""".trimIndent()
+
+val notificationWithoutConditionsOrActions = """
+ {
+ "id": "example_id_12344",
+ "schedule": {
+ "type": "StartUp"
+ },
+ "severity": "Critical",
+ "content": {
+ "en-US": {
+ "title": "Look at this!",
+ "description": "Some bug is there"
+ }
+ }
+ }
+
+
+""".trimIndent()
+
+val notificationsWithoutConditionsOrActionsData = NotificationData(
+ id = "example_id_12344",
+ schedule = NotificationSchedule(type = "StartUp"),
+ severity = "Critical",
+ condition = null,
+ actions = emptyList(),
+ content = NotificationContentDescriptionLocale(
+ NotificationContentDescription(
+ title = "Look at this!",
+ description = "Some bug is there"
+ )
+ )
+)
+
+val notificationWithConditionsOrActions = """
+ {
+ "id": "example_id_12344",
+ "schedule": {
+ "type": "StartUp"
+ },
+ "severity": "Critical",
+ "condition": {
+ "compute": {
+ "type": {
+
+ "==": "Local"
+
+ }
+ }
+ },
+ "actions": [
+ {
+ "type": "ShowMarketplace",
+ "content": {
+ "en-US": {
+ "title": "Go to market"
+ }
+ }
+ }
+ ],
+ "content": {
+ "en-US": {
+ "title": "Look at this!",
+ "description": "Some bug is there"
+ }
+ }
+ }
+
+""".trimIndent()
+
+val notificationWithConditionsOrActionsData = NotificationData(
+ id = "example_id_12344",
+ schedule = NotificationSchedule(type = "StartUp"),
+ severity = "Critical",
+ condition = NotificationDisplayCondition(
+ compute = ComputeType(type = NotificationExpression.ComparisonCondition("Local"), architecture = null),
+ os = null,
+ ide = null,
+ extension = null,
+ authx = null
+ ),
+ actions = listOf(
+ NotificationFollowupActions(
+ type = "ShowMarketplace",
+ content = NotificationFollowupActionsContent(
+ NotificationActionDescription(
+ title = "Go to market",
+ url = null
+ )
+ )
+ )
+ ),
+ content = NotificationContentDescriptionLocale(
+ NotificationContentDescription(
+ title = "Look at this!",
+ description = "Some bug is there"
+ )
+ )
+)
+
+val notificationWithValidConnection = """{
+ "id": "example_id_12344",
+ "schedule": {
+ "type": "StartUp"
+},
+ "severity": "Critical",
+ "condition": {
+ "authx": [{
+ "feature" : "q",
+ "type": {
+ "anyOf": [
+ "Idc",
+ "BuilderId"
+ ]
+ },
+ "region": {
+ "==": "us-west-2"
+ },
+ "connectionState": {
+ "==": "Connected"
+ }
+ } ]
+},
+ "content": {
+ "en-US": {
+ "title": "Look at this!",
+ "description": "Some bug is there"
+}
+}
+}
+""".trimIndent()
+
+val notificationWithValidConnectionData = NotificationData(
+ id = "example_id_12344",
+ schedule = NotificationSchedule(type = "StartUp"),
+ severity = "Critical",
+ condition = NotificationDisplayCondition(
+ compute = null,
+ os = null,
+ ide = null,
+ extension = null,
+ authx = listOf(
+ AuthxType(
+ feature = "q",
+ type = NotificationExpression.AnyOfCondition(listOf("Idc", "BuilderId")),
+ region = NotificationExpression.ComparisonCondition("us-west-2"),
+ connectionState = NotificationExpression.ComparisonCondition("Connected"),
+ ssoScopes = null
+ )
+ )
+ ),
+ actions = emptyList(),
+ content = NotificationContentDescriptionLocale(
+ NotificationContentDescription(
+ title = "Look at this!",
+ description = "Some bug is there"
+ )
+ )
+)
+
+val invalidIdeTypeAndVersion = """{
+ "id": "example_id_12344",
+ "schedule": {
+ "type": "StartUp"
+},
+ "severity": "Critical",
+ "condition": {
+ "ide": {
+ "type": {"noneOf": ["IC","IU","RD"]},
+ "version": {"!=": "1.3334"}
+}
+},
+ "content": {
+ "en-US": {
+ "title": "Look at this!",
+ "description": "Some bug is there"
+}
+}
+}
+""".trimIndent()
+
+val invalidIdeTypeAndVersionData = NotificationData(
+ id = "example_id_12344",
+ schedule = NotificationSchedule(type = "StartUp"),
+ severity = "Critical",
+ condition = NotificationDisplayCondition(
+ compute = null,
+ os = null,
+ ide = SystemType(
+ type = NotificationExpression.NoneOfCondition(listOf("IC", "IU", "RD")),
+ version = NotificationExpression.NotEqualsCondition("1.3334")
+ ),
+ extension = null,
+ authx = null
+
+ ),
+ actions = emptyList(),
+ content = NotificationContentDescriptionLocale(
+ NotificationContentDescription(
+ title = "Look at this!",
+ description = "Some bug is there"
+ )
+ )
+)
+
+val pluginNotPresentData = NotificationData(
+ id = "example_id_12344",
+ schedule = NotificationSchedule(type = "StartUp"),
+ severity = "Critical",
+ condition = NotificationDisplayCondition(
+ compute = null,
+ os = null,
+ ide = null,
+ extension = mutableListOf(
+ ExtensionType(
+ "amazon.q",
+ version = NotificationExpression.NotEqualsCondition("1.3334")
+ )
+ ),
+ authx = null
+
+ ),
+ actions = emptyList(),
+ content = NotificationContentDescriptionLocale(
+ NotificationContentDescription(
+ title = "Look at this!",
+ description = "Some bug is there"
+ )
+ )
+)
diff --git a/plugins/core/jetbrains-community/tst/software/aws/toolkits/jetbrains/core/notifications/NotificationManagerTest.kt b/plugins/core/jetbrains-community/tst/software/aws/toolkits/jetbrains/core/notifications/NotificationManagerTest.kt
new file mode 100644
index 0000000000..3e112a2305
--- /dev/null
+++ b/plugins/core/jetbrains-community/tst/software/aws/toolkits/jetbrains/core/notifications/NotificationManagerTest.kt
@@ -0,0 +1,45 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.core.notifications
+
+import com.intellij.testFramework.ApplicationExtension
+import com.intellij.testFramework.ProjectRule
+import org.assertj.core.api.Assertions.assertThat
+import org.junit.jupiter.api.Test
+import org.junit.jupiter.api.extension.ExtendWith
+import org.junit.jupiter.api.extension.Extension
+import org.junit.jupiter.api.extension.RegisterExtension
+
+@ExtendWith(ApplicationExtension::class)
+class NotificationManagerTest {
+
+ val projectRule = ProjectRule()
+
+ @JvmField
+ @RegisterExtension
+ val testExtension = object : Extension {
+ fun getProject() = projectRule.project
+ }
+
+ @Test
+ fun `If no follow-up actions, expand action is present`() {
+ val sut = NotificationManager.createActions(projectRule.project, listOf(), "Dummy Test Action", "Dummy title")
+ assertThat(sut).isNotNull
+ assertThat(sut).hasSize(1)
+ assertThat(sut.first().title).isEqualTo("More...")
+ }
+
+ @Test
+ fun `Show Url action shows the option to learn more`() {
+ val followupActions = NotificationFollowupActions(
+ "UpdateExtension",
+ NotificationFollowupActionsContent(NotificationActionDescription("title", null))
+ )
+ val sut = NotificationManager.createActions(projectRule.project, listOf(followupActions), "Dummy Test Action", "Dummy title")
+ assertThat(sut).isNotNull
+ assertThat(sut).hasSize(2)
+ assertThat(sut.first().title).isEqualTo("Update")
+ assertThat(sut[1].title).isEqualTo("More...")
+ }
+}
diff --git a/plugins/core/jetbrains-community/tst/software/aws/toolkits/jetbrains/core/notifications/NotificationPollingServiceTest.kt b/plugins/core/jetbrains-community/tst/software/aws/toolkits/jetbrains/core/notifications/NotificationPollingServiceTest.kt
new file mode 100644
index 0000000000..e4922519c8
--- /dev/null
+++ b/plugins/core/jetbrains-community/tst/software/aws/toolkits/jetbrains/core/notifications/NotificationPollingServiceTest.kt
@@ -0,0 +1,83 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.core.notifications
+
+import com.intellij.testFramework.ApplicationExtension
+import io.mockk.Runs
+import io.mockk.every
+import io.mockk.just
+import io.mockk.mockk
+import io.mockk.verify
+import org.junit.jupiter.api.AfterEach
+import org.junit.jupiter.api.BeforeEach
+import org.junit.jupiter.api.Test
+import org.junit.jupiter.api.extension.ExtendWith
+import software.aws.toolkits.core.utils.RemoteResourceResolver
+import software.aws.toolkits.core.utils.UpdateCheckResult
+import software.aws.toolkits.jetbrains.core.RemoteResourceResolverProvider
+import java.nio.file.Path
+import java.util.concurrent.CompletableFuture
+
+@ExtendWith(ApplicationExtension::class)
+class NotificationPollingServiceTest {
+ private lateinit var sut: NotificationPollingService
+ private lateinit var mockResolver: RemoteResourceResolver
+ private lateinit var mockProvider: RemoteResourceResolverProvider
+ private lateinit var observer: () -> Unit
+ private val testPath = Path.of("/test/path")
+
+ @BeforeEach
+ fun setUp() {
+ sut = NotificationPollingService()
+
+ mockResolver = mockk {
+ every { resolve(any()) } returns CompletableFuture.completedFuture(testPath)
+ }
+
+ mockProvider = mockk {
+ every { get() } returns mockResolver
+ }
+
+ val providerField = NotificationPollingService::class.java
+ .getDeclaredField("resourceResolver")
+ providerField.isAccessible = true
+ providerField.set(sut, mockProvider)
+
+ // Create mock observers
+ observer = mockk<() -> Unit>()
+ every { observer.invoke() } just Runs
+
+ val observersField = NotificationPollingService::class.java
+ .getDeclaredField("observers")
+ .apply { isAccessible = true }
+
+ observersField.set(sut, mutableListOf(observer))
+ }
+
+ @AfterEach
+ fun tearDown() {
+ sut.dispose()
+ }
+
+ @Test
+ fun `test pollForNotifications when ETag matches - no new notifications`() {
+ every { mockResolver.checkForUpdates(any(), any()) } returns UpdateCheckResult.NoUpdates
+ sut.startPolling()
+ verify(exactly = 0) { observer.invoke() }
+ }
+
+ @Test
+ fun `test pollForNotifications when ETag matches on startup - notify observers`() {
+ every { mockResolver.checkForUpdates(any(), any()) } returns UpdateCheckResult.FirstPollCheck
+ sut.startPolling()
+ verify(exactly = 1) { observer.invoke() }
+ }
+
+ @Test
+ fun `test pollForNotifications when ETag different - notify observers`() {
+ every { mockResolver.checkForUpdates(any(), any()) } returns UpdateCheckResult.HasUpdates
+ sut.startPolling()
+ verify(exactly = 1) { observer.invoke() }
+ }
+}
diff --git a/plugins/core/jetbrains-community/tst/software/aws/toolkits/jetbrains/core/notifications/NotificationResourceResolverTest.kt b/plugins/core/jetbrains-community/tst/software/aws/toolkits/jetbrains/core/notifications/NotificationResourceResolverTest.kt
new file mode 100644
index 0000000000..a9c0ed8496
--- /dev/null
+++ b/plugins/core/jetbrains-community/tst/software/aws/toolkits/jetbrains/core/notifications/NotificationResourceResolverTest.kt
@@ -0,0 +1,78 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.core.notifications
+
+import com.intellij.testFramework.ApplicationExtension
+import io.mockk.every
+import io.mockk.mockk
+import org.assertj.core.api.Assertions.assertThat
+import org.junit.jupiter.api.BeforeEach
+import org.junit.jupiter.api.Test
+import org.junit.jupiter.api.extension.ExtendWith
+import org.junit.jupiter.api.io.TempDir
+import software.aws.toolkits.core.utils.DefaultRemoteResourceResolver
+import software.aws.toolkits.core.utils.UpdateCheckResult
+import software.aws.toolkits.core.utils.UrlFetcher
+import java.nio.file.Path
+import java.util.concurrent.Callable
+import java.util.concurrent.CompletableFuture
+
+@ExtendWith(ApplicationExtension::class)
+class NotificationResourceResolverTest {
+ private lateinit var urlFetcher: UrlFetcher
+ private lateinit var sut: DefaultRemoteResourceResolver
+
+ @TempDir
+ lateinit var tempDir: Path
+
+ @BeforeEach
+ fun setUp() {
+ urlFetcher = mockk()
+ sut = DefaultRemoteResourceResolver(
+ urlFetcher = urlFetcher,
+ cacheBasePath = tempDir,
+ executor = { callable: Callable -> CompletableFuture.completedFuture(callable.call()) }
+ )
+ }
+
+ @Test
+ fun `first poll with no ETag changes returns FirstPollCheck`() {
+ NotificationEtagState.getInstance().etag = "same-etag"
+ val expectedETag = "same-etag"
+ every { urlFetcher.getETag(any()) } returns expectedETag
+
+ val result = sut.checkForUpdates("http://notification.test", NotificationEtagState.getInstance())
+ assertThat(result).isEqualTo(UpdateCheckResult.FirstPollCheck)
+ }
+
+ @Test
+ fun `ETag changes returns HasUpdates`() {
+ NotificationEtagState.getInstance().etag = "old-etag"
+ val expectedETag = "new-etag"
+ every { urlFetcher.getETag(any()) } returns expectedETag
+
+ val result = sut.checkForUpdates("http://notification.test", NotificationEtagState.getInstance())
+ assertThat(result).isEqualTo(UpdateCheckResult.HasUpdates)
+ }
+
+ @Test
+ fun `no ETag changes returns NoUpdates after first poll`() {
+ NotificationEtagState.getInstance().etag = "same-etag"
+ val expectedETag = "same-etag"
+ every { urlFetcher.getETag(any()) } returns expectedETag
+
+ // sets isFirstPoll to false
+ val firstResult = sut.checkForUpdates("http://notification.test", NotificationEtagState.getInstance())
+ assertThat(firstResult).isEqualTo(UpdateCheckResult.FirstPollCheck)
+
+ val secondResult = sut.checkForUpdates("http://notification.test", NotificationEtagState.getInstance())
+ assertThat(secondResult).isEqualTo(UpdateCheckResult.NoUpdates)
+ }
+
+ @Test
+ fun `getLocalResourcePath returns null for non-existent file`() {
+ val result = sut.getLocalResourcePath("non-existent-file")
+ assertThat(result).isNull()
+ }
+}
diff --git a/plugins/core/jetbrains-community/tst/software/aws/toolkits/jetbrains/core/notifications/ProcessNotificationsBaseTest.kt b/plugins/core/jetbrains-community/tst/software/aws/toolkits/jetbrains/core/notifications/ProcessNotificationsBaseTest.kt
new file mode 100644
index 0000000000..01b77f2234
--- /dev/null
+++ b/plugins/core/jetbrains-community/tst/software/aws/toolkits/jetbrains/core/notifications/ProcessNotificationsBaseTest.kt
@@ -0,0 +1,161 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.core.notifications
+
+import com.intellij.openapi.project.Project
+import com.intellij.testFramework.ApplicationExtension
+import io.mockk.every
+import io.mockk.mockk
+import io.mockk.mockkObject
+import io.mockk.spyk
+import io.mockk.unmockkAll
+import io.mockk.verify
+import org.junit.jupiter.api.AfterEach
+import org.junit.jupiter.api.BeforeEach
+import org.junit.jupiter.api.Test
+import org.junit.jupiter.api.extension.ExtendWith
+import java.util.concurrent.atomic.AtomicBoolean
+
+@ExtendWith(ApplicationExtension::class)
+class ProcessNotificationsBaseTest {
+ private lateinit var sut: ProcessNotificationsBase
+ private lateinit var project: Project
+ private lateinit var dismissalState: NotificationDismissalState
+
+ @BeforeEach
+ fun setUp() {
+ project = mockk()
+ dismissalState = spyk(NotificationDismissalState())
+
+ mockkObject(NotificationDismissalState)
+ every { NotificationDismissalState.getInstance() } returns dismissalState
+
+ sut = spyk(
+ objToCopy = ProcessNotificationsBase(project)
+ )
+ }
+
+ @Test
+ fun `startup notifications are only processed on first poll`() {
+ resetIsStartup()
+ val startupNotification = createNotification("startup-1", NotificationScheduleType.STARTUP)
+ every { sut["getNotificationsFromFile"]() } returns createNotificationsList(startupNotification)
+ every { dismissalState.isDismissed(any()) } returns false
+
+ sut.retrieveStartupAndEmergencyNotifications()
+
+ verify(exactly = 1) { sut.processNotification(project, startupNotification) }
+
+ // Second poll
+ sut.retrieveStartupAndEmergencyNotifications()
+
+ // Verify processNotification wasn't called again
+ verify(exactly = 1) { sut.processNotification(project, any()) }
+ }
+
+ @Test
+ fun `non startup notifications are processed on every poll`() {
+ val emergencyNotification = createNotification("emergency-1", NotificationScheduleType.EMERGENCY)
+ every { sut["getNotificationsFromFile"]() } returns createNotificationsList(emergencyNotification)
+ every { dismissalState.isDismissed(any()) } returns false
+
+ // First poll
+ sut.retrieveStartupAndEmergencyNotifications()
+ // Second poll
+ sut.retrieveStartupAndEmergencyNotifications()
+
+ verify(exactly = 2) { sut.processNotification(project, emergencyNotification) }
+ }
+
+ @Test
+ fun `dismissed notifications are not processed`() {
+ val notification = createNotification("toBeDismissed-1", NotificationScheduleType.EMERGENCY)
+ every { sut["getNotificationsFromFile"]() } returns createNotificationsList(notification)
+
+ // first poll results in showing/dismissal
+ sut.retrieveStartupAndEmergencyNotifications()
+ NotificationDismissalState.getInstance().dismissNotification(notification.id)
+
+ // second poll skips processing
+ sut.retrieveStartupAndEmergencyNotifications()
+
+ verify(exactly = 1) { sut.processNotification(project, any()) }
+ }
+
+ @Test
+ fun `null notifications list is handled gracefully`() {
+ every { sut["getNotificationsFromFile"]() } returns null
+
+ sut.retrieveStartupAndEmergencyNotifications()
+
+ verify(exactly = 0) { sut.processNotification(project, any()) }
+ }
+
+ @Test
+ fun `empty notifications list is handled gracefully`() {
+ every { sut["getNotificationsFromFile"]() } returns createNotificationsList()
+
+ sut.retrieveStartupAndEmergencyNotifications()
+
+ verify(exactly = 0) { sut.processNotification(project, any()) }
+ }
+
+ @Test
+ fun `multiple notifications are processed correctly`() {
+ val startupNotification = createNotification("startup-1", NotificationScheduleType.STARTUP)
+ val emergencyNotification = createNotification("emergency-1", NotificationScheduleType.EMERGENCY)
+
+ every { sut["getNotificationsFromFile"]() } returns createNotificationsList(
+ startupNotification,
+ emergencyNotification
+ )
+ every { dismissalState.isDismissed(any()) } returns false
+
+ // First poll - both should be processed
+ sut.retrieveStartupAndEmergencyNotifications()
+
+ verify(exactly = 1) { sut.processNotification(project, startupNotification) }
+ verify(exactly = 1) { sut.processNotification(project, emergencyNotification) }
+
+ // Second poll - only emergency should be processed
+ sut.retrieveStartupAndEmergencyNotifications()
+
+ verify(exactly = 1) { sut.processNotification(project, startupNotification) }
+ verify(exactly = 2) { sut.processNotification(project, emergencyNotification) }
+ }
+
+ // Helper functions to create test data
+ private fun createNotification(id: String, type: NotificationScheduleType) = NotificationData(
+ id = id,
+ schedule = NotificationSchedule(type = type),
+ severity = "INFO",
+ condition = null,
+ content = NotificationContentDescriptionLocale(
+ NotificationContentDescription(
+ title = "Look at this!",
+ description = "Some bug is there"
+ )
+ ),
+ actions = emptyList()
+ )
+
+ private fun createNotificationsList(vararg notifications: NotificationData) = NotificationsList(
+ schema = Schema("1.0"),
+ notifications = notifications.toList()
+ )
+
+ private fun resetIsStartup() {
+ val clazz = Class.forName("software.aws.toolkits.jetbrains.core.notifications.ProcessNotificationsBaseKt")
+ val field = clazz.getDeclaredField("isStartup")
+ field.isAccessible = true
+
+ val value = field.get(null) as AtomicBoolean
+ value.set(true)
+ }
+
+ @AfterEach
+ fun tearDown() {
+ unmockkAll()
+ }
+}
diff --git a/plugins/core/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/telemetry/PluginResolverTest.kt b/plugins/core/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/telemetry/PluginResolverTest.kt
index 6ee89f128d..eb0a6c867a 100644
--- a/plugins/core/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/telemetry/PluginResolverTest.kt
+++ b/plugins/core/jetbrains-community/tst/software/aws/toolkits/jetbrains/services/telemetry/PluginResolverTest.kt
@@ -6,28 +6,25 @@ package software.aws.toolkits.jetbrains.services.telemetry
import com.intellij.ide.plugins.IdeaPluginDescriptor
import com.intellij.ide.plugins.PluginManagerCore
import io.mockk.called
-import io.mockk.clearAllMocks
import io.mockk.every
+import io.mockk.junit5.MockKExtension
import io.mockk.mockk
import io.mockk.mockkStatic
import io.mockk.verify
import junit.framework.TestCase.assertEquals
-import org.junit.jupiter.api.AfterEach
import org.junit.jupiter.api.BeforeEach
import org.junit.jupiter.api.Test
+import org.junit.jupiter.api.extension.ExtendWith
import software.amazon.awssdk.services.toolkittelemetry.model.AWSProduct
+@ExtendWith(MockKExtension::class)
class PluginResolverTest {
@BeforeEach
fun setup() {
+ PluginResolver.setThreadLocal(null)
mockkStatic(PluginManagerCore::class)
}
- @AfterEach
- fun tearDown() {
- clearAllMocks()
- }
-
@Test
fun getsProductForAmazonQPlugin() {
val pluginDescriptor = mockk {
diff --git a/plugins/core/jetbrains-community/tst/software/aws/toolkits/jetbrains/utils/ThreadingUtilsKtTest.kt b/plugins/core/jetbrains-community/tst/software/aws/toolkits/jetbrains/utils/ThreadingUtilsKtTest.kt
index b9bb075d3c..140ae9d68a 100644
--- a/plugins/core/jetbrains-community/tst/software/aws/toolkits/jetbrains/utils/ThreadingUtilsKtTest.kt
+++ b/plugins/core/jetbrains-community/tst/software/aws/toolkits/jetbrains/utils/ThreadingUtilsKtTest.kt
@@ -10,7 +10,8 @@ import com.intellij.openapi.progress.ProgressManager
import com.intellij.testFramework.ApplicationRule
import com.intellij.util.concurrency.AppExecutorUtil
import io.mockk.every
-import io.mockk.mockk
+import io.mockk.impl.annotations.MockK
+import io.mockk.junit4.MockKRule
import org.assertj.core.api.Assertions.assertThatThrownBy
import org.junit.Assert.assertEquals
import org.junit.Rule
@@ -26,6 +27,12 @@ class ThreadingUtilsKtTest {
@JvmField
val application = ApplicationRule()
+ @get:Rule
+ val mockkRule = MockKRule(this)
+
+ @MockK
+ private lateinit var pluginResolver: PluginResolver
+
@Test
fun `computeOnEdt runs on edt`() {
computeOnEdt {
@@ -78,13 +85,13 @@ class ThreadingUtilsKtTest {
@Test
fun `pluginAwareExecuteOnPooledThread inherits plugin resolver`() {
- val pluginResolver = mockk {
- every { product } returns AWSProduct.AMAZON_Q_FOR_JET_BRAINS
- }
+ every { pluginResolver.product } returns AWSProduct.AMAZON_Q_FOR_JET_BRAINS
PluginResolver.setThreadLocal(pluginResolver)
pluginAwareExecuteOnPooledThread {
assertEquals(PluginResolver.fromCurrentThread().product, AWSProduct.AMAZON_Q_FOR_JET_BRAINS)
}.get()
+
+ PluginResolver.setThreadLocal(PluginResolver.fromStackTrace(Thread.currentThread().stackTrace))
}
}
diff --git a/plugins/core/resources/resources/software/aws/toolkits/resources/MessagesBundle.properties b/plugins/core/resources/resources/software/aws/toolkits/resources/MessagesBundle.properties
index cdab8276be..d23a8e879d 100644
--- a/plugins/core/resources/resources/software/aws/toolkits/resources/MessagesBundle.properties
+++ b/plugins/core/resources/resources/software/aws/toolkits/resources/MessagesBundle.properties
@@ -13,33 +13,68 @@ action.aws.caws.rebuildAction.text=Rebuild Dev Environment
action.aws.caws.updateDevfile.text=Update Devfile...
action.aws.toolkit.caws.logout.text=Sign out
action.aws.toolkit.caws.profile.text=View Profile...
-action.aws.toolkit.dynamoViewer.changeMaxResults.text=Max Results
action.aws.toolkit.dynamodb.delete_table.text=Delete Table...
action.aws.toolkit.ecr.repository.pull.text=Pull from Repository...
action.aws.toolkit.ecr.repository.push.text=Push to Repository...
-action.aws.toolkit.jetbrains.core.services.cwc.commands.ExplainCodeAction.description = Explains the selected code
-action.aws.toolkit.jetbrains.core.services.cwc.commands.ExplainCodeAction.text = Explain Code
-action.aws.toolkit.jetbrains.core.services.cwc.commands.FixCodeAction.description = Fixes the selected code
-action.aws.toolkit.jetbrains.core.services.cwc.commands.FixCodeAction.text = Fix Code
-action.aws.toolkit.jetbrains.core.services.cwc.commands.GenerateUnitTestsAction.description = Generates unit tests for the selected code
-action.aws.toolkit.jetbrains.core.services.cwc.commands.GenerateUnitTestsAction.text = Generate Tests (Beta)
-action.aws.toolkit.jetbrains.core.services.cwc.commands.OptimizeCodeAction.description = Optimizes the selected code
-action.aws.toolkit.jetbrains.core.services.cwc.commands.OptimizeCodeAction.text = Optimize Code
-action.aws.toolkit.jetbrains.core.services.cwc.commands.RefactorCodeAction.description = Refactors the selected code
-action.aws.toolkit.jetbrains.core.services.cwc.commands.RefactorCodeAction.text = Refactor Code
-action.aws.toolkit.jetbrains.core.services.cwc.commands.SendToPromptAction.description = Sends selected code to chat
-action.aws.toolkit.jetbrains.core.services.cwc.commands.SendToPromptAction.text = Send to Prompt
-action.aws.toolkit.jetbrains.core.services.cwc.inline.openChat.text = Inline Chat
+action.aws.toolkit.jetbrains.core.services.cwc.commands.ExplainCodeAction.description=Explains the selected code
+action.aws.toolkit.jetbrains.core.services.cwc.commands.ExplainCodeAction.text=Explain Code
+action.aws.toolkit.jetbrains.core.services.cwc.commands.FixCodeAction.description=Fixes the selected code
+action.aws.toolkit.jetbrains.core.services.cwc.commands.FixCodeAction.text=Fix Code
+action.aws.toolkit.jetbrains.core.services.cwc.commands.GenerateUnitTestsAction.description=Generates unit tests for the selected code
+action.aws.toolkit.jetbrains.core.services.cwc.commands.GenerateUnitTestsAction.text=Generate Tests
+action.aws.toolkit.jetbrains.core.services.cwc.commands.OptimizeCodeAction.description=Optimizes the selected code
+action.aws.toolkit.jetbrains.core.services.cwc.commands.OptimizeCodeAction.text=Optimize Code
+action.aws.toolkit.jetbrains.core.services.cwc.commands.RefactorCodeAction.description=Refactors the selected code
+action.aws.toolkit.jetbrains.core.services.cwc.commands.RefactorCodeAction.text=Refactor Code
+action.aws.toolkit.jetbrains.core.services.cwc.commands.SendToPromptAction.description=Sends selected code to chat
+action.aws.toolkit.jetbrains.core.services.cwc.commands.SendToPromptAction.text=Send to Prompt
+action.aws.toolkit.jetbrains.core.services.cwc.inline.openChat.text=Inline Chat
action.aws.toolkit.open.arn.browser.text=Open ARN in AWS Console
action.aws.toolkit.open.telemetry.viewer.text=View AWS Telemetry
action.aws.toolkit.s3.open.bucket.viewer.prefixed.text=View Bucket with Prefix...
action.aws.toolkit.s3.open.bucket.viewer.text=View Bucket
action.aws.toolkit.toolwindow.explorer.newConnection.text=Setup authentication to begin
action.aws.toolkit.toolwindow.newConnection.text=Add Another Connection...
-action.aws.toolkit.toolwindow.sso.signout.text=Sign out of SSO
action.dynamic.open.text=Open Resource...
action.q.openchat.text=Open Chat Panel
amazonqChat.project_context.index_in_progress=By the way, I'm still indexing this project for full context from your workspace. I may have a better response in a few minutes when it's complete if you'd like to try again then.
+amazonqDoc.answer.codeResult=You can accept the changes to your files, or describe any additional changes you'd like me to make.
+amazonqDoc.answer.readmeCreated=I've created a README for your code.
+amazonqDoc.answer.readmeUpdated=I've updated your README.
+amazonqDoc.edit.message=Okay, let's work on your README. Describe the changes you would like to make. For example, you can ask me to:\n- Correct something\n- Expand on something\n- Add a section\n- Remove a section
+amazonqDoc.edit.placeholder=Describe documentation changes
+amazonqDoc.error.generating=Unable to generate changes.
+amazonqDoc.error_text=I'm sorry, I ran into an issue while trying to generate your documentation. Please try again.
+amazonqDoc.exception.content_length_error=Your workspace is too large for me to review. Your workspace must be within the quota, even if you choose a smaller folder. For more information on quotas, see the Amazon Q Developer documentation.
+amazonqDoc.exception.no_change_required=I couldn't find any code changes to update in the README. Try another documentation task.
+amazonqDoc.exception.prompt_too_vague=I need more information to make changes to your README. Try providing some of the following details:\n- Which sections you want to modify\n- The content you want to add or remove\n- Specific issues that need correcting\n\nFor more information on prompt best practices, see the Amazon Q Developer documentation.
+amazonqDoc.exception.prompt_unrelated=These changes don't seem related to documentation. Try describing your changes again, using the following best practices:\n- Changes should relate to how project functionality is reflected in the README\n- Content you refer to should be available in your codebase\n\nFor more information on prompt best practices, see the Amazon Q Developer documentation.
+amazonqDoc.exception.readme_too_large=The README in your folder is too large for me to review. Try reducing the size of your README, or choose a folder with a smaller README. For more information on quotas, see the Amazon Q Developer documentation.
+amazonqDoc.exception.readme_update_too_large=The updated README is too large. Try reducing the size of your README, or asking for a smaller update. For more information on quotas, see the Amazon Q Developer documentation.
+amazonqDoc.exception.workspace_empty=The folder you chose did not contain any source files in a supported language. Choose another folder and try again. For more information on supported languages, see the Amazon Q Developer documentation.
+amazonqDoc.inprogress_message.generating=Generating documentation...
+amazonqDoc.progress_message.baseline=This might take a few minutes.
+amazonqDoc.progress_message.creating=Okay, I'm creating a README for your project.
+amazonqDoc.progress_message.generating=Generating documentation
+amazonqDoc.progress_message.scanning=Scanning source files
+amazonqDoc.progress_message.summarizing=Summarizing source files
+amazonqDoc.progress_message.updating=Okay, I'm updating the README.
+amazonqDoc.prompt.create=Create a README
+amazonqDoc.prompt.create.confirmation=Create a README for this project?
+amazonqDoc.prompt.folder.change=Change folder
+amazonqDoc.prompt.folder.proceed=Yes
+amazonqDoc.prompt.placeholder=Choose an option to continue
+amazonqDoc.prompt.reject.close_session=End session
+amazonqDoc.prompt.reject.message=Your changes have been discarded.
+amazonqDoc.prompt.reject.new_task=Start a new documentation task
+amazonqDoc.prompt.review.accept=Accept
+amazonqDoc.prompt.review.changes=Make changes
+amazonqDoc.prompt.review.message=Please review and accept the changes.
+amazonqDoc.prompt.update=Update an existing README
+amazonqDoc.prompt.update.follow_up.edit=Make a specific change
+amazonqDoc.prompt.update.follow_up.sync=Update README to reflect code
+amazonqDoc.session.create=Create documentation for a specific folder
+amazonqDoc.session.sync=Sync documentation
amazonqFeatureDev.chat_message.ask_for_new_task=What new task would you like to work on?
amazonqFeatureDev.chat_message.closed_session=Okay, I've ended this chat session. You can open a new tab to chat or start another workflow.
amazonqFeatureDev.chat_message.devFileInRepository=I noticed that your repository has a `devfile.yaml`. Would you like me to use the devfile to build and test your project as I generate code?\n\nFor more information on using devfiles to improve code generation, see the [Amazon Q Developer documentation](https://docs.aws.amazon.com/amazonq/latest/qdeveloper-ug/software-dev.html).
@@ -53,8 +88,9 @@ amazonqFeatureDev.code_generation.error_message=I'm sorry, I ran into an issue w
amazonqFeatureDev.code_generation.failed_generation=Code generation failed
amazonqFeatureDev.code_generation.generating_code=Generating code ...
amazonqFeatureDev.code_generation.iteration_counts=Would you like me to add this code to your project, or provide feedback for new code? You have {0} out of {1} code generations left.
+amazonqFeatureDev.code_generation.iteration_counts_ask_to_add_code=Would you like me to add this code to your project?
+amazonqFeatureDev.code_generation.iteration_counts_ask_to_add_code_or_feedback=Would you like me to add this code to your project, or provide feedback for new code?
amazonqFeatureDev.code_generation.iteration_limit.error_text=Sorry, you've reached the quota for number of iterations on code generation. You can insert this code in your files or discuss a new plan. For more information on quotas, see the [Amazon Q Developer documentation](https://docs.aws.amazon.com/amazonq/latest/qdeveloper-ug/software-dev.html#quotas).
-amazonqFeatureDev.code_generation.iteration_zero=Would you like me to add this code to your project?
amazonqFeatureDev.code_generation.no_file_changes=Unable to generate any file changes
amazonqFeatureDev.code_generation.no_retries.error_message=I'm sorry, I'm having trouble generating your code and can't continue at the moment. Please try again later, and share feedback to help me improve.
amazonqFeatureDev.code_generation.notification_message=Your code suggestions from Amazon Q are ready to review
@@ -62,6 +98,7 @@ amazonqFeatureDev.code_generation.notification_open_link=Open chat
amazonqFeatureDev.code_generation.notification_title=Amazon Q Developer Agent for software development
amazonqFeatureDev.code_generation.provide_code_feedback=How can I improve the code for your use case?
amazonqFeatureDev.code_generation.stopped_code_generation=I stopped generating your code. If you want to continue working on this task, provide another description. You have {0} out of {1} code generations left.
+amazonqFeatureDev.code_generation.stopped_code_generation_no_iteration_count_display=I stopped generating your code. If you want to continue working on this task, provide another description.
amazonqFeatureDev.code_generation.stopped_code_generation_no_iterations=I stopped generating your code. You don't have more iterations left, however, you can start a new session.
amazonqFeatureDev.code_generation.stopping_code_generation=Stopping code generation...
amazonqFeatureDev.code_generation.updated_code=Okay, I updated your code files. Would you like to work on another task?
@@ -230,6 +267,9 @@ aws.settings.aws_cli_settings=AWS CLI Settings
aws.settings.codewhisperer.allow_q_dev_build_test_commands=Amazon Q: Allow Q /dev to run code and test commands
aws.settings.codewhisperer.automatic_import_adder=Imports recommendation
aws.settings.codewhisperer.automatic_import_adder.tooltip=Amazon Q will add import statements with code suggestions when necessary
+aws.settings.codewhisperer.code_review=Code Review
+aws.settings.codewhisperer.code_review.description=Specifies a list of code issue identifiers(separated by ";") that Amazon Q should ignore when reviewing your workspace. Each item in the array should be a unique string identifier for a specific code issue. This allows you to suppress notifications for known issues that you've assessed and determined to be false positives or not applicable to your project. Use this setting with caution, as it may cause you to miss important security alerts.
+aws.settings.codewhisperer.code_review.title=Ignored Security Issues
aws.settings.codewhisperer.configurable.controlled_by_admin=\ Controlled by your admin
aws.settings.codewhisperer.configurable.opt_out.title=Share Amazon Q content with AWS
aws.settings.codewhisperer.configurable.opt_out.tooltip=When checked, your content processed by Amazon Q may be used for service improvement (except for content processed by the Amazon Q Developer Pro tier). Unchecking this box will cause AWS to delete any of your content used for that purpose. The information used to provide the Amazon Q service to you will not be affected. See the Service Terms for more detail.
@@ -572,7 +612,7 @@ codemodernizer.builderrordialog.description.title=Error occurred when building y
codemodernizer.chat.form.user_selection.item.choose_module=Choose a module to transform
codemodernizer.chat.form.user_selection.item.choose_one_or_multiple_diffs_option=Choose how to receive proposed changes
codemodernizer.chat.form.user_selection.item.choose_skip_tests_option=Choose to skip unit tests
-codemodernizer.chat.form.user_selection.item.choose_sql_metadata_file=Okay, I can convert the embedded SQL code for your Oracle to PostgreSQL transformation. To get started, upload the zipped metadata file from your schema conversion in AWS Data Migration Service (DMS). To retrieve the metadata file:\n1. Open your database migration project in the AWS DMS console.\n2. Open the schema conversion and choose **Convert the embedded SQL in your application**.\n3. Once you complete the conversion, close the project and go to the S3 bucket where your project is stored.\n4. Open the folder and find the project folder ("sct-project").\n5. Download the object inside the project folder. This will be a zip file.\n\nFor more info, refer to the [documentation](https://docs.aws.amazon.com/dms/latest/userguide/schema-conversion-save-apply.html#schema-conversion-save).
+codemodernizer.chat.form.user_selection.item.choose_sql_metadata_file=Okay, I can convert the embedded SQL code for your Oracle to PostgreSQL transformation. To get started, upload the zipped metadata file from your schema conversion in AWS Data Migration Service (DMS). To retrieve the metadata file:\n1. Open your database migration project in the AWS DMS console.\n2. Open the schema conversion and choose **Convert the embedded SQL in your application**.\n3. Once you complete the conversion, close the project and go to the S3 bucket where your project is stored.\n4. Open the folder and find the project folder ("sct-project").\n5. Download the object inside the project folder. This will be a zip file.\n\nFor more info, refer to the [documentation](https://docs.aws.amazon.com/dms/latest/userguide/schema-conversion-embedded-sql.html).
codemodernizer.chat.form.user_selection.item.choose_target_version=Choose the target code version
codemodernizer.chat.form.user_selection.title=Q - Code transformation
codemodernizer.chat.message.absolute_path_detected=I detected {0} potential absolute file path(s) in your {1} file: **{2}**. Absolute file paths might cause issues when I build your code. Any errors will show up in the build log.
@@ -587,10 +627,11 @@ codemodernizer.chat.message.button.select_sql_metadata=Select metadata file
codemodernizer.chat.message.button.stop_transform=Stop transformation
codemodernizer.chat.message.button.view_build=View build progress
codemodernizer.chat.message.button.view_diff=View diff
-codemodernizer.chat.message.button.view_failure_build_log=View Failure Build Log
+codemodernizer.chat.message.button.view_failure_build_log=View build log
codemodernizer.chat.message.button.view_summary=View summary
codemodernizer.chat.message.changes_applied=I applied the changes to your project.
-codemodernizer.chat.message.choose_objective=I can help you with the following tasks:\n- Upgrade your Java 8 and Java 11 codebases to Java 17, or upgrade Java 17 code with up to date libraries and other dependencies.\n- Convert embedded SQL code for Oracle to PostgreSQL database migrations in AWS DMS.\n\nWhat would you like to do? You can enter "language upgrade" or "sql conversion".
+codemodernizer.chat.message.choose_objective=I can help you with the following tasks:\n- Upgrade your Java 8 and Java 11 codebases to Java 17, or upgrade Java 17 code with up-to-date libraries and other dependencies.\n- Convert embedded SQL code for Oracle to PostgreSQL database migrations in AWS DMS. [Learn more](https://docs.aws.amazon.com/dms/latest/userguide/schema-conversion-embedded-sql.html).\n\nWhat would you like to do? You can enter "language upgrade" or "sql conversion".
+codemodernizer.chat.message.choose_objective_placeholder=Enter "language upgrade" or "sql conversion"
codemodernizer.chat.message.download_failed_client_instructions_expired=Your transformation is not available anymore. Your code and transformation summary are deleted 24 hours after the transformation completes. Please try starting the transformation again.
codemodernizer.chat.message.download_failed_invalid_artifact=Sorry, I was unable to find your {0}. Artifacts are deleted after 24 hours. Please try starting the transformation again.
codemodernizer.chat.message.download_failed_other=Sorry, I ran into an issue while trying to download your {0}. Please try again. {1}
@@ -641,6 +682,7 @@ codemodernizer.chat.message.sql_metadata_success=I found the following source da
codemodernizer.chat.message.sql_module_schema_prompt=To continue, choose the module and schema for this transformation.
codemodernizer.chat.message.transform_begin=I'm starting to transform your code. It can take 10 to 30 minutes to upgrade your code, depending on the size of your module. To monitor progress, go to the Transformation Hub.
codemodernizer.chat.message.transform_cancelled_by_user=I cancelled your transformation. If you want to start another transformation, choose **Start a new transformation**.
+codemodernizer.chat.message.transform_failed=I could not complete the transformation. {0}
codemodernizer.chat.message.transform_in_progress=If I run into any issues, I might pause the transformation to get input from you on how to proceed.
codemodernizer.chat.message.transform_stopped_by_user=I stopped your transformation. If you want to start another transformation, choose **Start a new transformation**.
codemodernizer.chat.message.transform_stopping=I'm stopping your transformation...
@@ -657,10 +699,10 @@ codemodernizer.chat.message.validation.error.invalid_target_db=I can only conver
codemodernizer.chat.message.validation.error.missing_sct_file=An .sct file is required for transformation. Make sure that you've uploaded the .zip file you retrieved from your schema conversion in AWS DMS.
codemodernizer.chat.message.validation.error.more_info=For more information, see the [Amazon Q documentation]({0}).
codemodernizer.chat.message.validation.error.no_java_project=Sorry, I could not find an open Java module with embedded Oracle SQL statements. Make sure you have a Java module open that has at least 1 content root.
-codemodernizer.chat.message.validation.error.no_pom=I couldn't find a module that I can upgrade. Your Java project must be built on Maven and contain a pom.xml file. For more information, see the [Amazon Q documentation]({0}).
-codemodernizer.chat.message.validation.error.other=I couldn't find a module that I can upgrade. Currently, I support Java 8, Java 11, and Java 17 projects built on Maven. Make sure your project is open in the IDE. If you have a Java 8, Java 11, or Java 17 module in your workspace, you might need to configure your project so that I can find it. Go to File and choose Project Structure. In the Projects tab, set the correct project JDK and the correct language level. In the Modules tab, set the correct module JDK and language level.
-codemodernizer.chat.message.validation.error.unsupported_java_version=I couldn't find a module that I can upgrade. Currently, I support Java 8, Java 11, and Java 17 projects built on Maven. Make sure your project is open in the IDE. If you have a Java 8, Java 11, or Java 17 in your workspace, you might need to configure your project so that I can find it. Go to File and choose Project Structure. In the Projects tab, set the correct project JDK and the correct language level. In the Modules tab, set the correct module JDK and language level.
-codemodernizer.chat.message.validation.no_jdk=I couldn't build your project with your current JDK configuration. To update your JDK, go to File and choose Project Structure. In the Projects tab, set the correct project JDK in the SDK field. In the Modules tab, set the correct module JDK in the SDK field. In Maven Runner settings, set the correct JDK in the JRE field.
+codemodernizer.chat.message.validation.error.no_pom=I couldn't find a module that I can upgrade. Your Java module must be built on Maven and contain a pom.xml file. For more information, see the [Amazon Q documentation]({0}).
+codemodernizer.chat.message.validation.error.other=I couldn't find a module that I can upgrade. Currently, I support Java 8, Java 11, and Java 17 modules built on Maven. Make sure your module is open in the IDE. If you have a Java 8, Java 11, or Java 17 module in your workspace, you might need to configure your module so that I can find it. Go to File and choose Project Structure. In the Projects tab, set the correct project JDK and the correct language level. In the Modules tab, set the correct module JDK and language level.
+codemodernizer.chat.message.validation.error.unsupported_java_version=I couldn't find a module that I can upgrade. Currently, I support Java 8, Java 11, and Java 17 projects built on Maven. Make sure your project is open in the IDE. If you have a Java 8, Java 11, or Java 17 module in your workspace, you might need to configure your module so that I can find it. Go to File and choose Project Structure. In the Projects tab, set the correct project JDK and the correct language level. In the Modules tab, set the correct module JDK and language level.
+codemodernizer.chat.message.validation.no_jdk=I couldn't build your module with your current JDK configuration. To update your JDK, go to File and choose Project Structure. In the Projects tab, set the correct project JDK in the SDK field. In the Modules tab, set the correct module JDK in the SDK field. In Maven Runner settings, set the correct JDK in the JRE field.
codemodernizer.chat.prompt.label.dependency_current_version=Current version
codemodernizer.chat.prompt.label.dependency_name=Dependency name
codemodernizer.chat.prompt.label.dependency_selected_version=Target version
@@ -698,7 +740,7 @@ codemodernizer.migration_plan.body.steps_name={0}
codemodernizer.migration_plan.body.steps_scroll_top=Scroll to top
codemodernizer.migration_plan.header.awsq=Amazon Q reviewed your code and generated a transformation plan. Amazon Q will suggest code changes according to the plan, and you can review the updated code before accepting changes to your files.
codemodernizer.migration_plan.header.billing_text= {0} lines of code were submitted for transformation. If you reach the quota for lines of code included in your subscription, you will be charged ${1} for each additional line of code. You might be charged up to ${2} for this transformation. To avoid being charged, stop the transformation job before it completes. For more information on pricing and quotas, see Amazon Q Developer pricing .
-codemodernizer.migration_plan.header.description=Plan to Transform your project
+codemodernizer.migration_plan.header.description=Plan to transform your module
codemodernizer.migration_plan.header.title=Code Transformation plan by Amazon Q
codemodernizer.migration_plan.substeps.description_failed=Build failed
codemodernizer.migration_plan.substeps.description_stopped=Job is stopped
@@ -708,7 +750,6 @@ codemodernizer.notification.info.download.started.content=Downloading the update
codemodernizer.notification.info.download.started.title=Download Started
codemodernizer.notification.info.modernize_complete.content=Amazon Q finished the transformation. You can review the diff to see the proposed changes and accept or reject them. The transformation summary has details about the files that were updated.
codemodernizer.notification.info.modernize_complete.title=Transform Complete
-codemodernizer.notification.info.modernize_complete.view_diff=View diff
codemodernizer.notification.info.modernize_complete.view_summary=View transformation summary
codemodernizer.notification.info.modernize_failed.connection_failed=Amazon Q could not complete the transformation. Try starting the transformation again. {0}
codemodernizer.notification.info.modernize_failed.title=Transformation failed
@@ -716,7 +757,7 @@ codemodernizer.notification.info.modernize_failed.unknown_failure_reason=Unknown
codemodernizer.notification.info.modernize_ongoing.view_status=View status
codemodernizer.notification.info.modernize_partial_complete.content=Amazon Q transformed part of your code. You can review the diff to see my proposed changes and accept or reject them. The transformation summary has details about the files I updated and the errors that prevented a complete transformation.
codemodernizer.notification.info.modernize_partial_complete.title=Transformation partially successful!
-codemodernizer.notification.info.transformation_resume.content=Amazon Q was unable to resume polling for the job you started before closing the project.
+codemodernizer.notification.info.transformation_resume.content=Amazon Q was unable to resume polling for the job you started before closing the module.
codemodernizer.notification.info.transformation_resume.title=Unable to resume polling for job updates.
codemodernizer.notification.info.transformation_start_stopping.as_no_response=Amazon Q could not stop the transformation.
codemodernizer.notification.info.transformation_start_stopping.content=Amazon Q is stopping your transformation. This might take a few seconds.
@@ -734,11 +775,9 @@ codemodernizer.notification.warn.download_failed_other.content=Amazon Q ran into
codemodernizer.notification.warn.download_failed_ssl.content=Please make sure all your certificates for your proxy client have been set up correctly for your IDE.
codemodernizer.notification.warn.download_failed_wildcard.content=Check your IDE proxy settings and remove any wildcard (*) references, and then try viewing the diff again.
codemodernizer.notification.warn.expired_credentials.title=Your connection to Q has expired
-codemodernizer.notification.warn.invalid_project.description.reason.invalid_jdk_versions=None of your open modules are supported for code transformation with Amazon Q. Amazon Q can upgrade Java 8, Java 11, and Java 17 projects built on Maven.
-codemodernizer.notification.warn.invalid_project.description.reason.missing_content_roots=None of your open modules are supported for code transformation with Amazon Q. Amazon Q can upgrade Java 8, Java 11, and Java 17 projects built on Maven, with content roots configured.
-codemodernizer.notification.warn.invalid_project.description.reason.no_valid_files=None of your open modules are supported for code transformation with Amazon Q. A pom.xml is required for transformation. Amazon Q can upgrade Java 8, Java 11, and Java 17 projects built on Maven.
+codemodernizer.notification.warn.invalid_project.description.reason.missing_content_roots=None of your open modules are supported for code transformation with Amazon Q. Amazon Q can upgrade Java 8, Java 11, and Java 17 modules built on Maven, with content roots configured.
codemodernizer.notification.warn.invalid_project.description.reason.not_logged_in=Amazon Q cannot start the transformation as you are not logged in with Identity Center or Builder ID. Also ensure that you are not using IntelliJ version 232.8660.185 and that you are not developing on a remote host (uncommon).
-codemodernizer.notification.warn.invalid_project.description.reason.remote_backend=None of your open modules are supported for code transformation with Amazon Q. Amazon Q cannot transform projects running on a remote host.
+codemodernizer.notification.warn.invalid_project.description.reason.remote_backend=None of your open modules are supported for code transformation with Amazon Q. Amazon Q cannot transform modules running on a remote host.
codemodernizer.notification.warn.maven_failed.content=Amazon Q could not run the Maven clean install command to build your module.
codemodernizer.notification.warn.maven_failed.title=Amazon Q Code Transform unable to zip dependencies
codemodernizer.notification.warn.on_resume.unknown_status_response.content=We received data from Amazon Q in a format that the plugin cannot handle. You may need to update the plugin and then try again.
@@ -747,9 +786,9 @@ codemodernizer.notification.warn.submit_feedback=Submit feedback
codemodernizer.notification.warn.unable_to_start_job=Amazon Q could not begin the transformation. Try starting the transformation again. {0}
codemodernizer.notification.warn.unknown_start_failure=Amazon Q could not begin the transformation. Try starting the transformation again.
codemodernizer.notification.warn.unknown_status_response=Amazon Q could not complete the transformation. Try starting the transformation again.
-codemodernizer.notification.warn.upload_failed=Amazon Q could not upload your project. Try starting the transformation again. {0}
+codemodernizer.notification.warn.upload_failed=Amazon Q could not upload your module. Try starting the transformation again. {0}
codemodernizer.notification.warn.upload_failed_expired_credentials.content=Unable to upload results as your credentials expired, please reauthenticate to Q and try again.
-codemodernizer.notification.warn.validation.no_jdk=Amazon Q couldn't build your project with your JDK configuration. Go to File and choose Project Structure to update your project SDK and module SDK.
+codemodernizer.notification.warn.validation.no_jdk=Amazon Q couldn't build your module with your JDK configuration. Go to File and choose Project Structure to update your project SDK and module SDK.
codemodernizer.notification.warn.view_build_log_failed.content=Unable to display the failure build log due to an error.
codemodernizer.notification.warn.view_build_log_failed.title=Unable to display failure build log
codemodernizer.notification.warn.view_diff_failed.content=Amazon Q could not download and parse the diff with your upgraded code. {0}
@@ -758,9 +797,8 @@ codemodernizer.notification.warn.view_summary_failed.content=Unable to display t
codemodernizer.notification.warn.view_summary_failed.title=Unable to display transformation summary
codemodernizer.notification.warn.zip_creation_failed=Amazon Q could not zip the selected module and begin the transformation. Try starting the transformation again. {0}
codemodernizer.notification.warn.zip_creation_failed.reasons.unknown=An unexpected error occurred
-codemodernizer.notification.warn.zip_too_large.content=Sorry, your project size exceeds the Amazon Q Code Transformation upload limit of 2GB.
-codemodernizer.notification.warn.zip_too_large.title=Project size exceeds limit
-codemodernizer.toolwindow.banner.action.diff=View diff
+codemodernizer.notification.warn.zip_too_large.content=Sorry, your module size exceeds the Amazon Q Code Transformation upload limit of 2GB.
+codemodernizer.notification.warn.zip_too_large.title=Module size exceeds limit
codemodernizer.toolwindow.banner.action.feedback=Provide Feedback
codemodernizer.toolwindow.banner.action.plan=View transformation plan
codemodernizer.toolwindow.banner.action.summary=View transformation summary
@@ -792,60 +830,93 @@ codemodernizer.toolwindow.scan_in_progress.stopping=Stopping the job...
codemodernizer.toolwindow.scan_in_progress.transforming=Amazon Q is transforming your code.
codemodernizer.toolwindow.stop_scan=Stop job
codemodernizer.toolwindow.table.header.date=Date
-codemodernizer.toolwindow.table.header.job_id=Job Id
+codemodernizer.toolwindow.table.header.job_id=Job ID
codemodernizer.toolwindow.table.header.module_name=Module name
codemodernizer.toolwindow.table.header.run_length=Job running time
codemodernizer.toolwindow.table.header.status=Status
codemodernizer.toolwindow.transformation.progress.header=Transformation progress
+codemodernizer.toolwindow.transformation.progress.job_id=Job ID: {0}
codemodernizer.toolwindow.transformation.progress.running_time=Running time: {0}
+codescan.chat.message.button.fileScan=Review active file
+codescan.chat.message.button.openIssues=View in Code Issues Panel
+codescan.chat.message.button.projectScan=Review project
+codescan.chat.message.error_request=Request failed
+codescan.chat.message.not_git_repo=Your workspace is not in a git repository. I'll review your project files for security issues, and your in-flight changes for code quality issues.
+codescan.chat.message.project_scan_failed=Sorry, I ran into an issue during the review. Please try again.
+codescan.chat.message.scan_begin_file=Okay, I'm reviewing your file for code issues.
+codescan.chat.message.scan_begin_project=Okay, I'm reviewing your project for code issues.
+codescan.chat.message.scan_begin_wait_time=This may take a few minutes. I'll share updates here as I work on this.
+codescan.chat.message.scan_file_in_progress=File review is in progress...
+codescan.chat.message.scan_project_in_progress=Project review is in progress...
+codescan.chat.message.scan_step_1=Initiating code review.
+codescan.chat.message.scan_step_2=Waiting for review to finish.
+codescan.chat.message.scan_step_3=Processing review results.
+codescan.chat.new_scan.input.message=Which type of review would you like to run?
+codescan.chat.placeholder.scan_in_progress=Reviewing code issues...
+codescan.chat.placeholder.waiting_for_inputs=Waiting on your inputs...
codewhisperer.actions.connect_github.title=Connect with Us on GitHub
codewhisperer.actions.open_settings.title=Open Settings
codewhisperer.actions.send_feedback.title=Send Feedback
codewhisperer.actions.view_documentation.title=View Documentation
+codewhisperer.codefix.code_fix_job_timed_out=Amazon Q: Timed out generating code fix
+codewhisperer.codefix.create_code_fix_error=Amazon Q: Failed to generate fix for the issue
+codewhisperer.codefix.invalid_zip_error=Amazon Q: Failed to create valid zip
codewhisperer.codescan.apply_fix_button_label=Apply fix
codewhisperer.codescan.apply_fix_button_tooltip=Apply suggested fix
codewhisperer.codescan.build_artifacts_not_found=Cannot find build artifacts for the project. Try rebuilding the Java project in IDE or specify compilation output path in File | Project Structure... | Project | Compiler output:
-codewhisperer.codescan.cancelled_by_user_exception=Code scan job cancelled by user.
+codewhisperer.codescan.cancelled_by_user_exception=Code review job cancelled by user.
codewhisperer.codescan.cannot_read_file=Amazon Q encountered an error while parsing a file.
+codewhisperer.codescan.clear_filters=Clear Filters
codewhisperer.codescan.cwe_label=Common Weakness Enumeration (CWE)
codewhisperer.codescan.detector_library_label=Detector library
-codewhisperer.codescan.explain_button_label=Amazon Q: Explain
-codewhisperer.codescan.file_ext_not_supported=File extension {0} is not supported for Amazon Q Security Scan feature. Please try again with a valid file format - java, python, javascript, typescript, csharp, yaml, json, tf, hcl, ruby, go.
+codewhisperer.codescan.explain_button_label=Explain
+codewhisperer.codescan.file_ext_not_supported=File extension {0} is not supported for the Amazon Q Code Review feature. Please try again with a valid file format - java, python, javascript, typescript, csharp, yaml, json, tf, hcl, ruby, go.
codewhisperer.codescan.file_name_issues_count= {0} {1} {2, choice, 1#1 issue|2#{2,number} issues}
codewhisperer.codescan.file_not_found=For file path {0} with error message: {0}
-codewhisperer.codescan.file_too_large=Amazon Q: The selected file exceeds the input artifact limit. Try again with a smaller file. For more information about scan limits, see the Amazon Q documentation.
+codewhisperer.codescan.file_path_label=File Path
+codewhisperer.codescan.file_too_large=Amazon Q: The selected file exceeds the input artifact limit. Try again with a smaller file. For more information about review limits, see the Amazon Q documentation.
codewhisperer.codescan.file_too_large_telemetry=Payload size limit reached
codewhisperer.codescan.fix_applied_fail=Apply fix command failed. {0}
codewhisperer.codescan.fix_available_label=Code fix available
codewhisperer.codescan.fix_button_label=Fix with Q
+codewhisperer.codescan.generate_fix_button_label=Generate Fix
+codewhisperer.codescan.ignore_all_button=Ignore All
+codewhisperer.codescan.ignore_button=Ignore
codewhisperer.codescan.invalid_source_zip_telemetry=Failed to create valid source zip.
-codewhisperer.codescan.java_module_not_found=Java plugin is required for scanning Java files, install Java plugin or perform the code scan in Intellij Idea instead.
-codewhisperer.codescan.no_file_open=Amazon Q: No file is open in an active editor. Open a file to start a Security Scan.
-codewhisperer.codescan.no_file_open_telemetry=Open a valid file to scan.
-codewhisperer.codescan.problems_window_not_found=Unable to display Security Scan results as the Problems View tool window cannot be fetched.
-codewhisperer.codescan.run_scan=Run Project Scan
-codewhisperer.codescan.run_scan_complete= Security Scan completed for {0, choice, 1#1 file|2#{0,number} files}. {1, choice, 0#No issues|1#1 issue|2#{1,number} issues} found in {2}. Last Run {4}
-codewhisperer.codescan.run_scan_error=Amazon Q encountered an error while scanning for security issues. Please try again later.
-codewhisperer.codescan.run_scan_error_telemetry=Security scan failed.
-codewhisperer.codescan.run_scan_info=Select 'Run' in toolbar to scan this package for security issues.
-codewhisperer.codescan.scan_display=Amazon Q Security Issues
-codewhisperer.codescan.scan_display_with_issues=Amazon Q Security Issues {0}
-codewhisperer.codescan.scan_in_progress=Scanning active project and its dependencies...
+codewhisperer.codescan.java_module_not_found=Java plugin is required for reviewing Java files, install Java plugin or perform the code review in Intellij Idea instead.
+codewhisperer.codescan.no_file_open=Amazon Q: No file is open in an active editor. Open a file to start a Code Review.
+codewhisperer.codescan.no_file_open_telemetry=Open a valid file to review.
+codewhisperer.codescan.problems_window_not_found=Unable to display Code Review results as the Problems View tool window cannot be fetched.
+codewhisperer.codescan.quota_exceeded=You've reached the monthly quota for Amazon Q Developer's agent capabilities. You can try again next month. For more information on usage limits, see the Amazon Q Developer pricing page.
+codewhisperer.codescan.regenerate_fix_button_label=Regenerate Fix
+codewhisperer.codescan.run_scan=Full Project Scan is now /review! Open in Chat Panel
+codewhisperer.codescan.run_scan_complete= Code Review completed for {0, choice, 1#1 file|2#{0,number} files}. {1, choice, 0#No issues|1#1 issue|2#{1,number} issues} found in {2}. Last Run {4}
+codewhisperer.codescan.run_scan_error=Amazon Q encountered an error while reviewing for code issues. Please try again later.
+codewhisperer.codescan.run_scan_error_telemetry=Code Review failed.
+codewhisperer.codescan.run_scan_info=Enter /review in Amazon Q Chat Panel to run code reviews.
+codewhisperer.codescan.scan_complete_count=- {0}: `{1, choice, 1#{1,number} issue|2#{1,number} issues}`
+codewhisperer.codescan.scan_complete_file=Reviewing your File is complete. Here's what I found:
+codewhisperer.codescan.scan_complete_project=Reviewing your Project is complete. Here's what I found:
+codewhisperer.codescan.scan_display=Amazon Q Code Issues
+codewhisperer.codescan.scan_display_with_issues=Amazon Q Code Issues {0}
+codewhisperer.codescan.scan_in_progress=Code review in progress...
codewhisperer.codescan.scan_recommendation= {0} {1}
-codewhisperer.codescan.scan_recommendation_invalid= {0} {1} [No longer valid: Re-scan to validate the fix]
-codewhisperer.codescan.scan_recommendation_invalid.tooltip_text=No longer valid. Re-scan to validate the fix.
-codewhisperer.codescan.scan_timed_out=Security Scan failed. Amazon Q timed out.
-codewhisperer.codescan.scanned_files_heading= {0} files were scanned during the last security scan.
-codewhisperer.codescan.stop_scan=Stop Security Scan
-codewhisperer.codescan.stop_scan_confirm_button=Stop scan
-codewhisperer.codescan.stop_scan_confirm_message=Are you sure you want to stop ongoing security scan? This scan will be counted as one complete scan towards your monthly security scan limits.
-codewhisperer.codescan.stopping_scan=Stopping Security Scan...
+codewhisperer.codescan.scan_recommendation_invalid= {0} {1} [No longer valid: Re-run the review to validate the fix]
+codewhisperer.codescan.scan_recommendation_invalid.tooltip_text=No longer valid. Re-run the review to validate the fix.
+codewhisperer.codescan.scan_results_hidden_by_filters=All code review results are hidden by current filters.
+codewhisperer.codescan.scan_timed_out=Code Review failed. Amazon Q timed out.
+codewhisperer.codescan.scanned_files_heading= {0} files were reviewed during the last code review.
+codewhisperer.codescan.severity_issues_count= {0} {1, choice, 1#{1,number} issue|2#{1,number} issues}
+codewhisperer.codescan.stop_scan=Stop Code Review
+codewhisperer.codescan.stop_scan_confirm_button=Stop review
+codewhisperer.codescan.stop_scan_confirm_message=Are you sure you want to stop ongoing code review? This review will be counted as one complete review towards your monthly code review limits.
+codewhisperer.codescan.stopping_scan=Stopping Code Review...
codewhisperer.codescan.suggested_fix_description=Why are we recommending this?
codewhisperer.codescan.suggested_fix_label=Suggested code fix preview
-codewhisperer.codescan.unsupported_language_error=Amazon Q: Project does not contain valid files to scan
-codewhisperer.codescan.unsupported_language_error_telemetry=Project does not contain valid files to scan
-codewhisperer.codescan.upload_to_s3_failed=Amazon Q is unable to upload your workspace artifacts to Amazon S3 for security scans. For more information, see the Amazon Q documentation.
-codewhisperer.codescan.view_scanned_files=View {0} scanned files
+codewhisperer.codescan.unsupported_language_error=Amazon Q: Project does not contain valid files to review
+codewhisperer.codescan.unsupported_language_error_telemetry=Project does not contain valid files to review
+codewhisperer.codescan.upload_to_s3_failed=Amazon Q is unable to upload your project artifacts to Amazon S3 for code reviews. For more information, see the Amazon Q documentation.
+codewhisperer.codescan.view_scanned_files=View {0} reviewed files
codewhisperer.credential.login.dialog.exception.cancel_login=Login cancelled
codewhisperer.credential.login.dialog.ok_button=Connect
codewhisperer.credential.login.dialog.prompt=Select a connection option to start using Amazon Q
@@ -873,11 +944,11 @@ codewhisperer.explorer.learn=Learn
codewhisperer.explorer.node.dismiss=Dismiss
codewhisperer.explorer.node.install_q=Install the Amazon Q Plugin
codewhisperer.explorer.pause_auto=Pause Auto-Suggestions
-codewhisperer.explorer.pause_auto_scans =Pause Auto-Scans
+codewhisperer.explorer.pause_auto_scans=Pause Auto-Reviews
codewhisperer.explorer.paused=\ Paused
codewhisperer.explorer.reconnect=Reconnect
codewhisperer.explorer.resume_auto=Resume Auto-Suggestions
-codewhisperer.explorer.resume_auto_scans =Resume Auto-Scans
+codewhisperer.explorer.resume_auto_scans=Resume Auto-Reviews
codewhisperer.explorer.tooltip.comment=Start with auto-suggestions and find more features here!
codewhisperer.explorer.tooltip.title=Get started with Amazon Q
codewhisperer.explorer.usage_limit_hit=\ Free tier limit met, paused until {0}
@@ -915,7 +986,7 @@ codewhisperer.notification.custom.simple.button.select_another_customization=Sel
codewhisperer.notification.custom.simple.button.select_customization=Select customization
codewhisperer.notification.remote.ide_unsupported.message=Please update your IDE backend to a 2023.3 or later version to continue using Amazon Q inline suggestions.
codewhisperer.notification.remote.ide_unsupported.title=Amazon Q inline suggestion not supported in this IDE version
-codewhisperer.notification.usage_limit.codescan.warn.content=Amazon Q: You have reached the monthly limit for project scans.
+codewhisperer.notification.usage_limit.codescan.warn.content=Amazon Q: You have reached the monthly limit for project reviews.
codewhisperer.notification.usage_limit.codesuggestion.warn.content=You have reached the monthly fair use limit of code recommendations.
codewhisperer.popup.button.accept= Insert Code \u21E5
codewhisperer.popup.button.next=Next→
@@ -931,10 +1002,10 @@ codewhisperer.statusbar.popup.title=Reconnect to Amazon Q?
codewhisperer.statusbar.sub_menu.connect_help.title=Connect / Help
codewhisperer.statusbar.sub_menu.inline.title=Inline Suggestions
codewhisperer.statusbar.sub_menu.other_features.title=Other Features
-codewhisperer.statusbar.sub_menu.security_scans.title=Security Scans
+codewhisperer.statusbar.sub_menu.security_scans.title=Code Reviews
codewhisperer.statusbar.tooltip=Amazon Q status
codewhisperer.toolwindow.entry.prefix=[{0}] ACCEPTED recommendation with the following code provided with reference under
-codewhisperer.toolwindow.entry.suffix= {1, choice, 0#|1#. Added to {0}} at line {2}
+codewhisperer.toolwindow.entry.suffix={1, choice, 0#|1#. Added to {0}} at line {2}
codewhisperer.toolwindow.popup.text=Reference code under the {0} license from repository {1}
codewhisperer.toolwindow.settings=Amazon Q Settings
codewhisperer.toolwindow.settings.prefix=Don't want suggestions that include code with references? Uncheck this option in
@@ -1264,9 +1335,11 @@ gateway.connection.workflow.start_ide=Start IDE
gateway.connection.workflow.step_failed=\nStep failed exceptionally\n
gateway.connection.workflow.step_skipped=Step skipped
gateway.connection.workflow.step_successful=\nStep completed successfully\n
+general.acknowledge=Acknowledge
general.add.another=Add another
general.auth.reauthenticate=Reauthenticate
general.cancel=Cancel
+general.canceling=Canceling
general.close_button=Close
general.configure_button=Configure
general.confirm=Confirm
@@ -1278,6 +1351,7 @@ general.default=Default
general.delete=Delete
general.delete_accessible_name=Delete confirmation box
general.details=(details)
+general.dismiss=Dismiss
general.execute_button=Execute
general.execution.canceled=canceled
general.execution.cli_error=Command did not exit successfully, exit code: {0}\n
@@ -1291,10 +1365,12 @@ general.in_progress_button=In progress
general.logs=Logs
general.message=Message
general.more=More
+general.more_dialog=More...
general.name.label=Name:
general.no_changes=No changes were provided
general.notification.action.hide_forever=Don't show again
general.notification.action.hide_once=Dismiss
+general.ok=OK
general.open.in.progress=Opening...
general.open_in_aws_console=Open in AWS Console
general.open_in_aws_console.error=Failed to open link in browser
@@ -1307,6 +1383,7 @@ general.save=Save
general.select_button=Select
general.step.canceled={0} has been canceled
general.step.failed={0} has failed: {1}
+general.success=Complete...
general.time=Time
general.time.five_minutes=Five Minutes
general.time.one_minute=One Minute
@@ -1381,6 +1458,7 @@ gettingstarted.setup.tabs.builderid=AWS Builder ID
gettingstarted.setup.tabs.iam=IAM Credentials
gettingstarted.setup.tabs.idc=IAM Identity Center
gettingstarted.setup.title=AWS Toolkit: Setup Authentication
+group.aws.toolkit.dynamoViewer.changeMaxResults.text=Max Results
group.aws.toolkit.dynamoViewer.toolbar.settings.text=Settings
group.aws.toolkit.jetbrains.core.services.cwc.actions.ContextMenuActions.text=Amazon Q
group.aws.toolkit.s3viewer.contextMenu.copyGroup.text=Copy
@@ -1537,6 +1615,10 @@ lambda.workflow.update_code.wait_for_updatable=Waiting for function to transitio
loading_resource.failed=Failed loading resources
loading_resource.loading=Loading...
loading_resource.still_loading=Resources are still loading
+notification.changelog=Changelog
+notification.expand=Expand
+notification.learn_more=Learn more
+notification.update=Update
plugin.incompatible.fix=Disable incompatible plugins and restart IDE
plugin.incompatible.message=The plugin versions for Amazon Q, AWS Toolkit, and AWS Toolkit Core must match or conflicts may occur.
plugin.incompatible.title=AWS Plugin Incompatibility
@@ -1974,6 +2056,19 @@ sqs.subscribe.sns.validation.empty_topic=Topic must be specified.
sqs.toolwindow=SQS
sqs.url.parse_error=Error parsing SQS queue URL
tags.title=Tags
+testgen.error.generic_error_message=Amazon Q encountered an error while generating tests. Try again later.
+testgen.error.generic_technical_error_message=I am experiencing technical difficulties at the moment. Please try again in a few minutes.
+testgen.error.maximum_generations_reach=You've reached the monthly quota for Amazon Q Developer's agent capabilities. You can try again next month. For more information on usage limits, see the Amazon Q Developer pricing page .
+testgen.message.cancelled=Unit test generation cancelled.
+testgen.message.failed=Sorry, Test generation failed. Please try again in few minutes.
+testgen.message.regenerate_input=Sure thing. Please provide new instructions for me to generate the tests, and select the function(s) you would like to test.
+testgen.message.success=Unit test generation completed.
+testgen.no_file_found=Sorry, there isn't a source file open right now that I can generate a test for. Make sure you open a source file so I can generate tests.
+testgen.placeholder.newtab=Ask any coding question or type \u0022/\u0022 for actions
+testgen.placeholder.select_an_option = Please select an action to proceed (Accept or Reject)
+testgen.placeholder.view_diff=Select View Diff to see the generated unit tests
+testgen.placeholder.waiting_on_your_inputs=Waiting on your inputs...
+testgen.progressbar.generate_unit_tests=Generating unit tests...
toolkit.login.aws_builder_id.already_connected.cancel=Use existing AWS Builder ID
toolkit.login.aws_builder_id.already_connected.message=You already signed in with an AWS Builder ID.\nSign out to add another?
toolkit.login.aws_builder_id.already_connected.reconnect=Sign out
diff --git a/plugins/core/sdk-codegen/codegen-resources/codewhispererruntime/service-2.json b/plugins/core/sdk-codegen/codegen-resources/codewhispererruntime/service-2.json
index 00b4bb2334..00e2305eb2 100644
--- a/plugins/core/sdk-codegen/codegen-resources/codewhispererruntime/service-2.json
+++ b/plugins/core/sdk-codegen/codegen-resources/codewhispererruntime/service-2.json
@@ -1267,7 +1267,8 @@
"required": ["conversationId", "codeGenerationId"],
"members": {
"conversationId": { "shape": "ConversationId" },
- "codeGenerationId": { "shape": "CodeGenerationId" }
+ "codeGenerationId": { "shape": "CodeGenerationId" },
+ "intent": { "shape": "String" }
}
},
"GetTaskAssistCodeGenerationResponse": {
diff --git a/plugins/core/sdk-codegen/codegen-resources/codewhispererstreaming/service-2.json b/plugins/core/sdk-codegen/codegen-resources/codewhispererstreaming/service-2.json
index 3934499739..1e7898a48d 100644
--- a/plugins/core/sdk-codegen/codegen-resources/codewhispererstreaming/service-2.json
+++ b/plugins/core/sdk-codegen/codegen-resources/codewhispererstreaming/service-2.json
@@ -288,7 +288,8 @@
"ExportContext":{
"type":"structure",
"members":{
- "transformationExportContext":{"shape":"TransformationExportContext"}
+ "transformationExportContext":{"shape":"TransformationExportContext"},
+ "unitTestGenerationExportContext":{"shape":"UnitTestGenerationExportContext"}
},
"union":true
},
@@ -296,7 +297,8 @@
"type":"string",
"enum":[
"TRANSFORMATION",
- "TASK_ASSIST"
+ "TASK_ASSIST",
+ "UNIT_TESTS"
]
},
"ExportResultArchiveRequest":{
@@ -625,6 +627,12 @@
"USAGE"
]
},
+ "TestGenerationJobGroupName":{
+ "type":"string",
+ "max":128,
+ "min":1,
+ "pattern":"[a-zA-Z0-9-_]+"
+ },
"TextDocument":{
"type":"structure",
"required":["relativeFilePath"],
@@ -706,6 +714,19 @@
"downloadArtifactType":{"shape":"TransformationDownloadArtifactType"}
}
},
+ "UUID":{
+ "type":"string",
+ "max":36,
+ "min":36
+ },
+ "UnitTestGenerationExportContext":{
+ "type":"structure",
+ "required":["testGenerationJobGroupName"],
+ "members":{
+ "testGenerationJobGroupName":{"shape":"TestGenerationJobGroupName"},
+ "testGenerationJobId":{"shape":"UUID"}
+ }
+ },
"UploadId":{
"type":"string",
"max":128,
diff --git a/plugins/core/src/main/resources/META-INF/plugin.xml b/plugins/core/src/main/resources/META-INF/plugin.xml
index dbc7e6b397..9703141c30 100644
--- a/plugins/core/src/main/resources/META-INF/plugin.xml
+++ b/plugins/core/src/main/resources/META-INF/plugin.xml
@@ -22,8 +22,11 @@
+
+
+
diff --git a/plugins/core/webview/package-lock.json b/plugins/core/webview/package-lock.json
index ea887ed6db..33e32e698a 100644
--- a/plugins/core/webview/package-lock.json
+++ b/plugins/core/webview/package-lock.json
@@ -1259,10 +1259,11 @@
"integrity": "sha512-dcKFX3jn0MpIaXjisoRvexIJVEKzaq7z2rZKxf+MSr9TkdmHmsU4m2lcLojrj/FHl8mk5VxMmYA+ftRkP/3oKQ=="
},
"node_modules/cross-spawn": {
- "version": "7.0.3",
- "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz",
- "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==",
+ "version": "7.0.6",
+ "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.6.tgz",
+ "integrity": "sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA==",
"dev": true,
+ "license": "MIT",
"dependencies": {
"path-key": "^3.1.0",
"shebang-command": "^2.0.0",
@@ -2446,12 +2447,13 @@
}
},
"node_modules/micromatch": {
- "version": "4.0.5",
- "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.5.tgz",
- "integrity": "sha512-DMy+ERcEW2q8Z2Po+WNXuw3c5YaUSFjAO5GsJqfEl7UjvtIuFKO6ZrKvcItdy98dwFI2N1tg3zNIdKaQT+aNdA==",
+ "version": "4.0.8",
+ "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.8.tgz",
+ "integrity": "sha512-PXwfBhYu0hBCPw8Dn0E+WDYb7af3dSLVWKi3HGv84IdF4TyFoC0ysxFd0Goxw7nSv4T/PzEJQxsYsEiFCKo2BA==",
"dev": true,
+ "license": "MIT",
"dependencies": {
- "braces": "^3.0.2",
+ "braces": "^3.0.3",
"picomatch": "^2.3.1"
},
"engines": {
@@ -2507,15 +2509,16 @@
"dev": true
},
"node_modules/nanoid": {
- "version": "3.3.7",
- "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.7.tgz",
- "integrity": "sha512-eSRppjcPIatRIMC1U6UngP8XFcz8MQWGQdt1MTBQ7NaAmvXDfvNxbvWV3x2y6CdEUciCSsDHDQZbhYaB8QEo2g==",
+ "version": "3.3.8",
+ "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.8.tgz",
+ "integrity": "sha512-WNLf5Sd8oZxOm+TzppcYk8gVOgP+l58xNy58D0nbUnOxOWRWvlcCV4kUF7ltmI6PsrLl/BgKEyS4mqsGChFN0w==",
"funding": [
{
"type": "github",
"url": "https://github.com/sponsors/ai"
}
],
+ "license": "MIT",
"bin": {
"nanoid": "bin/nanoid.cjs"
},
diff --git a/plugins/core/webview/src/q-ui/components/loginOptions.vue b/plugins/core/webview/src/q-ui/components/loginOptions.vue
index 9d55100d22..bce0f2a469 100644
--- a/plugins/core/webview/src/q-ui/components/loginOptions.vue
+++ b/plugins/core/webview/src/q-ui/components/loginOptions.vue
@@ -37,7 +37,7 @@ export default defineComponent({
login(type: LoginOption) {
this.$emit('login', type)
},
- emitUiClickTelemetry(elementId: String) {
+ emitUiClickTelemetry(elementId: string) {
this.$emit('emitUiClickTelemetry', elementId)
}
}
diff --git a/plugins/toolkit/jetbrains-core/resources/META-INF/plugin.xml b/plugins/toolkit/jetbrains-core/resources/META-INF/plugin.xml
index 5172c1fc68..39be6ec4e6 100644
--- a/plugins/toolkit/jetbrains-core/resources/META-INF/plugin.xml
+++ b/plugins/toolkit/jetbrains-core/resources/META-INF/plugin.xml
@@ -196,7 +196,6 @@
-
@@ -625,7 +624,7 @@
-
diff --git a/plugins/toolkit/jetbrains-core/src/software/aws/toolkits/jetbrains/core/credentials/actions/ExplorerNewConnectionAction.kt b/plugins/toolkit/jetbrains-core/src/software/aws/toolkits/jetbrains/core/credentials/actions/ExplorerNewConnectionAction.kt
index 0f05448be9..1f67c8c2fb 100644
--- a/plugins/toolkit/jetbrains-core/src/software/aws/toolkits/jetbrains/core/credentials/actions/ExplorerNewConnectionAction.kt
+++ b/plugins/toolkit/jetbrains-core/src/software/aws/toolkits/jetbrains/core/credentials/actions/ExplorerNewConnectionAction.kt
@@ -7,7 +7,7 @@ import com.intellij.icons.AllIcons
import com.intellij.openapi.actionSystem.AnActionEvent
import com.intellij.openapi.application.runInEdt
import com.intellij.openapi.project.DumbAwareAction
-import software.aws.toolkits.jetbrains.core.explorer.showWebview
+import software.aws.toolkits.jetbrains.core.explorer.ShowToolkitListener
import software.aws.toolkits.jetbrains.core.explorer.webview.ToolkitWebviewPanel
import software.aws.toolkits.jetbrains.core.gettingstarted.editor.GettingStartedPanel
import software.aws.toolkits.jetbrains.core.webview.BrowserState
@@ -25,7 +25,7 @@ class ExplorerNewConnectionAction : DumbAwareAction(AllIcons.General.Add) {
GettingStartedPanel.openPanel(it)
} else {
ToolkitWebviewPanel.getInstance(it).browser?.prepareBrowser(BrowserState(FeatureId.AwsExplorer, true))
- showWebview(it)
+ ShowToolkitListener.showWebview(it)
}
UiTelemetry.click(it, "devtools_connectToAws")
}
diff --git a/plugins/toolkit/jetbrains-core/src/software/aws/toolkits/jetbrains/core/credentials/actions/NewConnectionAction.kt b/plugins/toolkit/jetbrains-core/src/software/aws/toolkits/jetbrains/core/credentials/actions/NewConnectionAction.kt
index 67be9bcf01..b0a1c8601d 100644
--- a/plugins/toolkit/jetbrains-core/src/software/aws/toolkits/jetbrains/core/credentials/actions/NewConnectionAction.kt
+++ b/plugins/toolkit/jetbrains-core/src/software/aws/toolkits/jetbrains/core/credentials/actions/NewConnectionAction.kt
@@ -6,7 +6,7 @@ package software.aws.toolkits.jetbrains.core.credentials.actions
import com.intellij.openapi.actionSystem.AnActionEvent
import com.intellij.openapi.application.runInEdt
import com.intellij.openapi.project.DumbAwareAction
-import software.aws.toolkits.jetbrains.core.explorer.showWebview
+import software.aws.toolkits.jetbrains.core.explorer.ShowToolkitListener
import software.aws.toolkits.jetbrains.core.explorer.webview.ToolkitWebviewPanel
import software.aws.toolkits.jetbrains.core.gettingstarted.editor.GettingStartedPanel
import software.aws.toolkits.jetbrains.core.webview.BrowserState
@@ -22,7 +22,7 @@ class NewConnectionAction : DumbAwareAction() {
GettingStartedPanel.openPanel(it, connectionInitiatedFromExplorer = true)
} else {
ToolkitWebviewPanel.getInstance(it).browser?.prepareBrowser(BrowserState(FeatureId.AwsExplorer, true))
- showWebview(it)
+ ShowToolkitListener.showWebview(it)
}
UiTelemetry.click(e.project, "auth_gettingstarted_explorermenu")
}
diff --git a/plugins/toolkit/jetbrains-core/src/software/aws/toolkits/jetbrains/core/explorer/AwsToolkitExplorerFactory.kt b/plugins/toolkit/jetbrains-core/src/software/aws/toolkits/jetbrains/core/explorer/AwsToolkitExplorerFactory.kt
index 0174aef2e0..6179672b8c 100644
--- a/plugins/toolkit/jetbrains-core/src/software/aws/toolkits/jetbrains/core/explorer/AwsToolkitExplorerFactory.kt
+++ b/plugins/toolkit/jetbrains-core/src/software/aws/toolkits/jetbrains/core/explorer/AwsToolkitExplorerFactory.kt
@@ -6,12 +6,16 @@ package software.aws.toolkits.jetbrains.core.explorer
import com.intellij.icons.AllIcons
import com.intellij.openapi.actionSystem.ActionManager
import com.intellij.openapi.actionSystem.DefaultActionGroup
+import com.intellij.openapi.application.ApplicationManager
import com.intellij.openapi.application.runInEdt
import com.intellij.openapi.project.DumbAware
import com.intellij.openapi.project.Project
import com.intellij.openapi.wm.ToolWindow
import com.intellij.openapi.wm.ToolWindowFactory
import com.intellij.openapi.wm.ex.ToolWindowEx
+import com.intellij.ui.components.panels.Wrapper
+import com.intellij.util.messages.Topic
+import com.intellij.util.ui.components.BorderLayoutPanel
import software.aws.toolkits.core.utils.debug
import software.aws.toolkits.core.utils.getLogger
import software.aws.toolkits.jetbrains.AwsToolkit
@@ -32,15 +36,30 @@ import software.aws.toolkits.jetbrains.core.experiments.ExperimentsActionGroup
import software.aws.toolkits.jetbrains.core.explorer.webview.ToolkitWebviewPanel
import software.aws.toolkits.jetbrains.core.explorer.webview.shouldPromptToolkitReauth
import software.aws.toolkits.jetbrains.core.help.HelpIds
+import software.aws.toolkits.jetbrains.core.notifications.NotificationPanel
+import software.aws.toolkits.jetbrains.core.notifications.ProcessNotificationsBase
import software.aws.toolkits.jetbrains.core.webview.BrowserState
import software.aws.toolkits.jetbrains.utils.actions.OpenBrowserAction
import software.aws.toolkits.jetbrains.utils.isTookitConnected
import software.aws.toolkits.resources.message
import software.aws.toolkits.telemetry.FeatureId
+import java.util.EventListener
import javax.swing.JComponent
class AwsToolkitExplorerFactory : ToolWindowFactory, DumbAware {
+
override fun createToolWindowContent(project: Project, toolWindow: ToolWindow) {
+ val notificationPanel = NotificationPanel()
+ val toolkitPanel = Wrapper()
+ val mainPanel = BorderLayoutPanel()
+ mainPanel.addToTop(notificationPanel)
+ mainPanel.add(toolkitPanel)
+ val notifListener = ProcessNotificationsBase.getInstance(project)
+ notifListener.addListenerForNotification { bannerContent ->
+ runInEdt {
+ notificationPanel.updateNotificationPanel(bannerContent)
+ }
+ }
toolWindow.helpId = HelpIds.EXPLORER_WINDOW.id
if (toolWindow is ToolWindowEx) {
@@ -83,7 +102,9 @@ class AwsToolkitExplorerFactory : ToolWindowFactory, DumbAware {
AwsToolkitExplorerToolWindow.getInstance(project)
}
- val content = contentManager.factory.createContent(component, null, false).also {
+ toolkitPanel.setContent(component)
+
+ val content = contentManager.factory.createContent(mainPanel, null, false).also {
it.isCloseable = true
it.isPinnable = true
}
@@ -95,7 +116,7 @@ class AwsToolkitExplorerFactory : ToolWindowFactory, DumbAware {
ToolkitConnectionManagerListener.TOPIC,
object : ToolkitConnectionManagerListener {
override fun activeConnectionChanged(newConnection: ToolkitConnection?) {
- connectionChanged(project, newConnection)
+ connectionChanged(project, newConnection, toolkitPanel)
}
}
)
@@ -104,7 +125,7 @@ class AwsToolkitExplorerFactory : ToolWindowFactory, DumbAware {
AwsConnectionManager.CONNECTION_SETTINGS_STATE_CHANGED,
object : ConnectionSettingsStateChangeNotifier {
override fun settingsStateChanged(newState: ConnectionState) {
- settingsStateChanged(project, newState)
+ settingsStateChanged(project, newState, toolkitPanel)
}
}
)
@@ -116,18 +137,31 @@ class AwsToolkitExplorerFactory : ToolWindowFactory, DumbAware {
if (ToolkitConnectionManager.getInstance(project)
.connectionStateForFeature(CodeCatalystConnection.getInstance()) == BearerTokenAuthState.AUTHORIZED
) {
- showExplorerTree(project)
+ loadContent(AwsToolkitExplorerToolWindow.getInstance(project), toolkitPanel)
}
}
}
)
+
+ project.messageBus.connect().subscribe(
+ ShowToolkitListener.TOPIC,
+ object : ShowToolkitListener {
+ override fun showWebview(project: Project) {
+ loadContent(ToolkitWebviewPanel.getInstance(project).component, toolkitPanel)
+ }
+
+ override fun showExplorerTree(project: Project) {
+ loadContent(AwsToolkitExplorerToolWindow.getInstance(project), toolkitPanel)
+ }
+ }
+ )
}
override fun init(toolWindow: ToolWindow) {
toolWindow.stripeTitle = message("aws.notification.title")
}
- private fun connectionChanged(project: Project, newConnection: ToolkitConnection?) {
+ private fun connectionChanged(project: Project, newConnection: ToolkitConnection?, toolkitPanel: Wrapper) {
val isNewConnToolkitConnection = when (newConnection) {
is AwsConnectionManagerConnection -> {
LOG.debug { "IAM connection" }
@@ -148,16 +182,16 @@ class AwsToolkitExplorerFactory : ToolWindowFactory, DumbAware {
}
if (isNewConnToolkitConnection) {
- showExplorerTree(project)
+ loadContent(AwsToolkitExplorerToolWindow.getInstance(project), toolkitPanel)
} else if (!isTookitConnected(project) || shouldPromptToolkitReauth(project)) {
ToolkitWebviewPanel.getInstance(project).browser?.prepareBrowser(BrowserState(FeatureId.AwsExplorer))
- showWebview(project)
+ loadContent(ToolkitWebviewPanel.getInstance(project).component, toolkitPanel)
} else {
- showExplorerTree(project)
+ loadContent(AwsToolkitExplorerToolWindow.getInstance(project), toolkitPanel)
}
}
- private fun settingsStateChanged(project: Project, newState: ConnectionState) {
+ private fun settingsStateChanged(project: Project, newState: ConnectionState, toolkitPanel: Wrapper) {
val isToolkitConnected = if (newState is ConnectionState.ValidConnection) {
true
} else {
@@ -168,9 +202,15 @@ class AwsToolkitExplorerFactory : ToolWindowFactory, DumbAware {
if (!isToolkitConnected || shouldPromptToolkitReauth(project)) {
ToolkitWebviewPanel.getInstance(project).browser?.prepareBrowser(BrowserState(FeatureId.AwsExplorer))
- showWebview(project)
+ loadContent(ToolkitWebviewPanel.getInstance(project).component, toolkitPanel)
} else {
- showExplorerTree(project)
+ loadContent(AwsToolkitExplorerToolWindow.getInstance(project), toolkitPanel)
+ }
+ }
+
+ private fun loadContent(component: JComponent, toolkitPanel: Wrapper) {
+ runInEdt {
+ toolkitPanel.setContent(component)
}
}
@@ -180,22 +220,20 @@ class AwsToolkitExplorerFactory : ToolWindowFactory, DumbAware {
}
}
-fun showWebview(project: Project) {
- AwsToolkitExplorerToolWindow.toolWindow(project).loadContent(ToolkitWebviewPanel.getInstance(project).component)
-}
+interface ShowToolkitListener : EventListener {
+ fun showExplorerTree(project: Project)
+ fun showWebview(project: Project)
-fun showExplorerTree(project: Project) {
- AwsToolkitExplorerToolWindow.toolWindow(project).loadContent(AwsToolkitExplorerToolWindow.getInstance(project))
-}
+ companion object {
+ @Topic.AppLevel
+ val TOPIC = Topic.create("Show Explorer contents", ShowToolkitListener::class.java)
-private fun ToolWindow.loadContent(component: JComponent) {
- val content = contentManager.factory.createContent(component, null, false).also {
- it.isCloseable = true
- it.isPinnable = true
- }
+ fun showExplorerTree(project: Project) {
+ ApplicationManager.getApplication().messageBus.syncPublisher(TOPIC).showExplorerTree(project)
+ }
- runInEdt {
- contentManager.removeAllContents(true)
- contentManager.addContent(content)
+ fun showWebview(project: Project) {
+ ApplicationManager.getApplication().messageBus.syncPublisher(TOPIC).showWebview(project)
+ }
}
}
diff --git a/plugins/toolkit/jetbrains-core/src/software/aws/toolkits/jetbrains/core/explorer/ExplorerToolWindow.kt b/plugins/toolkit/jetbrains-core/src/software/aws/toolkits/jetbrains/core/explorer/ExplorerToolWindow.kt
index a3f3f3a456..c670548e7c 100644
--- a/plugins/toolkit/jetbrains-core/src/software/aws/toolkits/jetbrains/core/explorer/ExplorerToolWindow.kt
+++ b/plugins/toolkit/jetbrains-core/src/software/aws/toolkits/jetbrains/core/explorer/ExplorerToolWindow.kt
@@ -211,7 +211,7 @@ class ExplorerToolWindow(private val project: Project) :
GettingStartedPanel.openPanel(project)
} else {
ToolkitWebviewPanel.getInstance(project).browser?.prepareBrowser(BrowserState(FeatureId.AwsExplorer, true))
- showWebview(project)
+ ShowToolkitListener.showWebview(project)
}
}
}
diff --git a/plugins/toolkit/jetbrains-core/src/software/aws/toolkits/jetbrains/core/explorer/webview/OuterToolkitPanel.kt b/plugins/toolkit/jetbrains-core/src/software/aws/toolkits/jetbrains/core/explorer/webview/OuterToolkitPanel.kt
new file mode 100644
index 0000000000..82717b337d
--- /dev/null
+++ b/plugins/toolkit/jetbrains-core/src/software/aws/toolkits/jetbrains/core/explorer/webview/OuterToolkitPanel.kt
@@ -0,0 +1,36 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.core.explorer.webview
+
+import com.intellij.openapi.project.Project
+import com.intellij.ui.components.panels.Wrapper
+import com.intellij.util.ui.components.BorderLayoutPanel
+import software.aws.toolkits.core.utils.error
+import software.aws.toolkits.core.utils.getLogger
+import software.aws.toolkits.jetbrains.core.explorer.AwsToolkitExplorerToolWindow
+import software.aws.toolkits.jetbrains.utils.isTookitConnected
+import javax.swing.JComponent
+
+class OuterToolkitPanel(val project: Project) : BorderLayoutPanel() {
+ private val wrapper = Wrapper()
+ init {
+ isOpaque = false
+ addToCenter(wrapper)
+ val component = if (!isTookitConnected(project) || shouldPromptToolkitReauth(project)) {
+ ToolkitWebviewPanel.getInstance(project).component
+ } else {
+ AwsToolkitExplorerToolWindow.getInstance(project)
+ }
+
+ updateToolkitPanel(component)
+ }
+
+ fun updateToolkitPanel(content: JComponent) {
+ try {
+ wrapper.setContent(content)
+ } catch (e: Exception) {
+ getLogger().error { "Error while creating window" }
+ }
+ }
+}
diff --git a/plugins/toolkit/jetbrains-core/src/software/aws/toolkits/jetbrains/core/explorer/webview/ToolkitLoginWebview.kt b/plugins/toolkit/jetbrains-core/src/software/aws/toolkits/jetbrains/core/explorer/webview/ToolkitLoginWebview.kt
index 481e84e7f9..8ea17eceaa 100644
--- a/plugins/toolkit/jetbrains-core/src/software/aws/toolkits/jetbrains/core/explorer/webview/ToolkitLoginWebview.kt
+++ b/plugins/toolkit/jetbrains-core/src/software/aws/toolkits/jetbrains/core/explorer/webview/ToolkitLoginWebview.kt
@@ -46,7 +46,7 @@ import software.aws.toolkits.jetbrains.core.credentials.pinning.CodeCatalystConn
import software.aws.toolkits.jetbrains.core.credentials.sono.CODECATALYST_SCOPES
import software.aws.toolkits.jetbrains.core.credentials.sono.IDENTITY_CENTER_ROLE_ACCESS_SCOPE
import software.aws.toolkits.jetbrains.core.credentials.sono.isSono
-import software.aws.toolkits.jetbrains.core.explorer.showExplorerTree
+import software.aws.toolkits.jetbrains.core.explorer.ShowToolkitListener
import software.aws.toolkits.jetbrains.core.gettingstarted.IdcRolePopup
import software.aws.toolkits.jetbrains.core.gettingstarted.IdcRolePopupState
import software.aws.toolkits.jetbrains.core.region.AwsRegionProvider
@@ -211,7 +211,7 @@ class ToolkitWebviewBrowser(val project: Project, private val parentDisposable:
}
is BrowserMessage.ToggleBrowser -> {
- showExplorerTree(project)
+ ShowToolkitListener.showExplorerTree(project)
}
is BrowserMessage.CancelLogin -> {
diff --git a/plugins/toolkit/jetbrains-core/src/software/aws/toolkits/jetbrains/core/gettingstarted/ToolkitGettingStartedAuthUtils.kt b/plugins/toolkit/jetbrains-core/src/software/aws/toolkits/jetbrains/core/gettingstarted/ToolkitGettingStartedAuthUtils.kt
index 98da68929a..53b5479b47 100644
--- a/plugins/toolkit/jetbrains-core/src/software/aws/toolkits/jetbrains/core/gettingstarted/ToolkitGettingStartedAuthUtils.kt
+++ b/plugins/toolkit/jetbrains-core/src/software/aws/toolkits/jetbrains/core/gettingstarted/ToolkitGettingStartedAuthUtils.kt
@@ -6,7 +6,7 @@ package software.aws.toolkits.jetbrains.core.gettingstarted
import com.intellij.openapi.application.ApplicationInfo
import com.intellij.openapi.project.Project
import software.aws.toolkits.jetbrains.core.credentials.sono.CODECATALYST_SCOPES
-import software.aws.toolkits.jetbrains.core.explorer.showWebview
+import software.aws.toolkits.jetbrains.core.explorer.ShowToolkitListener
import software.aws.toolkits.jetbrains.core.explorer.webview.ToolkitWebviewPanel
import software.aws.toolkits.jetbrains.core.gettingstarted.editor.SourceOfEntry
import software.aws.toolkits.jetbrains.core.gettingstarted.editor.getConnectionCount
@@ -32,7 +32,7 @@ fun requestCredentialsForCodeCatalyst(
): Boolean? {
if (isQWebviewsAvailable() && project != null) {
ToolkitWebviewPanel.getInstance(project).browser?.prepareBrowser(BrowserState(FeatureId.Codecatalyst, true)) // TODO: consume data
- showWebview(project)
+ ShowToolkitListener.showWebview(project)
return null
}
@@ -128,7 +128,7 @@ fun requestCredentialsForExplorer(
): Boolean? {
if (isQWebviewsAvailable()) {
ToolkitWebviewPanel.getInstance(project).browser?.prepareBrowser(BrowserState(FeatureId.AwsExplorer, true)) // TODO: consume data
- showWebview(project)
+ ShowToolkitListener.showWebview(project)
return null
}
diff --git a/settings.gradle.kts b/settings.gradle.kts
index 21396263a7..c687218187 100644
--- a/settings.gradle.kts
+++ b/settings.gradle.kts
@@ -97,8 +97,15 @@ rootProject.name = "aws-toolkit-jetbrains"
include("detekt-rules")
include("ui-tests")
include("sandbox-all")
+include("ui-tests-starter")
when (providers.gradleProperty("ideProfileName").get()) {
- "2023.3", "2024.1" -> include("tmp-all")
+ // FIX_WHEN_MIN_IS_242: `tmp-all` test module no longer needed in 242+
+ "2023.3", "2024.1" -> {
+ include("tmp-all")
+
+ // only available 242+
+ project(":ui-tests-starter").projectDir = file("noop")
+ }
}
/*
diff --git a/ui-tests-starter/.gitignore b/ui-tests-starter/.gitignore
new file mode 100644
index 0000000000..0fb363385b
--- /dev/null
+++ b/ui-tests-starter/.gitignore
@@ -0,0 +1 @@
+allure-results/
diff --git a/ui-tests-starter/build.gradle.kts b/ui-tests-starter/build.gradle.kts
new file mode 100644
index 0000000000..a3cc765ef0
--- /dev/null
+++ b/ui-tests-starter/build.gradle.kts
@@ -0,0 +1,60 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+import org.jetbrains.intellij.platform.gradle.TestFrameworkType
+import software.aws.toolkits.gradle.findFolders
+import software.aws.toolkits.gradle.intellij.IdeVersions
+
+plugins {
+ id("toolkit-kotlin-conventions")
+ id("toolkit-intellij-plugin")
+
+ id("org.jetbrains.intellij.platform")
+}
+
+val ideProfile = IdeVersions.ideProfile(project)
+
+// Add our source sets per IDE profile version (i.e. src-211)
+sourceSets {
+ test {
+ java.srcDirs(findFolders(project, "tst", ideProfile))
+ resources.srcDirs(findFolders(project, "tst-resources", ideProfile))
+ }
+}
+
+intellijPlatform {
+ buildSearchableOptions = false
+ instrumentCode = false
+}
+
+val testPlugins by configurations.registering
+
+dependencies {
+ testImplementation(platform("com.jetbrains.intellij.tools:ide-starter-squashed"))
+ // should really be set by the BOM, but too much work to figure out right now
+ testImplementation("org.kodein.di:kodein-di-jvm:7.20.2")
+ intellijPlatform {
+ intellijIdeaCommunity(IdeVersions.ideProfile(providers).map { it.name })
+
+ testFramework(TestFrameworkType.Starter)
+ }
+
+ testPlugins(project(":plugin-amazonq", "pluginZip"))
+ testPlugins(project(":plugin-core", "pluginZip"))
+}
+
+tasks.test {
+ dependsOn(testPlugins)
+
+ useJUnitPlatform()
+
+ systemProperty("ui.test.plugins", testPlugins.get().asPath)
+}
+
+// hack to disable ui tests in ./gradlew check
+val action = Action {
+ if (hasTask(tasks.test.get())) {
+ tasks.test.get().enabled = false
+ }
+}
+gradle.taskGraph.whenReady(action)
diff --git a/ui-tests-starter/tst-241-242/software/aws/toolkits/jetbrains/uitests/TestCIServer.kt b/ui-tests-starter/tst-241-242/software/aws/toolkits/jetbrains/uitests/TestCIServer.kt
new file mode 100644
index 0000000000..625e02311d
--- /dev/null
+++ b/ui-tests-starter/tst-241-242/software/aws/toolkits/jetbrains/uitests/TestCIServer.kt
@@ -0,0 +1,29 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.uitests
+
+import com.intellij.ide.starter.ci.CIServer
+import java.nio.file.Path
+
+object TestCIServer : CIServer {
+ override val isBuildRunningOnCI: Boolean = System.getenv("CI").toBoolean() == true
+ override val buildNumber: String = ""
+ override val branchName: String = ""
+ override val buildParams: Map = mapOf()
+
+ override fun publishArtifact(source: Path, artifactPath: String, artifactName: String) {
+ }
+
+ override fun reportTestFailure(testName: String, message: String, details: String) {
+ println("test: $testName")
+ println("message: $message")
+ println("details: $details")
+ error(message)
+ }
+
+ override fun ignoreTestFailure(testName: String, message: String, details: String) {
+ }
+
+ override fun isTestFailureShouldBeIgnored(message: String) = false
+}
diff --git a/ui-tests-starter/tst-243+/software/aws/toolkits/jetbrains/uitests/TestCIServer.kt b/ui-tests-starter/tst-243+/software/aws/toolkits/jetbrains/uitests/TestCIServer.kt
new file mode 100644
index 0000000000..557f405925
--- /dev/null
+++ b/ui-tests-starter/tst-243+/software/aws/toolkits/jetbrains/uitests/TestCIServer.kt
@@ -0,0 +1,29 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+
+package software.aws.toolkits.jetbrains.uitests
+
+import com.intellij.ide.starter.ci.CIServer
+import java.nio.file.Path
+
+object TestCIServer : CIServer {
+ override val isBuildRunningOnCI: Boolean = System.getenv("CI").toBoolean() == true
+ override val buildNumber: String = ""
+ override val branchName: String = ""
+ override val buildParams: Map = mapOf()
+
+ override fun publishArtifact(source: Path, artifactPath: String, artifactName: String) {
+ }
+
+ override fun reportTestFailure(testName: String, message: String, details: String, linkToLogs: String?) {
+ println("test: $testName")
+ println("message: $message")
+ println("details: $details")
+ error(message)
+ }
+
+ override fun ignoreTestFailure(testName: String, message: String) {
+ }
+
+ override fun isTestFailureShouldBeIgnored(message: String) = false
+}
diff --git a/ui-tests-starter/tst/software/aws/toolkits/jetbrains/uitests/OfflineAmazonQInlineCompletionTest.kt b/ui-tests-starter/tst/software/aws/toolkits/jetbrains/uitests/OfflineAmazonQInlineCompletionTest.kt
new file mode 100644
index 0000000000..669a781deb
--- /dev/null
+++ b/ui-tests-starter/tst/software/aws/toolkits/jetbrains/uitests/OfflineAmazonQInlineCompletionTest.kt
@@ -0,0 +1,99 @@
+// Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved.
+// SPDX-License-Identifier: Apache-2.0
+package software.aws.toolkits.jetbrains.uitests
+
+import com.intellij.driver.sdk.openFile
+import com.intellij.driver.sdk.ui.ui
+import com.intellij.driver.sdk.waitForProjectOpen
+import com.intellij.ide.starter.ci.CIServer
+import com.intellij.ide.starter.di.di
+import com.intellij.ide.starter.driver.engine.runIdeWithDriver
+import com.intellij.ide.starter.ide.IdeProductProvider
+import com.intellij.ide.starter.junit5.hyphenateWithClass
+import com.intellij.ide.starter.models.TestCase
+import com.intellij.ide.starter.project.LocalProjectInfo
+import com.intellij.ide.starter.runner.CurrentTestMethod
+import com.intellij.ide.starter.runner.Starter
+import org.junit.jupiter.api.Test
+import org.kodein.di.DI
+import org.kodein.di.bindSingleton
+import java.io.File
+import java.nio.file.Path
+import java.nio.file.Paths
+import kotlin.io.path.createParentDirectories
+import kotlin.io.path.writeText
+
+class OfflineAmazonQInlineCompletionTest {
+ init {
+ di = DI {
+ extend(di)
+ bindSingleton(overrides = true) { TestCIServer }
+ }
+ }
+
+ @Test
+ fun `completion request with expired credentials does not freeze EDT`() {
+ val testCase = TestCase(
+ IdeProductProvider.IC,
+ LocalProjectInfo(
+ Paths.get("tstData", "Hello")
+ )
+ ).useRelease("2024.2")
+ Paths.get(System.getProperty("user.home"), ".aws", "sso", "cache", "ee1d2538cb8d358377d7661466c866af747a8a3f.json")
+ .createParentDirectories()
+ .writeText(
+ """
+ {
+ "clientId": "DummyId",
+ "clientSecret": "DummySecret",
+ "expiresAt": "3070-01-01T00:00:00Z",
+ "scopes": [
+ "scope1",
+ "scope2"
+ ],
+ "issuerUrl": "1",
+ "region": "2",
+ "clientType": "public",
+ "grantTypes": [
+ "authorization_code",
+ "refresh_token"
+ ],
+ "redirectUris": [
+ "http://127.0.0.1/oauth/callback"
+ ]
+ }
+ """.trimIndent()
+ )
+ Paths.get(System.getProperty("user.home"), ".aws", "sso", "cache", "d3b447f809607422aac1470dd17fbb32e358cdb3.json")
+ .writeText(
+ """
+ {
+ "issuerUrl": "https://example.awsapps.com/start",
+ "region": "us-east-1",
+ "accessToken": "DummyAccessToken",
+ "refreshToken": "RefreshToken",
+ "createdAt": "1970-01-01T00:00:00Z",
+ "expiresAt": "1970-01-01T00:00:00Z"
+ }
+ """.trimIndent()
+ )
+ Starter.newContext(CurrentTestMethod.hyphenateWithClass(), testCase).apply {
+ System.getProperty("ui.test.plugins").split(File.pathSeparator).forEach { path ->
+ pluginConfigurator.installPluginFromPath(
+ Path.of(path)
+ )
+ }
+
+ copyExistingConfig(Paths.get("tstData", "config"))
+ updateGeneralSettings()
+ }.runIdeWithDriver()
+ .useDriverAndCloseIde {
+ waitForProjectOpen()
+ openFile("Example.java")
+ ui.keyboard {
+ // left meta + c
+ repeat(5) { hotKey(18, 67) }
+ }
+ }
+ }
+}
diff --git a/ui-tests-starter/tstData/Hello/Example.java b/ui-tests-starter/tstData/Hello/Example.java
new file mode 100644
index 0000000000..e69de29bb2
diff --git a/ui-tests-starter/tstData/config/options/aws.xml b/ui-tests-starter/tstData/config/options/aws.xml
new file mode 100644
index 0000000000..66330aa8bb
--- /dev/null
+++ b/ui-tests-starter/tstData/config/options/aws.xml
@@ -0,0 +1,32 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/ui-tests-starter/tstData/config/options/ide.general.xml b/ui-tests-starter/tstData/config/options/ide.general.xml
new file mode 100644
index 0000000000..28da7b923d
--- /dev/null
+++ b/ui-tests-starter/tstData/config/options/ide.general.xml
@@ -0,0 +1,5 @@
+
+
+
+
+
diff --git a/ui-tests-starter/tstData/config/options/proxy.settings.xml b/ui-tests-starter/tstData/config/options/proxy.settings.xml
new file mode 100644
index 0000000000..f7b1482215
--- /dev/null
+++ b/ui-tests-starter/tstData/config/options/proxy.settings.xml
@@ -0,0 +1,8 @@
+
+
+
+
+
+
+
+
diff --git a/ui-tests/build.gradle.kts b/ui-tests/build.gradle.kts
index dcbf681de0..4888618c14 100644
--- a/ui-tests/build.gradle.kts
+++ b/ui-tests/build.gradle.kts
@@ -17,7 +17,6 @@ dependencies {
testImplementation(project(":plugin-core:core"))
testImplementation(project(path = ":plugin-core:core", configuration = "testArtifacts"))
testImplementation(libs.kotlin.coroutines)
- testImplementation(libs.junit5.jupiterApi)
testImplementation(libs.intellijRemoteFixtures)
testImplementation(libs.intellijRemoteRobot)
testImplementation(libs.aws.cloudformation)
@@ -29,8 +28,6 @@ dependencies {
testImplementation(libs.commons.io)
// match version declared by intellijRemoteRobot
testImplementation("com.squareup.okhttp3:okhttp:4.12.0")
-
- testRuntimeOnly(libs.junit5.jupiterEngine)
}
// don't run gui tests as part of check