Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

IEP-1247 Run CMake initial build for newly created project(idf.py reconfigure) #1051

Merged
merged 1 commit into from
Dec 11, 2024

Conversation

sigmaaa
Copy link
Collaborator

@sigmaaa sigmaaa commented Sep 16, 2024

Description

Running the idf.py reconfigure command after creating a new project. After that the sdkconfig should appear in the project without executing full build. Also, the problem related to executing idf.py commands on windows without previously building the project is fixes in this PR

Fixes # (IEP-1247)

Type of change

Please delete options that are not relevant.

  • New feature (non-breaking change which adds functionality)

How has this been tested?

  • Create a new project -> verify idf.py reconfigure job executed successfully
  • On windows create a new project -> open esp-idf terminal inside the project -> execute any esp-idf command

Test Configuration:

  • ESP-IDF Version:
  • OS (Windows,Linux and macOS):

Dependent components impacted by this PR:

  • Project
  • ESP-IDF terminal

Checklist

  • PR Self Reviewed
  • Applied Code formatting
  • Added Documentation
  • Added Unit Test
  • Verified on all platforms - Windows,Linux and macOS

Summary by CodeRabbit

Summary by CodeRabbit

  • New Features

    • Enhanced project wizard with asynchronous reconfiguration for the IoT Development Framework.
    • Real-time feedback displayed in a console during project setup.
    • Improved cross-platform compatibility for Python executable detection.
    • Added a new utility class for managing console instances within the IDE.
    • Introduced a command executor for executing IDF commands in project context.
    • New messaging for the reconfiguration process to improve user guidance.
    • Added a new entry for reconfiguration job feedback in the messaging system.
    • Introduced a checkbox option in the project creation wizard to enable reconfiguration.
    • New command for reconfiguration integrated into the UI.
    • Added a constant for enabling ccache in the IDF environment variables.
  • Bug Fixes

    • Implemented process status checks and logging for potential issues during command execution.
    • Refined environment variable management for improved command execution reliability.
    • Improved handling of environment variables to avoid conflicts during execution.

Copy link

coderabbitai bot commented Sep 16, 2024

Caution

Review failed

The pull request is closed.

Walkthrough

The changes in this pull request primarily focus on enhancing the functionality of the com.espressif.idf.core bundle and its associated classes. Key modifications include the addition of a new package to the MANIFEST.MF file, improvements in environment variable handling within the ProcessBuilderFactory and IDFConsoleLauncherDelegate classes, and the introduction of two new utility classes: ConsoleManager and IdfCommandExecutor. Additionally, the UI components have been updated to support new commands and messaging for project creation and reconfiguration.

Changes

Files Change Summary
bundles/com.espressif.idf.core/META-INF/MANIFEST.MF - Added org.eclipse.ui.console to Import-Package.
bundles/com.espressif.idf.core/src/com/espressif/idf/core/ProcessBuilderFactory.java - Modified run method to remove "Path" from environment variables.
- Removed commented-out line in runInBackground method.
bundles/com.espressif.idf.core/src/com/espressif/idf/core/util/ConsoleManager.java - Added new utility class ConsoleManager with methods to manage console instances.
bundles/com.espressif.idf.core/src/com/espressif/idf/core/util/IdfCommandExecutor.java - Added new class IdfCommandExecutor for executing IDF commands with methods for handling processes.
bundles/com.espressif.idf.terminal.connector/src/com/espressif/idf/terminal/connector/launcher/IDFConsoleLauncherDelegate.java - Updated environment variable handling logic, particularly for PATH.
bundles/com.espressif.idf.ui/src/com/espressif/idf/ui/wizard/Messages.java - Added new variable IdfReconfigureJobName and made formatting adjustments.
bundles/com.espressif.idf.ui/src/com/espressif/idf/ui/wizard/NewIDFProjectWizard.java - Added fields for IProject and MessageConsole, and integrated IdfCommandExecutor in performFinish.
bundles/com.espressif.idf.ui/src/com/espressif/idf/ui/wizard/messages.properties - Added entry for IdfReconfigureJobName and updated existing message formatting.
bundles/com.espressif.idf.ui/src/com/espressif/idf/ui/templates/Messages.java - Added new variable RunIdfCommandButtonTxt for localized message management.
bundles/com.espressif.idf.ui/src/com/espressif/idf/ui/templates/NewProjectCreationWizardPage.java - Added checkbox for reconfigure command execution and related methods.
bundles/com.espressif.idf.ui/src/com/espressif/idf/ui/templates/messages.properties - Added entry for RunIdfCommandButtonTxt related to project creation.
bundles/com.espressif.idf.ui/plugin.xml - Added command com.espressif.idf.ui.reconfigure and handler IdfReconfigureHandler.
bundles/com.espressif.idf.ui/src/com/espressif/idf/ui/handlers/IdfReconfigureHandler.java - Introduced handler for IDF reconfiguration with methods for command execution and workspace management.
bundles/com.espressif.idf.core/src/com/espressif/idf/core/IDFEnvironmentVariables.java - Added constant IDF_CCACHE_ENABLE and adjusted formatting for readability.

Possibly related PRs

Suggested reviewers

  • kolipakakondal
  • sigmaaa

Poem

In the wizard's realm, a change so bright,
Commands now run, bringing delight.
Python paths found, consoles aglow,
As projects take shape, watch them grow!
Hopping with joy, our code takes flight,
Thanks to these changes, all feels right! 🐰✨


📜 Recent review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Reviewing files that changed from the base of the PR and between 5e9d471 and 422cb58.

📒 Files selected for processing (14)
  • bundles/com.espressif.idf.core/META-INF/MANIFEST.MF (1 hunks)
  • bundles/com.espressif.idf.core/src/com/espressif/idf/core/IDFEnvironmentVariables.java (4 hunks)
  • bundles/com.espressif.idf.core/src/com/espressif/idf/core/ProcessBuilderFactory.java (2 hunks)
  • bundles/com.espressif.idf.core/src/com/espressif/idf/core/util/ConsoleManager.java (1 hunks)
  • bundles/com.espressif.idf.core/src/com/espressif/idf/core/util/IdfCommandExecutor.java (1 hunks)
  • bundles/com.espressif.idf.terminal.connector/src/com/espressif/idf/terminal/connector/launcher/IDFConsoleLauncherDelegate.java (1 hunks)
  • bundles/com.espressif.idf.ui/plugin.xml (2 hunks)
  • bundles/com.espressif.idf.ui/src/com/espressif/idf/ui/handlers/IdfReconfigureHandler.java (1 hunks)
  • bundles/com.espressif.idf.ui/src/com/espressif/idf/ui/templates/Messages.java (1 hunks)
  • bundles/com.espressif.idf.ui/src/com/espressif/idf/ui/templates/NewProjectCreationWizardPage.java (12 hunks)
  • bundles/com.espressif.idf.ui/src/com/espressif/idf/ui/templates/messages.properties (1 hunks)
  • bundles/com.espressif.idf.ui/src/com/espressif/idf/ui/wizard/Messages.java (2 hunks)
  • bundles/com.espressif.idf.ui/src/com/espressif/idf/ui/wizard/NewIDFProjectWizard.java (6 hunks)
  • bundles/com.espressif.idf.ui/src/com/espressif/idf/ui/wizard/messages.properties (1 hunks)

Thank you for using CodeRabbit. We offer it for free to the OSS community and would appreciate your support in helping us grow. If you find it useful, would you consider giving us a shout-out on your favorite social media?

❤️ Share
🪧 Tips

Chat

There are 3 ways to chat with CodeRabbit:

  • Review comments: Directly reply to a review comment made by CodeRabbit. Example:
    • I pushed a fix in commit <commit_id>, please review it.
    • Generate unit testing code for this file.
    • Open a follow-up GitHub issue for this discussion.
  • Files and specific lines of code (under the "Files changed" tab): Tag @coderabbitai in a new review comment at the desired location with your query. Examples:
    • @coderabbitai generate unit testing code for this file.
    • @coderabbitai modularize this function.
  • PR comments: Tag @coderabbitai in a new PR comment to ask questions about the PR branch. For the best results, please provide a very specific query, as very limited context is provided in this mode. Examples:
    • @coderabbitai gather interesting stats about this repository and render them as a table. Additionally, render a pie chart showing the language distribution in the codebase.
    • @coderabbitai read src/utils.ts and generate unit testing code.
    • @coderabbitai read the files in the src/scheduler package and generate a class diagram using mermaid and a README in the markdown format.
    • @coderabbitai help me debug CodeRabbit configuration file.

Note: Be mindful of the bot's finite context window. It's strongly recommended to break down tasks such as reading entire modules into smaller chunks. For a focused discussion, use review comments to chat about specific files and their changes, instead of using the PR comments.

CodeRabbit Commands (Invoked using PR comments)

  • @coderabbitai pause to pause the reviews on a PR.
  • @coderabbitai resume to resume the paused reviews.
  • @coderabbitai review to trigger an incremental review. This is useful when automatic reviews are disabled for the repository.
  • @coderabbitai full review to do a full review from scratch and review all the files again.
  • @coderabbitai summary to regenerate the summary of the PR.
  • @coderabbitai generate docstrings to generate docstrings for this PR. (Experiment)
  • @coderabbitai resolve resolve all the CodeRabbit review comments.
  • @coderabbitai configuration to show the current CodeRabbit configuration for the repository.
  • @coderabbitai help to get help.

Other keywords and placeholders

  • Add @coderabbitai ignore anywhere in the PR description to prevent this PR from being reviewed.
  • Add @coderabbitai summary to generate the high-level summary at a specific location in the PR description.
  • Add @coderabbitai anywhere in the PR title to generate the title automatically.

CodeRabbit Configuration File (.coderabbit.yaml)

  • You can programmatically configure CodeRabbit by adding a .coderabbit.yaml file to the root of your repository.
  • Please see the configuration documentation for more information.
  • If your editor has YAML language server enabled, you can add the path at the top of this file to enable auto-completion and validation: # yaml-language-server: $schema=https://coderabbit.ai/integrations/schema.v2.json

