From 9eac08d67215a615c748e03bd103281544f09491 Mon Sep 17 00:00:00 2001 From: Ahmed Elrefaey <68241710+a7medev@users.noreply.github.com> Date: Mon, 12 Aug 2024 14:19:20 +0300 Subject: [PATCH 1/8] test: add tests for navigator observer (#501) --- pubspec.yaml | 1 + .../instabug_navigator_observer_test.dart | 98 +++++++++++++++++++ 2 files changed, 99 insertions(+) create mode 100644 test/utils/instabug_navigator_observer_test.dart diff --git a/pubspec.yaml b/pubspec.yaml index 76c665ab3..466933d8c 100644 --- a/pubspec.yaml +++ b/pubspec.yaml @@ -15,6 +15,7 @@ dependencies: dev_dependencies: build_runner: ^2.0.3 + fake_async: '>=1.2.0 <1.4.0' flutter_test: sdk: flutter lint: ^1.0.0 diff --git a/test/utils/instabug_navigator_observer_test.dart b/test/utils/instabug_navigator_observer_test.dart new file mode 100644 index 000000000..0eca4ca2e --- /dev/null +++ b/test/utils/instabug_navigator_observer_test.dart @@ -0,0 +1,98 @@ +import 'package:fake_async/fake_async.dart'; +import 'package:flutter/material.dart'; +import 'package:flutter_test/flutter_test.dart'; +import 'package:instabug_flutter/instabug_flutter.dart'; +import 'package:instabug_flutter/src/generated/instabug.api.g.dart'; +import 'package:instabug_flutter/src/utils/screen_loading/screen_loading_manager.dart'; +import 'package:mockito/annotations.dart'; +import 'package:mockito/mockito.dart'; + +import 'instabug_navigator_observer_test.mocks.dart'; + +@GenerateMocks([ + InstabugHostApi, + ScreenLoadingManager, +]) +void main() { + TestWidgetsFlutterBinding.ensureInitialized(); + WidgetsFlutterBinding.ensureInitialized(); + + final mHost = MockInstabugHostApi(); + final mScreenLoadingManager = MockScreenLoadingManager(); + + late InstabugNavigatorObserver observer; + const screen = '/screen'; + const previousScreen = '/previousScreen'; + late Route route; + late Route previousRoute; + + setUpAll(() { + Instabug.$setHostApi(mHost); + ScreenLoadingManager.setInstance(mScreenLoadingManager); + }); + + setUp(() { + observer = InstabugNavigatorObserver(); + route = createRoute(screen); + previousRoute = createRoute(previousScreen); + }); + + test('should report screen change when a route is pushed', () { + fakeAsync((async) { + observer.didPush(route, previousRoute); + + async.elapse(const Duration(milliseconds: 1000)); + + verify( + mScreenLoadingManager.startUiTrace(screen), + ).called(1); + + verify( + mHost.reportScreenChange(screen), + ).called(1); + }); + }); + + test( + 'should report screen change when a route is popped and previous is known', + () { + fakeAsync((async) { + observer.didPop(route, previousRoute); + + async.elapse(const Duration(milliseconds: 1000)); + + verify( + mScreenLoadingManager.startUiTrace(previousScreen), + ).called(1); + + verify( + mHost.reportScreenChange(previousScreen), + ).called(1); + }); + }); + + test( + 'should not report screen change when a route is popped and previous is not known', + () { + fakeAsync((async) { + observer.didPop(route, null); + + async.elapse(const Duration(milliseconds: 1000)); + + verifyNever( + mScreenLoadingManager.startUiTrace(any), + ); + + verifyNever( + mHost.reportScreenChange(any), + ); + }); + }); +} + +Route createRoute(String? name) { + return MaterialPageRoute( + builder: (_) => Container(), + settings: RouteSettings(name: name), + ); +} From 40d38f9ef2e07d981ca72b03bd457b31dd32ad7b Mon Sep 17 00:00:00 2001 From: Ahmed Elrefaey <68241710+a7medev@users.noreply.github.com> Date: Thu, 5 Sep 2024 16:45:23 +0300 Subject: [PATCH 2/8] feat: support masking screen names (#500) --- CHANGELOG.md | 6 + lib/instabug_flutter.dart | 1 + lib/src/models/instabug_route.dart | 11 ++ lib/src/modules/instabug.dart | 9 ++ .../utils/instabug_navigator_observer.dart | 28 ++-- .../screen_loading_manager.dart | 26 +++- lib/src/utils/screen_loading/ui_trace.dart | 28 +++- lib/src/utils/screen_name_masker.dart | 45 ++++++ test/instabug_test.dart | 14 ++ .../instabug_navigator_observer_test.dart | 29 +++- .../screen_loading_manager_test.dart | 145 +++++++++++++++++- test/utils/screen_name_masker_test.dart | 77 ++++++++++ 12 files changed, 389 insertions(+), 30 deletions(-) create mode 100644 lib/src/models/instabug_route.dart create mode 100644 lib/src/utils/screen_name_masker.dart create mode 100644 test/utils/screen_name_masker_test.dart diff --git a/CHANGELOG.md b/CHANGELOG.md index d0b3bf7bc..34e09b698 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,5 +1,11 @@ # Changelog +## [Unreleased](https://github.com/Instabug/Instabug-Flutter/compare/v13.3.0...dev) + +### Added + +- Add support for masking screen names captured by Instabug through the `Instabug.setScreenNameMaskingCallback` API ([#500](https://github.com/Instabug/Instabug-Flutter/pull/500)). + ## [13.3.0](https://github.com/Instabug/Instabug-Flutter/compare/v13.2.0...v13.3.0) (August 5, 2024) ### Added diff --git a/lib/instabug_flutter.dart b/lib/instabug_flutter.dart index 783fb1350..6dc949d1b 100644 --- a/lib/instabug_flutter.dart +++ b/lib/instabug_flutter.dart @@ -19,3 +19,4 @@ export 'src/modules/surveys.dart'; export 'src/utils/instabug_navigator_observer.dart'; export 'src/utils/screen_loading/instabug_capture_screen_loading.dart'; export 'src/utils/screen_loading/route_matcher.dart'; +export 'src/utils/screen_name_masker.dart' show ScreenNameMaskingCallback; diff --git a/lib/src/models/instabug_route.dart b/lib/src/models/instabug_route.dart new file mode 100644 index 000000000..8c1539426 --- /dev/null +++ b/lib/src/models/instabug_route.dart @@ -0,0 +1,11 @@ +import 'package:flutter/material.dart'; + +class InstabugRoute { + final Route route; + final String name; + + const InstabugRoute({ + required this.route, + required this.name, + }); +} diff --git a/lib/src/modules/instabug.dart b/lib/src/modules/instabug.dart index 9d578e030..bf457a4fb 100644 --- a/lib/src/modules/instabug.dart +++ b/lib/src/modules/instabug.dart @@ -19,6 +19,7 @@ import 'package:instabug_flutter/src/generated/instabug.api.g.dart'; import 'package:instabug_flutter/src/utils/enum_converter.dart'; import 'package:instabug_flutter/src/utils/ibg_build_info.dart'; import 'package:instabug_flutter/src/utils/instabug_logger.dart'; +import 'package:instabug_flutter/src/utils/screen_name_masker.dart'; import 'package:meta/meta.dart'; enum InvocationEvent { @@ -191,6 +192,14 @@ class Instabug { ); } + /// Sets a [callback] to be called wehenever a screen name is captured to mask + /// sensitive information in the screen name. + static void setScreenNameMaskingCallback( + ScreenNameMaskingCallback? callback, + ) { + ScreenNameMasker.I.setMaskingCallback(callback); + } + /// Shows the welcome message in a specific mode. /// [welcomeMessageMode] is an enum to set the welcome message mode to live, or beta. static Future showWelcomeMessageWithMode( diff --git a/lib/src/utils/instabug_navigator_observer.dart b/lib/src/utils/instabug_navigator_observer.dart index 3cb39cc05..57550765c 100644 --- a/lib/src/utils/instabug_navigator_observer.dart +++ b/lib/src/utils/instabug_navigator_observer.dart @@ -1,32 +1,40 @@ import 'package:flutter/material.dart'; import 'package:instabug_flutter/instabug_flutter.dart'; +import 'package:instabug_flutter/src/models/instabug_route.dart'; import 'package:instabug_flutter/src/modules/instabug.dart'; import 'package:instabug_flutter/src/utils/instabug_logger.dart'; import 'package:instabug_flutter/src/utils/screen_loading/screen_loading_manager.dart'; +import 'package:instabug_flutter/src/utils/screen_name_masker.dart'; class InstabugNavigatorObserver extends NavigatorObserver { - final List _steps = []; + final List _steps = []; void screenChanged(Route newRoute) { try { final screenName = newRoute.settings.name.toString(); + final maskedScreenName = ScreenNameMasker.I.mask(screenName); + + final route = InstabugRoute( + route: newRoute, + name: maskedScreenName, + ); + // Starts a the new UI trace which is exclusive to screen loading - ScreenLoadingManager.I.startUiTrace(screenName); + ScreenLoadingManager.I.startUiTrace(maskedScreenName, screenName); // If there is a step that hasn't been pushed yet if (_steps.isNotEmpty) { // Report the last step and remove it from the list - Instabug.reportScreenChange( - _steps[_steps.length - 1].settings.name.toString(), - ); - _steps.remove(_steps[_steps.length - 1]); + Instabug.reportScreenChange(_steps.last.name); + _steps.removeLast(); } + // Add the new step to the list - _steps.add(newRoute); + _steps.add(route); Future.delayed(const Duration(milliseconds: 1000), () { // If this route is in the array, report it and remove it from the list - if (_steps.contains(newRoute)) { - Instabug.reportScreenChange(screenName); - _steps.remove(newRoute); + if (_steps.contains(route)) { + Instabug.reportScreenChange(route.name); + _steps.remove(route); } }); } catch (e) { diff --git a/lib/src/utils/screen_loading/screen_loading_manager.dart b/lib/src/utils/screen_loading/screen_loading_manager.dart index 2af57f09f..e95543ee9 100644 --- a/lib/src/utils/screen_loading/screen_loading_manager.dart +++ b/lib/src/utils/screen_loading/screen_loading_manager.dart @@ -126,8 +126,16 @@ class ScreenLoadingManager { return sanitizedScreenName; } + /// Starts a new UI trace with [screenName] as the public screen name and + /// [matchingScreenName] as the screen name used for matching the UI trace + /// with a Screen Loading trace. @internal - Future startUiTrace(String screenName) async { + Future startUiTrace( + String screenName, [ + String? matchingScreenName, + ]) async { + matchingScreenName ??= screenName; + try { resetDidStartScreenLoading(); @@ -151,10 +159,19 @@ class ScreenLoadingManager { } final sanitizedScreenName = sanitizeScreenName(screenName); + final sanitizedMatchingScreenName = + sanitizeScreenName(matchingScreenName); + final microTimeStamp = IBGDateTime.I.now().microsecondsSinceEpoch; final uiTraceId = IBGDateTime.I.now().millisecondsSinceEpoch; + APM.startCpUiTrace(sanitizedScreenName, microTimeStamp, uiTraceId); - currentUiTrace = UiTrace(sanitizedScreenName, traceId: uiTraceId); + + currentUiTrace = UiTrace( + screenName: sanitizedScreenName, + matchingScreenName: sanitizedMatchingScreenName, + traceId: uiTraceId, + ); } catch (error, stackTrace) { _logExceptionErrorAndStackTrace(error, stackTrace); } @@ -183,10 +200,7 @@ class ScreenLoadingManager { return; } - final isSameScreen = RouteMatcher.I.match( - routePath: trace.screenName, - actualPath: currentUiTrace?.screenName, - ); + final isSameScreen = currentUiTrace?.matches(trace.screenName) == true; final didStartLoading = currentUiTrace?.didStartScreenLoading == true; diff --git a/lib/src/utils/screen_loading/ui_trace.dart b/lib/src/utils/screen_loading/ui_trace.dart index 7fd03c9fd..17ef41046 100644 --- a/lib/src/utils/screen_loading/ui_trace.dart +++ b/lib/src/utils/screen_loading/ui_trace.dart @@ -1,25 +1,45 @@ +import 'package:instabug_flutter/src/utils/screen_loading/route_matcher.dart'; + class UiTrace { final String screenName; + + /// The screen name used while matching the UI trace with a Screen Loading + /// trace. + /// + /// For example, this is set to the original screen name before masking when + /// screen names masking is enabled. + final String _matchingScreenName; + final int traceId; bool didStartScreenLoading = false; bool didReportScreenLoading = false; bool didExtendScreenLoading = false; - UiTrace( - this.screenName, { + UiTrace({ + required this.screenName, required this.traceId, - }); + String? matchingScreenName, + }) : _matchingScreenName = matchingScreenName ?? screenName; UiTrace copyWith({ String? screenName, + String? matchingScreenName, int? traceId, }) { return UiTrace( - screenName ?? this.screenName, + screenName: screenName ?? this.screenName, + matchingScreenName: matchingScreenName ?? _matchingScreenName, traceId: traceId ?? this.traceId, ); } + bool matches(String routePath) { + return RouteMatcher.I.match( + routePath: routePath, + actualPath: _matchingScreenName, + ); + } + @override String toString() { return 'UiTrace{screenName: $screenName, traceId: $traceId, isFirstScreenLoadingReported: $didReportScreenLoading, isFirstScreenLoading: $didStartScreenLoading}'; diff --git a/lib/src/utils/screen_name_masker.dart b/lib/src/utils/screen_name_masker.dart new file mode 100644 index 000000000..ef228697f --- /dev/null +++ b/lib/src/utils/screen_name_masker.dart @@ -0,0 +1,45 @@ +import 'package:flutter/material.dart'; + +typedef ScreenNameMaskingCallback = String Function(String screen); + +/// Mockable [ScreenNameMasker] responsible for masking screen names +/// before they are sent to the native SDKs. +class ScreenNameMasker { + ScreenNameMasker._(); + + static ScreenNameMasker _instance = ScreenNameMasker._(); + + static ScreenNameMasker get instance => _instance; + + /// Shorthand for [instance] + static ScreenNameMasker get I => instance; + + static const emptyScreenNameFallback = "N/A"; + + ScreenNameMaskingCallback? _screenNameMaskingCallback; + + @visibleForTesting + // ignore: use_setters_to_change_properties + static void setInstance(ScreenNameMasker instance) { + _instance = instance; + } + + // ignore: use_setters_to_change_properties + void setMaskingCallback(ScreenNameMaskingCallback? callback) { + _screenNameMaskingCallback = callback; + } + + String mask(String screen) { + if (_screenNameMaskingCallback == null) { + return screen; + } + + final maskedScreen = _screenNameMaskingCallback!(screen).trim(); + + if (maskedScreen.isEmpty) { + return emptyScreenNameFallback; + } + + return maskedScreen; + } +} diff --git a/test/instabug_test.dart b/test/instabug_test.dart index f525dc02c..0e6f0f421 100644 --- a/test/instabug_test.dart +++ b/test/instabug_test.dart @@ -6,6 +6,7 @@ import 'package:instabug_flutter/instabug_flutter.dart'; import 'package:instabug_flutter/src/generated/instabug.api.g.dart'; import 'package:instabug_flutter/src/utils/enum_converter.dart'; import 'package:instabug_flutter/src/utils/ibg_build_info.dart'; +import 'package:instabug_flutter/src/utils/screen_name_masker.dart'; import 'package:mockito/annotations.dart'; import 'package:mockito/mockito.dart'; @@ -14,6 +15,7 @@ import 'instabug_test.mocks.dart'; @GenerateMocks([ InstabugHostApi, IBGBuildInfo, + ScreenNameMasker, ]) void main() { TestWidgetsFlutterBinding.ensureInitialized(); @@ -21,10 +23,12 @@ void main() { final mHost = MockInstabugHostApi(); final mBuildInfo = MockIBGBuildInfo(); + final mScreenNameMasker = MockScreenNameMasker(); setUpAll(() { Instabug.$setHostApi(mHost); IBGBuildInfo.setInstance(mBuildInfo); + ScreenNameMasker.setInstance(mScreenNameMasker); }); test('[setEnabled] should call host method', () async { @@ -76,6 +80,16 @@ void main() { ).called(1); }); + test( + '[setScreenNameMaskingCallback] should set masking callback on screen name masker', + () async { + String callback(String screen) => 'REDACTED/$screen'; + + Instabug.setScreenNameMaskingCallback(callback); + + verify(mScreenNameMasker.setMaskingCallback(callback)).called(1); + }); + test('[show] should call host method', () async { await Instabug.show(); diff --git a/test/utils/instabug_navigator_observer_test.dart b/test/utils/instabug_navigator_observer_test.dart index 0eca4ca2e..f7a6f6a7f 100644 --- a/test/utils/instabug_navigator_observer_test.dart +++ b/test/utils/instabug_navigator_observer_test.dart @@ -4,6 +4,7 @@ import 'package:flutter_test/flutter_test.dart'; import 'package:instabug_flutter/instabug_flutter.dart'; import 'package:instabug_flutter/src/generated/instabug.api.g.dart'; import 'package:instabug_flutter/src/utils/screen_loading/screen_loading_manager.dart'; +import 'package:instabug_flutter/src/utils/screen_name_masker.dart'; import 'package:mockito/annotations.dart'; import 'package:mockito/mockito.dart'; @@ -35,6 +36,8 @@ void main() { observer = InstabugNavigatorObserver(); route = createRoute(screen); previousRoute = createRoute(previousScreen); + + ScreenNameMasker.I.setMaskingCallback(null); }); test('should report screen change when a route is pushed', () { @@ -44,7 +47,7 @@ void main() { async.elapse(const Duration(milliseconds: 1000)); verify( - mScreenLoadingManager.startUiTrace(screen), + mScreenLoadingManager.startUiTrace(screen, screen), ).called(1); verify( @@ -62,7 +65,7 @@ void main() { async.elapse(const Duration(milliseconds: 1000)); verify( - mScreenLoadingManager.startUiTrace(previousScreen), + mScreenLoadingManager.startUiTrace(previousScreen, previousScreen), ).called(1); verify( @@ -80,7 +83,7 @@ void main() { async.elapse(const Duration(milliseconds: 1000)); verifyNever( - mScreenLoadingManager.startUiTrace(any), + mScreenLoadingManager.startUiTrace(any, any), ); verifyNever( @@ -88,6 +91,26 @@ void main() { ); }); }); + + test('should mask screen name when masking callback is set', () { + const maskedScreen = 'maskedScreen'; + + ScreenNameMasker.I.setMaskingCallback((_) => maskedScreen); + + fakeAsync((async) { + observer.didPush(route, previousRoute); + + async.elapse(const Duration(milliseconds: 1000)); + + verify( + mScreenLoadingManager.startUiTrace(maskedScreen, screen), + ).called(1); + + verify( + mHost.reportScreenChange(maskedScreen), + ).called(1); + }); + }); } Route createRoute(String? name) { diff --git a/test/utils/screen_loading/screen_loading_manager_test.dart b/test/utils/screen_loading/screen_loading_manager_test.dart index 14e33c663..c008b8bcc 100644 --- a/test/utils/screen_loading/screen_loading_manager_test.dart +++ b/test/utils/screen_loading/screen_loading_manager_test.dart @@ -81,7 +81,7 @@ void main() { '[resetDidStartScreenLoading] should set _currentUITrace?.didStartScreenLoading to false', () async { const expected = false; - final uiTrace = UiTrace('screen1', traceId: 1); + final uiTrace = UiTrace(screenName: 'screen1', traceId: 1); uiTrace.didStartScreenLoading = true; mScreenLoadingManager.currentUiTrace = uiTrace; @@ -103,7 +103,7 @@ void main() { '[resetDidReportScreenLoading] should set _currentUITrace?.didReportScreenLoading to false', () async { const expected = false; - final uiTrace = UiTrace('screen1', traceId: 1); + final uiTrace = UiTrace(screenName: 'screen1', traceId: 1); uiTrace.didReportScreenLoading = true; mScreenLoadingManager.currentUiTrace = uiTrace; @@ -125,7 +125,7 @@ void main() { '[resetDidExtendScreenLoading] should set _currentUITrace?.didExtendScreenLoading to false', () async { const expected = false; - final uiTrace = UiTrace('screen1', traceId: 1); + final uiTrace = UiTrace(screenName: 'screen1', traceId: 1); mScreenLoadingManager.currentUiTrace = uiTrace; ScreenLoadingManager.I.resetDidExtendScreenLoading(); @@ -160,7 +160,8 @@ void main() { setUp(() { time = DateTime.now(); - uiTrace = UiTrace(screenName, traceId: time.millisecondsSinceEpoch); + uiTrace = + UiTrace(screenName: screenName, traceId: time.millisecondsSinceEpoch); ScreenLoadingManager.setInstance(ScreenLoadingManagerNoResets.init()); mScreenLoadingManager.currentUiTrace = uiTrace; when(mDateTime.now()).thenReturn(time); @@ -230,6 +231,50 @@ void main() { ), ).called(1); }); + + test( + '[startUiTrace] with APM enabled should create a UI trace with the matching screen name', + () async { + when(FlagsConfig.apm.isEnabled()).thenAnswer((_) async => true); + when(IBGBuildInfo.I.isIOS).thenReturn(false); + when( + RouteMatcher.I.match( + routePath: anyNamed('routePath'), + actualPath: anyNamed('actualPath'), + ), + ).thenReturn(false); + + const matchingScreenName = 'matching_screen_name'; + + await ScreenLoadingManager.I.startUiTrace(screenName, matchingScreenName); + + final actualUiTrace = ScreenLoadingManager.I.currentUiTrace!; + + actualUiTrace.matches(screenName); + + verify( + mRouteMatcher.match( + routePath: screenName, + actualPath: matchingScreenName, + ), + ).called(1); + + verifyNever( + mRouteMatcher.match( + routePath: anyNamed('routePath'), + actualPath: screenName, + ), + ); + + expect(actualUiTrace.screenName, screenName); + verify( + mApmHost.startCpUiTrace( + screenName, + any, + any, + ), + ).called(1); + }); }); group('startScreenLoadingTrace tests', () { @@ -241,7 +286,7 @@ void main() { mScreenLoadingManager = ScreenLoadingManagerNoResets.init(); time = DateTime.now(); traceId = time.millisecondsSinceEpoch; - uiTrace = UiTrace(screenName, traceId: traceId); + uiTrace = UiTrace(screenName: screenName, traceId: traceId); mScreenLoadingManager.currentUiTrace = uiTrace; when(mDateTime.now()).thenReturn(time); @@ -409,6 +454,92 @@ void main() { ), ).called(1); }); + + test( + '[startScreenLoadingTrace] should start screen loading trace when screen loading trace matches UI trace matching screen name', + () async { + const isSameScreen = true; + const matchingScreenName = 'matching_screen_name'; + mScreenLoadingManager = ScreenLoadingManagerNoResets.init(); + when(FlagsConfig.screenLoading.isEnabled()).thenAnswer((_) async => true); + when(IBGBuildInfo.I.isIOS).thenReturn(false); + when(mDateTime.now()).thenReturn(time); + + // Match on matching screen name + when( + RouteMatcher.I.match( + routePath: screenName, + actualPath: matchingScreenName, + ), + ).thenReturn(isSameScreen); + + when( + RouteMatcher.I.match( + routePath: screenName, + actualPath: screenName, + ), + ).thenReturn(!isSameScreen); + + ScreenLoadingManager.I.currentUiTrace = uiTrace.copyWith( + matchingScreenName: matchingScreenName, + ); + + await ScreenLoadingManager.I.startScreenLoadingTrace(screenLoadingTrace); + + final actualUiTrace = ScreenLoadingManager.I.currentUiTrace!; + + expect( + ScreenLoadingManager.I.currentScreenLoadingTrace, + equals(screenLoadingTrace), + ); + expect( + actualUiTrace.didStartScreenLoading, + isTrue, + ); + }); + + test( + '[startScreenLoadingTrace] should not start screen loading trace when screen loading trace does not matches UI trace matching screen name', + () async { + const isSameScreen = false; + const matchingScreenName = 'matching_screen_name'; + mScreenLoadingManager = ScreenLoadingManagerNoResets.init(); + when(FlagsConfig.screenLoading.isEnabled()).thenAnswer((_) async => true); + when(IBGBuildInfo.I.isIOS).thenReturn(false); + when(mDateTime.now()).thenReturn(time); + + // Don't match on matching screen name + when( + RouteMatcher.I.match( + routePath: screenName, + actualPath: matchingScreenName, + ), + ).thenReturn(isSameScreen); + + when( + RouteMatcher.I.match( + routePath: screenName, + actualPath: screenName, + ), + ).thenReturn(!isSameScreen); + + ScreenLoadingManager.I.currentUiTrace = uiTrace.copyWith( + matchingScreenName: matchingScreenName, + ); + + await ScreenLoadingManager.I.startScreenLoadingTrace(screenLoadingTrace); + + final actualUiTrace = ScreenLoadingManager.I.currentUiTrace!; + + expect( + ScreenLoadingManager.I.currentScreenLoadingTrace, + isNull, + ); + expect( + actualUiTrace.didStartScreenLoading, + isFalse, + ); + }); }); group('reportScreenLoading tests', () { @@ -421,7 +552,7 @@ void main() { setUp(() { time = DateTime.now(); traceId = time.millisecondsSinceEpoch; - uiTrace = UiTrace(screenName, traceId: traceId); + uiTrace = UiTrace(screenName: screenName, traceId: traceId); mScreenLoadingManager.currentUiTrace = uiTrace; when(mDateTime.now()).thenReturn(time); screenLoadingTrace = ScreenLoadingTrace( @@ -719,7 +850,7 @@ void main() { setUp(() { time = DateTime.now(); traceId = time.millisecondsSinceEpoch; - uiTrace = UiTrace(screenName, traceId: traceId); + uiTrace = UiTrace(screenName: screenName, traceId: traceId); duration = 1000; extendedMonotonic = 500; endTime = time.add(Duration(microseconds: duration ?? 0)); diff --git a/test/utils/screen_name_masker_test.dart b/test/utils/screen_name_masker_test.dart new file mode 100644 index 000000000..bfdd81c98 --- /dev/null +++ b/test/utils/screen_name_masker_test.dart @@ -0,0 +1,77 @@ +import 'package:flutter/widgets.dart'; +import 'package:flutter_test/flutter_test.dart'; +import 'package:instabug_flutter/src/utils/screen_name_masker.dart'; + +void main() { + TestWidgetsFlutterBinding.ensureInitialized(); + WidgetsFlutterBinding.ensureInitialized(); + + setUp(() { + ScreenNameMasker.I.setMaskingCallback(null); + }); + + test('[mask] should return same screen name when no masking callback is set', + () { + const screen = 'home'; + + final result = ScreenNameMasker.I.mask(screen); + + expect(result, equals(screen)); + }); + + test( + '[mask] should mask screen name when [setMaskingCallback] has set a callback', + () { + const screen = '/documents/314159265'; + const masked = '/documents/REDACTED'; + + ScreenNameMasker.I.setMaskingCallback((screen) { + if (screen.startsWith('/documents/')) { + return masked; + } + + return screen; + }); + + final result = ScreenNameMasker.I.mask(screen); + + expect(result, equals(masked)); + }); + + test('[mask] should fallback to "N/A" when callback returns an empty string', + () { + const fallback = 'N/A'; + const screen = '/documents/314159265'; + const masked = ''; + + ScreenNameMasker.I.setMaskingCallback((screen) { + if (screen.startsWith('/documents/')) { + return masked; + } + + return screen; + }); + + final result = ScreenNameMasker.I.mask(screen); + + expect(result, equals(fallback)); + }); + + test('[mask] should trim masked screen name', () { + const screen = '/documents/314159265'; + const masked = ' /documents/REDACTED '; + const expected = '/documents/REDACTED'; + + ScreenNameMasker.I.setMaskingCallback((screen) { + if (screen.startsWith('/documents/')) { + return masked; + } + + return screen; + }); + + final result = ScreenNameMasker.I.mask(screen); + + expect(result, equals(expected)); + }); +} From 4be7117a7a675a3a2346c75f7e9f19626ebb0aea Mon Sep 17 00:00:00 2001 From: Ahmed Elrefaey <68241710+a7medev@users.noreply.github.com> Date: Fri, 13 Sep 2024 14:27:11 +0300 Subject: [PATCH 3/8] fix: use a fallback for empty screen names (#505) * fix: fallback to N/A when for empty screen names * test: add test for empty screen fallback * chore: add changelog item --- CHANGELOG.md | 4 ++++ .../utils/instabug_navigator_observer.dart | 6 +++++- lib/src/utils/repro_steps_constants.dart | 3 +++ .../screen_loading_manager.dart | 5 ++--- lib/src/utils/screen_name_masker.dart | 5 ++--- .../instabug_navigator_observer_test.dart | 19 +++++++++++++++++++ 6 files changed, 35 insertions(+), 7 deletions(-) create mode 100644 lib/src/utils/repro_steps_constants.dart diff --git a/CHANGELOG.md b/CHANGELOG.md index 34e09b698..49278fba8 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -6,6 +6,10 @@ - Add support for masking screen names captured by Instabug through the `Instabug.setScreenNameMaskingCallback` API ([#500](https://github.com/Instabug/Instabug-Flutter/pull/500)). +### Fixed + +- Fixed an issue with empty screen names captured in `InstabugNavigatorObserver` and fallback to `N/A` when the screen name is empty ([#505](https://github.com/Instabug/Instabug-Flutter/pull/505)), closes [#504](https://github.com/Instabug/Instabug-Flutter/issues/504). + ## [13.3.0](https://github.com/Instabug/Instabug-Flutter/compare/v13.2.0...v13.3.0) (August 5, 2024) ### Added diff --git a/lib/src/utils/instabug_navigator_observer.dart b/lib/src/utils/instabug_navigator_observer.dart index 57550765c..d9d6b02db 100644 --- a/lib/src/utils/instabug_navigator_observer.dart +++ b/lib/src/utils/instabug_navigator_observer.dart @@ -3,6 +3,7 @@ import 'package:instabug_flutter/instabug_flutter.dart'; import 'package:instabug_flutter/src/models/instabug_route.dart'; import 'package:instabug_flutter/src/modules/instabug.dart'; import 'package:instabug_flutter/src/utils/instabug_logger.dart'; +import 'package:instabug_flutter/src/utils/repro_steps_constants.dart'; import 'package:instabug_flutter/src/utils/screen_loading/screen_loading_manager.dart'; import 'package:instabug_flutter/src/utils/screen_name_masker.dart'; @@ -11,7 +12,10 @@ class InstabugNavigatorObserver extends NavigatorObserver { void screenChanged(Route newRoute) { try { - final screenName = newRoute.settings.name.toString(); + final rawScreenName = newRoute.settings.name.toString().trim(); + final screenName = rawScreenName.isEmpty + ? ReproStepsConstants.emptyScreenFallback + : rawScreenName; final maskedScreenName = ScreenNameMasker.I.mask(screenName); final route = InstabugRoute( diff --git a/lib/src/utils/repro_steps_constants.dart b/lib/src/utils/repro_steps_constants.dart new file mode 100644 index 000000000..c26dc205c --- /dev/null +++ b/lib/src/utils/repro_steps_constants.dart @@ -0,0 +1,3 @@ +class ReproStepsConstants { + static const emptyScreenFallback = 'N/A'; +} diff --git a/lib/src/utils/screen_loading/screen_loading_manager.dart b/lib/src/utils/screen_loading/screen_loading_manager.dart index e95543ee9..816ffebef 100644 --- a/lib/src/utils/screen_loading/screen_loading_manager.dart +++ b/lib/src/utils/screen_loading/screen_loading_manager.dart @@ -110,16 +110,15 @@ class ScreenLoadingManager { @internal String sanitizeScreenName(String screenName) { const characterToBeRemoved = '/'; - final lastIndex = screenName.length - 1; var sanitizedScreenName = screenName; if (screenName == characterToBeRemoved) { return 'ROOT_PAGE'; } - if (screenName[0] == characterToBeRemoved) { + if (screenName.startsWith(characterToBeRemoved)) { sanitizedScreenName = sanitizedScreenName.substring(1); } - if (screenName[lastIndex] == characterToBeRemoved) { + if (screenName.endsWith(characterToBeRemoved)) { sanitizedScreenName = sanitizedScreenName.substring(0, sanitizedScreenName.length - 1); } diff --git a/lib/src/utils/screen_name_masker.dart b/lib/src/utils/screen_name_masker.dart index ef228697f..1c2c4a972 100644 --- a/lib/src/utils/screen_name_masker.dart +++ b/lib/src/utils/screen_name_masker.dart @@ -1,4 +1,5 @@ import 'package:flutter/material.dart'; +import 'package:instabug_flutter/src/utils/repro_steps_constants.dart'; typedef ScreenNameMaskingCallback = String Function(String screen); @@ -14,8 +15,6 @@ class ScreenNameMasker { /// Shorthand for [instance] static ScreenNameMasker get I => instance; - static const emptyScreenNameFallback = "N/A"; - ScreenNameMaskingCallback? _screenNameMaskingCallback; @visibleForTesting @@ -37,7 +36,7 @@ class ScreenNameMasker { final maskedScreen = _screenNameMaskingCallback!(screen).trim(); if (maskedScreen.isEmpty) { - return emptyScreenNameFallback; + return ReproStepsConstants.emptyScreenFallback; } return maskedScreen; diff --git a/test/utils/instabug_navigator_observer_test.dart b/test/utils/instabug_navigator_observer_test.dart index f7a6f6a7f..ebf541137 100644 --- a/test/utils/instabug_navigator_observer_test.dart +++ b/test/utils/instabug_navigator_observer_test.dart @@ -92,6 +92,25 @@ void main() { }); }); + test('should fallback to "N/A" when the screen name is empty', () { + fakeAsync((async) { + final route = createRoute(''); + const fallback = 'N/A'; + + observer.didPush(route, previousRoute); + + async.elapse(const Duration(milliseconds: 1000)); + + verify( + mScreenLoadingManager.startUiTrace(fallback, fallback), + ).called(1); + + verify( + mHost.reportScreenChange(fallback), + ).called(1); + }); + }); + test('should mask screen name when masking callback is set', () { const maskedScreen = 'maskedScreen'; From a0a154ded78d4240c2141e0e6655145b93971d22 Mon Sep 17 00:00:00 2001 From: Ahmed Elrefaey <68241710+a7medev@users.noreply.github.com> Date: Fri, 13 Sep 2024 14:48:35 +0300 Subject: [PATCH 4/8] release: v13.4.0 (#506) * chore(ios): bump sdk to v13.4.0 * chore(android): bump sdk to v13.4.0 * chore: bump sdk version to v13.4.0 * chore: update changelog * fix(ios): update native network logging method signature (#507) * fix(ios): update native network logging method signature * feat(example): add network logging in example --------- Co-authored-by: Ahmed alaa --- CHANGELOG.md | 7 ++- android/build.gradle | 4 +- example/ios/InstabugTests/InstabugApiTests.m | 8 ++- example/ios/Podfile.lock | 10 ++-- example/ios/Runner.xcodeproj/project.pbxproj | 2 - example/lib/main.dart | 2 +- .../lib/src/components/network_content.dart | 2 + example/pubspec.lock | 10 +++- example/pubspec.yaml | 5 ++ ios/Classes/Modules/InstabugApi.m | 51 +++++++++---------- ios/Classes/Util/IBGNetworkLogger+CP.h | 22 ++++++++ ios/instabug_flutter.podspec | 4 +- pubspec.yaml | 2 +- 13 files changed, 84 insertions(+), 45 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 49278fba8..f14747606 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,11 +1,16 @@ # Changelog -## [Unreleased](https://github.com/Instabug/Instabug-Flutter/compare/v13.3.0...dev) +## [13.4.0](https://github.com/Instabug/Instabug-Flutter/compare/v13.3.0...v13.4.0) (September 15, 2024) ### Added - Add support for masking screen names captured by Instabug through the `Instabug.setScreenNameMaskingCallback` API ([#500](https://github.com/Instabug/Instabug-Flutter/pull/500)). +### Changed + +- Bump Instabug Android SDK to v13.4.0 ([#506](https://github.com/Instabug/Instabug-Flutter/pull/506)). [See release notes](https://github.com/Instabug/Instabug-Android/releases/tag/v13.4.0). +- Bump Instabug iOS SDK to v13.4.0 ([#506](https://github.com/Instabug/Instabug-Flutter/pull/506)). [See release notes](https://github.com/Instabug/Instabug-iOS/releases/tag/13.4.0). + ### Fixed - Fixed an issue with empty screen names captured in `InstabugNavigatorObserver` and fallback to `N/A` when the screen name is empty ([#505](https://github.com/Instabug/Instabug-Flutter/pull/505)), closes [#504](https://github.com/Instabug/Instabug-Flutter/issues/504). diff --git a/android/build.gradle b/android/build.gradle index f16d516e5..cc2b1d037 100644 --- a/android/build.gradle +++ b/android/build.gradle @@ -1,5 +1,5 @@ group 'com.instabug.flutter' -version '13.3.0' +version '13.4.0' buildscript { repositories { @@ -41,7 +41,7 @@ android { } dependencies { - api 'com.instabug.library:instabug:13.3.0' + api 'com.instabug.library:instabug:13.4.0' testImplementation 'junit:junit:4.13.2' testImplementation "org.mockito:mockito-inline:3.12.1" diff --git a/example/ios/InstabugTests/InstabugApiTests.m b/example/ios/InstabugTests/InstabugApiTests.m index 886e22ece..3e83aec1c 100644 --- a/example/ios/InstabugTests/InstabugApiTests.m +++ b/example/ios/InstabugTests/InstabugApiTests.m @@ -4,7 +4,7 @@ #import "InstabugApi.h" #import "Instabug/Instabug.h" #import "Util/Instabug+Test.h" -#import "Util/IBGNetworkLogger+Test.h" +#import "IBGNetworkLogger+CP.h" #import "Flutter/Flutter.h" @interface InstabugTests : XCTestCase @@ -435,7 +435,11 @@ - (void)testNetworkLog { duration:duration.integerValue gqlQueryName:nil serverErrorMessage:nil - ]); + isW3cCaughted:nil + partialID:nil + timestamp:nil + generatedW3CTraceparent:nil + caughtedW3CTraceparent:nil]); } - (void)testWillRedirectToAppStore { diff --git a/example/ios/Podfile.lock b/example/ios/Podfile.lock index 17a3f845c..8f27d4856 100644 --- a/example/ios/Podfile.lock +++ b/example/ios/Podfile.lock @@ -1,9 +1,9 @@ PODS: - Flutter (1.0.0) - - Instabug (13.3.0) - - instabug_flutter (13.3.0): + - Instabug (13.4.0) + - instabug_flutter (13.4.0): - Flutter - - Instabug (= 13.3.0) + - Instabug (= 13.4.0) - OCMock (3.6) DEPENDENCIES: @@ -24,8 +24,8 @@ EXTERNAL SOURCES: SPEC CHECKSUMS: Flutter: e0871f40cf51350855a761d2e70bf5af5b9b5de7 - Instabug: 4f26295103a330ec0236918359eef7ccaa74e2fa - instabug_flutter: 6be22be13b3dda72b293151d2c009952bb20b940 + Instabug: 183aa1e038d01ddc000f06835f46a9ea1f6c992b + instabug_flutter: 2cae2dc6f1d9e2442154886b3740a47f3429d239 OCMock: 5ea90566be239f179ba766fd9fbae5885040b992 PODFILE CHECKSUM: 8f7552fd115ace1988c3db54a69e4a123c448f84 diff --git a/example/ios/Runner.xcodeproj/project.pbxproj b/example/ios/Runner.xcodeproj/project.pbxproj index f40f40e79..858ba01e5 100644 --- a/example/ios/Runner.xcodeproj/project.pbxproj +++ b/example/ios/Runner.xcodeproj/project.pbxproj @@ -87,7 +87,6 @@ CC359DB82937720C0067A924 /* ApmApiTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = ApmApiTests.m; sourceTree = ""; }; CC3D69E6293F47FC000DCE54 /* ArgsRegistryTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = ArgsRegistryTests.m; sourceTree = ""; }; CC78720E293CA8EE008CB2A5 /* Instabug+Test.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "Instabug+Test.h"; sourceTree = ""; }; - CC787211293CAB28008CB2A5 /* IBGNetworkLogger+Test.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "IBGNetworkLogger+Test.h"; sourceTree = ""; }; CC9925D1293DEB0B001FD3EE /* CrashReportingApiTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = CrashReportingApiTests.m; sourceTree = ""; }; CC9925D4293DF534001FD3EE /* FeatureRequestsApiTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = FeatureRequestsApiTests.m; sourceTree = ""; }; CC9925D6293DFB03001FD3EE /* InstabugLogApiTests.m */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.objc; path = InstabugLogApiTests.m; sourceTree = ""; }; @@ -216,7 +215,6 @@ children = ( BE26C80C2BD55575009FECCF /* IBGCrashReporting+CP.h */, CC78720E293CA8EE008CB2A5 /* Instabug+Test.h */, - CC787211293CAB28008CB2A5 /* IBGNetworkLogger+Test.h */, CC198C62293E2392007077C8 /* IBGSurvey+Test.h */, ); path = Util; diff --git a/example/lib/main.dart b/example/lib/main.dart index 7749ed02d..cda7ff3ec 100644 --- a/example/lib/main.dart +++ b/example/lib/main.dart @@ -2,10 +2,10 @@ import 'dart:async'; import 'dart:developer'; import 'dart:io'; import 'dart:convert'; -import 'package:http/http.dart' as http; import 'package:flutter/material.dart'; import 'package:instabug_flutter/instabug_flutter.dart'; +import 'package:instabug_http_client/instabug_http_client.dart'; import 'package:instabug_flutter_example/src/app_routes.dart'; import 'package:instabug_flutter_example/src/widget/nested_view.dart'; diff --git a/example/lib/src/components/network_content.dart b/example/lib/src/components/network_content.dart index c364ff89d..f55af5c95 100644 --- a/example/lib/src/components/network_content.dart +++ b/example/lib/src/components/network_content.dart @@ -10,6 +10,8 @@ class NetworkContent extends StatefulWidget { } class _NetworkContentState extends State { + final http = InstabugHttpClient(); + final endpointUrlController = TextEditingController(); @override diff --git a/example/pubspec.lock b/example/pubspec.lock index 5b56a50f9..31cb6f5dd 100644 --- a/example/pubspec.lock +++ b/example/pubspec.lock @@ -115,7 +115,15 @@ packages: path: ".." relative: true source: path - version: "13.3.0" + version: "13.4.0" + instabug_http_client: + dependency: "direct main" + description: + name: instabug_http_client + sha256: "7d52803c0dd639f6dddbe07333418eb251ae02f3f9f4d30402517533ca692784" + url: "https://pub.dev" + source: hosted + version: "2.4.0" leak_tracker: dependency: transitive description: diff --git a/example/pubspec.yaml b/example/pubspec.yaml index 69fed929d..7f3e9e622 100644 --- a/example/pubspec.yaml +++ b/example/pubspec.yaml @@ -26,6 +26,7 @@ dependencies: http: ^0.13.0 instabug_flutter: path: ../ + instabug_http_client: ^2.4.0 dev_dependencies: espresso: 0.2.0+5 @@ -35,6 +36,10 @@ dev_dependencies: sdk: flutter flutter_lints: 1.0.4 +dependency_overrides: + instabug_flutter: + path: ../ + # For information on the generic Dart part of this file, see the # following page: https://dart.dev/tools/pub/pubspec diff --git a/ios/Classes/Modules/InstabugApi.m b/ios/Classes/Modules/InstabugApi.m index 44f356685..11ea09354 100644 --- a/ios/Classes/Modules/InstabugApi.m +++ b/ios/Classes/Modules/InstabugApi.m @@ -263,10 +263,10 @@ - (void)networkLogData:(NSDictionary *)data error:(FlutterError NSString *method = data[@"method"]; NSString *requestBody = data[@"requestBody"]; NSString *responseBody = data[@"responseBody"]; - int32_t responseCode = [data[@"responseCode"] integerValue]; + int32_t responseCode = (int32_t) [data[@"responseCode"] integerValue]; int64_t requestBodySize = [data[@"requestBodySize"] integerValue]; int64_t responseBodySize = [data[@"responseBodySize"] integerValue]; - int32_t errorCode = [data[@"errorCode"] integerValue]; + int32_t errorCode = (int32_t) [data[@"errorCode"] integerValue]; NSString *errorDomain = data[@"errorDomain"]; NSDictionary *requestHeaders = data[@"requestHeaders"]; if ([requestHeaders count] == 0) { @@ -286,32 +286,27 @@ - (void)networkLogData:(NSDictionary *)data error:(FlutterError serverErrorMessage = data[@"serverErrorMessage"]; } - SEL networkLogSEL = NSSelectorFromString(@"addNetworkLogWithUrl:method:requestBody:requestBodySize:responseBody:responseBodySize:responseCode:requestHeaders:responseHeaders:contentType:errorDomain:errorCode:startTime:duration:gqlQueryName:serverErrorMessage:"); - - if ([[IBGNetworkLogger class] respondsToSelector:networkLogSEL]) { - NSInvocation *inv = [NSInvocation invocationWithMethodSignature:[[IBGNetworkLogger class] methodSignatureForSelector:networkLogSEL]]; - [inv setSelector:networkLogSEL]; - [inv setTarget:[IBGNetworkLogger class]]; - - [inv setArgument:&(url) atIndex:2]; - [inv setArgument:&(method) atIndex:3]; - [inv setArgument:&(requestBody) atIndex:4]; - [inv setArgument:&(requestBodySize) atIndex:5]; - [inv setArgument:&(responseBody) atIndex:6]; - [inv setArgument:&(responseBodySize) atIndex:7]; - [inv setArgument:&(responseCode) atIndex:8]; - [inv setArgument:&(requestHeaders) atIndex:9]; - [inv setArgument:&(responseHeaders) atIndex:10]; - [inv setArgument:&(contentType) atIndex:11]; - [inv setArgument:&(errorDomain) atIndex:12]; - [inv setArgument:&(errorCode) atIndex:13]; - [inv setArgument:&(startTime) atIndex:14]; - [inv setArgument:&(duration) atIndex:15]; - [inv setArgument:&(gqlQueryName) atIndex:16]; - [inv setArgument:&(serverErrorMessage) atIndex:17]; - - [inv invoke]; - } + [IBGNetworkLogger addNetworkLogWithUrl:url + method:method + requestBody:requestBody + requestBodySize:requestBodySize + responseBody:responseBody + responseBodySize:responseBodySize + responseCode:responseCode + requestHeaders:requestHeaders + responseHeaders:responseHeaders + contentType:contentType + errorDomain:errorDomain + errorCode:errorCode + startTime:startTime + duration:duration + gqlQueryName:gqlQueryName + serverErrorMessage:serverErrorMessage + isW3cCaughted:nil + partialID:nil + timestamp:nil + generatedW3CTraceparent:nil + caughtedW3CTraceparent:nil]; } - (void)willRedirectToStoreWithError:(FlutterError * _Nullable __autoreleasing *)error { diff --git a/ios/Classes/Util/IBGNetworkLogger+CP.h b/ios/Classes/Util/IBGNetworkLogger+CP.h index ae5d32d66..764524fb2 100644 --- a/ios/Classes/Util/IBGNetworkLogger+CP.h +++ b/ios/Classes/Util/IBGNetworkLogger+CP.h @@ -6,6 +6,28 @@ NS_ASSUME_NONNULL_BEGIN + (void)disableAutomaticCapturingOfNetworkLogs; ++ (void)addNetworkLogWithUrl:(NSString *_Nonnull)url + method:(NSString *_Nonnull)method + requestBody:(NSString *_Nonnull)request + requestBodySize:(int64_t)requestBodySize + responseBody:(NSString *_Nonnull)response + responseBodySize:(int64_t)responseBodySize + responseCode:(int32_t)code + requestHeaders:(NSDictionary *_Nonnull)requestHeaders + responseHeaders:(NSDictionary *_Nonnull)responseHeaders + contentType:(NSString *_Nonnull)contentType + errorDomain:(NSString *_Nullable)errorDomain + errorCode:(int32_t)errorCode + startTime:(int64_t)startTime + duration:(int64_t) duration + gqlQueryName:(NSString * _Nullable)gqlQueryName + serverErrorMessage:(NSString * _Nullable)serverErrorMessage + isW3cCaughted:(NSNumber * _Nullable)isW3cCaughted + partialID:(NSNumber * _Nullable)partialID + timestamp:(NSNumber * _Nullable)timestamp + generatedW3CTraceparent:(NSString * _Nullable)generatedW3CTraceparent + caughtedW3CTraceparent:(NSString * _Nullable)caughtedW3CTraceparent; + @end NS_ASSUME_NONNULL_END diff --git a/ios/instabug_flutter.podspec b/ios/instabug_flutter.podspec index 31f7d8782..fa08814fd 100644 --- a/ios/instabug_flutter.podspec +++ b/ios/instabug_flutter.podspec @@ -1,6 +1,6 @@ Pod::Spec.new do |s| s.name = 'instabug_flutter' - s.version = '13.3.0' + s.version = '13.4.0' s.summary = 'Flutter plugin for integrating the Instabug SDK.' s.author = 'Instabug' s.homepage = 'https://www.instabug.com/platforms/flutter' @@ -17,6 +17,6 @@ Pod::Spec.new do |s| s.pod_target_xcconfig = { 'OTHER_LDFLAGS' => '-framework "Flutter" -framework "Instabug"'} s.dependency 'Flutter' - s.dependency 'Instabug', '13.3.0' + s.dependency 'Instabug', '13.4.0' end diff --git a/pubspec.yaml b/pubspec.yaml index 466933d8c..ba6dd0be2 100644 --- a/pubspec.yaml +++ b/pubspec.yaml @@ -1,5 +1,5 @@ name: instabug_flutter -version: 13.3.0 +version: 13.4.0 description: >- Instabug empowers mobile teams to monitor, prioritize, and debug performance and stability issues throughout the app development lifecycle. From d0d9720c3619b1587bdae3b254ffc0768c98682a Mon Sep 17 00:00:00 2001 From: ahmed alaa <154802748+ahmedAlaaInstabug@users.noreply.github.com> Date: Wed, 18 Sep 2024 12:59:28 +0300 Subject: [PATCH 5/8] chore(android): bump sdk to v13.4.1 (#509) --- CHANGELOG.md | 2 +- android/build.gradle | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index f14747606..d6bc190aa 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -8,7 +8,7 @@ ### Changed -- Bump Instabug Android SDK to v13.4.0 ([#506](https://github.com/Instabug/Instabug-Flutter/pull/506)). [See release notes](https://github.com/Instabug/Instabug-Android/releases/tag/v13.4.0). +- Bump Instabug Android SDK to v13.4.1 ([#509](https://github.com/Instabug/Instabug-Flutter/pull/509)). [See release notes](https://github.com/Instabug/Instabug-Android/releases/tag/v13.4.0). - Bump Instabug iOS SDK to v13.4.0 ([#506](https://github.com/Instabug/Instabug-Flutter/pull/506)). [See release notes](https://github.com/Instabug/Instabug-iOS/releases/tag/13.4.0). ### Fixed diff --git a/android/build.gradle b/android/build.gradle index cc2b1d037..92654b91b 100644 --- a/android/build.gradle +++ b/android/build.gradle @@ -41,7 +41,7 @@ android { } dependencies { - api 'com.instabug.library:instabug:13.4.0' + api 'com.instabug.library:instabug:13.4.1' testImplementation 'junit:junit:4.13.2' testImplementation "org.mockito:mockito-inline:3.12.1" From 1ab49ceb2257f298debd49869ffea3e22f2e2137 Mon Sep 17 00:00:00 2001 From: Ahmed Elrefaey <68241710+a7medev@users.noreply.github.com> Date: Sun, 22 Sep 2024 15:33:08 +0300 Subject: [PATCH 6/8] chore(ios): bump sdk to v13.4.1 (#510) * chore(ios): bump sdk to v13.4.1 * chore: update changelog --- CHANGELOG.md | 2 +- example/ios/Podfile.lock | 8 ++++---- ios/instabug_flutter.podspec | 2 +- 3 files changed, 6 insertions(+), 6 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index d6bc190aa..f03a1e4a2 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -9,7 +9,7 @@ ### Changed - Bump Instabug Android SDK to v13.4.1 ([#509](https://github.com/Instabug/Instabug-Flutter/pull/509)). [See release notes](https://github.com/Instabug/Instabug-Android/releases/tag/v13.4.0). -- Bump Instabug iOS SDK to v13.4.0 ([#506](https://github.com/Instabug/Instabug-Flutter/pull/506)). [See release notes](https://github.com/Instabug/Instabug-iOS/releases/tag/13.4.0). +- Bump Instabug iOS SDK to v13.4.1 ([#510](https://github.com/Instabug/Instabug-Flutter/pull/510)). See release notes for [13.4.0](https://github.com/Instabug/Instabug-iOS/releases/tag/13.4.0) and [13.4.1](https://github.com/Instabug/Instabug-iOS/releases/tag/13.4.1). ### Fixed diff --git a/example/ios/Podfile.lock b/example/ios/Podfile.lock index 8f27d4856..34cc30d6a 100644 --- a/example/ios/Podfile.lock +++ b/example/ios/Podfile.lock @@ -1,9 +1,9 @@ PODS: - Flutter (1.0.0) - - Instabug (13.4.0) + - Instabug (13.4.1) - instabug_flutter (13.4.0): - Flutter - - Instabug (= 13.4.0) + - Instabug (= 13.4.1) - OCMock (3.6) DEPENDENCIES: @@ -24,8 +24,8 @@ EXTERNAL SOURCES: SPEC CHECKSUMS: Flutter: e0871f40cf51350855a761d2e70bf5af5b9b5de7 - Instabug: 183aa1e038d01ddc000f06835f46a9ea1f6c992b - instabug_flutter: 2cae2dc6f1d9e2442154886b3740a47f3429d239 + Instabug: 896a318fb64b282832e0eda6fce489dac74b5d48 + instabug_flutter: b9f4503c3788c7346649bfe3396d6ca6e9711e96 OCMock: 5ea90566be239f179ba766fd9fbae5885040b992 PODFILE CHECKSUM: 8f7552fd115ace1988c3db54a69e4a123c448f84 diff --git a/ios/instabug_flutter.podspec b/ios/instabug_flutter.podspec index fa08814fd..e57559048 100644 --- a/ios/instabug_flutter.podspec +++ b/ios/instabug_flutter.podspec @@ -17,6 +17,6 @@ Pod::Spec.new do |s| s.pod_target_xcconfig = { 'OTHER_LDFLAGS' => '-framework "Flutter" -framework "Instabug"'} s.dependency 'Flutter' - s.dependency 'Instabug', '13.4.0' + s.dependency 'Instabug', '13.4.1' end From 3b3fb1302c167b53e31e95c20387f696ece0f5f5 Mon Sep 17 00:00:00 2001 From: Ahmed Elrefaey <68241710+a7medev@users.noreply.github.com> Date: Sun, 29 Sep 2024 14:50:47 +0300 Subject: [PATCH 7/8] chore: update release changelog (#514) --- CHANGELOG.md | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index f03a1e4a2..ff35d8df1 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,6 +1,6 @@ # Changelog -## [13.4.0](https://github.com/Instabug/Instabug-Flutter/compare/v13.3.0...v13.4.0) (September 15, 2024) +## [13.4.0](https://github.com/Instabug/Instabug-Flutter/compare/v13.3.0...v13.4.0) (September 29, 2024) ### Added @@ -8,7 +8,7 @@ ### Changed -- Bump Instabug Android SDK to v13.4.1 ([#509](https://github.com/Instabug/Instabug-Flutter/pull/509)). [See release notes](https://github.com/Instabug/Instabug-Android/releases/tag/v13.4.0). +- Bump Instabug Android SDK to v13.4.1 ([#509](https://github.com/Instabug/Instabug-Flutter/pull/509)). See release notes for [13.4.0](https://github.com/Instabug/Instabug-Android/releases/tag/v13.4.0) and [13.4.1](https://github.com/Instabug/Instabug-Android/releases/tag/v13.4.1). - Bump Instabug iOS SDK to v13.4.1 ([#510](https://github.com/Instabug/Instabug-Flutter/pull/510)). See release notes for [13.4.0](https://github.com/Instabug/Instabug-iOS/releases/tag/13.4.0) and [13.4.1](https://github.com/Instabug/Instabug-iOS/releases/tag/13.4.1). ### Fixed @@ -45,7 +45,7 @@ - Bump Instabug Android SDK to v13.2.0 ([#482](https://github.com/Instabug/Instabug-Flutter/pull/482)). [See release notes](https://github.com/Instabug/Instabug-Android/releases/tag/v13.2.0). - Bump Instabug iOS SDK to v13.2.0 ([#483](https://github.com/Instabug/Instabug-Flutter/pull/483)). [See release notes](https://github.com/Instabug/Instabug-iOS/releases/tag/13.2.0). -## [13.1.1](https://github.com/Instabug/Instabug-Flutter/compare/v13.0.0...dev) (Jun,11 2024) +## [13.1.1](https://github.com/Instabug/Instabug-Flutter/compare/v13.0.0...v13.1.1) (June 11, 2024) ### Added From 7ae921c543076ef8d999147da8f225cc774b9ecc Mon Sep 17 00:00:00 2001 From: Ahmed Elrefaey <68241710+a7medev@users.noreply.github.com> Date: Sun, 29 Sep 2024 16:07:13 +0300 Subject: [PATCH 8/8] chore(ios): bump sdk to v13.4.2 (#515) * chore(ios): bump sdk to v13.4.2 * chore: update changelog --- CHANGELOG.md | 2 +- example/ios/Podfile.lock | 8 ++++---- ios/instabug_flutter.podspec | 2 +- 3 files changed, 6 insertions(+), 6 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index ff35d8df1..418e317c8 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -9,7 +9,7 @@ ### Changed - Bump Instabug Android SDK to v13.4.1 ([#509](https://github.com/Instabug/Instabug-Flutter/pull/509)). See release notes for [13.4.0](https://github.com/Instabug/Instabug-Android/releases/tag/v13.4.0) and [13.4.1](https://github.com/Instabug/Instabug-Android/releases/tag/v13.4.1). -- Bump Instabug iOS SDK to v13.4.1 ([#510](https://github.com/Instabug/Instabug-Flutter/pull/510)). See release notes for [13.4.0](https://github.com/Instabug/Instabug-iOS/releases/tag/13.4.0) and [13.4.1](https://github.com/Instabug/Instabug-iOS/releases/tag/13.4.1). +- Bump Instabug iOS SDK to v13.4.2 ([#515](https://github.com/Instabug/Instabug-Flutter/pull/515)). See release notes for [13.4.0](https://github.com/Instabug/Instabug-iOS/releases/tag/13.4.0), [13.4.1](https://github.com/Instabug/Instabug-iOS/releases/tag/13.4.1) and [13.4.2](https://github.com/Instabug/Instabug-iOS/releases/tag/13.4.2). ### Fixed diff --git a/example/ios/Podfile.lock b/example/ios/Podfile.lock index 34cc30d6a..b0ee46890 100644 --- a/example/ios/Podfile.lock +++ b/example/ios/Podfile.lock @@ -1,9 +1,9 @@ PODS: - Flutter (1.0.0) - - Instabug (13.4.1) + - Instabug (13.4.2) - instabug_flutter (13.4.0): - Flutter - - Instabug (= 13.4.1) + - Instabug (= 13.4.2) - OCMock (3.6) DEPENDENCIES: @@ -24,8 +24,8 @@ EXTERNAL SOURCES: SPEC CHECKSUMS: Flutter: e0871f40cf51350855a761d2e70bf5af5b9b5de7 - Instabug: 896a318fb64b282832e0eda6fce489dac74b5d48 - instabug_flutter: b9f4503c3788c7346649bfe3396d6ca6e9711e96 + Instabug: 7a71890217b97b1e32dbca96661845396b66da2f + instabug_flutter: a2df87e3d4d9e410785e0b1ffef4bc64d1f4b787 OCMock: 5ea90566be239f179ba766fd9fbae5885040b992 PODFILE CHECKSUM: 8f7552fd115ace1988c3db54a69e4a123c448f84 diff --git a/ios/instabug_flutter.podspec b/ios/instabug_flutter.podspec index e57559048..1d362767f 100644 --- a/ios/instabug_flutter.podspec +++ b/ios/instabug_flutter.podspec @@ -17,6 +17,6 @@ Pod::Spec.new do |s| s.pod_target_xcconfig = { 'OTHER_LDFLAGS' => '-framework "Flutter" -framework "Instabug"'} s.dependency 'Flutter' - s.dependency 'Instabug', '13.4.1' + s.dependency 'Instabug', '13.4.2' end