| // Copyright (c) 2017, the Dart project authors.  Please see the AUTHORS file | 
 | // for details. All rights reserved. Use of this source code is governed by a | 
 | // BSD-style license that can be found in the LICENSE file. | 
 |  | 
 | import 'dart:async'; | 
 |  | 
 | import 'package:test/test.dart'; | 
 | import 'package:vm_service/vm_service.dart'; | 
 |  | 
 | import 'common/service_test_common.dart'; | 
 | import 'common/test_helper.dart'; | 
 |  | 
 | // AUTOGENERATED START | 
 | // | 
 | // Update these constants by running: | 
 | // | 
 | // dart pkg/vm_service/test/update_line_numbers.dart pkg/vm_service/test/add_breakpoint_rpc_kernel_test.dart | 
 | // | 
 | const LINE_A = 30; | 
 | // AUTOGENERATED END | 
 |  | 
 | int value = 0; | 
 |  | 
 | int incValue(int amount) { | 
 |   value += amount; | 
 |   return amount; | 
 | } | 
 |  | 
 | Future<void> testMain() async { | 
 |   incValue(incValue(1)); // LINE_A. | 
 | } | 
 |  | 
 | final tests = <IsolateTest>[ | 
 |   hasPausedAtStart, | 
 |  | 
 |   (VmService service, IsolateRef isolateRef) async { | 
 |     final isolateId = isolateRef.id!; | 
 |     final isolate = await service.getIsolate(isolateId); | 
 |     final rootLib = | 
 |         await service.getObject(isolateId, isolate.rootLib!.id!) as Library; | 
 |     final rootLibId = rootLib.id!; | 
 |     final scriptId = rootLib.scripts![0].id!; | 
 |  | 
 |     final bpt1 = await service.addBreakpoint(isolateId, scriptId, LINE_A); | 
 |     expect(bpt1.breakpointNumber, 1); | 
 |     expect(bpt1.resolved, true); | 
 |     expect(await bpt1.location!.line!, LINE_A); | 
 |     expect(await bpt1.location!.column, 12); | 
 |  | 
 |     // Breakpoint with specific column. | 
 |     final bpt2 = | 
 |         await service.addBreakpoint(isolateId, scriptId, LINE_A, column: 3); | 
 |     expect(bpt2.breakpointNumber, 2); | 
 |     expect(bpt2.resolved, true); | 
 |     expect(await bpt2.location!.line!, LINE_A); | 
 |     expect(await bpt2.location!.column!, 3); | 
 |  | 
 |     await service.resume(isolateId); | 
 |     await hasStoppedAtBreakpoint(service, isolate); | 
 |     // The first breakpoint hits before value is modified. | 
 |     InstanceRef result = | 
 |         await service.evaluate(isolateId, rootLibId, 'value') as InstanceRef; | 
 |     expect(result.valueAsString, '0'); | 
 |  | 
 |     await service.resume(isolateId); | 
 |     await hasStoppedAtBreakpoint(service, isolate); | 
 |     // The second breakpoint hits after value has been modified once. | 
 |     result = | 
 |         await service.evaluate(isolateId, rootLibId, 'value') as InstanceRef; | 
 |     expect(result.valueAsString, '1'); | 
 |  | 
 |     // Remove the breakpoints. | 
 |     expect( | 
 |       (await service.removeBreakpoint(isolateId, bpt1.id!)).type, | 
 |       'Success', | 
 |     ); | 
 |     expect( | 
 |       (await service.removeBreakpoint(isolateId, bpt2.id!)).type, | 
 |       'Success', | 
 |     ); | 
 |   }, | 
 |  | 
 |   // Test resolution of column breakpoints. | 
 |   (VmService service, IsolateRef isolateRef) async { | 
 |     final isolateId = isolateRef.id!; | 
 |     final isolate = await service.getIsolate(isolateId); | 
 |     final rootLibId = isolate.rootLib!.id!; | 
 |     final rootLib = await service.getObject(isolateId, rootLibId) as Library; | 
 |  | 
 |     final scriptId = rootLib.scripts![0].id!; | 
 |     final script = await service.getObject(isolateId, scriptId) as Script; | 
 |  | 
 |     // Try all valid column arguments. | 
 |     for (int col = 1; col <= 36; col++) { | 
 |       final bpt = | 
 |           await service.addBreakpoint(isolateId, scriptId, LINE_A, column: col); | 
 |       expect(bpt.resolved, isTrue); | 
 |       final int resolvedLine = script.getLineNumberFromTokenPos( | 
 |         bpt.location!.tokenPos!, | 
 |       )!; | 
 |       final int resolvedCol = script.getColumnNumberFromTokenPos( | 
 |         bpt.location!.tokenPos!, | 
 |       )!; | 
 |       print('$LINE_A:$col -> $resolvedLine:$resolvedCol'); | 
 |       if (col < 12) { | 
 |         // The second 'incValue' begins at column 12. | 
 |         expect(resolvedLine, LINE_A); | 
 |         expect(bpt.location!.line, LINE_A); | 
 |         expect(resolvedCol, 3); | 
 |         expect(bpt.location!.column, 3); | 
 |       } else { | 
 |         // The newline character at the end of LINE_A is at column 36. | 
 |         expect(resolvedLine, LINE_A); | 
 |         expect(bpt.location!.line, LINE_A); | 
 |         expect(resolvedCol, 12); | 
 |         expect(bpt.location!.column, 12); | 
 |       } | 
 |       expect( | 
 |         (await service.removeBreakpoint(isolateId, bpt.id!)).type, | 
 |         'Success', | 
 |       ); | 
 |     } | 
 |  | 
 |     // Ensure that an error is thrown when 0 is passed as the column argument. | 
 |     try { | 
 |       await service.addBreakpoint(isolateId, scriptId, LINE_A, column: 0); | 
 |       fail('Expected to catch an RPC error'); | 
 |     } on RPCError catch (e) { | 
 |       expect(e.code, RPCErrorKind.kInvalidParams.code); | 
 |       expect(e.details, "addBreakpoint: invalid 'column' parameter: 0"); | 
 |     } | 
 |  | 
 |     // Ensure that an error is thrown when a number greater than the number of | 
 |     // columns on the specified line is passed as the column argument. | 
 |     try { | 
 |       await service.addBreakpoint(isolateId, scriptId, LINE_A, column: 37); | 
 |       fail('Expected to catch an RPC error'); | 
 |     } on RPCError catch (e) { | 
 |       expect(e.code, RPCErrorKind.kCannotAddBreakpoint.code); | 
 |       expect( | 
 |         e.details, | 
 |         'addBreakpoint: Cannot add breakpoint at $LINE_A:37. Error occurred ' | 
 |         'when resolving breakpoint location: No debuggable code where ' | 
 |         'breakpoint was requested.', | 
 |       ); | 
 |     } | 
 |   }, | 
 | ]; | 
 |  | 
 | Future<void> main(args) => runIsolateTests( | 
 |       args, | 
 |       tests, | 
 |       'add_breakpoint_rpc_kernel_test.dart', | 
 |       testeeConcurrent: testMain, | 
 |       pauseOnStart: true, | 
 |     ); |