Documentation and Community

  • Visit our Documentation for detailed information on how to use CodeRabbit.
  • Join our Discord Community to get help, request features, and share feedback.
  • Follow us on X/Twitter for updates and announcements.

Copy link

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 0

Outside diff range and nitpick comments (2)
bundles/com.espressif.idf.ui/src/com/espressif/idf/ui/wizard/NewIDFProjectWizard.java (2)

181-239: LGTM! The changes improve the user experience and automate the reconfiguration process.

The method runCommandIdfPyInIdfEnv is well-structured and handles the execution of the idf.py reconfigure command effectively. It captures the command output and displays it in the console, providing real-time feedback to the user.

A few suggestions for further improvement:

  1. Consider extracting the timeout value (waitCount) into a constant for better readability and maintainability.
  2. Handle the case when the IDF environment variables are not set correctly, as the method relies on them for constructing the command.

266-287: LGTM! The method enhances cross-platform compatibility by determining the correct Python executable path.

The pythonVirtualExecutablePath method is well-structured and handles the construction of the Python executable path based on the operating system. It enhances cross-platform compatibility by appending the appropriate directory and filename.

One suggestion for further improvement:

  1. Consider adding error handling or a fallback mechanism in case the IDF_PYTHON_ENV_PATH environment variable is not set, as the method relies on it to retrieve the Python virtual environment path.
Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

Commits

Files that changed from the base of the PR and between ac7f36b and 989243c.

Files selected for processing (1)
  • bundles/com.espressif.idf.ui/src/com/espressif/idf/ui/wizard/NewIDFProjectWizard.java (5 hunks)
Additional comments not posted (1)
bundles/com.espressif.idf.ui/src/com/espressif/idf/ui/wizard/NewIDFProjectWizard.java (1)

241-264: LGTM! The method ensures that a console with the given name exists and is visible to the user.

The openConsole method is well-implemented and handles the creation and activation of the console effectively. It avoids creating duplicate consoles by searching for an existing one before creating a new one.

Copy link

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 4

🧹 Outside diff range and nitpick comments (11)
bundles/com.espressif.idf.core/src/com/espressif/idf/core/util/ConsoleManager.java (2)

12-18: Add Javadoc to explain utility class pattern.

Consider adding class-level documentation to explain that this is a utility class and why instantiation is prevented.

+/**
+ * Utility class for managing Eclipse console instances.
+ * This class should not be instantiated.
+ */
 public class ConsoleManager
 {
     private ConsoleManager()
     {
+        // Prevents instantiation of utility class
     }

31-41: Consider safer type casting and modern Java practices.

The method could benefit from these improvements:

  1. Add method documentation
  2. Add type checking before casting
  3. Consider using Optional for better null handling
+/**
+ * Finds an existing MessageConsole with the specified name.
+ *
+ * @param consoleName the name of the console to find
+ * @return Optional<MessageConsole> containing the found console, or empty if not found
+ */
-private static MessageConsole findConsole(String consoleName)
+private static Optional<MessageConsole> findConsole(String consoleName)
 {
     for (IConsole existing : ConsolePlugin.getDefault().getConsoleManager().getConsoles())
     {
-        if (consoleName.equals(existing.getName()))
+        if (existing instanceof MessageConsole && consoleName.equals(existing.getName()))
         {
-            return (MessageConsole) existing;
+            return Optional.of((MessageConsole) existing);
         }
     }
-    return null;
+    return Optional.empty();
 }

Then update the getConsole method accordingly:

 public static MessageConsole getConsole(String consoleName)
 {
-    MessageConsole console = findConsole(consoleName);
-    if (console == null)
+    return findConsole(consoleName).orElseGet(() ->
     {
-        console = new MessageConsole(consoleName, null);
-        ConsolePlugin.getDefault().getConsoleManager().addConsoles(new IConsole[] { console });
-    }
-    ConsolePlugin.getDefault().getConsoleManager().showConsoleView(console);
-    return console;
+        MessageConsole newConsole = new MessageConsole(consoleName, null);
+        ConsolePlugin.getDefault().getConsoleManager().addConsoles(new IConsole[] { newConsole });
+        ConsolePlugin.getDefault().getConsoleManager().showConsoleView(newConsole);
+        return newConsole;
+    });
 }
bundles/com.espressif.idf.ui/src/com/espressif/idf/ui/wizard/messages.properties (1)

30-30: Consider enhancing the reconfigure message for clarity.

While the message is functional, it could be more informative about the purpose of reconfiguration.

Consider this alternative:

-IdfReconfigureJobName=Running idf.py reconfigure command...
+IdfReconfigureJobName=Running idf.py reconfigure to initialize CMake build...
bundles/com.espressif.idf.core/src/com/espressif/idf/core/util/IdfCommandExecutor.java (3)

27-37: Add class documentation and null validation.

Consider adding:

  1. JavaDoc documentation for the class explaining its purpose and responsibilities
  2. Documentation for the fields explaining their roles
  3. Null validation in the constructor
