diff --git a/misc/debug-adapter/modules/debug-adapter-core/src/main/java/org/ballerinalang/debugadapter/BreakpointProcessor.java b/misc/debug-adapter/modules/debug-adapter-core/src/main/java/org/ballerinalang/debugadapter/BreakpointProcessor.java index 9d0b8b935ad2..8433049daa2f 100644 --- a/misc/debug-adapter/modules/debug-adapter-core/src/main/java/org/ballerinalang/debugadapter/BreakpointProcessor.java +++ b/misc/debug-adapter/modules/debug-adapter-core/src/main/java/org/ballerinalang/debugadapter/BreakpointProcessor.java @@ -22,17 +22,15 @@ import com.sun.jdi.ThreadReference; import com.sun.jdi.event.BreakpointEvent; import com.sun.jdi.request.BreakpointRequest; -import com.sun.jdi.request.EventRequest; import com.sun.jdi.request.StepRequest; import org.ballerinalang.debugadapter.breakpoint.BalBreakpoint; import org.ballerinalang.debugadapter.breakpoint.LogMessage; import org.ballerinalang.debugadapter.breakpoint.TemplateLogMessage; -import org.ballerinalang.debugadapter.config.ClientConfigHolder; -import org.ballerinalang.debugadapter.config.ClientLaunchConfigHolder; import org.ballerinalang.debugadapter.evaluation.BExpressionValue; import org.ballerinalang.debugadapter.evaluation.DebugExpressionEvaluator; import org.ballerinalang.debugadapter.evaluation.EvaluationException; import org.ballerinalang.debugadapter.evaluation.EvaluationExceptionKind; +import org.ballerinalang.debugadapter.jdi.JDIUtils; import org.ballerinalang.debugadapter.jdi.JdiProxyException; import org.ballerinalang.debugadapter.jdi.StackFrameProxyImpl; import org.ballerinalang.debugadapter.jdi.ThreadReferenceProxyImpl; @@ -48,6 +46,7 @@ import java.util.LinkedHashMap; import java.util.List; import java.util.Map; +import java.util.Objects; import java.util.Optional; import java.util.concurrent.CompletableFuture; import java.util.concurrent.ConcurrentHashMap; @@ -107,9 +106,10 @@ void processBreakpointEvent(BreakpointEvent bpEvent) { // need to internally step out if we are at the last line of a function, in order to ignore having debug hits // on the last line. if (requireStepOut(bpEvent)) { - activateDynamicBreakPoints((int) bpEvent.thread().uniqueID(), DynamicBreakpointMode.CALLER); + activateDynamicBreakPoints((int) bpEvent.thread().uniqueID(), DynamicBreakpointMode.CALLER, true); + context.setPrevInstruction(DebugInstruction.STEP_OVER); context.getDebuggeeVM().resume(); - } else if (context.getLastInstruction() != null && context.getLastInstruction() != DebugInstruction.CONTINUE) { + } else if (context.getPrevInstruction() != null && context.getPrevInstruction() != DebugInstruction.CONTINUE) { jdiEventProcessor.notifyStopEvent(bpEvent); } else if (fileBreakpoints == null || !fileBreakpoints.containsKey(lineNumber)) { jdiEventProcessor.notifyStopEvent(bpEvent); @@ -168,19 +168,13 @@ private void processAdvanceBreakpoints(BreakpointEvent event, BalBreakpoint brea /** * Responsible for clearing dynamic(temporary) breakpoints used for step over instruction and for restoring the * original user breakpoints before proceeding with the other debug instructions. - * - * @param instruction debug instruction */ - void restoreUserBreakpoints(DebugInstruction instruction) { + void restoreUserBreakpoints() { if (context.getDebuggeeVM() == null) { return; } - context.getEventManager().deleteAllBreakpoints(); - if (instruction == DebugInstruction.CONTINUE || instruction == DebugInstruction.STEP_OVER) { - context.getDebuggeeVM().allClasses().forEach(referenceType -> - activateUserBreakPoints(referenceType, false)); - } + context.getDebuggeeVM().allClasses().forEach(classRef -> activateUserBreakPoints(classRef, false)); } /** @@ -191,13 +185,6 @@ void restoreUserBreakpoints(DebugInstruction instruction) { */ void activateUserBreakPoints(ReferenceType referenceType, boolean shouldNotify) { try { - // avoids setting break points if the server is running in 'no-debug' mode. - ClientConfigHolder configHolder = context.getAdapter().getClientConfigHolder(); - if (configHolder instanceof ClientLaunchConfigHolder - && ((ClientLaunchConfigHolder) configHolder).isNoDebugMode()) { - return; - } - String qualifiedClassName = getQualifiedClassName(referenceType); if (!userBreakpoints.containsKey(qualifiedClassName)) { return; @@ -206,6 +193,7 @@ void activateUserBreakPoints(ReferenceType referenceType, boolean shouldNotify) for (BalBreakpoint breakpoint : breakpoints.values()) { List locations = referenceType.locationsOfLine(breakpoint.getLine()); if (!locations.isEmpty()) { + // TODO: should we consider the last location instead? Location loc = locations.get(0); BreakpointRequest bpReq = context.getEventManager().createBreakpointRequest(loc); bpReq.enable(); @@ -230,23 +218,32 @@ void activateUserBreakPoints(ReferenceType referenceType, boolean shouldNotify) * Activates dynamic/temporary breakpoints (which will be used to process the STEP-OVER instruction) via Java Debug * Interface(JDI). * - * @param threadId ID of the active java thread in oder to configure dynamic breakpoint on the active stack trace - * @param mode dynamic breakpoint mode + * @param threadId ID of the active java thread in oder to configure dynamic breakpoint on the active stack + * trace. + * @param mode dynamic breakpoint mode. + * @param validate If true, validates whether the dynamic breakpoints are already applied before activating dynamic + * breakpoints. This is to optimize the performance by avoiding redundant breakpoint activations. */ - void activateDynamicBreakPoints(int threadId, DynamicBreakpointMode mode) { - ThreadReferenceProxyImpl threadReference = context.getAdapter().getAllThreads().get(threadId); + void activateDynamicBreakPoints(int threadId, DynamicBreakpointMode mode, boolean validate) { try { + ThreadReferenceProxyImpl threadReference = context.getAdapter().getAllThreads().get(threadId); List jStackFrames = threadReference.frames(); List validFrames = jdiEventProcessor.filterValidBallerinaFrames(jStackFrames); if (mode == DynamicBreakpointMode.CURRENT && !validFrames.isEmpty()) { - configureBreakpointsForMethod(validFrames.get(0)); + Location currentLocation = validFrames.get(0).getJStackFrame().location(); + Optional prevLocation = context.getPrevLocation(); + if (!validate || prevLocation.isEmpty() || !isWithinSameSource(currentLocation, prevLocation.get())) { + doActivateDynamicBreakPoints(currentLocation); + } + context.setPrevLocation(currentLocation); } + // If the current function is invoked within another ballerina function, we need to explicitly set another // temporary breakpoint on the location of its invocation. This is supposed to handle the situations where // the user wants to step over on an exit point of the current function. if (mode == DynamicBreakpointMode.CALLER && validFrames.size() > 1) { - configureBreakpointsForMethod(validFrames.get(1)); + doActivateDynamicBreakPoints(validFrames.get(1).getJStackFrame().location()); } } catch (JdiProxyException e) { LOGGER.error(e.getMessage()); @@ -257,16 +254,35 @@ void activateDynamicBreakPoints(int threadId, DynamicBreakpointMode mode) { } } + private void doActivateDynamicBreakPoints(Location location) { + context.getEventManager().deleteAllBreakpoints(); + configureBreakpointsForMethod(location); + context.setPrevLocation(location); + } + + /** + * Checks whether the given two locations are within the same source file. + * + * @param currentLocation current location + * @param prevLocation previous location + * @return true if the given two locations are within the same source file, false otherwise + */ + private boolean isWithinSameSource(Location currentLocation, Location prevLocation) { + try { + return Objects.equals(currentLocation.sourcePath(), prevLocation.sourcePath()); + } catch (AbsentInformationException e) { + return false; + } + } + /** * Configures temporary(dynamic) breakpoints for all the lines within the method, which encloses the given stack * frame location. This strategy is used when processing STEP_OVER requests. * - * @param balStackFrame stack frame which contains the method information + * @param currentLocation current stack frame location */ - private void configureBreakpointsForMethod(BallerinaStackFrame balStackFrame) { + private void configureBreakpointsForMethod(Location currentLocation) { try { - Location currentLocation = balStackFrame.getJStackFrame().location(); - ReferenceType referenceType = currentLocation.declaringType(); List allLocations = currentLocation.method().allLineLocations(); Optional firstLocation = allLocations.stream() .filter(location -> location.lineNumber() > 0) @@ -278,7 +294,7 @@ private void configureBreakpointsForMethod(BallerinaStackFrame balStackFrame) { int nextStepPoint = firstLocation.get().lineNumber(); do { - List locations = referenceType.locationsOfLine(nextStepPoint); + List locations = currentLocation.method().locationsOfLine(nextStepPoint); if (!locations.isEmpty() && (locations.get(0).lineNumber() > firstLocation.get().lineNumber())) { // Checks whether there are any user breakpoint configured for the same location, before adding the // dynamic breakpoint. @@ -291,7 +307,7 @@ private void configureBreakpointsForMethod(BallerinaStackFrame balStackFrame) { } nextStepPoint++; } while (nextStepPoint <= lastLocation.get().lineNumber()); - } catch (AbsentInformationException | JdiProxyException e) { + } catch (AbsentInformationException e) { LOGGER.error(e.getMessage()); } } @@ -362,8 +378,7 @@ private BExpressionValue evaluateExpressionSafely(String expression, ThreadRefer // the new event. If this new EventSet is in 'SUSPEND_ALL' mode, then a deadlock will occur because no one // will resume the EventSet. Therefore to avoid this, we are disabling possible event requests before doing // the condition evaluation. - context.getEventManager().classPrepareRequests().forEach(EventRequest::disable); - context.getEventManager().breakpointRequests().forEach(BreakpointRequest::disable); + JDIUtils.disableJDIRequests(context); ThreadReferenceProxyImpl thread = context.getAdapter().getAllThreads().get((int) threadReference.uniqueID()); List validFrames = jdiEventProcessor.filterValidBallerinaFrames(thread.frames()); @@ -377,16 +392,16 @@ private BExpressionValue evaluateExpressionSafely(String expression, ThreadRefer evaluator.setExpression(expression); BExpressionValue evaluationResult = evaluator.evaluate(); - // As we are disabling all the breakpoint requests before evaluating the user's conditional + // As we disabled all the breakpoint requests before evaluating the user's conditional // expression, need to re-enable all the breakpoints before continuing the remote VM execution. - restoreUserBreakpoints(context.getLastInstruction()); + JDIUtils.enableJDIRequests(context); return evaluationResult; } private boolean requireStepOut(BreakpointEvent event) { try { - if (context.getLastInstruction() != DebugInstruction.STEP_OVER - && context.getLastInstruction() != DebugInstruction.STEP_OUT) { + if (context.getPrevInstruction() != DebugInstruction.STEP_OVER + && context.getPrevInstruction() != DebugInstruction.STEP_OUT) { return false; } Location currentLocation = event.location(); @@ -415,7 +430,7 @@ private void notifyBreakPointChangesToClient(BalBreakpoint balBreakpoint) { /** * Dynamic Breakpoint Options. */ - enum DynamicBreakpointMode { + public enum DynamicBreakpointMode { /** * Configures dynamic breakpoints only for the current method (active stack frame). */ diff --git a/misc/debug-adapter/modules/debug-adapter-core/src/main/java/org/ballerinalang/debugadapter/ExecutionContext.java b/misc/debug-adapter/modules/debug-adapter-core/src/main/java/org/ballerinalang/debugadapter/ExecutionContext.java index 9fcdba5595ae..04d26d2870f7 100644 --- a/misc/debug-adapter/modules/debug-adapter-core/src/main/java/org/ballerinalang/debugadapter/ExecutionContext.java +++ b/misc/debug-adapter/modules/debug-adapter-core/src/main/java/org/ballerinalang/debugadapter/ExecutionContext.java @@ -16,6 +16,7 @@ package org.ballerinalang.debugadapter; +import com.sun.jdi.Location; import com.sun.jdi.request.EventRequestManager; import io.ballerina.projects.Project; import org.ballerinalang.debugadapter.jdi.VirtualMachineProxyImpl; @@ -39,14 +40,16 @@ public class ExecutionContext { private String sourceProjectRoot; private final DebugProjectCache projectCache; private Process launchedProcess; - private DebugInstruction lastInstruction; private boolean terminateRequestReceived; private boolean supportsRunInTerminalRequest; + private DebugInstruction prevInstruction; + private Location prevLocation; ExecutionContext(JBallerinaDebugServer adapter) { this.adapter = adapter; this.projectCache = new DebugProjectCache(); - this.lastInstruction = DebugInstruction.CONTINUE; + this.prevInstruction = DebugInstruction.CONTINUE; + this.prevLocation = null; } public Optional getLaunchedProcess() { @@ -102,12 +105,20 @@ public BufferedReader getErrorStream() { return new BufferedReader(new InputStreamReader(launchedProcess.getErrorStream(), StandardCharsets.UTF_8)); } - public DebugInstruction getLastInstruction() { - return lastInstruction; + public DebugInstruction getPrevInstruction() { + return prevInstruction; } - public void setLastInstruction(DebugInstruction lastInstruction) { - this.lastInstruction = lastInstruction; + public void setPrevInstruction(DebugInstruction prevInstruction) { + this.prevInstruction = prevInstruction; + } + + public Optional getPrevLocation() { + return Optional.ofNullable(prevLocation); + } + + public void setPrevLocation(Location prevLocation) { + this.prevLocation = prevLocation; } public DebugMode getDebugMode() { diff --git a/misc/debug-adapter/modules/debug-adapter-core/src/main/java/org/ballerinalang/debugadapter/JBallerinaDebugServer.java b/misc/debug-adapter/modules/debug-adapter-core/src/main/java/org/ballerinalang/debugadapter/JBallerinaDebugServer.java index b6d854e42bf6..0d84b61b4a4c 100755 --- a/misc/debug-adapter/modules/debug-adapter-core/src/main/java/org/ballerinalang/debugadapter/JBallerinaDebugServer.java +++ b/misc/debug-adapter/modules/debug-adapter-core/src/main/java/org/ballerinalang/debugadapter/JBallerinaDebugServer.java @@ -30,6 +30,7 @@ import io.ballerina.identifier.Utils; import io.ballerina.projects.Project; import io.ballerina.projects.directory.SingleFileProject; +import org.ballerinalang.debugadapter.BreakpointProcessor.DynamicBreakpointMode; import org.ballerinalang.debugadapter.breakpoint.BalBreakpoint; import org.ballerinalang.debugadapter.completion.CompletionGenerator; import org.ballerinalang.debugadapter.completion.context.CompletionContext; @@ -215,6 +216,11 @@ public CompletableFuture initialize(InitializeRequestArguments arg @Override public CompletableFuture setBreakpoints(SetBreakpointsArguments args) { return CompletableFuture.supplyAsync(() -> { + SetBreakpointsResponse bpResponse = new SetBreakpointsResponse(); + if (isNoDebugMode()) { + return bpResponse; + } + BalBreakpoint[] balBreakpoints = Arrays.stream(args.getBreakpoints()) .map((SourceBreakpoint sourceBreakpoint) -> toBreakpoint(sourceBreakpoint, args.getSource())) .toArray(BalBreakpoint[]::new); @@ -224,7 +230,6 @@ public CompletableFuture setBreakpoints(SetBreakpointsAr breakpointsMap.put(bp.getLine(), bp); } - SetBreakpointsResponse bpResponse = new SetBreakpointsResponse(); String sourcePathUri = args.getSource().getPath(); Optional qualifiedClassName = getQualifiedClassName(context, sourcePathUri); if (qualifiedClassName.isEmpty()) { @@ -425,7 +430,7 @@ public CompletableFuture source(SourceArguments args) { @Override public CompletableFuture continue_(ContinueArguments args) { - prepareFor(DebugInstruction.CONTINUE); + prepareFor(DebugInstruction.CONTINUE, args.getThreadId()); context.getDebuggeeVM().resume(); ContinueResponse continueResponse = new ContinueResponse(); continueResponse.setAllThreadsContinued(true); @@ -434,29 +439,25 @@ public CompletableFuture continue_(ContinueArguments args) { @Override public CompletableFuture next(NextArguments args) { - prepareFor(DebugInstruction.STEP_OVER); + prepareFor(DebugInstruction.STEP_OVER, args.getThreadId()); eventProcessor.sendStepRequest(args.getThreadId(), StepRequest.STEP_OVER); return CompletableFuture.completedFuture(null); } @Override public CompletableFuture stepIn(StepInArguments args) { - prepareFor(DebugInstruction.STEP_IN); + prepareFor(DebugInstruction.STEP_IN, args.getThreadId()); eventProcessor.sendStepRequest(args.getThreadId(), StepRequest.STEP_INTO); return CompletableFuture.completedFuture(null); } @Override public CompletableFuture stepOut(StepOutArguments args) { - stepOut(args.getThreadId()); + prepareFor(DebugInstruction.STEP_OUT, args.getThreadId()); + eventProcessor.sendStepRequest(args.getThreadId(), StepRequest.STEP_OUT); return CompletableFuture.completedFuture(null); } - void stepOut(int threadId) { - prepareFor(DebugInstruction.STEP_OUT); - eventProcessor.sendStepRequest(threadId, StepRequest.STEP_OUT); - } - @Override public CompletableFuture setExceptionBreakpoints(SetExceptionBreakpointsArguments args) { return CompletableFuture.completedFuture(null); @@ -1119,10 +1120,23 @@ private void attachToRemoteVM(String hostName, int portName) throws IOException, /** * Clears previous state information and prepares for the given debug instruction type execution. */ - private void prepareFor(DebugInstruction instruction) { + private void prepareFor(DebugInstruction instruction, int threadId) { clearState(); - eventProcessor.getBreakpointProcessor().restoreUserBreakpoints(instruction); - context.setLastInstruction(instruction); + BreakpointProcessor bpProcessor = eventProcessor.getBreakpointProcessor(); + DebugInstruction prevInstruction = context.getPrevInstruction(); + if (prevInstruction == DebugInstruction.STEP_OVER && instruction == DebugInstruction.CONTINUE) { + bpProcessor.restoreUserBreakpoints(); + } else if (prevInstruction == DebugInstruction.CONTINUE && instruction == DebugInstruction.STEP_OVER) { + bpProcessor.activateDynamicBreakPoints(threadId, DynamicBreakpointMode.CURRENT, false); + } else if (instruction == DebugInstruction.STEP_OVER) { + bpProcessor.activateDynamicBreakPoints(threadId, DynamicBreakpointMode.CURRENT, true); + } + context.setPrevInstruction(instruction); + } + + private boolean isNoDebugMode() { + ClientConfigHolder confHolder = context.getAdapter().getClientConfigHolder(); + return confHolder instanceof ClientLaunchConfigHolder launchConfigHolder && launchConfigHolder.isNoDebugMode(); } /** diff --git a/misc/debug-adapter/modules/debug-adapter-core/src/main/java/org/ballerinalang/debugadapter/JDIEventProcessor.java b/misc/debug-adapter/modules/debug-adapter-core/src/main/java/org/ballerinalang/debugadapter/JDIEventProcessor.java index 37b8a123888a..b538daf2e89b 100755 --- a/misc/debug-adapter/modules/debug-adapter-core/src/main/java/org/ballerinalang/debugadapter/JDIEventProcessor.java +++ b/misc/debug-adapter/modules/debug-adapter-core/src/main/java/org/ballerinalang/debugadapter/JDIEventProcessor.java @@ -44,7 +44,6 @@ import java.util.concurrent.CompletableFuture; import java.util.concurrent.CopyOnWriteArrayList; -import static org.ballerinalang.debugadapter.BreakpointProcessor.DynamicBreakpointMode; import static org.ballerinalang.debugadapter.JBallerinaDebugServer.isBalStackFrame; import static org.ballerinalang.debugadapter.utils.PackageUtils.BAL_FILE_EXT; @@ -101,7 +100,7 @@ void startListening() { private void processEvent(EventSet eventSet, Event event) { if (event instanceof ClassPrepareEvent evt) { - if (context.getLastInstruction() != DebugInstruction.STEP_OVER) { + if (context.getPrevInstruction() != DebugInstruction.STEP_OVER) { breakpointProcessor.activateUserBreakPoints(evt.referenceType(), true); } eventSet.resume(); @@ -136,9 +135,7 @@ void enableBreakpoints(String qualifiedClassName, LinkedHashMap + * Therefore, to avoid this, we are disabling possible event requests (only class prepare and breakpoint requests + * for now) before invoking the method. + */ + public static void disableJDIRequests(ExecutionContext context) { + EventRequestManager eventManager = context.getEventManager(); + eventManager.classPrepareRequests().forEach(EventRequest::disable); + eventManager.breakpointRequests().forEach(EventRequest::disable); + } + + /** + * Used to re-enable the event requests after the method invocation is completed. + */ + public static void enableJDIRequests(ExecutionContext context) { + EventRequestManager eventManager = context.getEventManager(); + eventManager.breakpointRequests().forEach(EventRequest::enable); + } +} diff --git a/tests/jballerina-debugger-integration-test/src/test/java/org/ballerinalang/debugger/test/adapter/DebugInstructionTest.java b/tests/jballerina-debugger-integration-test/src/test/java/org/ballerinalang/debugger/test/adapter/DebugInstructionTest.java index 041df59f7acb..af1b60f54820 100644 --- a/tests/jballerina-debugger-integration-test/src/test/java/org/ballerinalang/debugger/test/adapter/DebugInstructionTest.java +++ b/tests/jballerina-debugger-integration-test/src/test/java/org/ballerinalang/debugger/test/adapter/DebugInstructionTest.java @@ -73,12 +73,8 @@ public void breakpointInBetweenStepOverTest() throws BallerinaTestException { debugTestRunner.initDebugSession(DebugUtils.DebuggeeExecutionKind.RUN); Pair debugHitInfo = debugTestRunner.waitForDebugHit(25000); - // Debugger should hit the breakpoint inside the function invocation when trying to step over. - debugTestRunner.resumeProgram(debugHitInfo.getRight(), DebugTestRunner.DebugResumeKind.STEP_OVER); - debugHitInfo = debugTestRunner.waitForDebugHit(10000); - Assert.assertEquals(debugHitInfo.getLeft(), new BallerinaTestDebugPoint(debugTestRunner.testEntryFilePath, 46)); - - // Should go back to the caller function when stepping over again. + // Debugger should not hit the breakpoint inside function invocations when trying to step over the caller + // function. debugTestRunner.resumeProgram(debugHitInfo.getRight(), DebugTestRunner.DebugResumeKind.STEP_OVER); debugHitInfo = debugTestRunner.waitForDebugHit(10000); Assert.assertEquals(debugHitInfo.getLeft(), new BallerinaTestDebugPoint(debugTestRunner.testEntryFilePath, 31)); diff --git a/tests/jballerina-debugger-integration-test/src/test/java/org/ballerinalang/debugger/test/adapter/RecursiveDebugTest.java b/tests/jballerina-debugger-integration-test/src/test/java/org/ballerinalang/debugger/test/adapter/RecursiveDebugTest.java index c9cb706cc6d2..2e2363d8eec6 100644 --- a/tests/jballerina-debugger-integration-test/src/test/java/org/ballerinalang/debugger/test/adapter/RecursiveDebugTest.java +++ b/tests/jballerina-debugger-integration-test/src/test/java/org/ballerinalang/debugger/test/adapter/RecursiveDebugTest.java @@ -63,12 +63,12 @@ public void testStepRequestsOnRecursiveFunctions() throws BallerinaTestException debugHitInfo = debugTestRunner.waitForDebugHit(10000); Assert.assertEquals(debugHitInfo.getLeft(), new BallerinaTestDebugPoint(debugTestRunner.testEntryFilePath, 26)); - // tries STEP_OVER request on the recursive function call line, which should results in a debug hit on the - // user configured breakpoint (since STEP_OVER request should honor any breakpoints which get reached during - // the program execution, before reaching to the next line of the same method). + // tries STEP_OVER request on the recursive function call line, which should not results in a debug hit on the + // user configured breakpoint (since STEP requests should not depend on any breakpoints which get reached + // during the program execution, before reaching to the next line of the same method). debugTestRunner.resumeProgram(debugHitInfo.getRight(), DebugTestRunner.DebugResumeKind.STEP_OVER); debugHitInfo = debugTestRunner.waitForDebugHit(10000); - Assert.assertEquals(debugHitInfo.getLeft(), debugTestRunner.testBreakpoints.get(0)); + Assert.assertEquals(debugHitInfo.getLeft(), new BallerinaTestDebugPoint(debugTestRunner.testEntryFilePath, 26)); // Todo - enable after fixing https://github.com/ballerina-platform/ballerina-lang/issues/34847 // since now the breakpoint is removed, STEP_OVER request should result in a debug hit on next immediate