diff --git a/.gitignore b/.gitignore index 829b1e5..a9a49f3 100644 --- a/.gitignore +++ b/.gitignore @@ -38,4 +38,7 @@ build/ launch.json #Coverage -coverage/ \ No newline at end of file +coverage/ + +# FVM Version Cache +.fvm/ \ No newline at end of file diff --git a/packages/core/lib/analytics_pigeon.dart b/packages/core/lib/analytics_pigeon.dart index 8824b36..93fb7db 100644 --- a/packages/core/lib/analytics_pigeon.dart +++ b/packages/core/lib/analytics_pigeon.dart @@ -1,3 +1,4 @@ + import 'package:segment_analytics/analytics_platform_interface.dart'; import 'package:flutter/services.dart'; @@ -7,11 +8,13 @@ import 'native_context.dart'; class AnalyticsPlatformImpl extends AnalyticsPlatform { static const EventChannel _eChannel = EventChannel('analytics/deep_link_events'); - final NativeContextApi _api = NativeContextApi(); + NativeContextApi api; + + AnalyticsPlatformImpl({NativeContextApi? api}) : api = api ?? NativeContextApi(); @override Future getContext({bool collectDeviceId = false}) { - return _api.getContext(collectDeviceId); + return api.getContext(collectDeviceId); } @override diff --git a/packages/core/lib/flush_policies/count_flush_policy.dart b/packages/core/lib/flush_policies/count_flush_policy.dart index 7807b54..f5d771f 100644 --- a/packages/core/lib/flush_policies/count_flush_policy.dart +++ b/packages/core/lib/flush_policies/count_flush_policy.dart @@ -1,3 +1,4 @@ +import 'package:flutter/foundation.dart'; import 'package:segment_analytics/event.dart'; import 'package:segment_analytics/flush_policies/flush_policy.dart'; @@ -7,6 +8,9 @@ class CountFlushPolicy extends FlushPolicy { CountFlushPolicy(this._flushAt, {int? count}) : _count = count ?? 0; + @visibleForTesting + int get count => _count; + @override void start() { _count = 0; diff --git a/packages/core/lib/plugins/segment_destination.dart b/packages/core/lib/plugins/segment_destination.dart index 57cb869..501d213 100644 --- a/packages/core/lib/plugins/segment_destination.dart +++ b/packages/core/lib/plugins/segment_destination.dart @@ -15,10 +15,10 @@ class SegmentDestination extends DestinationPlugin with Flushable { String? _apiHost; SegmentDestination() : super(segmentDestinationKey) { - _queuePlugin = QueueFlushingPlugin(_sendEvents); + _queuePlugin = QueueFlushingPlugin(sendEvents); } - Future _sendEvents(List events) async { + Future sendEvents(List events) async { if (events.isEmpty) { return; } diff --git a/packages/core/lib/utils/lifecycle/fgbg.dart b/packages/core/lib/utils/lifecycle/fgbg.dart index 3dc256b..ec55e25 100644 --- a/packages/core/lib/utils/lifecycle/fgbg.dart +++ b/packages/core/lib/utils/lifecycle/fgbg.dart @@ -4,12 +4,14 @@ import 'package:segment_analytics/utils/lifecycle/lifecycle.dart'; import 'package:flutter_fgbg/flutter_fgbg.dart'; class FGBGLifecycle extends LifeCycle { - final _stream = FGBGEvents.instance.stream; + final Stream stream; + + FGBGLifecycle(this.stream); @override StreamSubscription listen(void Function(AppStatus event)? onData, {Function? onError, void Function()? onDone, bool? cancelOnError}) { - return _stream + return stream .map((event) => (event == FGBGType.foreground) ? AppStatus.foreground : AppStatus.background) diff --git a/packages/core/lib/utils/lifecycle/lifecycle.dart b/packages/core/lib/utils/lifecycle/lifecycle.dart index 323c00d..cb17436 100644 --- a/packages/core/lib/utils/lifecycle/lifecycle.dart +++ b/packages/core/lib/utils/lifecycle/lifecycle.dart @@ -1,6 +1,7 @@ import 'dart:async'; import 'dart:io'; +import 'package:flutter_fgbg/flutter_fgbg.dart'; import 'package:segment_analytics/utils/lifecycle/fgbg.dart'; import 'package:segment_analytics/utils/lifecycle/widget_observer.dart'; import 'package:flutter/foundation.dart'; @@ -9,11 +10,12 @@ enum AppStatus { foreground, background } abstract class LifeCycle extends Stream {} -LifeCycle _getLifecycleStream() { +@visibleForTesting +LifeCycle getLifecycleStream() { if (!kIsWeb && (Platform.isAndroid || Platform.isIOS)) { // For iOS and Android we will use the FgBg Lifecycle listener as it reports directly from native level // ignoring native popups - return FGBGLifecycle(); + return FGBGLifecycle(FGBGEvents.instance.stream); } else { // For Web and Desktop we use the WidgetObserver implementation directly from Flutter // TBF Flutter doesn't have a very reliable background signal for those platforms @@ -21,4 +23,4 @@ LifeCycle _getLifecycleStream() { } } -final LifeCycle lifecycle = _getLifecycleStream(); +final LifeCycle lifecycle = getLifecycleStream(); diff --git a/packages/core/lib/utils/store/io.dart b/packages/core/lib/utils/store/io.dart index e4314f9..9159415 100644 --- a/packages/core/lib/utils/store/io.dart +++ b/packages/core/lib/utils/store/io.dart @@ -1,3 +1,4 @@ +// coverage:ignore-file import 'dart:async'; import 'dart:convert'; import 'dart:io'; diff --git a/packages/core/pubspec.yaml b/packages/core/pubspec.yaml index 991a5b2..71650f2 100644 --- a/packages/core/pubspec.yaml +++ b/packages/core/pubspec.yaml @@ -29,10 +29,11 @@ dev_dependencies: build_runner: ^2.4.7 flutter_test: sdk: flutter + fake_async: ^1.0.0 flutter_lints: ^4.0.0 json_serializable: ^6.8.0 pigeon: ^7.2.1 - mockito: ^5.4.4 + mockito: ^5.3.2 flutter: plugin: diff --git a/packages/core/test/analytics_pigeon_test.dart b/packages/core/test/analytics_pigeon_test.dart new file mode 100644 index 0000000..98a02b1 --- /dev/null +++ b/packages/core/test/analytics_pigeon_test.dart @@ -0,0 +1,33 @@ +// test/analytics_platform_impl_test.dart +import 'package:flutter_test/flutter_test.dart'; +import 'package:mockito/mockito.dart'; +import 'package:segment_analytics/analytics_pigeon.dart'; + +import 'package:segment_analytics/native_context.dart'; +import 'mocks/mocks.mocks.dart'; + +void main() { + TestWidgetsFlutterBinding.ensureInitialized(); + + group('AnalyticsPlatformImpl Tests', () { + late AnalyticsPlatformImpl analyticsPlatform; + late MockNativeContextApi mockNativeContextApi; + + setUp(() { + mockNativeContextApi = MockNativeContextApi(); + analyticsPlatform = AnalyticsPlatformImpl(); + analyticsPlatform.api = mockNativeContextApi; + }); + + test('getContext returns NativeContext', () async { + final nativeContext = NativeContext(); + when(mockNativeContextApi.getContext(any)) + .thenAnswer((_) async => nativeContext); + + final result = await analyticsPlatform.getContext(collectDeviceId: true); + + expect(result, isA()); + verify(mockNativeContextApi.getContext(true)).called(1); + }); + }); +} diff --git a/packages/core/test/analytics_test.dart b/packages/core/test/analytics_test.dart index e6e54ac..8d1e42b 100644 --- a/packages/core/test/analytics_test.dart +++ b/packages/core/test/analytics_test.dart @@ -1,7 +1,11 @@ import 'package:segment_analytics/analytics.dart'; import 'package:segment_analytics/analytics_platform_interface.dart'; +import 'package:segment_analytics/client.dart'; import 'package:segment_analytics/event.dart'; +import 'package:segment_analytics/flush_policies/count_flush_policy.dart'; +import 'package:segment_analytics/flush_policies/flush_policy.dart'; import 'package:segment_analytics/logger.dart'; +import 'package:segment_analytics/plugins/event_logger.dart'; import 'package:segment_analytics/state.dart'; import 'package:flutter/widgets.dart'; import 'package:flutter_test/flutter_test.dart'; @@ -9,6 +13,7 @@ import 'package:mockito/mockito.dart'; import 'package:shared_preferences/shared_preferences.dart'; import 'mocks/mocks.dart'; +import 'mocks/mocks.mocks.dart'; void main() { WidgetsFlutterBinding.ensureInitialized(); @@ -21,55 +26,118 @@ void main() { ]; group("analytics", () { - - setUp(() { + late Analytics analytics; + late MockHTTPClient httpClient; + setUp(() async { AnalyticsPlatform.instance = MockPlatform(); - // Prevents spamming the test console. Eventually logging info will be behind a debug flag so this won't be needed LogFactory.logger = Mocks.logTarget(); - SharedPreferences.setMockInitialValues({}); - }); - test( - "it fetches settings but does not fire track event when not tracking lifecycle events", - () async { - final httpClient = Mocks.httpClient(); + httpClient = Mocks.httpClient(); when(httpClient.settingsFor(writeKey)) .thenAnswer((_) => Future.value(SegmentAPISettings({}))); when(httpClient.startBatchUpload(writeKey, batch)) .thenAnswer((_) => Future.value(true)); - Analytics analytics = Analytics( + analytics = Analytics( Configuration("123", trackApplicationLifecycleEvents: false, - appStateStream: () => Mocks.streamSubscription()), + token: "abcdef12345"), Mocks.store(), httpClient: (_) => httpClient); await analytics.init(); + }); + test( + "it fetches settings but does not fire track event when not tracking lifecycle events", + () async { + verify(httpClient.settingsFor(writeKey)); verifyNever(httpClient.startBatchUpload(writeKey, batch)); }); test( "it fetches settings and fires track event when tracking lifecycle events", () async { - final httpClient = Mocks.httpClient(); - when(httpClient.settingsFor(writeKey)) - .thenAnswer((_) => Future.value(SegmentAPISettings({}))); - when(httpClient.startBatchUpload(writeKey, batch)) - .thenAnswer((_) => Future.value(true)); - - Analytics analytics = Analytics( - Configuration("123", - trackApplicationLifecycleEvents: true, - appStateStream: () => Mocks.streamSubscription()), - Mocks.store(), - httpClient: (_) => httpClient); - await analytics.init(); verify(httpClient.settingsFor(writeKey)); verifyNever(httpClient.startBatchUpload(writeKey, batch)); }); + + test('it analytics track should be callable', () { + analytics.track("test track"); + }); + test('it analytics screen should be callable', () { + analytics.screen("test screem"); + }); + test('it analytics identify should be callable', () { + analytics.identify(); + }); + test('it analytics group should be callable', () { + analytics.group("test group"); + }); + test('it analytics alias should be callable', () { + analytics.alias("test alias"); + }); + test('it analytics cleanup should be callable', () { + analytics.cleanup(); + }); + test('it analytics reset should be callable', () { + analytics.reset(); + }); + test('it analytics addFlushPolicy should be callable', () { + List policies = []; + policies.add(CountFlushPolicy(5)); + analytics.addFlushPolicy(policies); + }); + test('it analytics getFlushPolicies should be callable', () { + analytics.getFlushPolicies(); + }); + test('it analytics removeFlushPolicy should be callable', () { + List policies = []; + policies.add(CountFlushPolicy(5)); + analytics.removeFlushPolicy(policies); + }); + test('it analytics removePlugin should be callable', () { + analytics.addPlugin(EventLogger(), settings: {"event":"Track Event"}); + }); + test('it analytics removePlugin should be callable', () { + analytics.removePlugin(EventLogger()); + }); + test('it analytics onContextLoaded should be callable', () { + analytics.onContextLoaded((p0) { }); + }); + test('it analytics onPluginLoaded should be callable', () { + analytics.onPluginLoaded((p0) { }); + }); + + test("Test analytics platform getContext", () { + AnalyticsPlatform analyticsPlatform = MockAnalyticsPlatform(); + + expect( + () async => await analyticsPlatform.getContext(), + throwsA(isA()), + ); + }); + test("Test analytics platform linkStream", () { + AnalyticsPlatform analyticsPlatform = MockAnalyticsPlatform(); + + expect( + () async => analyticsPlatform.linkStream, + throwsA(isA()), + ); + }); + + test("it createClient", () async { + Analytics analytics = createClient(Configuration("123", + debug: false, + trackApplicationLifecycleEvents: true, + trackDeeplinks: true, + token: "abcdef12345") + ); + expect(analytics, isA()); + }); }); } + +class MockAnalyticsPlatform extends AnalyticsPlatform { } \ No newline at end of file diff --git a/packages/core/test/client_test.dart b/packages/core/test/client_test.dart new file mode 100644 index 0000000..d4f7145 --- /dev/null +++ b/packages/core/test/client_test.dart @@ -0,0 +1,43 @@ +import 'package:flutter/material.dart'; +import 'package:flutter_test/flutter_test.dart'; +import 'dart:async'; + +import 'package:segment_analytics/client.dart'; + + +void main() { + late ScreenObserver screenObserver; + late Stream screenStream; + late List events; + + setUp(() { + screenObserver = ScreenObserver(); + screenStream = screenObserver.screenStream; + events = []; + screenStream.listen((event) { + events.add(event); + }); + }); + + test('didPop adds previous route name to stream', () { + final route = MaterialPageRoute(settings: const RouteSettings(name: '/new'), builder: (_) => Container()); + final previousRoute = MaterialPageRoute(settings: const RouteSettings(name: '/old'), builder: (_) => Container()); + screenObserver.didPop(route, previousRoute); + }); + + test('didPush adds new route name to stream', () { + final route = MaterialPageRoute(settings: const RouteSettings(name: '/new'), builder: (_) => Container()); + screenObserver.didPush(route, null); + }); + + test('didRemove adds route name to stream', () { + final route = MaterialPageRoute(settings: const RouteSettings(name: '/remove'), builder: (_) => Container()); + screenObserver.didRemove(route, null); + }); + + test('didReplace adds new route name to stream', () { + final oldRoute = MaterialPageRoute(settings: const RouteSettings(name: '/old'), builder: (_) => Container()); + final newRoute = MaterialPageRoute(settings: const RouteSettings(name: '/new'), builder: (_) => Container()); + screenObserver.didReplace(newRoute: newRoute, oldRoute: oldRoute); + }); +} diff --git a/packages/core/test/errors_test.dart b/packages/core/test/errors_test.dart new file mode 100644 index 0000000..05d68f0 --- /dev/null +++ b/packages/core/test/errors_test.dart @@ -0,0 +1,79 @@ +import 'package:flutter_test/flutter_test.dart'; +import 'package:segment_analytics/errors.dart'; + +void main() { + group('errors classes', () { + + test('StorageUnableToCreate error validation', () { + const message = 'Test error message'; + final exception = StorageUnableToCreate(message); + final result = exception.toString(); + expect(result, 'Unable to create storage: $message'); + }); + + test('StorageUnableToWrite error validation', () { + const message = 'Test error message'; + final exception = StorageUnableToWrite(message); + final result = exception.toString(); + expect(result, 'Unable to write to storage: $message'); + }); + + test('StorageUnableToRename error validation', () { + const message = 'Test error message'; + final exception = StorageUnableToRename(message); + final result = exception.toString(); + expect(result, 'Unable to rename storage: $message'); + }); + + test('StorageUnableToOpen error validation', () { + const message = 'Test error message'; + final exception = StorageUnableToOpen(message); + final result = exception.toString(); + expect(result, 'Unable to open storage: $message'); + }); + + test('StorageUnableToClose error validation', () { + const message = 'Test error message'; + final exception = StorageUnableToClose(message); + final result = exception.toString(); + expect(result, 'Unable to close storage: $message'); + }); + + test('StorageInvalid error validation', () { + const message = 'Test error message'; + final exception = StorageInvalid(message); + final result = exception.toString(); + expect(result, 'Invalide storage: $message'); + }); + + test('StorageUnknown error validation', () { + const message = 'Test error message'; + final exception = StorageUnknown(message); + final result = exception.toString(); + expect(result, 'Unknown storage error: $message'); + }); + + test('JSONUnableToDeserialize error validation', () { + const message = 'Test error message'; + const type = 'Data'; + final exception = JSONUnableToDeserialize(type, message); + final result = exception.toString(); + expect(result, 'Unable to deserialize JSON to $type: $message'); + }); + + test('InconsistentStateError error validation', () { + const key = 'Data'; + final exception = InconsistentStateError(key); + final result = exception.toString(); + expect(result, 'Store for $key is in an inconsistent state'); + }); + + test('ErrorLoadingStorage error validation', () { + const innerError = 'Data error'; + final exception = ErrorLoadingStorage(innerError); + final result = exception.toString(); + expect(result, 'Error loading storage: $innerError'); + }); + + }); +} diff --git a/packages/core/test/events_test.dart b/packages/core/test/events_test.dart index c8cf6db..0453566 100644 --- a/packages/core/test/events_test.dart +++ b/packages/core/test/events_test.dart @@ -1,5 +1,6 @@ import 'dart:convert'; +import 'package:flutter/widgets.dart'; import 'package:segment_analytics/analytics_platform_interface.dart'; import 'package:segment_analytics/event.dart'; import 'package:segment_analytics/native_context.dart'; @@ -23,14 +24,70 @@ class MockPlatform extends AnalyticsPlatform { return Future.value(mockNativeContext); } + + final List contextObj = [ + "build", + "name", + "namespace", + "version" + ]; + + final List deviceObj = [ + "id", + "manufacturer", + "model", + "name", + "type", + false, + "advertisingId", + "trackingStatus", + "token" + ]; + + final List libraryObj = [ + "name", + "version" + ]; + + final List networkObj = [ + true, + false, + false + ]; + + final List osObj = [ + "name", + "version" + ]; + + final List screenObj = [ + 100, + 100, + 5.5 + ]; + + Object buildObject() { + final List encondeObj = [ + contextObj, + deviceObj, + libraryObj, + "en_EN", + networkObj, + osObj, + screenObj, + "timezone", + "userAgent" + ]; + return encondeObj; + } } void main() { group("Context", () { setUp(() { + WidgetsFlutterBinding.ensureInitialized(); AnalyticsPlatform.instance = MockPlatform(); }); - // }); test("It gets native context", () async { final context = await AnalyticsPlatform.instance.getContext(); expect(context.app, isNotNull); @@ -88,5 +145,47 @@ void main() { expect(context.traits.custom["custom"], reverseContext.traits.custom["custom"]); }); + + test("Test encode method on NativeContext", () async { + final context = await AnalyticsPlatform.instance.getContext(); + final contextEncode = context.encode(); + expect(contextEncode.toString() != context.toString(), true); + }); + + test("Test decode method on NativeContext", () async { + final encodeObject = MockPlatform().buildObject(); + final contextDecode = NativeContext.decode(encodeObject); + expect(contextDecode.toString() != encodeObject.toString(), true); + }); + + test("Test encode method on NativeContextApp", () async { + final context = await AnalyticsPlatform.instance.getContext(); + final contextEncode = context.app?.encode(); + expect(contextEncode.toString() != context.toString(), true); + }); + + test("Test encode method on NativeContextDevice", () async { + final context = NativeContextDevice(); + final contextEncode = context.encode(); + expect(contextEncode.toString() != context.toString(), true); + }); + + test("Test encode method on NativeContextLibrary", () async { + final context = NativeContextLibrary(); + final contextEncode = context.encode(); + expect(contextEncode.toString() != context.toString(), true); + }); + + test("Test encode method on NativeContextNetwork", () async { + final context = NativeContextNetwork(); + final contextEncode = context.encode(); + expect(contextEncode.toString() != context.toString(), true); + }); + + test("Test writeValue of NativeContextApi", () { + final nativeContextApi = NativeContextApi(); + nativeContextApi.getContext(true); + }); + }); } diff --git a/packages/core/test/flush_policies/count_flush_policies_test.dart b/packages/core/test/flush_policies/count_flush_policies_test.dart new file mode 100644 index 0000000..13e2924 --- /dev/null +++ b/packages/core/test/flush_policies/count_flush_policies_test.dart @@ -0,0 +1,38 @@ +import 'package:flutter_test/flutter_test.dart'; +import 'package:segment_analytics/event.dart'; +import 'package:segment_analytics/flush_policies/count_flush_policy.dart'; + + +void main() { + group('CountFlushPolicy Tests', () { + test('Initial count is set correctly', () { + final policy = CountFlushPolicy(5); + expect(policy.count, 0); + }); + + test('Start method resets count to 0', () { + final policy = CountFlushPolicy(5, count: 3); + policy.start(); + expect(policy.count, 0); + }); + + test('onEvent increments count and sets shouldFlush to true when _flushAt is reached', () { + final policy = CountFlushPolicy(2); + final event = TrackEvent("Test"); + + policy.onEvent(event); + expect(policy.count, 1); + expect(policy.shouldFlush, false); + + policy.onEvent(event); + expect(policy.count, 2); + expect(policy.shouldFlush, true); + }); + + test('reset method calls super.reset and resets count to 0', () { + final policy = CountFlushPolicy(5, count: 3); + policy.reset(); + expect(policy.count, 0); + }); + }); +} diff --git a/packages/core/test/flush_policies/flush_policy_executor_test.dart b/packages/core/test/flush_policies/flush_policy_executor_test.dart new file mode 100644 index 0000000..7f60097 --- /dev/null +++ b/packages/core/test/flush_policies/flush_policy_executor_test.dart @@ -0,0 +1,89 @@ +import 'package:flutter_test/flutter_test.dart'; +import 'package:mockito/mockito.dart'; +import 'package:segment_analytics/event.dart'; +import 'package:segment_analytics/flush_policies/flush_policy_executor.dart'; + +import '../mocks/mocks.mocks.dart'; + +void main() { + group('FlushPolicyExecuter Tests', () { + test('start method starts all policies', () { + final policy1 = MockFlushPolicy(); + final policy2 = MockFlushPolicy(); + final policies = [policy1, policy2]; + onFlush() {} + final executer = FlushPolicyExecuter(policies, onFlush); + + executer.start(); + + expect(executer.policies, containsAll(policies)); + }); + + test('add method adds and starts a new policy', () { + final policy1 = MockFlushPolicy(); + onFlush() {} + final executer = FlushPolicyExecuter([], onFlush); + + executer.add(policy1); + + expect(executer.policies, contains(policy1)); + }); + + test('remove method removes a policy', () { + final policy1 = MockFlushPolicy(); + final policies = [policy1]; + onFlush() {} + final executer = FlushPolicyExecuter(policies, onFlush); + + final result = executer.remove(policy1); + + expect(result, isTrue); + expect(executer.policies, isNot(contains(policy1))); + }); + + test('manualFlush method triggers onFlush if any policy should flush', () { + final policy1 = MockFlushPolicy(); + final onFlushCalled = []; + onFlush() { + onFlushCalled.add(true); + } + final executer = FlushPolicyExecuter([policy1], onFlush); + + when(policy1.shouldFlush).thenReturn(true); + executer.manualFlush(); + expect(onFlushCalled, [true]); + }); + + test('notify method notifies all policies of an event', () { + final policy1 = MockFlushPolicy(); + final policy2 = MockFlushPolicy(); + final policies = [policy1, policy2]; + onFlush() {} + final executer = FlushPolicyExecuter(policies, onFlush); + final event = TrackEvent("Test"); + + executer.notify(event); + }); + + test('reset method resets all policies', () { + final policy1 = MockFlushPolicy(); + final policy2 = MockFlushPolicy(); + final policies = [policy1, policy2]; + onFlush() {} + final executer = FlushPolicyExecuter(policies, onFlush); + + executer.reset(); + }); + + test('cleanup method unsubscribes all observers', () { + final policy1 = MockFlushPolicy(); + onFlush() {} + final executer = FlushPolicyExecuter([policy1], onFlush); + + executer.start(); + expect(executer.policies, contains(policy1)); + + executer.cleanup(); + }); + }); +} diff --git a/packages/core/test/flush_policies/flush_policy_test.dart b/packages/core/test/flush_policies/flush_policy_test.dart new file mode 100644 index 0000000..ef02d1f --- /dev/null +++ b/packages/core/test/flush_policies/flush_policy_test.dart @@ -0,0 +1,42 @@ +import 'package:flutter_test/flutter_test.dart'; +import 'package:mockito/mockito.dart'; +import 'package:segment_analytics/event.dart'; + +import '../mocks/mocks.mocks.dart'; + +void main() { + group('FlushPolicy Tests', () { + test('Initial state is false', () { + final policy = MockFlushPolicy(); + expect(policy.shouldFlush, isFalse); + }); + + test('shouldFlush can be set and retrieved', () { + final policy = MockFlushPolicy(); + expect(policy.shouldFlush, isFalse); + + when(policy.shouldFlush).thenReturn(true); + expect(policy.shouldFlush, isTrue); + }); + + test('start method sets startCalled to true', () { + final policy = MockFlushPolicy(); + policy.start(); + verify(policy.start()).called(1); + }); + + test('onEvent method sets onEventCalled to true', () { + final policy = MockFlushPolicy(); + final event = TrackEvent("Test"); + policy.onEvent(event); + verify(policy.onEvent(event)).called(1); + }); + + test('reset method sets shouldFlush to false', () { + final policy = MockFlushPolicy(); + policy.shouldFlush = true; + policy.reset(); + expect(policy.shouldFlush, isFalse); + }); + }); +} diff --git a/packages/core/test/flush_policies/startup_flush_policy_test.dart b/packages/core/test/flush_policies/startup_flush_policy_test.dart new file mode 100644 index 0000000..62960f1 --- /dev/null +++ b/packages/core/test/flush_policies/startup_flush_policy_test.dart @@ -0,0 +1,14 @@ +import 'package:flutter_test/flutter_test.dart'; +import 'package:segment_analytics/flush_policies/startup_flush_policy.dart'; + +void main() { + group('StartupFlushPolicy Tests', () { + test('start method sets shouldFlush to true', () { + final policy = StartupFlushPolicy(); + expect(policy.shouldFlush, isFalse); + + policy.start(); + expect(policy.shouldFlush, isTrue); + }); + }); +} \ No newline at end of file diff --git a/packages/core/test/flush_policies/timer_flush_policy_test.dart b/packages/core/test/flush_policies/timer_flush_policy_test.dart new file mode 100644 index 0000000..8c00151 --- /dev/null +++ b/packages/core/test/flush_policies/timer_flush_policy_test.dart @@ -0,0 +1,56 @@ +import 'package:flutter_test/flutter_test.dart'; +import 'package:fake_async/fake_async.dart'; +import 'package:segment_analytics/event.dart'; +import 'package:segment_analytics/flush_policies/timer_flush_policy.dart'; + + +void main() { + group('TimerFlushPolicy Tests', () { + test('start method starts the timer and sets shouldFlush to true after interval', () { + fakeAsync((async) { + final policy = TimerFlushPolicy(1000); // 1000 milliseconds = 1 second + policy.start(); + + expect(policy.shouldFlush, isFalse); // Initially shouldFlush is false + + async.elapse(const Duration(milliseconds: 1000)); // Advance time by 1 second + expect(policy.shouldFlush, isTrue); // After 1 second, shouldFlush should be true + }); + }); + + test('onEvent method resets the timer', () { + fakeAsync((async) { + final policy = TimerFlushPolicy(1000); // 1000 milliseconds = 1 second + final event = TrackEvent("Test"); // Asegúrate de definir RawEvent o usa un mock si es una clase compleja + policy.start(); + + async.elapse(const Duration(milliseconds: 500)); // Advance time by 0.5 second + expect(policy.shouldFlush, isFalse); // shouldFlush is still false + + policy.onEvent(event); // Reset the timer + async.elapse(const Duration(milliseconds: 500)); // Advance time by another 0.5 second + expect(policy.shouldFlush, isFalse); // shouldFlush is still false because the timer was reset + + async.elapse(const Duration(milliseconds: 500)); // Advance time by another 0.5 second + expect(policy.shouldFlush, isTrue); // After 1 second from the reset, shouldFlush should be true + }); + }); + + test('reset method resets the timer', () { + fakeAsync((async) { + final policy = TimerFlushPolicy(1000); // 1000 milliseconds = 1 second + policy.start(); + + async.elapse(const Duration(milliseconds: 500)); // Advance time by 0.5 second + expect(policy.shouldFlush, isFalse); // shouldFlush is still false + + policy.reset(); // Reset the timer + async.elapse(const Duration(milliseconds: 500)); // Advance time by another 0.5 second + expect(policy.shouldFlush, isFalse); // shouldFlush is still false because the timer was reset + + async.elapse(const Duration(milliseconds: 500)); // Advance time by another 0.5 second + expect(policy.shouldFlush, isTrue); // After 1 second from the reset, shouldFlush should be true + }); + }); + }); +} diff --git a/packages/core/test/logger_test.dart b/packages/core/test/logger_test.dart new file mode 100644 index 0000000..522594d --- /dev/null +++ b/packages/core/test/logger_test.dart @@ -0,0 +1,83 @@ +import 'package:flutter_test/flutter_test.dart'; +import 'package:logger/web.dart'; +import 'package:segment_analytics/logger.dart'; + +class MockLogger extends Logger { + List logs = []; + + @override + void d(dynamic message, { + DateTime? time, + Object? error, + StackTrace? stackTrace}) { + logs.add('Debug: $message'); + } + + @override + void w(dynamic message, { + DateTime? time, + Object? error, + StackTrace? stackTrace, + }) { + logs.add('Warning: $message'); + } + + @override + void e(dynamic message, { + DateTime? time, + Object? error, + StackTrace? stackTrace, + }) { + logs.add('Error: $message'); + } +} + +void main() { + group('SystemLogger', () { + test('should log debug messages correctly', () { + final mockLogger = MockLogger(); + final logger = SystemLogger()..logger = mockLogger; + + final logMessage = LogMessage(LogFilterKind.debug, 'Debug message', LogDestination.log); + logger.parseLog(logMessage); + expect(mockLogger.logs, contains('Debug: Segment: Debug message')); + }); + + test('should log warning messages correctly', () { + final mockLogger = MockLogger(); + final logger = SystemLogger()..logger = mockLogger; + + final logMessage = LogMessage(LogFilterKind.warning, 'Warning message', LogDestination.log); + logger.parseLog(logMessage); + expect(mockLogger.logs, contains('Warning: Segment: Warning message')); + }); + + test('should log error messages correctly', () { + final mockLogger = MockLogger(); + final logger = SystemLogger()..logger = mockLogger; + + final logMessage = LogMessage(LogFilterKind.error, 'Error message', LogDestination.log); + logger.parseLog(logMessage); + + expect(mockLogger.logs, contains('Error: Segment: Error message')); + }); + }); + + test('should return LogFilterKind', () { + expect(LogFilterKind.debug.toString(), "Debug"); + expect(LogFilterKind.warning.toString(), "Warning"); + expect(LogFilterKind.error.toString(), "ERROR"); + }); + + test('MetricType methods', () { + expect(MetricType.counter.toString(), "Counter"); + expect(MetricType.fromString("Gauge"), MetricType.gauge); + }); + + test('should LogFactory buildLog Method', () { + LogMessage lmMetric = LogFactory.buildLog(LogDestination.metric, "Test metric", LogFilterKind.debug); + LogMessage lmHistory = LogFactory.buildLog(LogDestination.history, "Test history", LogFilterKind.debug); + expect(lmMetric.logType, LogDestination.metric); + expect(lmHistory.logType, LogDestination.history); + }); +} diff --git a/packages/core/test/mocks/mocks.dart b/packages/core/test/mocks/mocks.dart index d1abfc5..dff3751 100644 --- a/packages/core/test/mocks/mocks.dart +++ b/packages/core/test/mocks/mocks.dart @@ -1,6 +1,9 @@ import 'dart:async'; +import 'package:flutter/widgets.dart'; +import 'package:logger/logger.dart'; import 'package:segment_analytics/analytics_platform_interface.dart'; +import 'package:segment_analytics/flush_policies/flush_policy.dart'; import 'package:segment_analytics/logger.dart'; import 'package:segment_analytics/native_context.dart'; import 'package:segment_analytics/utils/http_client.dart'; @@ -13,7 +16,11 @@ import 'package:mockito/annotations.dart'; MockSpec(), MockSpec(), MockSpec(), - MockSpec() + MockSpec(), + MockSpec(), + MockSpec(), + MockSpec(), + MockSpec() ]) import 'mocks.mocks.dart'; @@ -37,4 +44,5 @@ class Mocks { MockStreamSubscription(); static MockHTTPClient httpClient() => MockHTTPClient(); static MockStore store() => MockStore(); + static MockFlushPolicy flushPolicy() => MockFlushPolicy(); } diff --git a/packages/core/test/mocks/mocks.mocks.dart b/packages/core/test/mocks/mocks.mocks.dart index ea913b2..d3d5cb8 100644 --- a/packages/core/test/mocks/mocks.mocks.dart +++ b/packages/core/test/mocks/mocks.mocks.dart @@ -5,17 +5,32 @@ // ignore_for_file: no_leading_underscores_for_library_prefixes import 'dart:async' as _i4; import 'dart:convert' as _i2; -import 'dart:typed_data' as _i6; +import 'dart:developer' as _i27; +import 'dart:typed_data' as _i15; +import 'dart:ui' as _i9; +import 'package:flutter/foundation.dart' as _i8; +import 'package:flutter/gestures.dart' as _i11; +import 'package:flutter/rendering.dart' as _i12; +import 'package:flutter/scheduler.dart' as _i26; +import 'package:flutter/services.dart' as _i10; +import 'package:flutter/src/widgets/binding.dart' as _i25; +import 'package:flutter/src/widgets/focus_manager.dart' as _i7; +import 'package:flutter/src/widgets/framework.dart' as _i13; +import 'package:flutter/src/widgets/platform_menu_bar.dart' as _i6; import 'package:http/http.dart' as _i3; -import 'package:http/src/byte_stream.dart' as _i8; +import 'package:http/src/byte_stream.dart' as _i17; +import 'package:logger/logger.dart' as _i24; import 'package:mockito/mockito.dart' as _i1; -import 'package:mockito/src/dummies.dart' as _i7; -import 'package:segment_analytics/event.dart' as _i10; -import 'package:segment_analytics/logger.dart' as _i5; -import 'package:segment_analytics/state.dart' as _i11; -import 'package:segment_analytics/utils/http_client.dart' as _i9; -import 'package:segment_analytics/utils/store/store.dart' as _i12; +import 'package:mockito/src/dummies.dart' as _i16; +import 'package:segment_analytics/event.dart' as _i19; +import 'package:segment_analytics/flush_policies/flush_policy.dart' as _i22; +import 'package:segment_analytics/logger.dart' as _i14; +import 'package:segment_analytics/native_context.dart' as _i5; +import 'package:segment_analytics/state.dart' as _i20; +import 'package:segment_analytics/utils/http_client.dart' as _i18; +import 'package:segment_analytics/utils/store/store.dart' as _i21; +import 'package:state_notifier/state_notifier.dart' as _i23; // ignore_for_file: type=lint // ignore_for_file: avoid_redundant_argument_values @@ -71,12 +86,286 @@ class _FakeFuture_3 extends _i1.SmartFake implements _i4.Future { ); } +class _FakeNativeContext_4 extends _i1.SmartFake implements _i5.NativeContext { + _FakeNativeContext_4( + Object parent, + Invocation parentInvocation, + ) : super( + parent, + parentInvocation, + ); +} + +class _FakePlatformMenuDelegate_5 extends _i1.SmartFake + implements _i6.PlatformMenuDelegate { + _FakePlatformMenuDelegate_5( + Object parent, + Invocation parentInvocation, + ) : super( + parent, + parentInvocation, + ); +} + +class _FakeFocusManager_6 extends _i1.SmartFake implements _i7.FocusManager { + _FakeFocusManager_6( + Object parent, + Invocation parentInvocation, + ) : super( + parent, + parentInvocation, + ); + + @override + String toString({_i8.DiagnosticLevel? minLevel = _i8.DiagnosticLevel.info}) => + super.toString(); +} + +class _FakeSingletonFlutterWindow_7 extends _i1.SmartFake + implements _i9.SingletonFlutterWindow { + _FakeSingletonFlutterWindow_7( + Object parent, + Invocation parentInvocation, + ) : super( + parent, + parentInvocation, + ); +} + +class _FakePlatformDispatcher_8 extends _i1.SmartFake + implements _i9.PlatformDispatcher { + _FakePlatformDispatcher_8( + Object parent, + Invocation parentInvocation, + ) : super( + parent, + parentInvocation, + ); +} + +class _FakeValueNotifier_9 extends _i1.SmartFake + implements _i8.ValueNotifier { + _FakeValueNotifier_9( + Object parent, + Invocation parentInvocation, + ) : super( + parent, + parentInvocation, + ); +} + +class _FakeHardwareKeyboard_10 extends _i1.SmartFake + implements _i10.HardwareKeyboard { + _FakeHardwareKeyboard_10( + Object parent, + Invocation parentInvocation, + ) : super( + parent, + parentInvocation, + ); +} + +class _FakeKeyEventManager_11 extends _i1.SmartFake + implements _i10.KeyEventManager { + _FakeKeyEventManager_11( + Object parent, + Invocation parentInvocation, + ) : super( + parent, + parentInvocation, + ); +} + +class _FakeBinaryMessenger_12 extends _i1.SmartFake + implements _i10.BinaryMessenger { + _FakeBinaryMessenger_12( + Object parent, + Invocation parentInvocation, + ) : super( + parent, + parentInvocation, + ); +} + +class _FakeChannelBuffers_13 extends _i1.SmartFake + implements _i9.ChannelBuffers { + _FakeChannelBuffers_13( + Object parent, + Invocation parentInvocation, + ) : super( + parent, + parentInvocation, + ); +} + +class _FakeRestorationManager_14 extends _i1.SmartFake + implements _i10.RestorationManager { + _FakeRestorationManager_14( + Object parent, + Invocation parentInvocation, + ) : super( + parent, + parentInvocation, + ); +} + +class _FakeDuration_15 extends _i1.SmartFake implements Duration { + _FakeDuration_15( + Object parent, + Invocation parentInvocation, + ) : super( + parent, + parentInvocation, + ); +} + +class _FakePointerRouter_16 extends _i1.SmartFake + implements _i11.PointerRouter { + _FakePointerRouter_16( + Object parent, + Invocation parentInvocation, + ) : super( + parent, + parentInvocation, + ); +} + +class _FakeGestureArenaManager_17 extends _i1.SmartFake + implements _i11.GestureArenaManager { + _FakeGestureArenaManager_17( + Object parent, + Invocation parentInvocation, + ) : super( + parent, + parentInvocation, + ); +} + +class _FakePointerSignalResolver_18 extends _i1.SmartFake + implements _i11.PointerSignalResolver { + _FakePointerSignalResolver_18( + Object parent, + Invocation parentInvocation, + ) : super( + parent, + parentInvocation, + ); +} + +class _FakeSamplingClock_19 extends _i1.SmartFake + implements _i11.SamplingClock { + _FakeSamplingClock_19( + Object parent, + Invocation parentInvocation, + ) : super( + parent, + parentInvocation, + ); +} + +class _FakePipelineOwner_20 extends _i1.SmartFake + implements _i12.PipelineOwner { + _FakePipelineOwner_20( + Object parent, + Invocation parentInvocation, + ) : super( + parent, + parentInvocation, + ); + + @override + String toString({_i8.DiagnosticLevel? minLevel = _i8.DiagnosticLevel.info}) => + super.toString(); +} + +class _FakeRenderView_21 extends _i1.SmartFake implements _i12.RenderView { + _FakeRenderView_21( + Object parent, + Invocation parentInvocation, + ) : super( + parent, + parentInvocation, + ); + + @override + String toString({_i8.DiagnosticLevel? minLevel = _i8.DiagnosticLevel.info}) => + super.toString(); +} + +class _FakeMouseTracker_22 extends _i1.SmartFake implements _i12.MouseTracker { + _FakeMouseTracker_22( + Object parent, + Invocation parentInvocation, + ) : super( + parent, + parentInvocation, + ); +} + +class _FakeAccessibilityFeatures_23 extends _i1.SmartFake + implements _i9.AccessibilityFeatures { + _FakeAccessibilityFeatures_23( + Object parent, + Invocation parentInvocation, + ) : super( + parent, + parentInvocation, + ); +} + +class _FakeWidget_24 extends _i1.SmartFake implements _i13.Widget { + _FakeWidget_24( + Object parent, + Invocation parentInvocation, + ) : super( + parent, + parentInvocation, + ); + + @override + String toString({_i8.DiagnosticLevel? minLevel = _i8.DiagnosticLevel.info}) => + super.toString(); +} + +class _FakeViewConfiguration_25 extends _i1.SmartFake + implements _i12.ViewConfiguration { + _FakeViewConfiguration_25( + Object parent, + Invocation parentInvocation, + ) : super( + parent, + parentInvocation, + ); +} + +class _FakeSemanticsHandle_26 extends _i1.SmartFake + implements _i12.SemanticsHandle { + _FakeSemanticsHandle_26( + Object parent, + Invocation parentInvocation, + ) : super( + parent, + parentInvocation, + ); +} + +class _FakeSemanticsUpdateBuilder_27 extends _i1.SmartFake + implements _i9.SemanticsUpdateBuilder { + _FakeSemanticsUpdateBuilder_27( + Object parent, + Invocation parentInvocation, + ) : super( + parent, + parentInvocation, + ); +} + /// A class which mocks [LogTarget]. /// /// See the documentation for Mockito's code generation for more information. -class MockLogTarget extends _i1.Mock implements _i5.LogTarget { +class MockLogTarget extends _i1.Mock implements _i14.LogTarget { @override - void parseLog(_i5.LogMessage? log) => super.noSuchMethod( + void parseLog(_i14.LogMessage? log) => super.noSuchMethod( Invocation.method( #parseLog, [log], @@ -128,11 +417,11 @@ class MockRequest extends _i1.Mock implements _i3.Request { ); @override - _i6.Uint8List get bodyBytes => (super.noSuchMethod( + _i15.Uint8List get bodyBytes => (super.noSuchMethod( Invocation.getter(#bodyBytes), - returnValue: _i6.Uint8List(0), - returnValueForMissingStub: _i6.Uint8List(0), - ) as _i6.Uint8List); + returnValue: _i15.Uint8List(0), + returnValueForMissingStub: _i15.Uint8List(0), + ) as _i15.Uint8List); @override set bodyBytes(List? value) => super.noSuchMethod( @@ -146,11 +435,11 @@ class MockRequest extends _i1.Mock implements _i3.Request { @override String get body => (super.noSuchMethod( Invocation.getter(#body), - returnValue: _i7.dummyValue( + returnValue: _i16.dummyValue( this, Invocation.getter(#body), ), - returnValueForMissingStub: _i7.dummyValue( + returnValueForMissingStub: _i16.dummyValue( this, Invocation.getter(#body), ), @@ -184,11 +473,11 @@ class MockRequest extends _i1.Mock implements _i3.Request { @override String get method => (super.noSuchMethod( Invocation.getter(#method), - returnValue: _i7.dummyValue( + returnValue: _i16.dummyValue( this, Invocation.getter(#method), ), - returnValueForMissingStub: _i7.dummyValue( + returnValueForMissingStub: _i16.dummyValue( this, Invocation.getter(#method), ), @@ -270,26 +559,26 @@ class MockRequest extends _i1.Mock implements _i3.Request { ) as bool); @override - _i8.ByteStream finalize() => (super.noSuchMethod( + _i17.ByteStream finalize() => (super.noSuchMethod( Invocation.method( #finalize, [], ), - returnValue: _i7.dummyValue<_i8.ByteStream>( + returnValue: _i16.dummyValue<_i17.ByteStream>( this, Invocation.method( #finalize, [], ), ), - returnValueForMissingStub: _i7.dummyValue<_i8.ByteStream>( + returnValueForMissingStub: _i16.dummyValue<_i17.ByteStream>( this, Invocation.method( #finalize, [], ), ), - ) as _i8.ByteStream); + ) as _i17.ByteStream); @override _i4.Future<_i3.StreamedResponse> send() => (super.noSuchMethod( @@ -389,8 +678,8 @@ class MockStreamSubscription extends _i1.Mock #asFuture, [futureValue], ), - returnValue: _i7.ifNotNull( - _i7.dummyValueOrNull( + returnValue: _i16.ifNotNull( + _i16.dummyValueOrNull( this, Invocation.method( #asFuture, @@ -406,8 +695,8 @@ class MockStreamSubscription extends _i1.Mock [futureValue], ), ), - returnValueForMissingStub: _i7.ifNotNull( - _i7.dummyValueOrNull( + returnValueForMissingStub: _i16.ifNotNull( + _i16.dummyValueOrNull( this, Invocation.method( #asFuture, @@ -429,11 +718,11 @@ class MockStreamSubscription extends _i1.Mock /// A class which mocks [HTTPClient]. /// /// See the documentation for Mockito's code generation for more information. -class MockHTTPClient extends _i1.Mock implements _i9.HTTPClient { +class MockHTTPClient extends _i1.Mock implements _i18.HTTPClient { @override _i4.Future startBatchUpload( String? writeKey, - List<_i10.RawEvent>? batch, { + List<_i19.RawEvent>? batch, { String? host, }) => (super.noSuchMethod( @@ -450,21 +739,21 @@ class MockHTTPClient extends _i1.Mock implements _i9.HTTPClient { ) as _i4.Future); @override - _i4.Future<_i11.SegmentAPISettings?> settingsFor(String? writeKey) => + _i4.Future<_i20.SegmentAPISettings?> settingsFor(String? writeKey) => (super.noSuchMethod( Invocation.method( #settingsFor, [writeKey], ), - returnValue: _i4.Future<_i11.SegmentAPISettings?>.value(), - returnValueForMissingStub: _i4.Future<_i11.SegmentAPISettings?>.value(), - ) as _i4.Future<_i11.SegmentAPISettings?>); + returnValue: _i4.Future<_i20.SegmentAPISettings?>.value(), + returnValueForMissingStub: _i4.Future<_i20.SegmentAPISettings?>.value(), + ) as _i4.Future<_i20.SegmentAPISettings?>); } /// A class which mocks [Store]. /// /// See the documentation for Mockito's code generation for more information. -class MockStore extends _i1.Mock implements _i12.Store { +class MockStore extends _i1.Mock implements _i21.Store { @override _i4.Future get ready => (super.noSuchMethod( Invocation.getter(#ready), @@ -509,3 +798,1914 @@ class MockStore extends _i1.Mock implements _i12.Store { returnValueForMissingStub: null, ); } + +/// A class which mocks [FlushPolicy]. +/// +/// See the documentation for Mockito's code generation for more information. +class MockFlushPolicy extends _i1.Mock implements _i22.FlushPolicy { + @override + bool get shouldFlush => (super.noSuchMethod( + Invocation.getter(#shouldFlush), + returnValue: false, + returnValueForMissingStub: false, + ) as bool); + + @override + set shouldFlush(bool? shouldFlush) => super.noSuchMethod( + Invocation.setter( + #shouldFlush, + shouldFlush, + ), + returnValueForMissingStub: null, + ); + + @override + set onError(_i23.ErrorListener? _onError) => super.noSuchMethod( + Invocation.setter( + #onError, + _onError, + ), + returnValueForMissingStub: null, + ); + + @override + bool get mounted => (super.noSuchMethod( + Invocation.getter(#mounted), + returnValue: false, + returnValueForMissingStub: false, + ) as bool); + + @override + _i4.Stream get stream => (super.noSuchMethod( + Invocation.getter(#stream), + returnValue: _i4.Stream.empty(), + returnValueForMissingStub: _i4.Stream.empty(), + ) as _i4.Stream); + + @override + bool get state => (super.noSuchMethod( + Invocation.getter(#state), + returnValue: false, + returnValueForMissingStub: false, + ) as bool); + + @override + set state(bool? value) => super.noSuchMethod( + Invocation.setter( + #state, + value, + ), + returnValueForMissingStub: null, + ); + + @override + bool get debugState => (super.noSuchMethod( + Invocation.getter(#debugState), + returnValue: false, + returnValueForMissingStub: false, + ) as bool); + + @override + bool get hasListeners => (super.noSuchMethod( + Invocation.getter(#hasListeners), + returnValue: false, + returnValueForMissingStub: false, + ) as bool); + + @override + void start() => super.noSuchMethod( + Invocation.method( + #start, + [], + ), + returnValueForMissingStub: null, + ); + + @override + void onEvent(_i19.RawEvent? event) => super.noSuchMethod( + Invocation.method( + #onEvent, + [event], + ), + returnValueForMissingStub: null, + ); + + @override + void reset() => super.noSuchMethod( + Invocation.method( + #reset, + [], + ), + returnValueForMissingStub: null, + ); + + @override + bool updateShouldNotify( + bool? old, + bool? current, + ) => + (super.noSuchMethod( + Invocation.method( + #updateShouldNotify, + [ + old, + current, + ], + ), + returnValue: false, + returnValueForMissingStub: false, + ) as bool); + + @override + _i23.RemoveListener addListener( + _i23.Listener? listener, { + bool? fireImmediately = true, + }) => + (super.noSuchMethod( + Invocation.method( + #addListener, + [listener], + {#fireImmediately: fireImmediately}, + ), + returnValue: () {}, + returnValueForMissingStub: () {}, + ) as _i23.RemoveListener); + + @override + void dispose() => super.noSuchMethod( + Invocation.method( + #dispose, + [], + ), + returnValueForMissingStub: null, + ); +} + +/// A class which mocks [Logger]. +/// +/// See the documentation for Mockito's code generation for more information. +class MockLogger extends _i1.Mock implements _i24.Logger { + @override + _i4.Future get init => (super.noSuchMethod( + Invocation.getter(#init), + returnValue: _i4.Future.value(), + returnValueForMissingStub: _i4.Future.value(), + ) as _i4.Future); + + @override + void v( + dynamic message, { + DateTime? time, + Object? error, + StackTrace? stackTrace, + }) => + super.noSuchMethod( + Invocation.method( + #v, + [message], + { + #time: time, + #error: error, + #stackTrace: stackTrace, + }, + ), + returnValueForMissingStub: null, + ); + + @override + void t( + dynamic message, { + DateTime? time, + Object? error, + StackTrace? stackTrace, + }) => + super.noSuchMethod( + Invocation.method( + #t, + [message], + { + #time: time, + #error: error, + #stackTrace: stackTrace, + }, + ), + returnValueForMissingStub: null, + ); + + @override + void d( + dynamic message, { + DateTime? time, + Object? error, + StackTrace? stackTrace, + }) => + super.noSuchMethod( + Invocation.method( + #d, + [message], + { + #time: time, + #error: error, + #stackTrace: stackTrace, + }, + ), + returnValueForMissingStub: null, + ); + + @override + void i( + dynamic message, { + DateTime? time, + Object? error, + StackTrace? stackTrace, + }) => + super.noSuchMethod( + Invocation.method( + #i, + [message], + { + #time: time, + #error: error, + #stackTrace: stackTrace, + }, + ), + returnValueForMissingStub: null, + ); + + @override + void w( + dynamic message, { + DateTime? time, + Object? error, + StackTrace? stackTrace, + }) => + super.noSuchMethod( + Invocation.method( + #w, + [message], + { + #time: time, + #error: error, + #stackTrace: stackTrace, + }, + ), + returnValueForMissingStub: null, + ); + + @override + void e( + dynamic message, { + DateTime? time, + Object? error, + StackTrace? stackTrace, + }) => + super.noSuchMethod( + Invocation.method( + #e, + [message], + { + #time: time, + #error: error, + #stackTrace: stackTrace, + }, + ), + returnValueForMissingStub: null, + ); + + @override + void wtf( + dynamic message, { + DateTime? time, + Object? error, + StackTrace? stackTrace, + }) => + super.noSuchMethod( + Invocation.method( + #wtf, + [message], + { + #time: time, + #error: error, + #stackTrace: stackTrace, + }, + ), + returnValueForMissingStub: null, + ); + + @override + void f( + dynamic message, { + DateTime? time, + Object? error, + StackTrace? stackTrace, + }) => + super.noSuchMethod( + Invocation.method( + #f, + [message], + { + #time: time, + #error: error, + #stackTrace: stackTrace, + }, + ), + returnValueForMissingStub: null, + ); + + @override + void log( + _i24.Level? level, + dynamic message, { + DateTime? time, + Object? error, + StackTrace? stackTrace, + }) => + super.noSuchMethod( + Invocation.method( + #log, + [ + level, + message, + ], + { + #time: time, + #error: error, + #stackTrace: stackTrace, + }, + ), + returnValueForMissingStub: null, + ); + + @override + bool isClosed() => (super.noSuchMethod( + Invocation.method( + #isClosed, + [], + ), + returnValue: false, + returnValueForMissingStub: false, + ) as bool); + + @override + _i4.Future close() => (super.noSuchMethod( + Invocation.method( + #close, + [], + ), + returnValue: _i4.Future.value(), + returnValueForMissingStub: _i4.Future.value(), + ) as _i4.Future); +} + +/// A class which mocks [NativeContextApi]. +/// +/// See the documentation for Mockito's code generation for more information. +class MockNativeContextApi extends _i1.Mock implements _i5.NativeContextApi { + @override + _i4.Future<_i5.NativeContext> getContext(bool? arg_collectDeviceId) => + (super.noSuchMethod( + Invocation.method( + #getContext, + [arg_collectDeviceId], + ), + returnValue: _i4.Future<_i5.NativeContext>.value(_FakeNativeContext_4( + this, + Invocation.method( + #getContext, + [arg_collectDeviceId], + ), + )), + returnValueForMissingStub: + _i4.Future<_i5.NativeContext>.value(_FakeNativeContext_4( + this, + Invocation.method( + #getContext, + [arg_collectDeviceId], + ), + )), + ) as _i4.Future<_i5.NativeContext>); +} + +/// A class which mocks [WidgetsBinding]. +/// +/// See the documentation for Mockito's code generation for more information. +class MockWidgetsBinding extends _i1.Mock implements _i25.WidgetsBinding { + @override + _i6.PlatformMenuDelegate get platformMenuDelegate => (super.noSuchMethod( + Invocation.getter(#platformMenuDelegate), + returnValue: _FakePlatformMenuDelegate_5( + this, + Invocation.getter(#platformMenuDelegate), + ), + returnValueForMissingStub: _FakePlatformMenuDelegate_5( + this, + Invocation.getter(#platformMenuDelegate), + ), + ) as _i6.PlatformMenuDelegate); + + @override + set platformMenuDelegate(_i6.PlatformMenuDelegate? _platformMenuDelegate) => + super.noSuchMethod( + Invocation.setter( + #platformMenuDelegate, + _platformMenuDelegate, + ), + returnValueForMissingStub: null, + ); + + @override + bool get debugBuildingDirtyElements => (super.noSuchMethod( + Invocation.getter(#debugBuildingDirtyElements), + returnValue: false, + returnValueForMissingStub: false, + ) as bool); + + @override + set debugBuildingDirtyElements(bool? _debugBuildingDirtyElements) => + super.noSuchMethod( + Invocation.setter( + #debugBuildingDirtyElements, + _debugBuildingDirtyElements, + ), + returnValueForMissingStub: null, + ); + + @override + _i7.FocusManager get focusManager => (super.noSuchMethod( + Invocation.getter(#focusManager), + returnValue: _FakeFocusManager_6( + this, + Invocation.getter(#focusManager), + ), + returnValueForMissingStub: _FakeFocusManager_6( + this, + Invocation.getter(#focusManager), + ), + ) as _i7.FocusManager); + + @override + bool get firstFrameRasterized => (super.noSuchMethod( + Invocation.getter(#firstFrameRasterized), + returnValue: false, + returnValueForMissingStub: false, + ) as bool); + + @override + _i4.Future get waitUntilFirstFrameRasterized => (super.noSuchMethod( + Invocation.getter(#waitUntilFirstFrameRasterized), + returnValue: _i4.Future.value(), + returnValueForMissingStub: _i4.Future.value(), + ) as _i4.Future); + + @override + bool get debugDidSendFirstFrameEvent => (super.noSuchMethod( + Invocation.getter(#debugDidSendFirstFrameEvent), + returnValue: false, + returnValueForMissingStub: false, + ) as bool); + + @override + bool get framesEnabled => (super.noSuchMethod( + Invocation.getter(#framesEnabled), + returnValue: false, + returnValueForMissingStub: false, + ) as bool); + + @override + bool get isRootWidgetAttached => (super.noSuchMethod( + Invocation.getter(#isRootWidgetAttached), + returnValue: false, + returnValueForMissingStub: false, + ) as bool); + + @override + _i9.SingletonFlutterWindow get window => (super.noSuchMethod( + Invocation.getter(#window), + returnValue: _FakeSingletonFlutterWindow_7( + this, + Invocation.getter(#window), + ), + returnValueForMissingStub: _FakeSingletonFlutterWindow_7( + this, + Invocation.getter(#window), + ), + ) as _i9.SingletonFlutterWindow); + + @override + _i9.PlatformDispatcher get platformDispatcher => (super.noSuchMethod( + Invocation.getter(#platformDispatcher), + returnValue: _FakePlatformDispatcher_8( + this, + Invocation.getter(#platformDispatcher), + ), + returnValueForMissingStub: _FakePlatformDispatcher_8( + this, + Invocation.getter(#platformDispatcher), + ), + ) as _i9.PlatformDispatcher); + + @override + bool get locked => (super.noSuchMethod( + Invocation.getter(#locked), + returnValue: false, + returnValueForMissingStub: false, + ) as bool); + + @override + _i8.ValueNotifier get accessibilityFocus => (super.noSuchMethod( + Invocation.getter(#accessibilityFocus), + returnValue: _FakeValueNotifier_9( + this, + Invocation.getter(#accessibilityFocus), + ), + returnValueForMissingStub: _FakeValueNotifier_9( + this, + Invocation.getter(#accessibilityFocus), + ), + ) as _i8.ValueNotifier); + + @override + _i10.HardwareKeyboard get keyboard => (super.noSuchMethod( + Invocation.getter(#keyboard), + returnValue: _FakeHardwareKeyboard_10( + this, + Invocation.getter(#keyboard), + ), + returnValueForMissingStub: _FakeHardwareKeyboard_10( + this, + Invocation.getter(#keyboard), + ), + ) as _i10.HardwareKeyboard); + + @override + _i10.KeyEventManager get keyEventManager => (super.noSuchMethod( + Invocation.getter(#keyEventManager), + returnValue: _FakeKeyEventManager_11( + this, + Invocation.getter(#keyEventManager), + ), + returnValueForMissingStub: _FakeKeyEventManager_11( + this, + Invocation.getter(#keyEventManager), + ), + ) as _i10.KeyEventManager); + + @override + _i10.BinaryMessenger get defaultBinaryMessenger => (super.noSuchMethod( + Invocation.getter(#defaultBinaryMessenger), + returnValue: _FakeBinaryMessenger_12( + this, + Invocation.getter(#defaultBinaryMessenger), + ), + returnValueForMissingStub: _FakeBinaryMessenger_12( + this, + Invocation.getter(#defaultBinaryMessenger), + ), + ) as _i10.BinaryMessenger); + + @override + _i9.ChannelBuffers get channelBuffers => (super.noSuchMethod( + Invocation.getter(#channelBuffers), + returnValue: _FakeChannelBuffers_13( + this, + Invocation.getter(#channelBuffers), + ), + returnValueForMissingStub: _FakeChannelBuffers_13( + this, + Invocation.getter(#channelBuffers), + ), + ) as _i9.ChannelBuffers); + + @override + _i10.RestorationManager get restorationManager => (super.noSuchMethod( + Invocation.getter(#restorationManager), + returnValue: _FakeRestorationManager_14( + this, + Invocation.getter(#restorationManager), + ), + returnValueForMissingStub: _FakeRestorationManager_14( + this, + Invocation.getter(#restorationManager), + ), + ) as _i10.RestorationManager); + + @override + _i26.SchedulingStrategy get schedulingStrategy => (super.noSuchMethod( + Invocation.getter(#schedulingStrategy), + returnValue: ({ + required int priority, + required _i26.SchedulerBinding scheduler, + }) => + false, + returnValueForMissingStub: ({ + required int priority, + required _i26.SchedulerBinding scheduler, + }) => + false, + ) as _i26.SchedulingStrategy); + + @override + set schedulingStrategy(_i26.SchedulingStrategy? _schedulingStrategy) => + super.noSuchMethod( + Invocation.setter( + #schedulingStrategy, + _schedulingStrategy, + ), + returnValueForMissingStub: null, + ); + + @override + int get transientCallbackCount => (super.noSuchMethod( + Invocation.getter(#transientCallbackCount), + returnValue: 0, + returnValueForMissingStub: 0, + ) as int); + + @override + _i4.Future get endOfFrame => (super.noSuchMethod( + Invocation.getter(#endOfFrame), + returnValue: _i4.Future.value(), + returnValueForMissingStub: _i4.Future.value(), + ) as _i4.Future); + + @override + bool get hasScheduledFrame => (super.noSuchMethod( + Invocation.getter(#hasScheduledFrame), + returnValue: false, + returnValueForMissingStub: false, + ) as bool); + + @override + _i26.SchedulerPhase get schedulerPhase => (super.noSuchMethod( + Invocation.getter(#schedulerPhase), + returnValue: _i26.SchedulerPhase.idle, + returnValueForMissingStub: _i26.SchedulerPhase.idle, + ) as _i26.SchedulerPhase); + + @override + Duration get currentFrameTimeStamp => (super.noSuchMethod( + Invocation.getter(#currentFrameTimeStamp), + returnValue: _FakeDuration_15( + this, + Invocation.getter(#currentFrameTimeStamp), + ), + returnValueForMissingStub: _FakeDuration_15( + this, + Invocation.getter(#currentFrameTimeStamp), + ), + ) as Duration); + + @override + Duration get currentSystemFrameTimeStamp => (super.noSuchMethod( + Invocation.getter(#currentSystemFrameTimeStamp), + returnValue: _FakeDuration_15( + this, + Invocation.getter(#currentSystemFrameTimeStamp), + ), + returnValueForMissingStub: _FakeDuration_15( + this, + Invocation.getter(#currentSystemFrameTimeStamp), + ), + ) as Duration); + + @override + _i11.PointerRouter get pointerRouter => (super.noSuchMethod( + Invocation.getter(#pointerRouter), + returnValue: _FakePointerRouter_16( + this, + Invocation.getter(#pointerRouter), + ), + returnValueForMissingStub: _FakePointerRouter_16( + this, + Invocation.getter(#pointerRouter), + ), + ) as _i11.PointerRouter); + + @override + _i11.GestureArenaManager get gestureArena => (super.noSuchMethod( + Invocation.getter(#gestureArena), + returnValue: _FakeGestureArenaManager_17( + this, + Invocation.getter(#gestureArena), + ), + returnValueForMissingStub: _FakeGestureArenaManager_17( + this, + Invocation.getter(#gestureArena), + ), + ) as _i11.GestureArenaManager); + + @override + _i11.PointerSignalResolver get pointerSignalResolver => (super.noSuchMethod( + Invocation.getter(#pointerSignalResolver), + returnValue: _FakePointerSignalResolver_18( + this, + Invocation.getter(#pointerSignalResolver), + ), + returnValueForMissingStub: _FakePointerSignalResolver_18( + this, + Invocation.getter(#pointerSignalResolver), + ), + ) as _i11.PointerSignalResolver); + + @override + bool get resamplingEnabled => (super.noSuchMethod( + Invocation.getter(#resamplingEnabled), + returnValue: false, + returnValueForMissingStub: false, + ) as bool); + + @override + set resamplingEnabled(bool? _resamplingEnabled) => super.noSuchMethod( + Invocation.setter( + #resamplingEnabled, + _resamplingEnabled, + ), + returnValueForMissingStub: null, + ); + + @override + Duration get samplingOffset => (super.noSuchMethod( + Invocation.getter(#samplingOffset), + returnValue: _FakeDuration_15( + this, + Invocation.getter(#samplingOffset), + ), + returnValueForMissingStub: _FakeDuration_15( + this, + Invocation.getter(#samplingOffset), + ), + ) as Duration); + + @override + set samplingOffset(Duration? _samplingOffset) => super.noSuchMethod( + Invocation.setter( + #samplingOffset, + _samplingOffset, + ), + returnValueForMissingStub: null, + ); + + @override + _i11.SamplingClock get samplingClock => (super.noSuchMethod( + Invocation.getter(#samplingClock), + returnValue: _FakeSamplingClock_19( + this, + Invocation.getter(#samplingClock), + ), + returnValueForMissingStub: _FakeSamplingClock_19( + this, + Invocation.getter(#samplingClock), + ), + ) as _i11.SamplingClock); + + @override + _i12.PipelineOwner get pipelineOwner => (super.noSuchMethod( + Invocation.getter(#pipelineOwner), + returnValue: _FakePipelineOwner_20( + this, + Invocation.getter(#pipelineOwner), + ), + returnValueForMissingStub: _FakePipelineOwner_20( + this, + Invocation.getter(#pipelineOwner), + ), + ) as _i12.PipelineOwner); + + @override + _i12.RenderView get renderView => (super.noSuchMethod( + Invocation.getter(#renderView), + returnValue: _FakeRenderView_21( + this, + Invocation.getter(#renderView), + ), + returnValueForMissingStub: _FakeRenderView_21( + this, + Invocation.getter(#renderView), + ), + ) as _i12.RenderView); + + @override + _i12.MouseTracker get mouseTracker => (super.noSuchMethod( + Invocation.getter(#mouseTracker), + returnValue: _FakeMouseTracker_22( + this, + Invocation.getter(#mouseTracker), + ), + returnValueForMissingStub: _FakeMouseTracker_22( + this, + Invocation.getter(#mouseTracker), + ), + ) as _i12.MouseTracker); + + @override + _i12.PipelineOwner get rootPipelineOwner => (super.noSuchMethod( + Invocation.getter(#rootPipelineOwner), + returnValue: _FakePipelineOwner_20( + this, + Invocation.getter(#rootPipelineOwner), + ), + returnValueForMissingStub: _FakePipelineOwner_20( + this, + Invocation.getter(#rootPipelineOwner), + ), + ) as _i12.PipelineOwner); + + @override + Iterable<_i12.RenderView> get renderViews => (super.noSuchMethod( + Invocation.getter(#renderViews), + returnValue: <_i12.RenderView>[], + returnValueForMissingStub: <_i12.RenderView>[], + ) as Iterable<_i12.RenderView>); + + @override + bool get sendFramesToEngine => (super.noSuchMethod( + Invocation.getter(#sendFramesToEngine), + returnValue: false, + returnValueForMissingStub: false, + ) as bool); + + @override + bool get semanticsEnabled => (super.noSuchMethod( + Invocation.getter(#semanticsEnabled), + returnValue: false, + returnValueForMissingStub: false, + ) as bool); + + @override + int get debugOutstandingSemanticsHandles => (super.noSuchMethod( + Invocation.getter(#debugOutstandingSemanticsHandles), + returnValue: 0, + returnValueForMissingStub: 0, + ) as int); + + @override + _i9.AccessibilityFeatures get accessibilityFeatures => (super.noSuchMethod( + Invocation.getter(#accessibilityFeatures), + returnValue: _FakeAccessibilityFeatures_23( + this, + Invocation.getter(#accessibilityFeatures), + ), + returnValueForMissingStub: _FakeAccessibilityFeatures_23( + this, + Invocation.getter(#accessibilityFeatures), + ), + ) as _i9.AccessibilityFeatures); + + @override + bool get disableAnimations => (super.noSuchMethod( + Invocation.getter(#disableAnimations), + returnValue: false, + returnValueForMissingStub: false, + ) as bool); + + @override + void initInstances() => super.noSuchMethod( + Invocation.method( + #initInstances, + [], + ), + returnValueForMissingStub: null, + ); + + @override + void initServiceExtensions() => super.noSuchMethod( + Invocation.method( + #initServiceExtensions, + [], + ), + returnValueForMissingStub: null, + ); + + @override + void addObserver(_i25.WidgetsBindingObserver? observer) => super.noSuchMethod( + Invocation.method( + #addObserver, + [observer], + ), + returnValueForMissingStub: null, + ); + + @override + bool removeObserver(_i25.WidgetsBindingObserver? observer) => + (super.noSuchMethod( + Invocation.method( + #removeObserver, + [observer], + ), + returnValue: false, + returnValueForMissingStub: false, + ) as bool); + + @override + _i4.Future<_i9.AppExitResponse> handleRequestAppExit() => (super.noSuchMethod( + Invocation.method( + #handleRequestAppExit, + [], + ), + returnValue: + _i4.Future<_i9.AppExitResponse>.value(_i9.AppExitResponse.exit), + returnValueForMissingStub: + _i4.Future<_i9.AppExitResponse>.value(_i9.AppExitResponse.exit), + ) as _i4.Future<_i9.AppExitResponse>); + + @override + void handleMetricsChanged() => super.noSuchMethod( + Invocation.method( + #handleMetricsChanged, + [], + ), + returnValueForMissingStub: null, + ); + + @override + void handleTextScaleFactorChanged() => super.noSuchMethod( + Invocation.method( + #handleTextScaleFactorChanged, + [], + ), + returnValueForMissingStub: null, + ); + + @override + void handlePlatformBrightnessChanged() => super.noSuchMethod( + Invocation.method( + #handlePlatformBrightnessChanged, + [], + ), + returnValueForMissingStub: null, + ); + + @override + void handleAccessibilityFeaturesChanged() => super.noSuchMethod( + Invocation.method( + #handleAccessibilityFeaturesChanged, + [], + ), + returnValueForMissingStub: null, + ); + + @override + void handleLocaleChanged() => super.noSuchMethod( + Invocation.method( + #handleLocaleChanged, + [], + ), + returnValueForMissingStub: null, + ); + + @override + void dispatchLocalesChanged(List<_i9.Locale>? locales) => super.noSuchMethod( + Invocation.method( + #dispatchLocalesChanged, + [locales], + ), + returnValueForMissingStub: null, + ); + + @override + void dispatchAccessibilityFeaturesChanged() => super.noSuchMethod( + Invocation.method( + #dispatchAccessibilityFeaturesChanged, + [], + ), + returnValueForMissingStub: null, + ); + + @override + _i4.Future handlePopRoute() => (super.noSuchMethod( + Invocation.method( + #handlePopRoute, + [], + ), + returnValue: _i4.Future.value(), + returnValueForMissingStub: _i4.Future.value(), + ) as _i4.Future); + + @override + _i4.Future handlePushRoute(String? route) => (super.noSuchMethod( + Invocation.method( + #handlePushRoute, + [route], + ), + returnValue: _i4.Future.value(), + returnValueForMissingStub: _i4.Future.value(), + ) as _i4.Future); + + @override + void handleAppLifecycleStateChanged(_i9.AppLifecycleState? state) => + super.noSuchMethod( + Invocation.method( + #handleAppLifecycleStateChanged, + [state], + ), + returnValueForMissingStub: null, + ); + + @override + void handleMemoryPressure() => super.noSuchMethod( + Invocation.method( + #handleMemoryPressure, + [], + ), + returnValueForMissingStub: null, + ); + + @override + void drawFrame() => super.noSuchMethod( + Invocation.method( + #drawFrame, + [], + ), + returnValueForMissingStub: null, + ); + + @override + _i13.Widget wrapWithDefaultView(_i13.Widget? rootWidget) => + (super.noSuchMethod( + Invocation.method( + #wrapWithDefaultView, + [rootWidget], + ), + returnValue: _FakeWidget_24( + this, + Invocation.method( + #wrapWithDefaultView, + [rootWidget], + ), + ), + returnValueForMissingStub: _FakeWidget_24( + this, + Invocation.method( + #wrapWithDefaultView, + [rootWidget], + ), + ), + ) as _i13.Widget); + + @override + void scheduleAttachRootWidget(_i13.Widget? rootWidget) => super.noSuchMethod( + Invocation.method( + #scheduleAttachRootWidget, + [rootWidget], + ), + returnValueForMissingStub: null, + ); + + @override + void attachRootWidget(_i13.Widget? rootWidget) => super.noSuchMethod( + Invocation.method( + #attachRootWidget, + [rootWidget], + ), + returnValueForMissingStub: null, + ); + + @override + void attachToBuildOwner(_i25.RootWidget? widget) => super.noSuchMethod( + Invocation.method( + #attachToBuildOwner, + [widget], + ), + returnValueForMissingStub: null, + ); + + @override + _i4.Future performReassemble() => (super.noSuchMethod( + Invocation.method( + #performReassemble, + [], + ), + returnValue: _i4.Future.value(), + returnValueForMissingStub: _i4.Future.value(), + ) as _i4.Future); + + @override + _i9.Locale? computePlatformResolvedLocale( + List<_i9.Locale>? supportedLocales) => + (super.noSuchMethod( + Invocation.method( + #computePlatformResolvedLocale, + [supportedLocales], + ), + returnValueForMissingStub: null, + ) as _i9.Locale?); + + @override + bool debugCheckZone(String? entryPoint) => (super.noSuchMethod( + Invocation.method( + #debugCheckZone, + [entryPoint], + ), + returnValue: false, + returnValueForMissingStub: false, + ) as bool); + + @override + _i4.Future lockEvents(_i4.Future Function()? callback) => + (super.noSuchMethod( + Invocation.method( + #lockEvents, + [callback], + ), + returnValue: _i4.Future.value(), + returnValueForMissingStub: _i4.Future.value(), + ) as _i4.Future); + + @override + void unlocked() => super.noSuchMethod( + Invocation.method( + #unlocked, + [], + ), + returnValueForMissingStub: null, + ); + + @override + _i4.Future reassembleApplication() => (super.noSuchMethod( + Invocation.method( + #reassembleApplication, + [], + ), + returnValue: _i4.Future.value(), + returnValueForMissingStub: _i4.Future.value(), + ) as _i4.Future); + + @override + void registerSignalServiceExtension({ + required String? name, + required _i8.AsyncCallback? callback, + }) => + super.noSuchMethod( + Invocation.method( + #registerSignalServiceExtension, + [], + { + #name: name, + #callback: callback, + }, + ), + returnValueForMissingStub: null, + ); + + @override + void registerBoolServiceExtension({ + required String? name, + required _i8.AsyncValueGetter? getter, + required _i8.AsyncValueSetter? setter, + }) => + super.noSuchMethod( + Invocation.method( + #registerBoolServiceExtension, + [], + { + #name: name, + #getter: getter, + #setter: setter, + }, + ), + returnValueForMissingStub: null, + ); + + @override + void registerNumericServiceExtension({ + required String? name, + required _i8.AsyncValueGetter? getter, + required _i8.AsyncValueSetter? setter, + }) => + super.noSuchMethod( + Invocation.method( + #registerNumericServiceExtension, + [], + { + #name: name, + #getter: getter, + #setter: setter, + }, + ), + returnValueForMissingStub: null, + ); + + @override + void postEvent( + String? eventKind, + Map? eventData, + ) => + super.noSuchMethod( + Invocation.method( + #postEvent, + [ + eventKind, + eventData, + ], + ), + returnValueForMissingStub: null, + ); + + @override + void registerStringServiceExtension({ + required String? name, + required _i8.AsyncValueGetter? getter, + required _i8.AsyncValueSetter? setter, + }) => + super.noSuchMethod( + Invocation.method( + #registerStringServiceExtension, + [], + { + #name: name, + #getter: getter, + #setter: setter, + }, + ), + returnValueForMissingStub: null, + ); + + @override + void registerServiceExtension({ + required String? name, + required _i8.ServiceExtensionCallback? callback, + }) => + super.noSuchMethod( + Invocation.method( + #registerServiceExtension, + [], + { + #name: name, + #callback: callback, + }, + ), + returnValueForMissingStub: null, + ); + + @override + _i10.BinaryMessenger createBinaryMessenger() => (super.noSuchMethod( + Invocation.method( + #createBinaryMessenger, + [], + ), + returnValue: _FakeBinaryMessenger_12( + this, + Invocation.method( + #createBinaryMessenger, + [], + ), + ), + returnValueForMissingStub: _FakeBinaryMessenger_12( + this, + Invocation.method( + #createBinaryMessenger, + [], + ), + ), + ) as _i10.BinaryMessenger); + + @override + _i4.Future handleSystemMessage(Object? systemMessage) => + (super.noSuchMethod( + Invocation.method( + #handleSystemMessage, + [systemMessage], + ), + returnValue: _i4.Future.value(), + returnValueForMissingStub: _i4.Future.value(), + ) as _i4.Future); + + @override + void initLicenses() => super.noSuchMethod( + Invocation.method( + #initLicenses, + [], + ), + returnValueForMissingStub: null, + ); + + @override + void evict(String? asset) => super.noSuchMethod( + Invocation.method( + #evict, + [asset], + ), + returnValueForMissingStub: null, + ); + + @override + void readInitialLifecycleStateFromNativeWindow() => super.noSuchMethod( + Invocation.method( + #readInitialLifecycleStateFromNativeWindow, + [], + ), + returnValueForMissingStub: null, + ); + + @override + _i4.Future<_i9.AppExitResponse> exitApplication( + _i9.AppExitType? exitType, [ + int? exitCode = 0, + ]) => + (super.noSuchMethod( + Invocation.method( + #exitApplication, + [ + exitType, + exitCode, + ], + ), + returnValue: + _i4.Future<_i9.AppExitResponse>.value(_i9.AppExitResponse.exit), + returnValueForMissingStub: + _i4.Future<_i9.AppExitResponse>.value(_i9.AppExitResponse.exit), + ) as _i4.Future<_i9.AppExitResponse>); + + @override + _i10.RestorationManager createRestorationManager() => (super.noSuchMethod( + Invocation.method( + #createRestorationManager, + [], + ), + returnValue: _FakeRestorationManager_14( + this, + Invocation.method( + #createRestorationManager, + [], + ), + ), + returnValueForMissingStub: _FakeRestorationManager_14( + this, + Invocation.method( + #createRestorationManager, + [], + ), + ), + ) as _i10.RestorationManager); + + @override + void setSystemUiChangeCallback(_i10.SystemUiChangeCallback? callback) => + super.noSuchMethod( + Invocation.method( + #setSystemUiChangeCallback, + [callback], + ), + returnValueForMissingStub: null, + ); + + @override + _i4.Future initializationComplete() => (super.noSuchMethod( + Invocation.method( + #initializationComplete, + [], + ), + returnValue: _i4.Future.value(), + returnValueForMissingStub: _i4.Future.value(), + ) as _i4.Future); + + @override + void addTimingsCallback(_i9.TimingsCallback? callback) => super.noSuchMethod( + Invocation.method( + #addTimingsCallback, + [callback], + ), + returnValueForMissingStub: null, + ); + + @override + void removeTimingsCallback(_i9.TimingsCallback? callback) => + super.noSuchMethod( + Invocation.method( + #removeTimingsCallback, + [callback], + ), + returnValueForMissingStub: null, + ); + + @override + void resetLifecycleState() => super.noSuchMethod( + Invocation.method( + #resetLifecycleState, + [], + ), + returnValueForMissingStub: null, + ); + + @override + _i4.Future scheduleTask( + _i26.TaskCallback? task, + _i26.Priority? priority, { + String? debugLabel, + _i27.Flow? flow, + }) => + (super.noSuchMethod( + Invocation.method( + #scheduleTask, + [ + task, + priority, + ], + { + #debugLabel: debugLabel, + #flow: flow, + }, + ), + returnValue: _i16.ifNotNull( + _i16.dummyValueOrNull( + this, + Invocation.method( + #scheduleTask, + [ + task, + priority, + ], + { + #debugLabel: debugLabel, + #flow: flow, + }, + ), + ), + (T v) => _i4.Future.value(v), + ) ?? + _FakeFuture_3( + this, + Invocation.method( + #scheduleTask, + [ + task, + priority, + ], + { + #debugLabel: debugLabel, + #flow: flow, + }, + ), + ), + returnValueForMissingStub: _i16.ifNotNull( + _i16.dummyValueOrNull( + this, + Invocation.method( + #scheduleTask, + [ + task, + priority, + ], + { + #debugLabel: debugLabel, + #flow: flow, + }, + ), + ), + (T v) => _i4.Future.value(v), + ) ?? + _FakeFuture_3( + this, + Invocation.method( + #scheduleTask, + [ + task, + priority, + ], + { + #debugLabel: debugLabel, + #flow: flow, + }, + ), + ), + ) as _i4.Future); + + @override + bool handleEventLoopCallback() => (super.noSuchMethod( + Invocation.method( + #handleEventLoopCallback, + [], + ), + returnValue: false, + returnValueForMissingStub: false, + ) as bool); + + @override + int scheduleFrameCallback( + _i26.FrameCallback? callback, { + bool? rescheduling = false, + }) => + (super.noSuchMethod( + Invocation.method( + #scheduleFrameCallback, + [callback], + {#rescheduling: rescheduling}, + ), + returnValue: 0, + returnValueForMissingStub: 0, + ) as int); + + @override + void cancelFrameCallbackWithId(int? id) => super.noSuchMethod( + Invocation.method( + #cancelFrameCallbackWithId, + [id], + ), + returnValueForMissingStub: null, + ); + + @override + bool debugAssertNoTransientCallbacks(String? reason) => (super.noSuchMethod( + Invocation.method( + #debugAssertNoTransientCallbacks, + [reason], + ), + returnValue: false, + returnValueForMissingStub: false, + ) as bool); + + @override + bool debugAssertNoPendingPerformanceModeRequests(String? reason) => + (super.noSuchMethod( + Invocation.method( + #debugAssertNoPendingPerformanceModeRequests, + [reason], + ), + returnValue: false, + returnValueForMissingStub: false, + ) as bool); + + @override + bool debugAssertNoTimeDilation(String? reason) => (super.noSuchMethod( + Invocation.method( + #debugAssertNoTimeDilation, + [reason], + ), + returnValue: false, + returnValueForMissingStub: false, + ) as bool); + + @override + void addPersistentFrameCallback(_i26.FrameCallback? callback) => + super.noSuchMethod( + Invocation.method( + #addPersistentFrameCallback, + [callback], + ), + returnValueForMissingStub: null, + ); + + @override + void addPostFrameCallback( + _i26.FrameCallback? callback, { + String? debugLabel = r'callback', + }) => + super.noSuchMethod( + Invocation.method( + #addPostFrameCallback, + [callback], + {#debugLabel: debugLabel}, + ), + returnValueForMissingStub: null, + ); + + @override + void ensureFrameCallbacksRegistered() => super.noSuchMethod( + Invocation.method( + #ensureFrameCallbacksRegistered, + [], + ), + returnValueForMissingStub: null, + ); + + @override + void ensureVisualUpdate() => super.noSuchMethod( + Invocation.method( + #ensureVisualUpdate, + [], + ), + returnValueForMissingStub: null, + ); + + @override + void scheduleFrame() => super.noSuchMethod( + Invocation.method( + #scheduleFrame, + [], + ), + returnValueForMissingStub: null, + ); + + @override + void scheduleForcedFrame() => super.noSuchMethod( + Invocation.method( + #scheduleForcedFrame, + [], + ), + returnValueForMissingStub: null, + ); + + @override + void scheduleWarmUpFrame() => super.noSuchMethod( + Invocation.method( + #scheduleWarmUpFrame, + [], + ), + returnValueForMissingStub: null, + ); + + @override + void resetEpoch() => super.noSuchMethod( + Invocation.method( + #resetEpoch, + [], + ), + returnValueForMissingStub: null, + ); + + @override + void handleBeginFrame(Duration? rawTimeStamp) => super.noSuchMethod( + Invocation.method( + #handleBeginFrame, + [rawTimeStamp], + ), + returnValueForMissingStub: null, + ); + + @override + _i26.PerformanceModeRequestHandle? requestPerformanceMode( + _i9.DartPerformanceMode? mode) => + (super.noSuchMethod( + Invocation.method( + #requestPerformanceMode, + [mode], + ), + returnValueForMissingStub: null, + ) as _i26.PerformanceModeRequestHandle?); + + @override + void handleDrawFrame() => super.noSuchMethod( + Invocation.method( + #handleDrawFrame, + [], + ), + returnValueForMissingStub: null, + ); + + @override + void cancelPointer(int? pointer) => super.noSuchMethod( + Invocation.method( + #cancelPointer, + [pointer], + ), + returnValueForMissingStub: null, + ); + + @override + void handlePointerEvent(_i10.PointerEvent? event) => super.noSuchMethod( + Invocation.method( + #handlePointerEvent, + [event], + ), + returnValueForMissingStub: null, + ); + + @override + void hitTestInView( + _i11.HitTestResult? result, + _i9.Offset? position, + int? viewId, + ) => + super.noSuchMethod( + Invocation.method( + #hitTestInView, + [ + result, + position, + viewId, + ], + ), + returnValueForMissingStub: null, + ); + + @override + void hitTest( + _i11.HitTestResult? result, + _i9.Offset? position, + ) => + super.noSuchMethod( + Invocation.method( + #hitTest, + [ + result, + position, + ], + ), + returnValueForMissingStub: null, + ); + + @override + void dispatchEvent( + _i10.PointerEvent? event, + _i11.HitTestResult? hitTestResult, + ) => + super.noSuchMethod( + Invocation.method( + #dispatchEvent, + [ + event, + hitTestResult, + ], + ), + returnValueForMissingStub: null, + ); + + @override + void handleEvent( + _i10.PointerEvent? event, + _i11.HitTestEntry<_i11.HitTestTarget>? entry, + ) => + super.noSuchMethod( + Invocation.method( + #handleEvent, + [ + event, + entry, + ], + ), + returnValueForMissingStub: null, + ); + + @override + void resetGestureBinding() => super.noSuchMethod( + Invocation.method( + #resetGestureBinding, + [], + ), + returnValueForMissingStub: null, + ); + + @override + _i12.PipelineOwner createRootPipelineOwner() => (super.noSuchMethod( + Invocation.method( + #createRootPipelineOwner, + [], + ), + returnValue: _FakePipelineOwner_20( + this, + Invocation.method( + #createRootPipelineOwner, + [], + ), + ), + returnValueForMissingStub: _FakePipelineOwner_20( + this, + Invocation.method( + #createRootPipelineOwner, + [], + ), + ), + ) as _i12.PipelineOwner); + + @override + void addRenderView(_i12.RenderView? view) => super.noSuchMethod( + Invocation.method( + #addRenderView, + [view], + ), + returnValueForMissingStub: null, + ); + + @override + void removeRenderView(_i12.RenderView? view) => super.noSuchMethod( + Invocation.method( + #removeRenderView, + [view], + ), + returnValueForMissingStub: null, + ); + + @override + _i12.ViewConfiguration createViewConfigurationFor( + _i12.RenderView? renderView) => + (super.noSuchMethod( + Invocation.method( + #createViewConfigurationFor, + [renderView], + ), + returnValue: _FakeViewConfiguration_25( + this, + Invocation.method( + #createViewConfigurationFor, + [renderView], + ), + ), + returnValueForMissingStub: _FakeViewConfiguration_25( + this, + Invocation.method( + #createViewConfigurationFor, + [renderView], + ), + ), + ) as _i12.ViewConfiguration); + + @override + void initMouseTracker([_i12.MouseTracker? tracker]) => super.noSuchMethod( + Invocation.method( + #initMouseTracker, + [tracker], + ), + returnValueForMissingStub: null, + ); + + @override + void performSemanticsAction(_i9.SemanticsActionEvent? action) => + super.noSuchMethod( + Invocation.method( + #performSemanticsAction, + [action], + ), + returnValueForMissingStub: null, + ); + + @override + void deferFirstFrame() => super.noSuchMethod( + Invocation.method( + #deferFirstFrame, + [], + ), + returnValueForMissingStub: null, + ); + + @override + void allowFirstFrame() => super.noSuchMethod( + Invocation.method( + #allowFirstFrame, + [], + ), + returnValueForMissingStub: null, + ); + + @override + void resetFirstFrameSent() => super.noSuchMethod( + Invocation.method( + #resetFirstFrameSent, + [], + ), + returnValueForMissingStub: null, + ); + + @override + void addSemanticsEnabledListener(_i9.VoidCallback? listener) => + super.noSuchMethod( + Invocation.method( + #addSemanticsEnabledListener, + [listener], + ), + returnValueForMissingStub: null, + ); + + @override + void removeSemanticsEnabledListener(_i9.VoidCallback? listener) => + super.noSuchMethod( + Invocation.method( + #removeSemanticsEnabledListener, + [listener], + ), + returnValueForMissingStub: null, + ); + + @override + _i12.SemanticsHandle ensureSemantics() => (super.noSuchMethod( + Invocation.method( + #ensureSemantics, + [], + ), + returnValue: _FakeSemanticsHandle_26( + this, + Invocation.method( + #ensureSemantics, + [], + ), + ), + returnValueForMissingStub: _FakeSemanticsHandle_26( + this, + Invocation.method( + #ensureSemantics, + [], + ), + ), + ) as _i12.SemanticsHandle); + + @override + _i9.SemanticsUpdateBuilder createSemanticsUpdateBuilder() => + (super.noSuchMethod( + Invocation.method( + #createSemanticsUpdateBuilder, + [], + ), + returnValue: _FakeSemanticsUpdateBuilder_27( + this, + Invocation.method( + #createSemanticsUpdateBuilder, + [], + ), + ), + returnValueForMissingStub: _FakeSemanticsUpdateBuilder_27( + this, + Invocation.method( + #createSemanticsUpdateBuilder, + [], + ), + ), + ) as _i9.SemanticsUpdateBuilder); +} diff --git a/packages/core/test/plugin_test.dart b/packages/core/test/plugin_test.dart new file mode 100644 index 0000000..6439355 --- /dev/null +++ b/packages/core/test/plugin_test.dart @@ -0,0 +1,173 @@ +import 'package:flutter_test/flutter_test.dart'; +import 'package:mockito/mockito.dart'; +import 'package:segment_analytics/analytics.dart'; +import 'package:segment_analytics/analytics_platform_interface.dart'; +import 'package:segment_analytics/event.dart'; +import 'package:segment_analytics/plugin.dart'; +import 'package:segment_analytics/state.dart'; + +import 'mocks/mocks.dart'; +import 'mocks/mocks.mocks.dart'; + +void main() { + TestWidgetsFlutterBinding.ensureInitialized(); + // Define arguments + String writeKey = '123'; + List batch = [ + TrackEvent("Event 1"), + TrackEvent("Event 2"), + TrackEvent("Event 3"), + ]; + group('Plugin Tests', () { + late Plugin plugin; + late Analytics mockAnalytics; + late MockHTTPClient httpClient; + + setUp(() async { + plugin = MockPlugin(PluginType.after); // Ejemplo con un tipo arbitrario + AnalyticsPlatform.instance = MockPlatform(); + + httpClient = Mocks.httpClient(); + when(httpClient.settingsFor(writeKey)) + .thenAnswer((_) => Future.value(SegmentAPISettings({}))); + when(httpClient.startBatchUpload(writeKey, batch)) + .thenAnswer((_) => Future.value(true)); + mockAnalytics = Analytics( + Configuration("123", + trackApplicationLifecycleEvents: false, + token: "abcdef12345"), + Mocks.store(), + httpClient: (_) => httpClient); + await mockAnalytics.init(); + }); + + test('Plugin clear should set analytics to null', () { + plugin.configure(mockAnalytics); + plugin.clear(); + expect(plugin.analytics, isNull); + }); + + test('Plugin configure should set analytics', () { + plugin.configure(mockAnalytics); + expect(plugin.analytics, equals(mockAnalytics)); + }); + + test('Plugin execute should return the same event by default', () async { + final mockEvent = TrackEvent("track test event"); + final result = await plugin.execute(mockEvent); + expect(result, equals(mockEvent)); + }); + + test('Plugin shutdown should set analytics to null', () { + plugin.configure(mockAnalytics); + plugin.shutdown(); + expect(plugin.analytics, isNull); + }); + }); + + group('EventPlugin Tests', () { + late EventPlugin eventPlugin; + late Analytics mockAnalytics; + late MockHTTPClient httpClient; + + setUp(() async { + eventPlugin = MockEventPlugin(PluginType.after); + AnalyticsPlatform.instance = MockPlatform(); + + httpClient = Mocks.httpClient(); + when(httpClient.settingsFor(writeKey)) + .thenAnswer((_) => Future.value(SegmentAPISettings({}))); + when(httpClient.startBatchUpload(writeKey, batch)) + .thenAnswer((_) => Future.value(true)); + mockAnalytics = Analytics( + Configuration("123", + trackApplicationLifecycleEvents: false, + token: "abcdef12345"), + Mocks.store(), + httpClient: (_) => httpClient); + await mockAnalytics.init(); + }); + + test('EventPlugin execute should call the correct method based on event type', () async { + final mockIdentifyEvent = eventFromJson({"type":"identify"}); + final result = await eventPlugin.execute(mockIdentifyEvent); + expect(result, equals(mockIdentifyEvent)); + + final mockTrackEvent = eventFromJson({"type":"track", "event": "Test track json event"}); + final resultTrack = await eventPlugin.execute(mockTrackEvent); + expect(resultTrack, equals(mockTrackEvent)); + + final mockAliasEvent = eventFromJson({"type":"alias", "previousId": "Test track json event"}); + final resultAlias = await eventPlugin.execute(mockAliasEvent); + expect(resultAlias, equals(mockAliasEvent)); + + final mockGroupEvent = eventFromJson({"type":"group", "groupId": "Test track json event"}); + final resultGroup = await eventPlugin.execute(mockGroupEvent); + expect(resultGroup, equals(mockGroupEvent)); + + final mockScreenEvent = eventFromJson({"type":"screen", "name": "Test track json event"}); + final resultScreen = await eventPlugin.execute(mockScreenEvent); + expect(resultScreen, equals(mockScreenEvent)); + + }); + + test('EventPlugin flush should be callable', () async { + await eventPlugin.flush(); + }); + + test('EventPlugin reset should be callable', () { + eventPlugin.reset(); + }); + + }); + + group('DestinationPlugin Tests', () { + late DestinationPlugin destinationPlugin; + late Analytics mockAnalytics; + late MockHTTPClient httpClient; + + setUp(() async { + destinationPlugin = MockDestinationPlugin("1234567890"); + AnalyticsPlatform.instance = MockPlatform(); + + httpClient = Mocks.httpClient(); + when(httpClient.settingsFor(writeKey)) + .thenAnswer((_) => Future.value(SegmentAPISettings({}))); + when(httpClient.startBatchUpload(writeKey, batch)) + .thenAnswer((_) => Future.value(true)); + mockAnalytics = Analytics( + Configuration("123", + trackApplicationLifecycleEvents: false, + token: "abcdef12345"), + Mocks.store(), + httpClient: (_) => httpClient); + await mockAnalytics.init(); + }); + + test('DestinationPlugin execute should call the correct method based on event type', () async { + final mockScreenEvent = eventFromJson({"type":"screen", "name": "Test track json event"}); + final resultScreen = await destinationPlugin.execute(mockScreenEvent); + expect(resultScreen, isNull); + + }); + + test('DestinationPlugin flush should be callable', () async { + await destinationPlugin.flush(); + }); + + test('DestinationPlugin reset should be callable', () { + destinationPlugin.reset(); + }); + + }); +} + +class MockPlugin extends Plugin { + MockPlugin(super.type); +} +class MockEventPlugin extends EventPlugin { + MockEventPlugin(super.type); +} +class MockDestinationPlugin extends DestinationPlugin { + MockDestinationPlugin(super.type); +} diff --git a/packages/core/test/plugins/destination_metadata_enrichment_test.dart b/packages/core/test/plugins/destination_metadata_enrichment_test.dart new file mode 100644 index 0000000..1dc4790 --- /dev/null +++ b/packages/core/test/plugins/destination_metadata_enrichment_test.dart @@ -0,0 +1,52 @@ +import 'package:flutter_test/flutter_test.dart'; +import 'package:segment_analytics/analytics.dart'; +import 'package:segment_analytics/analytics_platform_interface.dart'; +import 'package:segment_analytics/event.dart'; +import 'package:mockito/mockito.dart'; +import 'package:segment_analytics/plugins/destination_metadata_enrichment.dart'; +import 'package:segment_analytics/state.dart'; + +import '../mocks/mocks.dart'; + +void main() { + String writeKey = '123'; + List batch = [ + TrackEvent("Event 1"), + TrackEvent("Event 2"), + TrackEvent("Event 3"), + ]; + final Map settings = { + 'integrations': "12345abcdef" + }; + group('DestinationMetadataEnrichment', () { + late DestinationMetadataEnrichment plugin; + late RawEvent event; + late Analytics mockAnalytics; + setUp(() async{ + AnalyticsPlatform.instance = MockPlatform(); + plugin = DestinationMetadataEnrichment('destinationKey'); + final httpClient = Mocks.httpClient(); + when(httpClient.settingsFor(writeKey)) + .thenAnswer((_) => Future.value(SegmentAPISettings({}))); + when(httpClient.startBatchUpload(writeKey, batch)) + .thenAnswer((_) => Future.value(true)); + mockAnalytics = Analytics( + Configuration("123", + trackApplicationLifecycleEvents: false, + appStateStream: () => Mocks.streamSubscription()), + Mocks.store(), + httpClient: (_) => httpClient); + await mockAnalytics.init(); + mockAnalytics.state.integrations.state = settings; + mockAnalytics.addPlugin(plugin); + // ignore: invalid_use_of_protected_member + plugin.pAnalytics = mockAnalytics; + event = TrackEvent('test_event'); + }); + + test('bundled keys are correctly added excluding the destinationKey', () async { + final resultEvent = await plugin.execute(event); + expect(resultEvent.metadata?.bundled, isNot(contains('unbundledIntegrations'))); + }); + }); +} diff --git a/packages/core/test/plugins/event_logger_test.dart b/packages/core/test/plugins/event_logger_test.dart new file mode 100644 index 0000000..80a8782 --- /dev/null +++ b/packages/core/test/plugins/event_logger_test.dart @@ -0,0 +1,39 @@ +import 'package:flutter_test/flutter_test.dart'; +import 'package:logger/logger.dart'; +import 'package:mockito/mockito.dart'; +import 'package:segment_analytics/analytics.dart'; +import 'package:segment_analytics/event.dart'; +import 'package:segment_analytics/plugins/event_logger.dart'; + +import '../mocks/mocks.mocks.dart'; + + +class MockAnalytics extends Mock implements Analytics {} + +void main() { + group('EventLogger Tests', () { + late EventLogger eventLogger; + late MockAnalytics mockAnalytics; + late MockLogger mockLogger; + + setUp(() { + eventLogger = EventLogger(); + mockAnalytics = MockAnalytics(); + mockLogger = MockLogger(); + // Logger.instance = mockLogger; // Usa el mock de Logger + }); + + test('should configure with analytics instance', () { + eventLogger.configure(mockAnalytics); + // ignore: invalid_use_of_protected_member + expect(eventLogger.pAnalytics, mockAnalytics); + }); + + test('should log track event correctly', () async { + final trackEvent = TrackEvent('Test Event'); + await eventLogger.execute(trackEvent); + final logMessage = verifyNever(mockLogger.log(Level.debug, "Test")).captured; + expect(logMessage, []); + }); + }); +} diff --git a/packages/core/test/plugins/inject_context_test.dart b/packages/core/test/plugins/inject_context_test.dart new file mode 100644 index 0000000..8a1077d --- /dev/null +++ b/packages/core/test/plugins/inject_context_test.dart @@ -0,0 +1,50 @@ +// test/inject_token_test.dart +import 'package:flutter_test/flutter_test.dart'; +import 'package:mockito/mockito.dart'; +import 'package:segment_analytics/analytics.dart'; +import 'package:segment_analytics/analytics_platform_interface.dart'; +import 'package:segment_analytics/event.dart'; +import 'package:segment_analytics/plugins/inject_context.dart'; +import 'package:segment_analytics/state.dart'; + +import '../mocks/mocks.dart'; + + +void main() { + String writeKey = '123'; + List batch = [ + TrackEvent("Event 1"), + TrackEvent("Event 2"), + TrackEvent("Event 3"), + ]; + group('InjectContext Tests', () { + late InjectContext injectContext; + + setUp(() async { + AnalyticsPlatform.instance = MockPlatform(); + }); + + test('should inject token into event context', () async { + final httpClient = Mocks.httpClient(); + when(httpClient.settingsFor(writeKey)) + .thenAnswer((_) => Future.value(SegmentAPISettings({}))); + when(httpClient.startBatchUpload(writeKey, batch)) + .thenAnswer((_) => Future.value(true)); + Analytics analytics = Analytics( + Configuration("123", + trackApplicationLifecycleEvents: false, + appStateStream: () => Mocks.streamSubscription()), + Mocks.store(), + httpClient: (_) => httpClient); + await analytics.init(); + + injectContext = InjectContext(); + // ignore: invalid_use_of_protected_member + injectContext.pAnalytics = analytics; + + final resultEvent = await injectContext.execute(TrackEvent("Test")); + + expect(resultEvent.context!.instanceId, isNotNull); + }); + }); +} diff --git a/packages/core/test/plugins/inject_token_test.dart b/packages/core/test/plugins/inject_token_test.dart new file mode 100644 index 0000000..e7c4185 --- /dev/null +++ b/packages/core/test/plugins/inject_token_test.dart @@ -0,0 +1,50 @@ +// test/inject_token_test.dart +import 'package:flutter_test/flutter_test.dart'; +import 'package:mockito/mockito.dart'; +import 'package:segment_analytics/analytics.dart'; +import 'package:segment_analytics/analytics_platform_interface.dart'; +import 'package:segment_analytics/event.dart'; +import 'package:segment_analytics/plugins/inject_token.dart'; +import 'package:segment_analytics/state.dart'; + +import '../mocks/mocks.dart'; + + +void main() { + String writeKey = '123'; + List batch = [ + TrackEvent("Event 1"), + TrackEvent("Event 2"), + TrackEvent("Event 3"), + ]; + group('InjectToken Tests', () { + late InjectToken injectToken; + + setUp(() async { + AnalyticsPlatform.instance = MockPlatform(); + }); + + test('should inject token into event context', () async { + final httpClient = Mocks.httpClient(); + when(httpClient.settingsFor(writeKey)) + .thenAnswer((_) => Future.value(SegmentAPISettings({}))); + when(httpClient.startBatchUpload(writeKey, batch)) + .thenAnswer((_) => Future.value(true)); + Analytics analytics = Analytics( + Configuration("123", + trackApplicationLifecycleEvents: false, + appStateStream: () => Mocks.streamSubscription()), + Mocks.store(), + httpClient: (_) => httpClient); + await analytics.init(); + + injectToken = InjectToken('test-token'); + // ignore: invalid_use_of_protected_member + injectToken.pAnalytics = analytics; + + final resultEvent = await injectToken.execute(TrackEvent("Test")); + + expect(resultEvent.context!.device.token, 'test-token'); + }); + }); +} diff --git a/packages/core/test/plugins/inject_user_info_test.dart b/packages/core/test/plugins/inject_user_info_test.dart new file mode 100644 index 0000000..e8cac90 --- /dev/null +++ b/packages/core/test/plugins/inject_user_info_test.dart @@ -0,0 +1,58 @@ +// test/inject_token_test.dart +import 'package:flutter_test/flutter_test.dart'; +import 'package:mockito/mockito.dart'; +import 'package:segment_analytics/analytics.dart'; +import 'package:segment_analytics/analytics_platform_interface.dart'; +import 'package:segment_analytics/event.dart'; +import 'package:segment_analytics/plugins/inject_user_info.dart'; +import 'package:segment_analytics/state.dart'; + +import '../mocks/mocks.dart'; + + +void main() { + String writeKey = '123'; + List batch = [ + TrackEvent("Event 1"), + TrackEvent("Event 2"), + TrackEvent("Event 3"), + ]; + group('Inject User Info Tests', () { + late InjectUserInfo userInfo; + final UserTraits userT = UserTraits(firstName: "Christy", custom: {"myCustomTrait": "customValue"}); + final GroupTraits groupT = GroupTraits(name: "abc"); + + setUp(() async { + AnalyticsPlatform.instance = MockPlatform(); + }); + + test('should inject user info into event context', () async { + final httpClient = Mocks.httpClient(); + when(httpClient.settingsFor(writeKey)) + .thenAnswer((_) => Future.value(SegmentAPISettings({}))); + when(httpClient.startBatchUpload(writeKey, batch)) + .thenAnswer((_) => Future.value(true)); + Analytics analytics = Analytics( + Configuration("123", + trackApplicationLifecycleEvents: false, + appStateStream: () => Mocks.streamSubscription()), + Mocks.store(), + httpClient: (_) => httpClient); + await analytics.init(); + + userInfo = InjectUserInfo(); + UserInfo user = UserInfo("abcdefg123456789", userTraits: userT, groupTraits: groupT); + analytics.state.userInfo.setState(user); + // ignore: invalid_use_of_protected_member + userInfo.pAnalytics = analytics; + + final resultEvent = await userInfo.execute(IdentifyEvent(traits: userT)); + final resultEvent1 = await userInfo.execute(AliasEvent("123")); + final resultEvent2 = await userInfo.execute(GroupEvent("abc", traits: groupT)); + + expect(resultEvent.anonymousId, 'abcdefg123456789'); + expect(resultEvent1.anonymousId, 'abcdefg123456789'); + expect(resultEvent2.anonymousId, 'abcdefg123456789'); + }); + }); +} diff --git a/packages/core/test/plugins/segment_destination_test.dart b/packages/core/test/plugins/segment_destination_test.dart new file mode 100644 index 0000000..f3c6511 --- /dev/null +++ b/packages/core/test/plugins/segment_destination_test.dart @@ -0,0 +1,65 @@ +import 'package:flutter_test/flutter_test.dart'; +import 'package:mockito/mockito.dart'; +import 'package:segment_analytics/analytics.dart'; +import 'package:segment_analytics/analytics_platform_interface.dart'; +import 'package:segment_analytics/event.dart'; +import 'package:segment_analytics/plugins/segment_destination.dart'; +import 'package:segment_analytics/state.dart'; +import '../mocks/mocks.dart'; +import '../mocks/mocks.mocks.dart'; + +void main() { + String writeKey = '123'; + List batch = [ + TrackEvent("Event 1"), + TrackEvent("Event 2"), + TrackEvent("Event 3"), + ]; + final Map settings = { + 'key': "12345abcdef" + }; + late SegmentDestination segmentDestination; + late MockHTTPClient httpClient; + setUp(() async { + AnalyticsPlatform.instance = MockPlatform(); + segmentDestination = SegmentDestination(); + httpClient = Mocks.httpClient(); + when(httpClient.settingsFor(writeKey)) + .thenAnswer((_) => Future.value(SegmentAPISettings({}))); + when(httpClient.startBatchUpload(writeKey, batch)) + .thenAnswer((_) => Future.value(true)); + Analytics analytics = Analytics( + Configuration("123", + trackApplicationLifecycleEvents: false, + appStateStream: () => Mocks.streamSubscription()), + Mocks.store(), + httpClient: (_) => httpClient); + analytics.state.integrations.state = settings; + await analytics.init(); + segmentDestination.configure(analytics); + }); + + test('Test _sendEvents with empty events list', () async { + await segmentDestination.sendEvents([]); + // No assertions needed since it should just return + }); + + test('Test _sendEvents with non-empty events list', () async { + final List events = [ + TrackEvent('test_event', properties: {}), + ]; + + when(httpClient.startBatchUpload(any, any, host: anyNamed('host'))) + .thenAnswer((_) async => true); + + await segmentDestination.sendEvents(events); + + verify(httpClient.startBatchUpload(any, any, host: anyNamed('host'))) + .called(1); + }); + + test('Test flush method', () async { + await segmentDestination.flush(); + // Assert the flush method calls the flush of the queue plugin + }); +} diff --git a/packages/core/test/state_test.dart b/packages/core/test/state_test.dart new file mode 100644 index 0000000..c0b239e --- /dev/null +++ b/packages/core/test/state_test.dart @@ -0,0 +1,43 @@ +import 'package:segment_analytics/state.dart'; +import 'package:flutter_test/flutter_test.dart'; + +void main() { + + group("State", () { + + test('UserInfo fromJson method', () { + final result = UserInfo.fromJson({"anonymousId":"1234567890"}); + expect(result.anonymousId, "1234567890"); + }); + + test('SegmentAPISettings fromJson method', () { + final result = SegmentAPISettings.fromJson({"integrations":{"integrations":"1234567890"}}); + expect(result.integrations, {"integrations":"1234567890"}); + }); + + test('RoutingRule fromJson method', () { + final result = RoutingRule.fromJson({"scope":"scope", "target_type":"1234567890"}); + expect(result.targetType, "1234567890"); + }); + + test('MatchTransformerer fromJson method', () { + final result = Transformer.fromJson({"type":"scope"}); + expect(result.type, "scope"); + }); + + test('TransformerConfig fromJson method', () { + final result = TransformerConfig.fromJson({"allow": {"name":["event"]}}); + expect(result.allow, {"name":["event"]}); + }); + + test('TransformerConfigSample fromJson method', () { + final result = TransformerConfigSample.fromJson({"percent": 1, "path":"/test"}); + expect(result.percent, 1); + }); + + test('TransformerConfigMap fromJson method', () { + final result = TransformerConfigMap.fromJson({"set": "/test"}); + expect(result.set, "/test"); + }); + }); +} diff --git a/packages/core/test/timeline_test.dart b/packages/core/test/timeline_test.dart new file mode 100644 index 0000000..7774ec5 --- /dev/null +++ b/packages/core/test/timeline_test.dart @@ -0,0 +1,76 @@ +import 'package:flutter_test/flutter_test.dart'; +import 'package:mockito/mockito.dart'; +import 'package:segment_analytics/plugin.dart'; +import 'package:segment_analytics/timeline.dart'; + +// Clases Mock +class MockPlugin extends Mock implements Plugin { + @override + final PluginType type; + + MockPlugin(this.type); +} + +void main() { + group('Timeline Tests', () { + late Timeline timeline; + late MockPlugin mockPluginBefore; + late MockPlugin mockPluginEnrichment; + + setUp(() { + timeline = Timeline(); + mockPluginBefore = MockPlugin(PluginType.before); + mockPluginEnrichment = MockPlugin(PluginType.enrichment); + }); + + test('add should add plugin to correct type', () { + timeline.add(mockPluginBefore); + expect(timeline.getPlugins(PluginType.before), contains(mockPluginBefore)); + }); + + test('remove should remove plugin from correct type', () { + timeline.add(mockPluginBefore); + timeline.remove(mockPluginBefore); + expect(timeline.getPlugins(PluginType.before), isNot(contains(mockPluginBefore))); + }); + + test('apply should execute closure on all plugins', () { + timeline.add(mockPluginBefore); + timeline.add(mockPluginEnrichment); + + int closureCallCount = 0; + timeline.apply((plugin) { + closureCallCount++; + }); + + expect(closureCallCount, 2); + }); + + test('getPlugins should return plugins of specified type', () { + timeline.add(mockPluginBefore); + timeline.add(mockPluginEnrichment); + + final beforePlugins = timeline.getPlugins(PluginType.before); + final enrichmentPlugins = timeline.getPlugins(PluginType.enrichment); + final emptyPlugins = timeline.getPlugins(null); + + expect(beforePlugins, contains(mockPluginBefore)); + expect(enrichmentPlugins, contains(mockPluginEnrichment)); + expect(emptyPlugins, contains(mockPluginBefore)); + }); + + test('getPluginsWithFlush sholudd return plugins', (){ + timeline.add(mockPluginBefore); + timeline.add(mockPluginEnrichment); + final list = getPluginsWithFlush(timeline); + expect(list.length, 0); + }); + + test('getPluginsWithReset sholudd return plugins', (){ + timeline.add(mockPluginBefore); + timeline.add(mockPluginEnrichment); + final list = getPluginsWithReset(timeline); + expect(list.length, 0); + }); + }); +} diff --git a/packages/core/test/utils/http_client_test.dart b/packages/core/test/utils/http_client_test.dart index 7d2cc9a..3a97474 100644 --- a/packages/core/test/utils/http_client_test.dart +++ b/packages/core/test/utils/http_client_test.dart @@ -46,5 +46,35 @@ void main() { verify(mockRequest.send()); verify((LogFactory.logger as MockLogTarget).parseLog(captureAny)); }); + + test("It logs on bad response 429", () async { + final mockRequest = Mocks.request(); + when(mockRequest.send()).thenAnswer( + (_) => Future.value(StreamedResponse(const Stream.empty(), 429))); + when(mockRequest.url).thenAnswer((_) => Uri.parse("http://segment.io")); + HTTPClient client = HTTPClient(Analytics( + Configuration("123", requestFactory: (_) => mockRequest), + Mocks.store())); + + await client.startBatchUpload("123", []); + + verify(mockRequest.send()); + verify((LogFactory.logger as MockLogTarget).parseLog(captureAny)); + }); + + test("It logs on bad response 500", () async { + final mockRequest = Mocks.request(); + when(mockRequest.send()).thenAnswer( + (_) => Future.value(StreamedResponse(const Stream.empty(), 500))); + when(mockRequest.url).thenAnswer((_) => Uri.parse("http://segment.io")); + HTTPClient client = HTTPClient(Analytics( + Configuration("123", requestFactory: (_) => mockRequest), + Mocks.store())); + + await client.startBatchUpload("123", []); + + verify(mockRequest.send()); + verify((LogFactory.logger as MockLogTarget).parseLog(captureAny)); + }); }); } diff --git a/packages/core/test/utils/lifecycle/fgbg_lifecycle_test.dart b/packages/core/test/utils/lifecycle/fgbg_lifecycle_test.dart new file mode 100644 index 0000000..1810441 --- /dev/null +++ b/packages/core/test/utils/lifecycle/fgbg_lifecycle_test.dart @@ -0,0 +1,81 @@ +import 'dart:async'; +import 'package:flutter_test/flutter_test.dart'; +import 'package:segment_analytics/utils/lifecycle/fgbg.dart'; +import 'package:segment_analytics/utils/lifecycle/lifecycle.dart'; +import 'package:flutter_fgbg/flutter_fgbg.dart'; + + +void main() { + group('FGBGLifecycle Tests', () { + late FGBGLifecycle lifecycle; + late StreamController controller; + + setUp(() { + controller = StreamController.broadcast(); + lifecycle = FGBGLifecycle(controller.stream); + }); + + tearDown(() { + controller.close(); + }); + + test('should map FGBGType.foreground to AppStatus.foreground', () async { + final events = []; + lifecycle.listen(events.add); + + controller.add(FGBGType.foreground); + + await Future.delayed(Duration.zero); // Allow stream event to propagate + + expect(events, [AppStatus.foreground]); + }); + + test('should map FGBGType.background to AppStatus.background', () async { + final events = []; + lifecycle.listen(events.add); + + controller.add(FGBGType.background); + + await Future.delayed(Duration.zero); // Allow stream event to propagate + + expect(events, [AppStatus.background]); + }); + + test('should handle multiple events', () async { + final events = []; + lifecycle.listen(events.add); + + controller.add(FGBGType.foreground); + controller.add(FGBGType.background); + controller.add(FGBGType.foreground); + + await Future.delayed(Duration.zero); // Allow stream events to propagate + + expect(events, [ + AppStatus.foreground, + AppStatus.background, + AppStatus.foreground, + ]); + }); + + test('should call onDone when stream is closed', () async { + bool isDone = false; + lifecycle.listen(null, onDone: () => isDone = true); + + await controller.close(); // Close the stream + + expect(isDone, true); + }); + + test('should call onError on stream error', () async { + dynamic error; + lifecycle.listen(null, onError: (e) => error = e); + + controller.addError(Exception('Test error')); + + await Future.delayed(Duration.zero); // Allow error to propagate + + expect(error, isA()); + }); + }); +} diff --git a/packages/core/test/utils/lifecycle/lifecycle_test.dart b/packages/core/test/utils/lifecycle/lifecycle_test.dart new file mode 100644 index 0000000..3c31758 --- /dev/null +++ b/packages/core/test/utils/lifecycle/lifecycle_test.dart @@ -0,0 +1,67 @@ +import 'package:flutter/foundation.dart'; +import 'package:flutter_test/flutter_test.dart'; +import 'package:segment_analytics/utils/lifecycle/lifecycle.dart'; +import 'package:segment_analytics/utils/lifecycle/widget_observer.dart'; + +enum TestPlatform { android, ios, web, macos, windows, linux } + +TestPlatform? testPlatform; +bool? testIsWeb; + +bool get isWeb => testIsWeb ?? kIsWeb; + + +void main() { + setUp(() { + testPlatform = null; + testIsWeb = null; + }); + + group('LifeCycle Tests', () { + // test('should return FGBGLifecycle for Android', () { + // when(Platform.isAndroid).thenReturn(true); + // when(Platform.isIOS).thenReturn(false); + // when(kIsWeb).thenReturn(false); + + // final lifecycle = getLifecycleStream(); + + // expect(lifecycle, isA()); + // }); + + // test('should return FGBGLifecycle for iOS', () { + // when(Platform.isAndroid).thenReturn(false); + // when(Platform.isIOS).thenReturn(true); + // when(kIsWeb).thenReturn(false); + + // final lifecycle = getLifecycleStream(); + // expect(lifecycle, isA()); + // }); + + test('should return WidgetObserverLifecycle for Web', () { + testIsWeb = true; + final lifecycle = getLifecycleStream(); + expect(lifecycle, isA()); + }); + + test('should return WidgetObserverLifecycle for macOS', () { + testPlatform = TestPlatform.macos; + testIsWeb = false; + final lifecycle = getLifecycleStream(); + expect(lifecycle, isA()); + }); + + test('should return WidgetObserverLifecycle for Windows', () { + testPlatform = TestPlatform.windows; + testIsWeb = false; + final lifecycle = getLifecycleStream(); + expect(lifecycle, isA()); + }); + + test('should return WidgetObserverLifecycle for Linux', () { + testPlatform = TestPlatform.linux; + testIsWeb = false; + final lifecycle = getLifecycleStream(); + expect(lifecycle, isA()); + }); + }); +} diff --git a/packages/core/test/utils/lifecycle/widget_observer_test.dart b/packages/core/test/utils/lifecycle/widget_observer_test.dart new file mode 100644 index 0000000..780327a --- /dev/null +++ b/packages/core/test/utils/lifecycle/widget_observer_test.dart @@ -0,0 +1,25 @@ +import 'package:flutter/widgets.dart'; +import 'package:flutter_test/flutter_test.dart'; +import 'package:mockito/mockito.dart'; +import 'package:segment_analytics/utils/lifecycle/widget_observer.dart'; + +import '../../mocks/mocks.mocks.dart'; + +void main() { + TestWidgetsFlutterBinding.ensureInitialized(); + + group('WidgetObserverLifecycle', () { + testWidgets('should add observer to WidgetsBinding', (WidgetTester tester) async { + final mockWidgetsBinding = MockWidgetsBinding(); + final observer = WidgetObserverLifecycle(); + + when(mockWidgetsBinding.addObserver(observer)).thenReturn(mockWidgetsBinding.initInstances()); + when(mockWidgetsBinding.removeObserver(observer)).thenReturn(true); + + observer.lifeCycleImpl(); + observer.didChangeAppLifecycleState(AppLifecycleState.paused); + observer.listen((event) { }); + verifyNever(mockWidgetsBinding.addObserver(observer)); + }); + }); +} diff --git a/packages/core/test/utils/store/unsupported_test.dart b/packages/core/test/utils/store/unsupported_test.dart new file mode 100644 index 0000000..73ed81b --- /dev/null +++ b/packages/core/test/utils/store/unsupported_test.dart @@ -0,0 +1,22 @@ +import 'package:flutter_test/flutter_test.dart'; +import 'package:segment_analytics/errors.dart'; +import 'package:segment_analytics/utils/store/impl.dart'; + +void main() { + group('StoreImpl unsupported Tests', () { + late StoreImpl store; + + setUp(() { + store = StoreImpl(); + }); + + test('getPersisted should throw PlatformNotSupportedError', () { + expect(() => store.getPersisted('test_key'), throwsA(isA())); + }); + + test('setPersisted should throw PlatformNotSupportedError', () { + expect(() => store.setPersisted('test_key', {'field': 'value'}), throwsA(isA())); + }); + + }); +}