+/**
+ * Executes IDF (IoT Development Framework) commands within a project context.
+ * This executor handles the reconfiguration of IDF projects using the specified target platform.
+ */
 public class IdfCommandExecutor
 {
+    /** The target platform for IDF configuration */
     private final String target;
+    /** Console for command output */
     private final MessageConsole console;

     public IdfCommandExecutor(String target, MessageConsole console)
     {
+        if (target == null || console == null) {
+            throw new IllegalArgumentException("Target and console must not be null");
+        }
         this.target = target;
         this.console = console;
     }

39-43: Add method documentation.

Add JavaDoc to explain the method's purpose, parameters, and return values.

+/**
+ * Executes the reconfigure command for the given project.
+ * Activates the console and runs the IDF reconfigure command.
+ *
+ * @param project the project to reconfigure
+ * @return IStatus indicating success or failure of the operation
+ */
 public IStatus executeReconfigure(IProject project)

1-93: Consider implementing a more robust command execution framework.

Given that this class is part of a larger IDF command execution system, consider:

  1. Creating an abstract base class or interface for command execution to standardize error handling, timeout management, and output processing across different commands
  2. Implementing a command pattern to encapsulate different IDF commands
  3. Adding a command result object that includes:
    • Exit code
    • Output/error streams
    • Execution duration
    • Resource usage
  4. Implementing proper cleanup of processes and resources
  5. Adding support for command cancellation

This would make the system more maintainable and extensible for future IDF commands.

bundles/com.espressif.idf.core/src/com/espressif/idf/core/ProcessBuilderFactory.java (1)

Line range hint 1-150: Consider enhancing error handling and documentation for CMake integration.

Since this class is now part of the CMake initial build functionality:

  1. Consider adding CMake-specific error detection and handling in processData
  2. Document the class's role in the CMake build process
  3. Consider adding debug logging for better troubleshooting of CMake issues

Would you like help with implementing any of these suggestions?

bundles/com.espressif.idf.ui/src/com/espressif/idf/ui/wizard/NewIDFProjectWizard.java (2)

68-69: Remove unused field console.

The console field is declared but never used throughout the class. Consider removing it to maintain code cleanliness.

 private IProject project;
-private MessageConsole console;

147-166: Consider moving the reconfigure job to a separate class.

The job implementation would be better placed in a separate class to improve maintainability and separation of concerns. This would also make it easier to test and reuse the functionality.

Consider creating a new class like IDFProjectReconfigureJob that extends Job and encapsulates all the reconfiguration logic. This would make the wizard code cleaner and more focused on its primary responsibility.

bundles/com.espressif.idf.terminal.connector/src/com/espressif/idf/terminal/connector/launcher/IDFConsoleLauncherDelegate.java (2)

397-400: Improve working directory setup readability.

The indentation is inconsistent with the rest of the file. Also, consider adding a comment to explain the fallback behavior.

Apply this diff to improve readability:

 String terminalWrkDir = getWorkingDir();
-if (StringUtil.isEmpty(terminalWrkDir)) {
-    terminalWrkDir = workingDir;
-}
+// Fall back to the provided working directory if getWorkingDir() returns empty
+if (StringUtil.isEmpty(terminalWrkDir)) {
+	terminalWrkDir = workingDir;
+}

Line range hint 437-449: Add null checks and improve method documentation.

The method needs several improvements:

  1. Add null checks to prevent NPE in the resource handling chain.
  2. Add method documentation to explain the logic and return value.
  3. Consider logging when falling back to IDF path.

Apply this diff to fix the issues:

+/**
+ * Returns the working directory for the terminal.
+ * First tries to get the directory from the selected resource's project.
+ * Falls back to IDF path if no resource is selected or if resource handling is unavailable.
+ *
+ * @return The working directory path as a string, never null
+ */
 protected String getWorkingDir() {
     Bundle bundle = Platform.getBundle("org.eclipse.core.resources"); //$NON-NLS-1$
     if (bundle != null && bundle.getState() != Bundle.UNINSTALLED && bundle.getState() != Bundle.STOPPING) {
         // if we have a IResource selection use the location for working directory
         IResource resource = EclipseUtil.getSelectionResource();
-        if (resource instanceof IResource) {
+        if (resource != null && resource.getProject() != null && resource.getProject().getLocation() != null) {
             String dir = ((IResource) resource).getProject().getLocation().toString();
             return dir;
         }
     }
+    // Fall back to IDF path when resource selection is not available
     return IDFUtil.getIDFPath();
 }
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Reviewing files that changed from the base of the PR and between 989243c and 167642b.

📒 Files selected for processing (8)
  • bundles/com.espressif.idf.core/META-INF/MANIFEST.MF (1 hunks)
  • bundles/com.espressif.idf.core/src/com/espressif/idf/core/ProcessBuilderFactory.java (2 hunks)
  • bundles/com.espressif.idf.core/src/com/espressif/idf/core/util/ConsoleManager.java (1 hunks)
  • bundles/com.espressif.idf.core/src/com/espressif/idf/core/util/IdfCommandExecutor.java (1 hunks)
  • bundles/com.espressif.idf.terminal.connector/src/com/espressif/idf/terminal/connector/launcher/IDFConsoleLauncherDelegate.java (2 hunks)
  • bundles/com.espressif.idf.ui/src/com/espressif/idf/ui/wizard/Messages.java (2 hunks)
  • bundles/com.espressif.idf.ui/src/com/espressif/idf/ui/wizard/NewIDFProjectWizard.java (5 hunks)
  • bundles/com.espressif.idf.ui/src/com/espressif/idf/ui/wizard/messages.properties (1 hunks)
✅ Files skipped from review due to trivial changes (1)
  • bundles/com.espressif.idf.ui/src/com/espressif/idf/ui/wizard/Messages.java
🔇 Additional comments (11)
bundles/com.espressif.idf.core/src/com/espressif/idf/core/util/ConsoleManager.java (2)

1-11: LGTM! File structure and imports are well-organized.

The copyright header, package declaration, and imports are properly structured and contain only the necessary dependencies.


12-42: Verify thread safety for concurrent console creation.

Since this utility class is used by NewIDFProjectWizard, multiple wizard instances could potentially create race conditions when creating/accessing consoles concurrently.

✅ Verification successful

Based on the search results, I can now provide a final response:

ConsoleManager is thread-safe for console creation and access

The ConsoleManager implementation is inherently thread-safe because:

  1. It delegates to Eclipse's ConsolePlugin which handles thread synchronization internally
  2. The only usage found shows ConsoleManager.getConsole() being called from the UI thread in NewIDFProjectWizard
  3. The findConsole() method performs a read-only operation on the console list
  4. Console creation is handled atomically through ConsolePlugin's addConsoles() method
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Check for potential concurrent usage of ConsoleManager

# Test: Search for concurrent access patterns
rg -A 5 "ConsoleManager.getConsole" 

# Test: Check if ConsoleManager is used in any async/parallel execution contexts
ast-grep --pattern 'new Thread($$$)' 
ast-grep --pattern 'CompletableFuture<$_>($$$)'
ast-grep --pattern 'Job($$$)'

Length of output: 20291

bundles/com.espressif.idf.ui/src/com/espressif/idf/ui/wizard/messages.properties (1)

29-29: LGTM!

The project name label text is clear and consistent with other labels in the file.

bundles/com.espressif.idf.core/META-INF/MANIFEST.MF (2)

66-67: LGTM! The package import addition is well-structured.

The addition of org.eclipse.ui.console to the Import-Package section is correctly formatted and aligns with the introduction of console management functionality.


67-67: Verify the ConsoleManager implementation.

Let's verify that the ConsoleManager class exists and properly utilizes this newly imported package.

✅ Verification successful

Let me generate another script to examine the ConsoleManager implementation more closely since we found its location.


ConsoleManager implementation properly utilizes org.eclipse.ui.console package

The ConsoleManager class is well-implemented and correctly uses the org.eclipse.ui.console package:

  • It properly imports required console classes (ConsolePlugin, IConsole, MessageConsole)
  • Implements standard console management patterns with findConsole() and getConsole() methods
  • Correctly uses ConsolePlugin for console registration and management
  • Has appropriate usage across multiple UI components as shown by the references
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Verify the existence and implementation of ConsoleManager
# Expected: Find ConsoleManager class with imports from org.eclipse.ui.console

# Search for ConsoleManager class definition
echo "Searching for ConsoleManager class definition:"
ast-grep --pattern 'class ConsoleManager {
  $$$
}'

# Search for console package usage
echo -e "\nVerifying org.eclipse.ui.console package usage:"
rg "import org\.eclipse\.ui\.console\." -A 3

Length of output: 14295


Script:

#!/bin/bash
# Let's examine the ConsoleManager implementation
echo "ConsoleManager implementation:"
cat bundles/com.espressif.idf.core/src/com/espressif/idf/core/util/ConsoleManager.java

echo -e "\nSearching for ConsoleManager usage:"
rg "ConsoleManager" -l

Length of output: 1904

bundles/com.espressif.idf.core/src/com/espressif/idf/core/util/IdfCommandExecutor.java (2)

1-26: LGTM! File structure and imports are well-organized.


72-92: ⚠️ Potential issue

Improve process execution robustness.

The current process execution implementation has several potential issues:

  1. No timeout handling for long-running processes
  2. Error stream is not consumed, which might cause process to hang
  3. Process exit code is not checked

Consider implementing these improvements:

 private IStatus runProcess(List<String> arguments, Map<String, String> environment,
         ProcessBuilderFactory processRunner, IProject project, MessageConsoleStream messageConsoleStream)
 {
     StringBuilder output = new StringBuilder();
-    try (BufferedReader reader = new BufferedReader(new InputStreamReader(
-            processRunner.run(arguments, project.getLocation(), environment).getInputStream())))
+    try
     {
-        String line;
-        while ((line = reader.readLine()) != null)
-        {
-            output.append(line).append(System.lineSeparator());
-            messageConsoleStream.println(line);
+        Process process = processRunner.run(arguments, project.getLocation(), environment);
+        
+        // Handle both output and error streams
+        Thread outputThread = startStreamReader(process.getInputStream(), output, messageConsoleStream);
+        Thread errorThread = startStreamReader(process.getErrorStream(), output, messageConsoleStream);
+        
+        // Wait for process with timeout
+        if (!process.waitFor(5, TimeUnit.MINUTES)) {
+            process.destroyForcibly();
+            return IDFCorePlugin.errorStatus("Process timed out after 5 minutes", null);
         }
-        return new Status(IStatus.OK, IDFCorePlugin.PLUGIN_ID, output.toString());
+        
+        // Wait for stream readers to complete
+        outputThread.join();
+        errorThread.join();
+        
+        int exitCode = process.exitValue();
+        if (exitCode != 0) {
+            return IDFCorePlugin.errorStatus("Process failed with exit code: " + exitCode, null);
+        }
+        
+        return new Status(IStatus.OK, IDFCorePlugin.PLUGIN_ID, output.toString());
     }
     catch (Exception e)
     {
         Logger.log(e);
         return IDFCorePlugin.errorStatus(e.getMessage(), e);
     }
 }

+private Thread startStreamReader(InputStream input, StringBuilder output, MessageConsoleStream console) {
+    Thread thread = new Thread(() -> {
+        try (BufferedReader reader = new BufferedReader(new InputStreamReader(input))) {
+            String line;
+            while ((line = reader.readLine()) != null) {
+                synchronized (output) {
+                    output.append(line).append(System.lineSeparator());
+                }
+                console.println(line);
+            }
+        } catch (IOException e) {
+            Logger.log(e);
+        }
+    });
+    thread.start();
+    return thread;
+}

Let's verify the process execution behavior in similar files:

bundles/com.espressif.idf.core/src/com/espressif/idf/core/ProcessBuilderFactory.java (2)

51-51: LGTM! Clean and concise process stream handling.

The code cleanup improves readability while maintaining the correct process stream handling functionality.


39-42: Verify Windows compatibility for PATH handling.

While standardizing on "PATH" is good practice, removing "Path" on Windows systems needs careful consideration. Please ensure:

  1. The "PATH" variable is properly populated before removing "Path"
  2. This change doesn't affect CMake's ability to find required tools on Windows

Consider adding a more detailed comment explaining:

  • Why we're removing "Path"
  • How this relates to cross-platform compatibility
  • Any prerequisites for PATH setup
✅ Verification successful

Let me analyze the search results to verify the PATH handling concern. I'll need to check how the environment variables are handled across the codebase, particularly in relation to the ProcessBuilderFactory.


Let me check one more aspect to understand the complete environment variable handling flow, particularly focusing on how the environment is populated before the Path removal.


Based on the comprehensive analysis of the codebase, I can now provide a definitive response about the PATH handling concern.

The PATH handling in ProcessBuilderFactory is correct and safe.

The code follows a deliberate standardization pattern where:

  1. The environment is first populated with all required variables via processBuilder.environment().putAll(environment)
  2. The Windows-specific "Path" is then removed because:
    • The system environment is already properly captured through IDFUtil.getSystemEnv()
    • The PATH variable (uppercase) contains the complete path information
    • This standardization ensures consistent behavior across platforms

The codebase shows this is part of a well-designed environment handling system where:

  • IDFUtil.getSystemEnv() properly captures all system environment variables
  • Multiple components (EspToolCommands, ExportIDFTools, etc.) use this same pattern
  • The PATH/Path handling is consistently managed across the entire toolchain
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Search for other PATH/Path handling in the codebase
rg -i 'path.*=|path.*remove' --type java

# Look for CMake-related environment setup
rg -i 'cmake.*path|path.*cmake' --type java

Length of output: 91194


Script:

#!/bin/bash
# Search for environment variable handling in ProcessBuilderFactory and related classes
ast-grep --pattern 'processBuilder.environment().$_($_)'

# Look for PATH/Path environment variable usage in the codebase
ast-grep --pattern '$_.environment().get("PATH")'
ast-grep --pattern '$_.environment().get("Path")'

Length of output: 1100


Script:

#!/bin/bash
# Look for the environment setup before ProcessBuilder creation
rg -B 5 -A 5 'processBuilder.environment\(\).remove\("Path"\)' --type java

# Check for environment variable setup in IDFUtil
rg -A 5 'getSystemEnv' --type java

Length of output: 35202

bundles/com.espressif.idf.ui/src/com/espressif/idf/ui/wizard/NewIDFProjectWizard.java (1)

118-121: LGTM: Project field usage is well-structured.

The conversion of the local project variable to a class field is a good improvement, making it accessible for both the Clang file updates and the new reconfigure functionality.

bundles/com.espressif.idf.terminal.connector/src/com/espressif/idf/terminal/connector/launcher/IDFConsoleLauncherDelegate.java (1)

Line range hint 1-449: Verify CMake command execution setup.

The changes set up environment variables and working directory for terminal operations, which are prerequisites for running CMake commands. However, we should verify that these changes properly support the CMake initial build functionality mentioned in the PR title.

✅ Verification successful

Let me check the CMake command execution setup more thoroughly.


Based on the findings, I can now generate the final response:

Terminal connector changes correctly support CMake build functionality

The changes in IDFConsoleLauncherDelegate.java properly support CMake build operations by:

  1. Setting up CDT build environment variables needed for CMake execution
  2. Establishing correct working directory for build operations through getWorkingDir()
  3. Ensuring proper PATH setup with IDF extras needed for CMake tools

The terminal connector acts as an environment provider rather than directly executing CMake commands, which aligns with its architectural role. The actual CMake execution is handled by IDFBuildConfiguration which uses this environment.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Verify CMake-related functionality
# Test 1: Check for CMake command execution in the codebase
echo "Checking for CMake command execution..."
rg -l "cmake|CMake" --type java

# Test 2: Check for IDF reconfigure command usage
echo "Checking for reconfigure command usage..."
rg -l "reconfigure" --type java

Length of output: 4052


Script:

#!/bin/bash
# Check for CMake command execution in IdfCommandExecutor
echo "Checking IdfCommandExecutor for CMake commands..."
rg -A 10 "cmake" bundles/com.espressif.idf.core/src/com/espressif/idf/core/util/IdfCommandExecutor.java

# Check for CMake build configuration
echo "Checking CMake build configuration..."
rg -A 10 "cmake" bundles/com.espressif.idf.core/src/com/espressif/idf/core/build/IDFBuildConfiguration.java

# Check terminal connector's role in build process
echo "Checking terminal connector's role..."
rg "build|cmake" bundles/com.espressif.idf.terminal.connector/src/com/espressif/idf/terminal/connector/launcher/IDFConsoleLauncherDelegate.java

Length of output: 5956

Comment on lines +19 to +29
public static MessageConsole getConsole(String consoleName)
{
MessageConsole console = findConsole(consoleName);
if (console == null)
{
console = new MessageConsole(consoleName, null);
ConsolePlugin.getDefault().getConsoleManager().addConsoles(new IConsole[] { console });
}
ConsolePlugin.getDefault().getConsoleManager().showConsoleView(console);
return console;
}
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🛠️ Refactor suggestion

Add documentation and parameter validation.

The method logic is correct, but could benefit from these improvements:

  1. Add Javadoc with parameter and return descriptions
  2. Add null check for consoleName parameter
+/**
+ * Gets or creates a MessageConsole with the specified name.
+ * If a console with the given name already exists, it will be returned.
+ * Otherwise, a new console will be created and added to the console manager.
+ *
+ * @param consoleName the name of the console to get or create
+ * @return the MessageConsole instance
+ * @throws IllegalArgumentException if consoleName is null or empty
+ */
 public static MessageConsole getConsole(String consoleName)
 {
+    if (consoleName == null || consoleName.trim().isEmpty()) {
+        throw new IllegalArgumentException("Console name cannot be null or empty");
+    }
     MessageConsole console = findConsole(consoleName);
📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
public static MessageConsole getConsole(String consoleName)
{
MessageConsole console = findConsole(consoleName);
if (console == null)
{
console = new MessageConsole(consoleName, null);
ConsolePlugin.getDefault().getConsoleManager().addConsoles(new IConsole[] { console });
}
ConsolePlugin.getDefault().getConsoleManager().showConsoleView(console);
return console;
}
/**
* Gets or creates a MessageConsole with the specified name.
* If a console with the given name already exists, it will be returned.
* Otherwise, a new console will be created and added to the console manager.
*
* @param consoleName the name of the console to get or create
* @return the MessageConsole instance
* @throws IllegalArgumentException if consoleName is null or empty
*/
public static MessageConsole getConsole(String consoleName)
{
if (consoleName == null || consoleName.trim().isEmpty()) {
throw new IllegalArgumentException("Console name cannot be null or empty");
}
MessageConsole console = findConsole(consoleName);
if (console == null)
{
console = new MessageConsole(consoleName, null);
ConsolePlugin.getDefault().getConsoleManager().addConsoles(new IConsole[] { console });
}
ConsolePlugin.getDefault().getConsoleManager().showConsoleView(console);
return console;
}

Comment on lines 45 to 98
private IStatus runIdfReconfigureCommand(IProject project)
{
ProcessBuilderFactory processRunner = new ProcessBuilderFactory();
List<String> arguments = prepareArguments();
Map<String, String> environment = new HashMap<>(new IDFEnvironmentVariables().getEnvMap());

try (MessageConsoleStream messageConsoleStream = console.newMessageStream())
{
return runProcess(arguments, environment, processRunner, project, messageConsoleStream);
}
catch (IOException e1)
{
Logger.log(e1);
return IDFCorePlugin.errorStatus(e1.getMessage(), e1);
}
}

private List<String> prepareArguments()
{
List<String> arguments = new ArrayList<>();
arguments.add(IDFUtil.getIDFPythonEnvPath());
arguments.add(IDFUtil.getIDFPythonScriptFile().getAbsolutePath());
arguments.add("-DIDF_TARGET=" + target); //$NON-NLS-1$
arguments.add("reconfigure"); //$NON-NLS-1$
return arguments;
}
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

⚠️ Potential issue

Add input validation and path verification.

The current implementation has several potential issues:

  1. Missing validation for the project parameter
  2. No verification if required paths (Python env, script file) exist
  3. Potential null pointer issues from utility methods

Consider adding these safety checks:

 private IStatus runIdfReconfigureCommand(IProject project)
 {
+    if (project == null || !project.exists()) {
+        return IDFCorePlugin.errorStatus("Invalid project", null);
+    }
+
     ProcessBuilderFactory processRunner = new ProcessBuilderFactory();
     List<String> arguments = prepareArguments();
     Map<String, String> environment = new HashMap<>(new IDFEnvironmentVariables().getEnvMap());

 private List<String> prepareArguments()
 {
     List<String> arguments = new ArrayList<>();
+    String pythonEnvPath = IDFUtil.getIDFPythonEnvPath();
+    if (pythonEnvPath == null || pythonEnvPath.isEmpty()) {
+        throw new IllegalStateException("Python environment path not found");
+    }
+    java.io.File scriptFile = IDFUtil.getIDFPythonScriptFile();
+    if (scriptFile == null || !scriptFile.exists()) {
+        throw new IllegalStateException("IDF Python script file not found");
+    }
-    arguments.add(IDFUtil.getIDFPythonEnvPath());
-    arguments.add(IDFUtil.getIDFPythonScriptFile().getAbsolutePath());
+    arguments.add(pythonEnvPath);
+    arguments.add(scriptFile.getAbsolutePath());
📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
private IStatus runIdfReconfigureCommand(IProject project)
{
ProcessBuilderFactory processRunner = new ProcessBuilderFactory();
List<String> arguments = prepareArguments();
Map<String, String> environment = new HashMap<>(new IDFEnvironmentVariables().getEnvMap());
try (MessageConsoleStream messageConsoleStream = console.newMessageStream())
{
return runProcess(arguments, environment, processRunner, project, messageConsoleStream);
}
catch (IOException e1)
{
Logger.log(e1);
return IDFCorePlugin.errorStatus(e1.getMessage(), e1);
}
}
private List<String> prepareArguments()
{
List<String> arguments = new ArrayList<>();
arguments.add(IDFUtil.getIDFPythonEnvPath());
arguments.add(IDFUtil.getIDFPythonScriptFile().getAbsolutePath());
arguments.add("-DIDF_TARGET=" + target); //$NON-NLS-1$
arguments.add("reconfigure"); //$NON-NLS-1$
return arguments;
}
private IStatus runIdfReconfigureCommand(IProject project)
{
if (project == null || !project.exists()) {
return IDFCorePlugin.errorStatus("Invalid project", null);
}
ProcessBuilderFactory processRunner = new ProcessBuilderFactory();
List<String> arguments = prepareArguments();
Map<String, String> environment = new HashMap<>(new IDFEnvironmentVariables().getEnvMap());
try (MessageConsoleStream messageConsoleStream = console.newMessageStream())
{
return runProcess(arguments, environment, processRunner, project, messageConsoleStream);
}
catch (IOException e1)
{
Logger.log(e1);
return IDFCorePlugin.errorStatus(e1.getMessage(), e1);
}
}
private List<String> prepareArguments()
{
List<String> arguments = new ArrayList<>();
String pythonEnvPath = IDFUtil.getIDFPythonEnvPath();
if (pythonEnvPath == null || pythonEnvPath.isEmpty()) {
throw new IllegalStateException("Python environment path not found");
}
java.io.File scriptFile = IDFUtil.getIDFPythonScriptFile();
if (scriptFile == null || !scriptFile.exists()) {
throw new IllegalStateException("IDF Python script file not found");
}
arguments.add(pythonEnvPath);
arguments.add(scriptFile.getAbsolutePath());
arguments.add("-DIDF_TARGET=" + target); //$NON-NLS-1$
arguments.add("reconfigure"); //$NON-NLS-1$
return arguments;
}

Comment on lines 147 to 170
Job job = new Job(Messages.IdfReconfigureJobName)
{

protected IStatus run(IProgressMonitor monitor)
{
IdfCommandExecutor executor = new IdfCommandExecutor(target,
ConsoleManager.getConsole("CDT Build Console")); //$NON-NLS-1$
IStatus status = executor.executeReconfigure(project);
try
{
IDEWorkbenchPlugin.getPluginWorkspace().getRoot().refreshLocal(IResource.DEPTH_INFINITE, null);
}
catch (CoreException e)
{
Logger.log(e);
}
return status;
}
};
job.schedule();
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

💡 Codebase verification

🛠️ Refactor suggestion

Console name should be defined as a constant and potentially externalized

The hardcoded console name "CDT Build Console" is used consistently across the codebase, but it should be defined as a constant in a central location since it's a shared resource identifier. The console name appears in both production and test code, making it more critical to maintain consistency through a single source of truth.

Consider:

  1. Define the console name as a constant in a shared location (e.g., UIPlugin or a dedicated constants class)
  2. Use the constant instead of the hardcoded string
  3. Consider externalizing the string to support internationalization if needed

Example refactoring:

// In UIPlugin or a dedicated constants class
public static final String CDT_BUILD_CONSOLE_NAME = "CDT Build Console"; //$NON-NLS-1$

// In NewIDFProjectWizard.java
ConsoleManager.getConsole(UIPlugin.CDT_BUILD_CONSOLE_NAME)
🔗 Analysis chain

Several improvements needed in the reconfigure job implementation.

  1. The job scheduling in a dispose listener could lead to race conditions if the wizard is closed before the project is fully created.
  2. The error handling could be improved to provide better user feedback.
  3. The console name is hardcoded.
  4. The progress monitor is not being utilized.

Consider implementing these improvements:

-Job job = new Job(Messages.IdfReconfigureJobName)
+private static final String CDT_BUILD_CONSOLE = "CDT Build Console";
+
+Job job = new Job(Messages.IdfReconfigureJobName) {
+    @Override
+    protected IStatus run(IProgressMonitor monitor) {
+        try {
+            monitor.beginTask(Messages.IdfReconfigureJobName, IProgressMonitor.UNKNOWN);
+            
+            if (project == null || !project.exists()) {
+                return new Status(IStatus.ERROR, UIPlugin.PLUGIN_ID, 
+                    "Project not found or not fully created");
+            }
+            
+            IdfCommandExecutor executor = new IdfCommandExecutor(target,
+                    ConsoleManager.getConsole(CDT_BUILD_CONSOLE));
+            IStatus status = executor.executeReconfigure(project);
+            
+            if (monitor.isCanceled()) {
+                return Status.CANCEL_STATUS;
+            }
+            
+            try {
+                IDEWorkbenchPlugin.getPluginWorkspace().getRoot()
+                    .refreshLocal(IResource.DEPTH_INFINITE, monitor);
+            } catch (CoreException e) {
+                Logger.log(e);
+                return new Status(IStatus.ERROR, UIPlugin.PLUGIN_ID,
+                    "Failed to refresh workspace", e);
+            }
+            
+            return status;
+        } finally {
+            monitor.done();
+        }
+    }
+};
+job.setUser(true);  // Show progress to user
 job.schedule();

Let's verify the console name usage across the codebase:

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Search for hardcoded console names to ensure consistency
rg -l "CDT Build Console"

Length of output: 467


Script:

#!/bin/bash
# Let's check the actual usage context of "CDT Build Console" in these files
rg "CDT Build Console" -C 2

# Also check if there's any constant defined for this console name
rg -i "console.*constant|constant.*console"

Length of output: 12234


Script:

#!/bin/bash
# Let's check if there are any other console names used with ConsoleManager.getConsole
rg "ConsoleManager\.getConsole" -C 2

# Also check for any constants or messages related to CDT Build Console
rg -l "CDT.*Build.*Console" bundles/com.espressif.idf.ui/src/com/espressif/idf/ui/messages/

Length of output: 877

Comment on lines 383 to 401
if (!StringUtil.isEmpty(idfExtraPaths)) {
envValue = envValue + ":" + idfExtraPaths; //$NON-NLS-1$
}
}
envpList.add(envKey + "=" + envValue); //$NON-NLS-1$
}

//Removing path, since we are using PATH
envMap.remove("Path"); //$NON-NLS-1$
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

⚠️ Potential issue

Fix path separator and environment variable handling issues.

Several issues need to be addressed:

  1. The path separator ":" is hardcoded, which won't work on Windows. Use File.pathSeparator instead.
  2. The removal of "Path" after adding all environment variables to envpList is redundant and potentially problematic.

Apply this diff to fix the issues:

 if (!StringUtil.isEmpty(idfExtraPaths)) {
-    envValue = envValue + ":" + idfExtraPaths; //$NON-NLS-1$
+    envValue = envValue + File.pathSeparator + idfExtraPaths;
 }

Also, consider moving the envMap.remove("Path") before the environment variables are added to envpList to ensure consistency.

Committable suggestion skipped: line range outside the PR's diff.

@sigmaaa sigmaaa changed the title WIP: IEP-1247 Run CMake initial build for newly created project(idf.py reconfigure) IEP-1247 Run CMake initial build for newly created project(idf.py reconfigure) Nov 11, 2024
@AndriiFilippov
Copy link
Collaborator

@alirana01 @kolipakakondal hi guys!
please, review.

Copy link
Collaborator

@alirana01 alirana01 left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

LGTM, just a small question

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I dont understand the need for this class, maybe I am not clear on this. Just a question from my side that why do we need a separate class to just execute this command when we can do that in the Job we are creating for the console?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hi @alirana01,

thanks for the review. I've created this class not only for this command but also for other idf commands that we have to execute in the console. My idea was that in the future we could refactor existing code and move other IDF commands and their preparation into this class to not scatter similar logic over the code base and to avoid duplicate code

@AndriiFilippov
Copy link
Collaborator

@sigmaaa hi !
Tested under:
OS - Windows 11
ESP-IDF: v5.3.1

got this error during tool installation:

Successfully installed annotated-types-0.7.0 bitarray-2.9.3 bitstring-4.2.3 certifi-2024.8.30 cffi-1.17.1 charset-normalizer-3.4.0 click-8.1.7 colorama-0.4.6 construct-2.10.70 cryptography-42.0.8 ecdsa-0.19.0 esp-coredump-1.12.0 esp-idf-kconfig-2.3.0 esp-idf-monitor-1.5.0 esp-idf-nvs-partition-gen-0.1.6 esp-idf-panic-decoder-1.2.1 esp-idf-size-1.6.1 esptool-4.8.1 freertos_gdb-1.0.3 idf-component-manager-2.0.4 idna-3.10 intelhex-2.3.0 jsonref-1.1.0 markdown-it-py-3.0.0 mdurl-0.1.2 packaging-24.2 pyclang-0.5.0 pycparser-2.22 pydantic-2.9.2 pydantic-core-2.23.4 pydantic-settings-2.6.1 pyelftools-0.31 pygdbmi-0.11.0.0 pygments-2.18.0 pyparsing-3.2.0 pyserial-3.5 python-dotenv-1.0.1 pyyaml-6.0.2 reedsolo-1.7.0 requests-2.32.3 requests-file-2.1.0 requests-toolbelt-1.0.0 rich-13.9.4 six-1.16.0 tqdm-4.67.0 typing-extensions-4.12.2 urllib3-1.26.20 windows-curses-2.4.0

Executing C:\Python311\python.exe C:\Users\AndriiFilippov\e\esp-idf-v5.3.1\tools\idf_tools.py export --format=key-value
Traceback (most recent call last):
  File "C:\Users\AndriiFilippov\e\esp-idf-v5.3.1\tools\idf_tools.py", line 3239, in <module>
    main(sys.argv[1:])
  File "C:\Users\AndriiFilippov\e\esp-idf-v5.3.1\tools\idf_tools.py", line 3231, in main
    action_func(args)
  File "C:\Users\AndriiFilippov\e\esp-idf-v5.3.1\tools\idf_tools.py", line 2211, in action_export
    paths_to_check = rf'{export_vars["PATH"]}{os.environ["PATH"]}'
                                              ~~~~~~~~~~^^^^^^^^
  File "<frozen os>", line 678, in __getitem__
KeyError: 'PATH'

image

@sigmaaa
Copy link
Collaborator Author

sigmaaa commented Nov 18, 2024

@sigmaaa hi ! Tested under: OS - Windows 11 ESP-IDF: v5.3.1

got this error during tool installation:

Successfully installed annotated-types-0.7.0 bitarray-2.9.3 bitstring-4.2.3 certifi-2024.8.30 cffi-1.17.1 charset-normalizer-3.4.0 click-8.1.7 colorama-0.4.6 construct-2.10.70 cryptography-42.0.8 ecdsa-0.19.0 esp-coredump-1.12.0 esp-idf-kconfig-2.3.0 esp-idf-monitor-1.5.0 esp-idf-nvs-partition-gen-0.1.6 esp-idf-panic-decoder-1.2.1 esp-idf-size-1.6.1 esptool-4.8.1 freertos_gdb-1.0.3 idf-component-manager-2.0.4 idna-3.10 intelhex-2.3.0 jsonref-1.1.0 markdown-it-py-3.0.0 mdurl-0.1.2 packaging-24.2 pyclang-0.5.0 pycparser-2.22 pydantic-2.9.2 pydantic-core-2.23.4 pydantic-settings-2.6.1 pyelftools-0.31 pygdbmi-0.11.0.0 pygments-2.18.0 pyparsing-3.2.0 pyserial-3.5 python-dotenv-1.0.1 pyyaml-6.0.2 reedsolo-1.7.0 requests-2.32.3 requests-file-2.1.0 requests-toolbelt-1.0.0 rich-13.9.4 six-1.16.0 tqdm-4.67.0 typing-extensions-4.12.2 urllib3-1.26.20 windows-curses-2.4.0

Executing C:\Python311\python.exe C:\Users\AndriiFilippov\e\esp-idf-v5.3.1\tools\idf_tools.py export --format=key-value
Traceback (most recent call last):
  File "C:\Users\AndriiFilippov\e\esp-idf-v5.3.1\tools\idf_tools.py", line 3239, in <module>
    main(sys.argv[1:])
  File "C:\Users\AndriiFilippov\e\esp-idf-v5.3.1\tools\idf_tools.py", line 3231, in main
    action_func(args)
  File "C:\Users\AndriiFilippov\e\esp-idf-v5.3.1\tools\idf_tools.py", line 2211, in action_export
    paths_to_check = rf'{export_vars["PATH"]}{os.environ["PATH"]}'
                                              ~~~~~~~~~~^^^^^^^^
  File "<frozen os>", line 678, in __getitem__
KeyError: 'PATH'

image

Hi @AndriiFilippov ,

thanks for reporting this. Fixed in the last commit

@AndriiFilippov
Copy link
Collaborator

AndriiFilippov commented Nov 18, 2024

@sigmaaa hi !

Now I am able to install tools, however this error occurs after Project Creation:

ERROR: C:\Espressif\espidf.constraints.v5.3.txt doesn't exist. Perhaps you've forgotten to run the install scripts. Please check the installation guide for more information.

ESP-IDF v5.3.1-dirty

unable to open Menu Config.
image

java.lang.reflect.InvocationTargetException: Build directory is not found: C:\Users\AndriiFilippov\workspaceDENYSTESTPRDELETELATERCMAKEBUILDreconfigure\wewewe\build
	at com.espressif.idf.sdk.config.ui.SDKConfigurationEditor.lambda$0(SDKConfigurationEditor.java:190)
	at org.eclipse.jface.operation.ModalContext$ModalContextThread.run(ModalContext.java:124)

Copy link

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 3

🧹 Outside diff range and nitpick comments (3)
bundles/com.espressif.idf.ui/src/com/espressif/idf/ui/templates/messages.properties (1)

11-11: Consider refining the button text for clarity and consistency.

While the implementation is correct, consider these improvements to the message text:

  1. Make it more concise
  2. Follow consistent capitalization
  3. Consider adding a tooltip property to explain the technical term

Consider this alternative text:

-RunIdfCommandButtonTxt=Execute idf.py reconfigure with Project Creation
+RunIdfCommandButtonTxt=Run initial project configuration

And optionally add a tooltip property:

RunIdfCommandButtonTooltip=Executes 'idf.py reconfigure' to initialize project configuration without performing a full build
bundles/com.espressif.idf.ui/src/com/espressif/idf/ui/templates/NewProjectCreationWizardPage.java (2)

135-140: Consider improving layout and accessibility

The checkbox configuration has several areas for improvement:

  1. The GridData spans 4 columns but the parent composite only has 2 columns (layout.numColumns = 2)
  2. Missing accessibility support (no mnemonic key)
  3. Default selection state (true) should be documented or configurable

Consider these improvements:

 runIdfReconfigureCheckBoxButton = new Button(projectNameGroup, SWT.CHECK | SWT.RIGHT);
 GridData buttonData = new GridData();
-buttonData.horizontalSpan = 4;
+buttonData.horizontalSpan = 2; // Match parent's column count
+buttonData.horizontalAlignment = GridData.FILL;
 runIdfReconfigureCheckBoxButton.setLayoutData(buttonData);
 runIdfReconfigureCheckBoxButton.setSelection(true);
-runIdfReconfigureCheckBoxButton.setText(Messages.RunIdfCommandButtonTxt);
+// Add mnemonic using '&' before the character
+runIdfReconfigureCheckBoxButton.setText(Messages.RunIdfCommandButtonTxt + " (&R)");
+// Consider making default state configurable
+boolean defaultState = IDFPreferenceStore.getDefault().getBoolean("default.reconfigure.enabled");
+runIdfReconfigureCheckBoxButton.setSelection(defaultState);

Line range hint 244-257: Consider adding Windows-specific validation

Given that this PR aims to address Windows-specific issues with idf.py commands, consider adding validation logic when the reconfigure option is disabled on Windows.

 protected boolean validatePage()
 {
     IWorkspace workspace = IDEWorkbenchPlugin.getPluginWorkspace();
+    
+    // Add Windows-specific validation for reconfigure option
+    if (!runIdfReconfigureCheckBoxButton.getSelection() && 
+        System.getProperty("os.name").toLowerCase().contains("windows")) {
+        setMessage("Disabling reconfigure on Windows may cause issues with sdkconfig", IStatus.WARNING);
+    }

     String worspaceLocation = workspace.getRoot().getLocation().toOSString();
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Reviewing files that changed from the base of the PR and between e6f987a and e938ba8.

📒 Files selected for processing (4)
  • bundles/com.espressif.idf.ui/src/com/espressif/idf/ui/templates/Messages.java (1 hunks)
  • bundles/com.espressif.idf.ui/src/com/espressif/idf/ui/templates/NewProjectCreationWizardPage.java (12 hunks)
  • bundles/com.espressif.idf.ui/src/com/espressif/idf/ui/templates/messages.properties (1 hunks)
  • bundles/com.espressif.idf.ui/src/com/espressif/idf/ui/wizard/NewIDFProjectWizard.java (6 hunks)
🔇 Additional comments (3)
bundles/com.espressif.idf.ui/src/com/espressif/idf/ui/templates/Messages.java (1)

26-26: LGTM! Verify messages.properties entry

The new message constant follows the established naming conventions and is properly integrated with the NLS framework.

Let's verify the corresponding entry exists in the messages.properties file:

✅ Verification successful

✓ Message property is properly defined

The corresponding entry RunIdfCommandButtonTxt=Execute idf.py reconfigure with Project Creation exists in the messages.properties file with appropriate descriptive text.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Check if RunIdfCommandButtonTxt is defined in messages.properties
# Expected: Entry should exist in the properties file

rg "RunIdfCommandButtonTxt" bundles/com.espressif.idf.ui/src/com/espressif/idf/ui/templates/messages.properties

Length of output: 183

bundles/com.espressif.idf.ui/src/com/espressif/idf/ui/wizard/NewIDFProjectWizard.java (1)

148-152: LGTM! Good placement of the reconfigure command execution

The conditional execution of the reconfigure command is well-placed within the dispose listener, ensuring it runs after all wizard-related tasks are complete.

bundles/com.espressif.idf.ui/src/com/espressif/idf/ui/templates/NewProjectCreationWizardPage.java (1)

51-51: LGTM: Field declaration follows conventions

The new field declaration follows proper Java naming conventions and has appropriate visibility.

@@ -62,6 +66,8 @@ public class NewIDFProjectWizard extends TemplateWizard
private static final String NEW_LAUNCH_CONFIG_EDIT_PAGE = "NewLaunchConfigEditPage"; //$NON-NLS-1$
public static final String TARGET_SWITCH_JOB = "TARGET SWITCH JOB"; //$NON-NLS-1$
private NewProjectCreationWizardPage projectCreationWizardPage;
private IProject project;
private MessageConsole console;
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🛠️ Refactor suggestion

Remove unused field console

The console field is declared but never used throughout the class.

-private MessageConsole console;
📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
private MessageConsole console;

Comment on lines +157 to +179
private void runIdfReconfigureCommandJob(final String target)
{
Job job = new Job(Messages.IdfReconfigureJobName)
{

protected IStatus run(IProgressMonitor monitor)
{
IdfCommandExecutor executor = new IdfCommandExecutor(target,
ConsoleManager.getConsole("CDT Build Console")); //$NON-NLS-1$
IStatus status = executor.executeReconfigure(project);
try
{
IDEWorkbenchPlugin.getPluginWorkspace().getRoot().refreshLocal(IResource.DEPTH_INFINITE, null);
}
catch (CoreException e)
{
Logger.log(e);
}
return status;
}
};
job.schedule();
}
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🛠️ Refactor suggestion

Enhance job implementation with proper progress monitoring and error handling

Several improvements are needed in the reconfigure job implementation:

  1. The progress monitor is not being utilized
  2. Error handling for workspace refresh could be improved
  3. The job's user property should be set to show progress to the user

Apply this diff to improve the implementation:

 private void runIdfReconfigureCommandJob(final String target)
 {
     Job job = new Job(Messages.IdfReconfigureJobName)
     {
         protected IStatus run(IProgressMonitor monitor)
         {
+            try {
+                monitor.beginTask(Messages.IdfReconfigureJobName, IProgressMonitor.UNKNOWN);
+                
                 IdfCommandExecutor executor = new IdfCommandExecutor(target,
                         ConsoleManager.getConsole("CDT Build Console")); //$NON-NLS-1$
                 IStatus status = executor.executeReconfigure(project);
+                
+                if (monitor.isCanceled()) {
+                    return Status.CANCEL_STATUS;
+                }
+                
                 try
                 {
-                    IDEWorkbenchPlugin.getPluginWorkspace().getRoot().refreshLocal(IResource.DEPTH_INFINITE, null);
+                    IDEWorkbenchPlugin.getPluginWorkspace().getRoot().refreshLocal(IResource.DEPTH_INFINITE, monitor);
                 }
                 catch (CoreException e)
                 {
                     Logger.log(e);
+                    return new Status(IStatus.ERROR, UIPlugin.PLUGIN_ID, 
+                        "Failed to refresh workspace after reconfigure", e);
                 }
                 return status;
+            } finally {
+                monitor.done();
+            }
         }
     };
+    job.setUser(true);  // Show progress to user
     job.schedule();
 }
📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
private void runIdfReconfigureCommandJob(final String target)
{
Job job = new Job(Messages.IdfReconfigureJobName)
{
protected IStatus run(IProgressMonitor monitor)
{
IdfCommandExecutor executor = new IdfCommandExecutor(target,
ConsoleManager.getConsole("CDT Build Console")); //$NON-NLS-1$
IStatus status = executor.executeReconfigure(project);
try
{
IDEWorkbenchPlugin.getPluginWorkspace().getRoot().refreshLocal(IResource.DEPTH_INFINITE, null);
}
catch (CoreException e)
{
Logger.log(e);
}
return status;
}
};
job.schedule();
}
private void runIdfReconfigureCommandJob(final String target)
{
Job job = new Job(Messages.IdfReconfigureJobName)
{
protected IStatus run(IProgressMonitor monitor)
{
try {
monitor.beginTask(Messages.IdfReconfigureJobName, IProgressMonitor.UNKNOWN);
IdfCommandExecutor executor = new IdfCommandExecutor(target,
ConsoleManager.getConsole("CDT Build Console")); //$NON-NLS-1$
IStatus status = executor.executeReconfigure(project);
if (monitor.isCanceled()) {
return Status.CANCEL_STATUS;
}
try
{
IDEWorkbenchPlugin.getPluginWorkspace().getRoot().refreshLocal(IResource.DEPTH_INFINITE, monitor);
}
catch (CoreException e)
{
Logger.log(e);
return new Status(IStatus.ERROR, UIPlugin.PLUGIN_ID,
"Failed to refresh workspace after reconfigure", e);
}
return status;
} finally {
monitor.done();
}
}
};
job.setUser(true); // Show progress to user
job.schedule();
}

Comment on lines +178 to +181
public boolean isRunIdfReconfigureEnabled()
{
return runIdfReconfigureCheckBoxButton.getSelection();
}
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🛠️ Refactor suggestion

Add null check for robustness

The getter method should handle cases where the button hasn't been initialized yet.

 public boolean isRunIdfReconfigureEnabled()
 {
-    return runIdfReconfigureCheckBoxButton.getSelection();
+    return runIdfReconfigureCheckBoxButton != null && runIdfReconfigureCheckBoxButton.getSelection();
 }
📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
public boolean isRunIdfReconfigureEnabled()
{
return runIdfReconfigureCheckBoxButton.getSelection();
}
public boolean isRunIdfReconfigureEnabled()
{
return runIdfReconfigureCheckBoxButton != null && runIdfReconfigureCheckBoxButton.getSelection();
}

@AndriiFilippov
Copy link
Collaborator

AndriiFilippov commented Nov 21, 2024

@sigmaaa hi !

Tested under:
OS - Windows 11 / Mac arm64

Once project is created - the "idf.py reconfigure" executes. ✅
Able to open Menu Config. ✅
Do see the option to turn off "idf.py reconfigure" after project created, works as expected ✅
image

  1. The other thing that bothers me is the execution time of the command on Windows 11 - between 25-40 seconds.

2.After "idf.py reconfigure" is done there is a weird behave - it execute full "confserver" command on every Menu Config opening. it is building 20 seconds every time on my machine. But once project is built, there is very brief execution and process takes 2-4 seconds only:

Executing action: confserver
Running ninja in directory C:\Users\AndriiFilippov\workspaceTESTdenysPR\bl1\build
Executing "ninja confserver"...
Server running, waiting for requests on stdin...
[0/1] cmd.exe /C "cd /D C:\Users\AndriiFilippov\workspaceTESTdenysPR\bl1\build && C:\Users\AndriiFilippov\........
  1. I was thinking perhaps we could improve the error message when a user tries to open the Menu Config without a built project, now it looks like this:

image

there is a better description in the Error logs, but it is not default window.
image

could we improve it ?

Copy link

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 2

🧹 Outside diff range and nitpick comments (2)
bundles/com.espressif.idf.ui/src/com/espressif/idf/ui/handlers/IdfReconfigureHandler.java (1)

14-14: Consider alternatives to internal Eclipse API usage.

The use of org.eclipse.ui.internal.ide.IDEWorkbenchPlugin with @SuppressWarnings("restriction") could lead to compatibility issues with future Eclipse versions. Consider using public APIs if available.

Also applies to: 25-25

bundles/com.espressif.idf.ui/plugin.xml (1)

479-484: Consider enhancing the command definition with visibility conditions and tooltip.

The command definition could be improved by:

  1. Adding visibility conditions to control when the command appears
  2. Including a tooltip to guide users
   <command
         commandId="com.espressif.idf.ui.reconfigure"
         icon="icons/espressif.png"
         label="Reconfigure"
-        style="push">
+        style="push"
+        tooltip="%command.tooltip.reconfigure">
+     <visibleWhen checkEnabled="false">
+       <iterate ifEmpty="false" operator="or">
+         <instanceof value="org.eclipse.core.resources.IProject">
+         </instanceof>
+         <test property="org.eclipse.core.resources.projectNature"
+               value="com.espressif.idf.core.idfNature">
+         </test>
+       </iterate>
+     </visibleWhen>
   </command>
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Reviewing files that changed from the base of the PR and between e938ba8 and 25b57f0.

📒 Files selected for processing (2)
  • bundles/com.espressif.idf.ui/plugin.xml (2 hunks)
  • bundles/com.espressif.idf.ui/src/com/espressif/idf/ui/handlers/IdfReconfigureHandler.java (1 hunks)
🔇 Additional comments (3)
bundles/com.espressif.idf.ui/src/com/espressif/idf/ui/handlers/IdfReconfigureHandler.java (2)

59-73: Verify the default target selection.

The method falls back to "esp32" as the default target. Please verify:

  1. Is "esp32" always a safe default?
  2. Should we consider reading the default from configuration?

Let's verify the target handling in the codebase:

✅ Verification successful

Let me gather more specific information about the default target handling.


Let me gather more specific information about target validation and configuration.


Based on the analysis of the codebase, I can now provide a final response:

Using "esp32" as default target is appropriate but could be more robust

The codebase reveals that:

  1. "esp32" is a safe default as it's the most basic target platform
  2. The system already has infrastructure to read target configuration:
    • Through SDKConfigJsonReader for project-specific targets
    • Via EspConfigParser which maintains a set of valid targets
    • Through launch target attributes

Recommendations:

  • Keep "esp32" as the fallback default
  • Consider reading from project's sdkconfig.json first if available
  • Add validation against EspConfigParser.getTargets() to ensure target validity
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Search for other default target definitions and target handling
rg -l "esp32" --type java

# Search for target configuration handling
ast-grep --pattern 'class $_ implements ILaunchTarget {
  $$$
}'

Length of output: 2286


Script:

#!/bin/bash
# Search for DEFAULT_TARGET constant definition
rg "DEFAULT_TARGET.*=.*esp32" --type java -C 2

# Search for target-related configuration handling
rg "getTarget|setTarget|TARGET" --type java bundles/com.espressif.idf.core/src/com/espressif/idf/core/ -C 2

# Look for target validation or supported targets list
rg "getSupportedTargets|validateTarget|isValidTarget" --type java

Length of output: 27744


Script:

#!/bin/bash
# Search for target validation or configuration in EspConfigParser
rg -A 5 "getTargets|validateTarget|configureTarget" bundles/com.espressif.idf.core/src/com/espressif/idf/core/util/EspConfigParser.java

# Look for sdkconfig.defaults handling
rg -A 5 "sdkconfig.defaults" --type java

# Search for IDF_TARGET in configuration files
rg "IDF_TARGET" --type properties --type json

Length of output: 454


41-43: Verify Windows compatibility for command execution.

Given the reported Windows-specific issues in the PR comments, please ensure:

  1. Command execution works correctly on Windows
  2. Console output is properly handled
  3. Performance concerns (25-40 seconds execution time) are addressed

Let's check the command execution implementation:

✅ Verification successful

Let me generate additional verification scripts to check the Windows-specific handling in the command execution:


Let me gather more information about the command execution and environment handling:


Let me check one final aspect about the command execution and environment setup:


Based on the gathered information, I can now provide the final response:

Windows compatibility is properly handled in the command execution

The implementation has proper Windows-specific handling:

  • ProcessBuilderFactory handles Windows PATH environment variable correctly by managing both "PATH" and "Path" cases
  • Environment variables are properly propagated through IDFEnvironmentVariables and ProcessBuilderFactory
  • Command execution uses platform-independent paths and file handling
  • Console output uses UTF-8 encoding and proper line separators
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Search for Windows-specific handling in command execution
rg -A 5 "windows|Windows|cmd.exe|ProcessBuilder" --type java

# Look for performance-related configurations
ast-grep --pattern 'class IdfCommandExecutor {
  $$$
}'

Length of output: 88192


Script:

#!/bin/bash
# Check IdfCommandExecutor implementation for Windows-specific handling
ast-grep --pattern 'class IdfCommandExecutor {
  $$$
  executeReconfigure($$$) {
    $$$
  }
  $$$
}'

# Look for Windows-specific environment variable handling
rg -A 5 "getSystemEnv|environment" bundles/com.espressif.idf.core/src/com/espressif/idf/core/util/IdfCommandExecutor.java

# Check ProcessBuilderFactory implementation for Windows handling
ast-grep --pattern 'class ProcessBuilderFactory {
  $$$
  runInBackground($$$) {
    $$$
  }
  $$$
}'

Length of output: 1244


Script:

#!/bin/bash
# Check IDFEnvironmentVariables implementation for Windows-specific handling
rg -A 10 "class IDFEnvironmentVariables" --type java

# Look for the actual reconfigure command construction
rg -A 5 "reconfigure" bundles/com.espressif.idf.ui/src/com/espressif/idf/ui/handlers/IdfReconfigureHandler.java

# Check ProcessBuilderFactory's run method
ast-grep --pattern 'class ProcessBuilderFactory {
  $$$
  run($$$) {
    $$$
  }
  $$$
}'

Length of output: 1535


Script:

#!/bin/bash
# Check the full implementation of IdfReconfigureHandler
rg -l "IdfReconfigureHandler" --type java
cat bundles/com.espressif.idf.ui/src/com/espressif/idf/ui/handlers/IdfReconfigureHandler.java

# Look for ProcessBuilderFactory's implementation
cat bundles/com.espressif.idf.core/src/com/espressif/idf/core/ProcessBuilderFactory.java

# Check for any Windows-specific environment setup
rg -A 5 "getSystemEnvMap|getSystemEnv" --type java

Length of output: 41398

bundles/com.espressif.idf.ui/plugin.xml (1)

668-671: LGTM!

The handler registration follows the standard pattern and correctly maps the command to its handler class.


public Object execute(ExecutionEvent event) throws ExecutionException
{
IProject selectedProject = EclipseUtil.getSelectedProjectInExplorer();
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

⚠️ Potential issue

Add null check for selected project.

The selected project could be null if no project is selected in the explorer. Add appropriate error handling.

 IProject selectedProject = EclipseUtil.getSelectedProjectInExplorer();
+if (selectedProject == null) {
+    return new Status(IStatus.ERROR, IDFCorePlugin.PLUGIN_ID, Messages.NoProjectSelectedError);
+}

Committable suggestion skipped: line range outside the PR's diff.

Comment on lines +38 to +53
protected IStatus run(IProgressMonitor monitor)
{

IdfCommandExecutor executor = new IdfCommandExecutor(getCurrentTarget(),
ConsoleManager.getConsole("CDT Build Console")); //$NON-NLS-1$
IStatus status = executor.executeReconfigure(selectedProject);
try
{
IDEWorkbenchPlugin.getPluginWorkspace().getRoot().refreshLocal(IResource.DEPTH_INFINITE, null);
}
catch (CoreException e)
{
Logger.log(e);
}
return status;
}
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🛠️ Refactor suggestion

Improve job implementation with progress monitoring and resource management.

Several improvements can be made to the job implementation:

  1. The progress monitor is not being used
  2. The console resource is not properly managed
  3. The workspace refresh could block UI

Consider this improved implementation:

 protected IStatus run(IProgressMonitor monitor)
 {
+    if (monitor.isCanceled()) {
+        return Status.CANCEL_STATUS;
+    }
+    monitor.beginTask(Messages.ReconfigureTaskName, 2);
 
     IdfCommandExecutor executor = new IdfCommandExecutor(getCurrentTarget(),
             ConsoleManager.getConsole("CDT Build Console")); //$NON-NLS-1$
     IStatus status = executor.executeReconfigure(selectedProject);
+    monitor.worked(1);
+
+    if (monitor.isCanceled()) {
+        return Status.CANCEL_STATUS;
+    }
+
     try
     {
-        IDEWorkbenchPlugin.getPluginWorkspace().getRoot().refreshLocal(IResource.DEPTH_INFINITE, null);
+        IDEWorkbenchPlugin.getPluginWorkspace().getRoot().refreshLocal(IResource.DEPTH_INFINITE, monitor);
     }
     catch (CoreException e)
     {
         Logger.log(e);
     }
+    monitor.worked(1);
+    monitor.done();
     return status;
 }
📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
protected IStatus run(IProgressMonitor monitor)
{
IdfCommandExecutor executor = new IdfCommandExecutor(getCurrentTarget(),
ConsoleManager.getConsole("CDT Build Console")); //$NON-NLS-1$
IStatus status = executor.executeReconfigure(selectedProject);
try
{
IDEWorkbenchPlugin.getPluginWorkspace().getRoot().refreshLocal(IResource.DEPTH_INFINITE, null);
}
catch (CoreException e)
{
Logger.log(e);
}
return status;
}
protected IStatus run(IProgressMonitor monitor)
{
if (monitor.isCanceled()) {
return Status.CANCEL_STATUS;
}
monitor.beginTask(Messages.ReconfigureTaskName, 2);
IdfCommandExecutor executor = new IdfCommandExecutor(getCurrentTarget(),
ConsoleManager.getConsole("CDT Build Console")); //$NON-NLS-1$
IStatus status = executor.executeReconfigure(selectedProject);
monitor.worked(1);
if (monitor.isCanceled()) {
return Status.CANCEL_STATUS;
}
try
{
IDEWorkbenchPlugin.getPluginWorkspace().getRoot().refreshLocal(IResource.DEPTH_INFINITE, monitor);
}
catch (CoreException e)
{
Logger.log(e);
}
monitor.worked(1);
monitor.done();
return status;
}

@AndriiFilippov
Copy link
Collaborator

@sigmaaa hi !

  1. still after idf.py reconfigure it compile full idf.py menuconfig command. I don't see this behave using terminal. In terminal it does not execute full configserver command

  2. Should this feature be available for the custom build folder ? I tried different approaches but it still use workspace build folder

Copy link

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 0

🧹 Outside diff range and nitpick comments (1)
bundles/com.espressif.idf.core/src/com/espressif/idf/core/IDFEnvironmentVariables.java (1)

Line range hint 1-170: Consider enhancing environment variable persistence and validation.

Given that this class plays a crucial role in managing environment variables for the ESP-IDF integration:

  1. Consider adding validation for critical environment variables, especially for Windows paths.
  2. Consider implementing a mechanism to detect and repair broken environment configurations.
  3. Think about adding logging for environment variable changes to help with debugging.
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Reviewing files that changed from the base of the PR and between 25b57f0 and 786e86f.

📒 Files selected for processing (2)
  • bundles/com.espressif.idf.core/src/com/espressif/idf/core/IDFEnvironmentVariables.java (4 hunks)
  • bundles/com.espressif.idf.core/src/com/espressif/idf/core/util/IdfCommandExecutor.java (1 hunks)
🚧 Files skipped from review as they are similar to previous changes (1)
  • bundles/com.espressif.idf.core/src/com/espressif/idf/core/util/IdfCommandExecutor.java
🔇 Additional comments (2)
bundles/com.espressif.idf.core/src/com/espressif/idf/core/IDFEnvironmentVariables.java (2)

47-50: LGTM! Verify path handling across platforms.

The addition of GIT_PATH and PYTHON_EXE_PATH constants follows the existing pattern and will help manage executable paths across different platforms.

Let's verify the usage of these constants:

✅ Verification successful

The new path constants are properly integrated and used across the codebase

The verification shows that both GIT_PATH and PYTHON_EXE_PATH constants are appropriately used:

  • In ToolsActivationJob.java: Used to set Git and Python executable paths during tools activation
  • In ListInstalledToolsHandler.java: Used to retrieve Git executable path
  • The constants follow consistent naming patterns with other path-related constants in the codebase
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Search for usage of the new path constants
rg "GIT_PATH|PYTHON_EXE_PATH" --type java

Length of output: 1774


53-53: Verify ccache integration with the build system.

The addition of IDF_CCACHE_ENABLE is good for build optimization. However, we should ensure proper integration with the ESP-IDF build system.

Let's verify the ccache integration:

✅ Verification successful

IDF_CCACHE_ENABLE integration is properly implemented

The constant is correctly integrated with the build system:

  • In JsonConfigServer.java, it's set to "false" in the build environment
  • In IdfCommandExecutor.java, it's used to control ccache via DCCACHE_ENABLE build argument
  • The implementation handles empty values by defaulting to "0"
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Search for ccache-related configurations and usage
rg "IDF_CCACHE|ccache" --type cmake

Length of output: 37


Script:

#!/bin/bash
# Let's search more broadly for ccache-related code and configurations
rg -i "IDF_CCACHE|ccache" -t java -t xml -t properties -A 3

Length of output: 96


Script:

#!/bin/bash
# Search for build system integration points
fd -e java -e xml -e properties | xargs grep -l "build.properties\|build.system"

Length of output: 305


Script:

#!/bin/bash
# Look for environment variable handling
ast-grep --pattern 'getenv($ENV_VAR)'

Length of output: 39


Script:

#!/bin/bash
# Check for build-related configuration files
fd -e cmake -e mk -e properties -e xml build

Length of output: 1140


Script:

#!/bin/bash
# Let's check the core bundle for environment variable usage
cat bundles/com.espressif.idf.core/src/com/espressif/idf/core/IDFEnvironmentVariables.java

Length of output: 5725


Script:

#!/bin/bash
# Search for environment variable usage in build-related files
rg -t java "IDF_CCACHE_ENABLE" -A 3

Length of output: 1623


Script:

#!/bin/bash
# Check for build system configuration files
fd -e java build.system

Length of output: 25

@sigmaaa
Copy link
Collaborator Author

sigmaaa commented Nov 29, 2024

@sigmaaa hi !

  1. still after idf.py reconfigure it compile full idf.py menuconfig command. I don't see this behave using terminal. In terminal it does not execute full configserver command
  2. Should this feature be available for the custom build folder ? I tried different approaches but it still use workspace build folder

Hi @AndriiFilippov,

In the last commit, I've replaced the idf.py reconfigure command with the corresponding cmake command to allow the -B argument.

@AndriiFilippov
Copy link
Collaborator

@sigmaaa hi !

OS - Windows 11

create project - select "execute idf.py reconfigure with Project Creation" -> error on re-configure

Cannot run program "cmake" (in directory "C:\Users\AndriiFilippov\workspaceTEW21w23\kkmk"): CreateProcess error=2, The system cannot find the file specified
Cannot run program "cmake" (in directory "C:\Users\AndriiFilippov\workspaceTEW21w23\kkmk"): CreateProcess error=2, The system cannot find the file specified

@AndriiFilippov
Copy link
Collaborator

AndriiFilippov commented Dec 11, 2024

@sigmaaa hi !
Tested:
OS - Windows 11
ESP-IDF: v5.4

create project with Execute idf.py reconfigure with Project Creation then create project without Execute idf.py reconfigure with Project Creation -> run MenuConfig on project without Execute idf.py reconfigure with Project Creation -> it takes build folder from 1 project:

image

@sigmaaa
Copy link
Collaborator Author

sigmaaa commented Dec 11, 2024

@sigmaaa hi ! Tested: OS - Windows 11 ESP-IDF: v5.4

create project with Execute idf.py reconfigure with Project Creation then create project without Execute idf.py reconfigure with Project Creation -> run MenuConfig on project without Execute idf.py reconfigure with Project Creation -> it takes build folder from 1 project:

image

Hi @AndriiFilippov,

seems it's not caused by this PR so we can handle this in a separate PR

@AndriiFilippov
Copy link
Collaborator

@sigmaaa hi !

ok, ticket created.

LGTM 👍

fix: fixing esp-idf terminal and idf.py reconfigure not working on windows
fix: fixed problem with installation tools and making remove Path more robust
fix: passing system env map instead env map before command execution
fix: adding option to turn off idf.py reconfigure
fix: adding reconfigure command to the menu
fix: changing idf.py reconfigure to cmake command to allow -B
fix: replace cmake with absolute cmake path from path
fix: fixed a non-nls comment
@AndriiFilippov AndriiFilippov merged commit b4631fb into master Dec 11, 2024
2 of 4 checks passed
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants