Skip to content
New issue

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

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

Already on GitHub? Sign in to your account

Merge with Develop 2024-1-14 #2640

Merged
merged 2 commits into from
Nov 14, 2024
Merged

Conversation

palisadoes
Copy link
Contributor

@palisadoes palisadoes commented Nov 14, 2024

Merge with Develop 2024-1-14

Summary by CodeRabbit

Release Notes

  • New Features

    • Introduced functionality for managing agenda items, including creating, updating, and deleting agenda items.
    • Added a new page for creating and editing agenda items with a user-friendly interface.
    • Enhanced event information page to display agenda sections based on user roles.
  • Bug Fixes

    • Improved error handling and validation for various input fields in agenda item forms.
  • Documentation

    • Updated documentation for new methods and classes related to agenda item management.
  • Tests

    • Added comprehensive tests for new features, including unit tests for models and widget tests for UI components.

Copy link
Contributor

coderabbitai bot commented Nov 14, 2024

Walkthrough

This pull request includes several changes across various files. Key modifications involve updating the Dependabot configuration to a monthly schedule, enhancing GitHub Actions workflows for better management of sensitive file changes and adding new functionalities related to agenda items in an event management context. Additionally, new models and view models have been introduced, and existing files have been cleaned up or modified to improve code organization and maintainability.

Changes

File Change Summary
.github/dependabot.yaml Updated schedule from weekly to monthly for Flutter's pub package manager.
.github/workflows/authorized-changes-detection.yml Deleted file that monitored unauthorized changes in specific files during pull requests.
.github/workflows/pull-request.yml Modified jobs: added steps for counting lines of code, renamed job, enhanced conditions for job execution, and added a new job for counting changed files.
.github/workflows/push.yml Updated job conditions to skip execution for dependabot[bot], removed documentation update jobs, and standardized job names.
.github/workflows/talawa_mobile_md_mdx_format_adjuster.py Introduced a script for adjusting Dart documentation to MDX syntax, with functions for escaping characters and processing files.
.gitignore Added entries to ignore .metadata/ and .metadata, updated position of yarn-error.log.
lib/locator.dart Added import and registration for EditAgendaItemViewModel.
lib/models/events/event_agenda_category.dart Introduced AgendaCategory class with properties and a factory method for JSON deserialization.
lib/models/events/event_agenda_item.dart Created EventAgendaItem class with properties and factory method for JSON deserialization.
lib/services/database_mutation_functions.dart Updated methods to include date conversion logic for GraphQL operations.
lib/services/event_service.dart Modified to change GraphQL operations and added new methods for managing agenda items.
lib/services/image_service.dart Removed console logging from convertToBase64 method.
lib/utils/event_queries.dart Added methods for managing agenda items in GraphQL queries.
lib/utils/time_conversion.dart Introduced utility functions for date and time conversions.
lib/view_model/after_auth_view_models/event_view_models/create_event_view_model.dart Simplified formatting logic in createEvent method.
lib/view_model/after_auth_view_models/event_view_models/edit_agenda_view_model.dart Added EditAgendaItemViewModel class for managing agenda item editing.
lib/views/after_auth_screens/events/create_agenda_item_page.dart Introduced a page for creating agenda items.
lib/views/after_auth_screens/events/edit_agenda_item_page.dart Introduced a page for editing agenda items.
lib/views/after_auth_screens/events/event_info_page.dart Updated to conditionally display tabs based on user role.
lib/views/after_auth_screens/events/manage_agenda_items_screen.dart Introduced a screen for managing agenda items.
lib/widgets/agenda_item_tile.dart Created a widget for displaying expandable agenda items.
pubspec.yaml Updated various dependencies and added a new dependency.
test/helpers/test_helpers.dart Added mock services and updated view model registrations for testing.
test/helpers/test_helpers.mocks.dart Enhanced MockEventService with new methods for agenda item management.
test/helpers/test_locator.dart Updated to register EditAgendaItemViewModel for testing.
test/model_tests/events/event_agenda_category_test.dart Added tests for AgendaCategory model.
test/model_tests/events/event_agenda_item_test.dart Added tests for EventAgendaItem model.
test/service_tests/event_service_test.dart Added tests for new agenda-related methods in EventService.
test/utils/event_queries_test.dart Added tests for new methods in EventQueries.
test/view_model_tests/after_auth_view_model_tests/event_view_model_tests/edit_agenda_view_model_test.dart Added tests for EditAgendaItemViewModel.
test/view_model_tests/after_auth_view_model_tests/event_view_model_tests/event_info_view_model_test.dart Added tests for agenda item management in EventInfoViewModel.
test/views/after_auth_screens/events/create_agenda_item_page_test.dart Created widget tests for CreateAgendaItemPage.
test/views/after_auth_screens/events/edit_agenda_item_page_test.dart Created widget tests for EditAgendaItemPage.
test/views/after_auth_screens/events/manage_agenda_items_screen_test.dart Created widget tests for ManageAgendaScreen.
test/widget_tests/after_auth_screens/events/event_info_page_test.dart Added tests for agenda section in EventInfoPage.
test/widget_tests/after_auth_screens/events/time_conversion_test.dart Created tests for time conversion utilities.
test/widget_tests/widgets/agenda_item_tile_test.dart Created tests for ExpandableAgendaItemTile.
test/views/helpers/test_locator.dart Deleted file that contained linting rule comments.

Possibly related issues

Possibly related PRs

  • Timezone in App #2581: Modifications in this PR involve date conversion logic in DataBaseMutationFunctions, which relates to the handling of time formats.
  • Added try-catch #2592: This PR enhances error handling in time conversion functions, aligning with the changes related to time management in this PR.

🐰 In the meadow, where changes bloom,
New features sprout, dispelling gloom.
With tests that guard our code so tight,
We hop along, in code's delight!
Let's create and edit with joy and cheer,
For every agenda, we hold dear! 🌼


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

❤️ Share
🪧 Tips

Chat

There are 3 ways to chat with CodeRabbit:

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

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

CodeRabbit Commands (Invoked using PR comments)

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

Other keywords and placeholders

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

Documentation and Community

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

Copy link

Our Pull Request Approval Process

We have these basic policies to make the approval process smoother for our volunteer team.

Testing Your Code

Please make sure your code passes all tests. Our test code coverage system will fail if either of these two conditions occur:

  1. The overall code coverage drops below the target threshold of the repository
  2. Any file in the pull request has code coverage levels below the repository threshold

The process helps maintain the overall reliability of the code base and is a prerequisite for getting your PR approved. Assigned reviewers regularly review the PR queue and tend to focus on PRs that are passing.

Reviewers

Do not assign reviewers. Our Queue Monitors will review your PR and assign them.
When your PR has been assigned reviewers contact them to get your code reviewed and approved via:

  1. comments in this PR or
  2. our slack channel

Reviewing Your Code

Your reviewer(s) will have the following roles:

  1. arbitrators of future discussions with other contributors about the validity of your changes
  2. point of contact for evaluating the validity of your work
  3. person who verifies matching issues by others that should be closed.
  4. person who gives general guidance in fixing your tests

Other

🎯 Please be considerate of our volunteers' time. Contacting the person who assigned the reviewers is not advised unless they ask for your input. Do not @ the person who did the assignment otherwise.

@palisadoes palisadoes added the ignore-sensitive-files-pr The contributor has a legitimate reason for editiing protected files label Nov 14, 2024
Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

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

Caution

Inline review comments failed to post. This is likely due to GitHub's limits when posting large numbers of comments.

🛑 Comments failed to post (45)
lib/models/events/event_agenda_category.dart (1)

9-16: 🛠️ Refactor suggestion

Enhance JSON handling with validation and serialization.

The JSON deserialization could be more robust with input validation and a corresponding serialization method.

+  /// Creates a new `AgendaCategory` instance from a JSON map.
+  ///
+  /// Throws [ArgumentError] if the JSON map is null or if required fields are missing.
 factory AgendaCategory.fromJson(Map<String, dynamic> json) {
+    if (json['name'] == null) {
+      throw ArgumentError('Name is required for AgendaCategory');
+    }
     return AgendaCategory(
       id: json['_id'] as String?,
       name: json['name'] as String?,
       description: json['description'] as String?,
     );
   }
+
+  /// Converts the category to a JSON map.
+  Map<String, dynamic> toJson() => {
+    'id': id,
+    'name': name,
+    'description': description,
+  };

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

lib/models/events/event_agenda_item.dart (1)

6-13: 💡 Codebase verification

ItemType enum is currently unused and should be removed

The verification shows that:

  • The ItemType enum is defined but not used anywhere in the codebase
  • Related properties itemType and isNote are commented out in the EventAgendaItem constructor
  • There are no references to ItemType values anywhere in the codebase

Since this enum is not being used and its related properties are commented out, it should be removed to avoid dead code.

🔗 Analysis chain

Verify the usage of ItemType enum

The enum is well-defined, but there are commented-out properties (itemType, isNote) in the constructor suggesting this enum might not be fully integrated yet.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Search for any usage of ItemType enum in the codebase
rg "ItemType\." --type dart

Length of output: 29


Script:

#!/bin/bash
# Let's try a broader search for ItemType without the dot
rg "ItemType" --type dart -A 2

Length of output: 214


Script:

#!/bin/bash
# Search for potential references to note or regular type in agenda items
rg "isNote|itemType" --type dart -A 2

Length of output: 742


Script:

#!/bin/bash
# Let's also check the actual file content to see the commented code mentioned
cat "lib/models/events/event_agenda_item.dart"

Length of output: 2886

.github/workflows/talawa_mobile_md_mdx_format_adjuster.py (3)

44-64: ⚠️ Potential issue

Fix aggressive comment removal in code blocks.

The current implementation has two issues:

  1. The comment removal regex (///|//)\s* might incorrectly modify code content containing // within strings
  2. There's no validation of code block syntax which could lead to malformed output

Apply this fix:

 def adjust_dart_code_blocks(text):
+    if not text:
+        return text
+
+    # Split into code blocks
+    parts = re.split(r'(```dart.*?```)', text, flags=re.DOTALL)
+    
+    for i in range(1, len(parts), 2):
+        # Process only dart code blocks
+        if '```dart' in parts[i]:
+            # Remove comments only at start of lines
+            parts[i] = re.sub(r'^\s*(///|//)\s*', '', parts[i], flags=re.MULTILINE)
+            
+            # Validate code block syntax
+            if not parts[i].strip().endswith('```'):
+                raise ValueError(f"Malformed code block detected: {parts[i][:50]}...")
+    
+    return ''.join(parts)
-    # Ensure the Dart code blocks use ```dart for proper syntax highlighting
-    text = re.sub(r"```(\s*)dart", r"```dart", text)
-
-    # Handle single-line comments in Dart (`//` or `///`)
-    text = re.sub(r"(///|//)\s*", "", text)
-
-    return text

21-43: 🛠️ Refactor suggestion

Consider enhancing the character escaping logic.

While the basic escaping functionality works, there are several edge cases that should be handled:

  1. Characters within code blocks should not be escaped
  2. The function should preserve existing escaped sequences
  3. Empty strings should be handled gracefully

Consider this improved implementation:

 def escape_mdx_characters(text):
+    if not text:
+        return text
+    
+    # Split text into code blocks and non-code blocks
+    parts = re.split(r'(```.*?```)', text, flags=re.DOTALL)
+    
     patterns = {
         "<": r"(?<!\\)<",
         ">": r"(?<!\\)>",
         "{": r"(?<!\\){",
         "}": r"(?<!\\)}"
     }
     
-    for char, pattern in patterns.items():
-        text = re.sub(pattern, f"\\{char}", text)
+    # Process only non-code block parts
+    for i in range(0, len(parts), 2):
+        for char, pattern in patterns.items():
+            parts[i] = re.sub(pattern, f"\\{char}", parts[i])
+    
+    return ''.join(parts)
📝 Committable suggestion

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

def escape_mdx_characters(text):
    """
    Escape special characters (<, >, {, }) in Dart docs to make them MDX compatible.

    Args:
        text (str): The text content to be processed.

    Returns:
        str: The modified string with escaped MDX characters.
    """
    if not text:
        return text
    
    # Split text into code blocks and non-code blocks
    parts = re.split(r'(```.*?```)', text, flags=re.DOTALL)
    
    patterns = {
        "<": r"(?<!\\)<",
        ">": r"(?<!\\)>",
        "{": r"(?<!\\){",
        "}": r"(?<!\\)}"
    }
    
    # Process only non-code block parts
    for i in range(0, len(parts), 2):
        for char, pattern in patterns.items():
            parts[i] = re.sub(pattern, f"\\{char}", parts[i])
    
    return ''.join(parts)

65-86: 🛠️ Refactor suggestion

Add error handling and optimization to file processing.

The current implementation lacks error handling and could be more efficient:

  1. Missing try-except blocks for file operations
  2. No backup mechanism before modification
  3. Writes back even when content hasn't changed

Consider this improved implementation:

 def process_file(filepath):
+    import shutil
+    from pathlib import Path
+    
+    # Create backup
+    backup_path = Path(filepath).with_suffix('.bak')
+    try:
+        shutil.copy2(filepath, backup_path)
+    except IOError as e:
+        print(f"Warning: Could not create backup of {filepath}: {e}")
+        return
+
+    try:
         with open(filepath, 'r', encoding='utf-8') as file:
             content = file.read()
 
-    # Escape MDX special characters
-    content = escape_mdx_characters(content)
-    # Adjust Dart code blocks for MDX
-    content = adjust_dart_code_blocks(content)
+        # Process content
+        new_content = adjust_dart_code_blocks(escape_mdx_characters(content))
+        
+        # Write only if content changed
+        if new_content != content:
+            with open(filepath, 'w', encoding='utf-8') as file:
+                file.write(new_content)
+            print(f"Updated: {filepath}")
+        else:
+            print(f"No changes needed: {filepath}")
 
-    # Write back to the file only if changes were made
-    with open(filepath, 'w', encoding='utf-8') as file:
-        file.write(content)
+        # Remove backup if successful
+        backup_path.unlink()
+    except Exception as e:
+        print(f"Error processing {filepath}: {e}")
+        # Restore from backup
+        try:
+            shutil.copy2(backup_path, filepath)
+            print(f"Restored backup for {filepath}")
+        except IOError as restore_error:
+            print(f"Error restoring backup: {restore_error}")
📝 Committable suggestion

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

def process_file(filepath):
    """
    Process a single Dart Markdown file for MDX compatibility.

    Args:
        filepath (str): The path to the Markdown file to be processed.

    Returns:
        None: Writes the processed content back to the file if any changes occur.
    """
    import shutil
    from pathlib import Path
    
    # Create backup
    backup_path = Path(filepath).with_suffix('.bak')
    try:
        shutil.copy2(filepath, backup_path)
    except IOError as e:
        print(f"Warning: Could not create backup of {filepath}: {e}")
        return

    try:
        with open(filepath, 'r', encoding='utf-8') as file:
            content = file.read()

        # Process content
        new_content = adjust_dart_code_blocks(escape_mdx_characters(content))
        
        # Write only if content changed
        if new_content != content:
            with open(filepath, 'w', encoding='utf-8') as file:
                file.write(new_content)
            print(f"Updated: {filepath}")
        else:
            print(f"No changes needed: {filepath}")

        # Remove backup if successful
        backup_path.unlink()
    except Exception as e:
        print(f"Error processing {filepath}: {e}")
        # Restore from backup
        try:
            shutil.copy2(backup_path, filepath)
            print(f"Restored backup for {filepath}")
        except IOError as restore_error:
            print(f"Error restoring backup: {restore_error}")
test/view_model_tests/after_auth_view_model_tests/event_view_model_tests/edit_agenda_view_model_test.dart (1)

34-117: 🛠️ Refactor suggestion

Enhance test coverage with error scenarios

The current test suite covers happy paths well, but should be expanded to include:

  1. Error handling scenarios
  2. Input validation
  3. Network failures
  4. Concurrent operations

Add these test cases:

test('updateAgendaItem() handles network failure', () async {
  model.initialize(testAgendaItem, testCategories);
  
  when(eventService.updateAgendaItem(any, any)).thenAnswer(
    (_) async => QueryResult(
      source: QueryResultSource.network,
      exception: OperationException(graphqlErrors: [
        GraphQLError(message: 'Network error'),
      ]),
      options: QueryOptions(document: gql('')),
    ),
  );
  
  expect(() => model.updateAgendaItem(), throwsException);
});

test('initialize() validates input data', () {
  final invalidAgendaItem = EventAgendaItem(
    id: '',
    title: '',
    description: '',
    duration: '-1',
    urls: ['invalid-url'],
    attachments: [],
    categories: [],
  );
  
  expect(
    () => model.initialize(invalidAgendaItem, []),
    throwsAssertionError,
  );
});

test('concurrent operations are handled correctly', () async {
  model.initialize(testAgendaItem, testCategories);
  
  // Simulate concurrent updates
  final future1 = model.updateAgendaItem();
  final future2 = model.updateAgendaItem();
  
  await Future.wait([future1, future2]);
  
  // Verify the service was called exactly once
  verify(eventService.updateAgendaItem(any, any)).called(1);
});
lib/views/after_auth_screens/events/event_info_page.dart (1)

32-37: 🛠️ Refactor suggestion

Consider adding null safety check for creator

The creator check might cause runtime errors if the creator field is null. Also, this logic is duplicated later in the build method.

Consider refactoring like this:

-    final bool isCreator = (widget.args["event"] as Event).creator!.id ==
-        userConfig.currentUser.id;
+    final Event event = widget.args["event"] as Event;
+    final bool isCreator = event.creator?.id == userConfig.currentUser.id;

Also, consider extracting this check into a method to avoid duplication:

bool isEventCreator(Event event) {
  return event.creator?.id == userConfig.currentUser.id;
}
test/views/after_auth_screens/events/edit_agenda_item_page_test.dart (2)

77-142: 🛠️ Refactor suggestion

Add test cases for error handling and edge cases.

The test suite would benefit from additional test cases covering:

  1. Error states (network errors, invalid inputs)
  2. Concurrent operations (multiple rapid updates)
  3. Form validation
  4. Navigation scenarios (back button handling)

Example test cases to add:

testWidgets('Shows error message on network failure', (tester) async {
  // Mock network failure
  // Verify error display
});

testWidgets('Handles invalid URL input', (tester) async {
  // Test with invalid URL
  // Verify validation message
});

testWidgets('Prevents concurrent updates', (tester) async {
  // Trigger multiple rapid updates
  // Verify proper handling
});

134-141: ⚠️ Potential issue

Add assertions to the Add Attachments test case.

The test for Add Attachments button lacks assertions to verify the expected behavior.

     testWidgets('Add Attachments button works correctly',
         (WidgetTester tester) async {
       await tester.pumpWidget(createEditAgendaItemScreen());
       await tester.pumpAndSettle();

       await tester.tap(find.text('Add Attachments'));
       await tester.pumpAndSettle();
+      
+      // Verify the file picker was shown
+      expect(find.byType(FilePickerDialog), findsOneWidget);
+      
+      // Verify attachment list updates after selection
+      // Mock file selection here
+      expect(find.byType(AttachmentList), findsOneWidget);
     });

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

.github/workflows/push.yml (1)

108-108: 🛠️ Refactor suggestion

Enhance release management strategy

The current release process has several areas for improvement:

  1. Using a static "automated" tag overwrites previous releases
  2. No versioning strategy or build numbering
  3. Limited release notes

Consider these improvements:

      - uses: ncipollo/release-action@v1
        with:
          name: "Automated Android Release"
          artifacts: "./build/app/outputs/flutter-apk/app-release.apk"
          allowUpdates: "true"
          generateReleaseNotes: false
-         tag: "automated"
+         tag: "android-${{ github.sha }}"
          body: |
            This is an automated release, triggered by a recent push. 
-           This may or may not be stable, so please have a look at the stable release(s).
+           Build Details:
+           - Commit: ${{ github.sha }}
+           - Branch: ${{ github.ref_name }}
+           - Workflow Run: ${{ github.server_url }}/${{ github.repository }}/actions/runs/${{ github.run_id }}

Also applies to: 137-146

test/views/after_auth_screens/events/create_agenda_item_page_test.dart (2)

121-216: 🛠️ Refactor suggestion

Add negative test cases for better coverage

The test suite focuses on happy paths. Consider adding tests for error scenarios:

  • Invalid URLs
  • Empty required fields
  • Invalid duration format
  • Category selection with empty categories

Would you like me to help generate these additional test cases?


191-207: 🛠️ Refactor suggestion

Avoid using hardcoded index in URL test

Using .at(3) to find the URL text field is fragile and could break if the form layout changes.

Use a key to identify the URL field:

+      await tester.enterText(
+        find.byKey(const Key('url_input_field')),
+        'https://example.com',
+      );
-      await tester.enterText(
-        find.byType(TextFormField).at(3),
-        'https://example.com',
-      );
📝 Committable suggestion

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

    testWidgets('Add URL works correctly', (WidgetTester tester) async {
      await tester.pumpWidget(createCreateAgendaItemScreen());
      await tester.pumpAndSettle();

      await tester.enterText(
        find.byKey(const Key('url_input_field')),
        'https://example.com',
      );
      await tester.pumpAndSettle();
      await tester.tap(
        find.byKey(const Key('add_url')),
      );
      await tester.pumpAndSettle();

      expect(find.byType(Chip), findsOneWidget);
      expect(find.text('https://example.com'), findsOneWidget);
    });
lib/widgets/agenda_item_tile.dart (2)

32-32: ⚠️ Potential issue

Add null safety check for item.sequence

Direct use of the null assertion operator (!) could lead to runtime errors if sequence is null.

- index: item.sequence! - 1,
+ index: (item.sequence ?? 1) - 1,
📝 Committable suggestion

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

          index: (item.sequence ?? 1) - 1,

31-34: ⚠️ Potential issue

Use a unique key for ReorderableDragStartListener

The current key ('reorder_icon') is not unique across multiple items, which could cause issues with Flutter's widget tree reconciliation.

- key: const Key('reorder_icon'),
+ key: Key('reorder_icon_${item.id}'),
📝 Committable suggestion

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

          key: Key('reorder_icon_${item.id}'),
          index: item.sequence! - 1,
          child: const Icon(Icons.drag_handle),
        ),
test/views/after_auth_screens/events/manage_agenda_items_screen_test.dart (3)

161-202: ⚠️ Potential issue

Incomplete verification in reorder test.

The reorder test performs the drag action but doesn't verify the final order of items. Add assertions to verify the reordering was successful.

Example enhancement:

testWidgets('Can reorder agenda items', (WidgetTester tester) async {
  // ... existing setup code ...

  // Perform reorder
  await tester.drag(
    find.byKey(const Key('reorder_icon')).first,
    const Offset(0, 200),
  );
  await tester.pumpAndSettle();

  // Verify new order
  final listItems = find.byType(ExpandableAgendaItemTile).evaluate().toList();
  expect(
    (listItems[0].widget as ExpandableAgendaItemTile).agendaItem.title,
    equals('Agenda 2'),
  );
  expect(
    (listItems[1].widget as ExpandableAgendaItemTile).agendaItem.title,
    equals('Agenda 1'),
  );
});

204-236: ⚠️ Potential issue

Incomplete verification in delete test.

The delete test doesn't verify that the item was actually removed from the UI after deletion.

Example enhancement:

testWidgets('Can delete agenda item', (WidgetTester tester) async {
  // ... existing setup code ...

  // Verify initial state
  expect(find.text('Agenda 1'), findsOneWidget);
  
  // Perform deletion
  await tester.tap(find.byKey(const Key("delete_agenda_item1")));
  await tester.pumpAndSettle();  // Use pumpAndSettle instead of pump

  // Verify item is removed
  expect(find.text('Agenda 1'), findsNothing);
  verify(eventService.deleteAgendaItem({"removeAgendaItemId": '1'})).called(1);
});

115-248: 🛠️ Refactor suggestion

Add tests for error cases and edge scenarios.

The test suite lacks coverage for:

  1. Error cases:
    • Failed API calls
    • Network errors
    • Invalid responses
  2. Edge cases:
    • Maximum number of agenda items
    • Invalid agenda item data
    • Concurrent modifications

Example additional test cases:

testWidgets('Shows error when API call fails', (WidgetTester tester) async {
  when(eventService.fetchAgendaItems('1'))
      .thenThrow(Exception('Network error'));

  await tester.pumpWidget(createManageAgendaScreen());
  await tester.pumpAndSettle();

  expect(find.text('Error loading agenda items'), findsOneWidget);
});

testWidgets('Handles invalid agenda item data', (WidgetTester tester) async {
  final mockResult = QueryResult(
    source: QueryResultSource.network,
    data: {
      'agendaItemByEvent': [
        {
          '_id': '1',
          'title': '', // Invalid empty title
          'duration': 'invalid',
          'sequence': -1, // Invalid sequence
        },
      ],
    },
    options: QueryOptions(
      document: gql(EventQueries().fetchAgendaItemsByEvent('1')),
    ),
  );

  when(eventService.fetchAgendaItems('1'))
      .thenAnswer((_) async => mockResult);

  await tester.pumpWidget(createManageAgendaScreen());
  await tester.pumpAndSettle();

  expect(find.text('Invalid agenda item'), findsOneWidget);
});
lib/services/database_mutation_functions.dart (2)

128-132: 🛠️ Refactor suggestion

Standardize error handling across query and mutation methods

The date conversion logic in mutations differs from queries:

  1. No error handling for date conversion failures
  2. Inconsistent approach compared to query methods
  3. Test coverage is excluded without explanation

Consider standardizing the approach:

-  // coverage:ignore-start
   if (variables != null) {
     traverseAndConvertDates(
       variables,
       convertLocalToUTC,
       splitDateTimeUTC,
+      onError: (error) {
+        print('Error converting dates in mutation variables: $error');
+        // Handle error appropriately
+      }
     );
   }
-  // coverage:ignore-end

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


173-177: 🛠️ Refactor suggestion

Reduce code duplication in mutation methods

The date conversion logic is duplicated between gqlAuthMutation and gqlNonAuthMutation.

Consider extracting the common logic:

+ private void convertVariableDates(Map<String, dynamic>? variables) {
+   if (variables != null) {
+     traverseAndConvertDates(
+       variables,
+       convertLocalToUTC,
+       splitDateTimeUTC,
+       onError: (error) {
+         print('Error converting dates in mutation variables: $error');
+         // Handle error appropriately
+       }
+     );
+   }
+ }

  Future<QueryResult<Object?>> gqlNonAuthMutation(...) async {
-   // coverage:ignore-start
-   if (variables != null) {
-     traverseAndConvertDates(variables, convertLocalToUTC, splitDateTimeUTC);
-   }
-   // coverage:ignore-end
+   convertVariableDates(variables);
    ...
  }

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

.github/workflows/pull-request.yml (1)

176-176: 💡 Codebase verification

Security risk: Label bypass affects critical branch protection

After analyzing the workflow file, this is a critical security concern because:

  1. The label bypass is used in the Branch-check job which enforces the requirement to merge into the develop branch
  2. The same workflow contains a Check-Sensitive-Files job that prevents unauthorized changes to critical repository files like workflows, security policies, and configuration
  3. The label is not protected as the GitHub API call failed to find it in repository settings

Recommendations:

  • Remove the label bypass condition from line 176
  • If bypass is necessary, implement it through protected environment or branch protection rules instead
  • At minimum, ensure the label can only be managed by repository administrators through branch protection settings
🔗 Analysis chain

Review security implications of branch check bypass

The addition of the ignore-sensitive-files-pr label bypass could potentially be misused. Ensure that:

  1. The label can only be applied by authorized team members
  2. There's a process to review PRs with this label
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Check if the label is protected in repository settings
gh api repos/$GITHUB_REPOSITORY/labels | jq '.[] | select(.name=="ignore-sensitive-files-pr")'

Length of output: 168


Script:

#!/bin/bash
# Let's check the workflow file to understand the context and implications
cat .github/workflows/pull-request.yml

# Also search for any other workflows that might reference this label
rg "ignore-sensitive-files-pr" .github/workflows/

Length of output: 10578

test/view_model_tests/after_auth_view_model_tests/event_view_model_tests/event_info_view_model_test.dart (1)

254-287: ⚠️ Potential issue

Separate concerns in updateAgendaItemSequence test

The test is validating both sequence and title updates, but the method name updateAgendaItemSequence suggests it should only handle sequence updates. Consider:

  1. Splitting into separate test cases for sequence and title updates
  2. Renaming the method if it's intended to handle both updates
lib/view_model/after_auth_view_models/event_view_models/create_event_view_model.dart (1)

228-231: ⚠️ Potential issue

Add time range validation.

The code should validate that the end time is after the start time to prevent invalid event creation.

Add this validation in the createEvent method before building the variables map:

+          // Validate time range
+          if (!isAllDay && endTime.isBefore(startTime)) {
+            throw Exception('End time must be after start time');
+          }
+
           // all required data for creating an event
           final Map<String, dynamic> variables = {

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

lib/utils/time_conversion.dart (5)

84-91: 🛠️ Refactor suggestion

Replace print statements with a proper logging mechanism

In the convertLocalToUTC function, replace print with a logging tool to handle errors effectively.

Apply this diff:

     } catch (e) {
-      print('Timezone Error converting local to UTC: $e $localTime');
+      logger.error('Timezone Error converting local to UTC: $e $localTime');
       return '';
     }

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


24-34: 🛠️ Refactor suggestion

Replace print statements with a proper logging mechanism

In the splitDateTimeUTC function, using print statements for error logging is not recommended in production code. Consider using a structured logging framework to handle error messages more effectively.

Apply this diff to replace print with a logging function:

     } catch (e) {
-      print('Timezone Error parsing UTC date time: $e $dateTimeStr');
+      logger.error('Timezone Error parsing UTC date time: $e $dateTimeStr');
       return {};
     }

Ensure that logger is properly configured in your application.

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


66-73: 🛠️ Refactor suggestion

Replace print statements with a proper logging mechanism

In the convertUTCToLocal function, use a logging framework instead of print for better error handling.

Apply this diff:

     } catch (e) {
-      print('Timezone Error converting UTC to local: $e $utcTime');
+      logger.error('Timezone Error converting UTC to local: $e $utcTime');
       return '';
     }

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


122-123: ⚠️ Potential issue

Avoid assigning to map keys with empty strings

In the traverseAndConvertDates function, using field['dateField'] ?? '' may result in empty string keys if field['dateField'] is null. This can lead to unexpected behavior when accessing map entries.

Consider adding null checks before assignment:

-    obj[field['dateField'] ?? ''] = splitDateTime['date'] ?? '';
-    obj[field['timeField'] ?? ''] = splitDateTime['time'] ?? '';
+    if (field['dateField'] != null && splitDateTime['date'] != null) {
+      obj[field['dateField']] = splitDateTime['date'];
+    }
+    if (field['timeField'] != null && splitDateTime['time'] != null) {
+      obj[field['timeField']] = splitDateTime['time'];
+    }
📝 Committable suggestion

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

          if (field['dateField'] != null && splitDateTime['date'] != null) {
            obj[field['dateField']] = splitDateTime['date'];
          }
          if (field['timeField'] != null && splitDateTime['time'] != null) {
            obj[field['timeField']] = splitDateTime['time'];
          }

45-55: 🛠️ Refactor suggestion

Replace print statements with a proper logging mechanism

In the splitDateTimeLocal function, replace print statements with a logging framework to handle errors appropriately in production environments.

Apply this diff:

     } catch (e) {
-      print('Timezone Error parsing local date time: $e $dateTimeStr');
+      logger.error('Timezone Error parsing local date time: $e $dateTimeStr');
       return {};
     }

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

lib/view_model/after_auth_view_models/event_view_models/edit_agenda_view_model.dart (1)

149-149: ⚠️ Potential issue

Undefined imageService in pickAttachment method

The variable imageService is used but not defined in the current scope:

final base64PickedFile = await imageService.convertToBase64(pickedFile);

This will cause a compile-time error. Ensure that imageService is properly declared and initialized before use.

Apply this diff to fix the issue:

+ final imageService = locator<ImageService>();
final base64PickedFile = await imageService.convertToBase64(pickedFile);
lib/view_model/after_auth_view_models/event_view_models/event_info_view_model.dart (5)

309-309: ⚠️ Potential issue

Remove placeholder text in documentation

The comment more_info_if_required appears to be a placeholder and should be removed or replaced with actual information.

Apply this diff:

- /// more_info_if_required
📝 Committable suggestion

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



36-37: ⚠️ Potential issue

Correct the documentation comment for agendaItems getter

The documentation comment incorrectly refers to volunteer groups instead of agenda items.

Apply this diff to fix the comment:

- /// List of volunteer groups of an event.
+ /// List of agenda items of an event.
📝 Committable suggestion

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

  /// List of agenda items of an event.
  List<EventAgendaItem> get agendaItems => _agendaItems;

216-216: ⚠️ Potential issue

Provide a description for the categories parameter

The parameter categories needs a proper description in the documentation.

Apply this diff:

- /// * `categories`: define_the_param
+ /// * `categories`: The list of selected agenda categories to set.
📝 Committable suggestion

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

  /// * `categories`: The list of selected agenda categories to set.

60-65: ⚠️ Potential issue

Adjust state management in the initialize method

The setState(ViewState.busy); should be called before the asynchronous operations to indicate that the view model is busy, and setState(ViewState.idle); should be called after all operations are complete.

Apply this diff:

Future<void> initialize({required Map<String, dynamic> args}) async {
  event = args["event"] as Event;
  exploreEventsInstance =
      args["exploreEventViewModel"] as ExploreEventsViewModel;
  fabTitle = getFabTitle();
+ setState(ViewState.busy);
  await fetchCategories();
  await fetchAgendaItems();
  selectedCategories.clear();
- setState(ViewState.busy);
  attendees = event.attendees ?? [];
  setState(ViewState.idle);
}
📝 Committable suggestion

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

    setState(ViewState.busy);
    await fetchCategories();
    await fetchAgendaItems();
    selectedCategories.clear();
    attendees = event.attendees ?? [];
    setState(ViewState.idle);

273-304: 🛠️ Refactor suggestion

Handle exceptions more robustly in createAgendaItem

Currently, exceptions are caught and printed but not rethrown or handled meaningfully. Consider providing feedback to the user or calling code about the failure.

Apply this diff:

} catch (e) {
-   print('Error creating agenda item: $e');
+   // Handle the error appropriately
+   setState(ViewState.error);
+   throw Exception('Error creating agenda item: $e');
}

This change ensures that the error state is reflected in the UI and that calling methods are aware of the failure.

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

lib/services/event_service.dart (4)

401-405: ⚠️ Potential issue

Handle null data when fetching agenda items.

Ensure that result.data and the expected keys are not null to prevent runtime errors.

Add null checks:

final result = await _dbFunctions.gqlAuthQuery(
  EventQueries().fetchAgendaItemsByEvent(eventId),
);

if (result.data == null || result.data!['agendaItems'] == null) {
  throw Exception('Unable to fetch agenda items');
}

final List itemsJson = result.data!['agendaItems'] as List;
// Process itemsJson...

327-333: ⚠️ Potential issue

Correct the method documentation for fetchAgendaCategories.

The documentation describes creating an agenda item, but the method fetches agenda categories. Updating the comments enhances clarity.

Apply this diff:

       /// This function is used to create an agenda item.
+      /// This function is used to fetch agenda categories for an organization.
       
       /// **params**:
       /// * `orgId`: ID of organisation to fetch categories.
       
       /// **returns**:
-      /// * `Future<dynamic>`: Information about the created agenda item.
+      /// * `Future<dynamic>`: Information about the fetched agenda categories.
📝 Committable suggestion

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

  /// This function is used to fetch agenda categories for an organization.
  ///
  /// **params**:
  /// * `orgId`: ID of organisation to fetch categories.
  ///
  /// **returns**:
  /// * `Future<dynamic>`: Information about the fetched agenda categories.

71-71: ⚠️ Potential issue

Initialize _events before populating to avoid duplication.

Ensure _events is cleared or initialized before adding new events to prevent duplication when the method is called multiple times.

Apply this diff:

     _events = []; // Add this line to initialize the list
     eventsJson.forEach((eventJsonData) {
       final Event event = Event.fromJson(eventJsonData as Map<String, dynamic>);
       event.isRegistered = event.attendees?.any(
             (attendee) => attendee.id == _userConfig.currentUser.id,
           ) ??
           false;
       _events.insert(0, event);
     });

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


335-337: ⚠️ Potential issue

Use gqlAuthQuery instead of gqlAuthMutation for data fetching.

Fetching data should use gqlAuthQuery to semantically represent a query operation.

Apply this diff:

       final result = await _dbFunctions.gqlAuthMutation(
+      final result = await _dbFunctions.gqlAuthQuery(
         EventQueries().fetchAgendaItemCategoriesByOrganization(orgId),
       );

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

lib/views/after_auth_screens/events/edit_agenda_item_page.dart (3)

106-137: 🛠️ Refactor suggestion

Use an appropriate widget for multi-select categories instead of DropdownButtonFormField

The current implementation uses DropdownButtonFormField for selecting multiple categories. However, DropdownButtonFormField is designed for single selection and may not provide the best user experience for multi-selection scenarios. This could lead to unexpected behavior or a confusing interface for users.

Consider using a widget that supports multiple selections, such as a DropdownButton with isExpanded and multiSelect capabilities, a MultiSelectFormField, or a custom dialog with checkboxes to allow users to select multiple categories more effectively.

Suggested Solution: Use a custom multi-select widget

You might implement a custom multi-select dialog:

// Trigger the multi-select dialog
ElevatedButton(
  onPressed: () async {
    final selectedCategories = await showDialog<List<AgendaCategory>>(
      context: context,
      builder: (context) {
        return MultiSelectDialog(
          categories: model.categories,
          initiallySelected: model.selectedCategories,
        );
      },
    );
    if (selectedCategories != null) {
      model.setSelectedCategories(selectedCategories);
    }
  },
  child: Text(AppLocalizations.of(context)!.strictTranslate('Select Categories')),
),

And create a MultiSelectDialog widget to handle multiple selections.


241-248: ⚠️ Potential issue

Implement validation for duration format in mm:ss

The validator for the duration field currently only checks if the field is empty. There's a comment indicating that additional validation for the mm:ss format is needed. Without proper validation, users might enter durations in an incorrect format, leading to errors in processing the duration.

Consider implementing validation to ensure that the duration entered adheres to the mm:ss format and represents a valid time duration.

Suggested Solution: Add regex validation for mm:ss format

validator: (value) {
  if (value == null || value.isEmpty) {
    return AppLocalizations.of(context)!.strictTranslate('Please enter a duration');
  }
  final regex = RegExp(r'^\d{1,2}:\d{2}$');
  if (!regex.hasMatch(value)) {
    return AppLocalizations.of(context)!.strictTranslate('Duration must be in mm:ss format');
  }
  // Optional: Further validate that seconds are less than 60
  final parts = value.split(':');
  if (int.parse(parts[1]) >= 60) {
    return AppLocalizations.of(context)!.strictTranslate('Seconds should be less than 60');
  }
  return null;
},

62-65: 🛠️ Refactor suggestion

Handle potential errors during agenda item update

When calling model.updateAgendaItem(), there might be scenarios where the update fails due to network issues or server errors. Currently, there is no error handling or user feedback for such cases, which could lead to a poor user experience.

Suggested Solution: Add error handling and user feedback

onPressed: () async {
  if (model.checkForChanges()) {
    try {
      await model.updateAgendaItem();
      if (context.mounted) {
        Navigator.of(context).pop(true);
      }
    } catch (e) {
      // Display an error message to the user
      DelightToastBar(
        snackbarDuration: const Duration(seconds: 2),
        builder: (context) {
          return ToastCard(
            title: Text(
              AppLocalizations.of(context)!.strictTranslate('Failed to update agenda item'),
              style: const TextStyle(
                color: Colors.white,
                fontSize: 12,
              ),
            ),
            leading: const Icon(
              Icons.error_outline,
              color: Colors.redAccent,
            ),
            color: Colors.black.withOpacity(0.8),
          );
        },
      ).show(context);
    }
  } else {
    // Existing code for no changes made
  }
},
lib/views/after_auth_screens/events/create_agenda_item_page.dart (5)

144-155: ⚠️ Potential issue

Add Error Handling for File Conversion in _pickAttachment

The method _pickAttachment lacks error handling for the file-to-base64 conversion process. If imageService.convertToBase64(pickedFile) fails or throws an exception, the app might crash or behave unexpectedly.

Consider adding a try-catch block to handle potential exceptions during the conversion:

Future<void> _pickAttachment({bool fromCamera = false}) async {
  final File? pickedFile =
      await _multiMediaPickerService.getPhotoFromGallery(camera: fromCamera);
  if (pickedFile != null) {
    try {
      final base64PickedFile = await imageService.convertToBase64(pickedFile);
      setState(() {
        attachments.add(base64PickedFile);
      });
    } catch (e) {
      // Handle the error, e.g., show a snackbar or log the error
      print('Error converting file to base64: $e');
    }
  }
}

196-203: ⚠️ Potential issue

Validate Inputs Before Submitting Agenda Item

Before calling widget.model.createAgendaItem, the inputs from the controllers should be validated to ensure they meet the required criteria. Currently, there's no check to prevent empty or invalid data from being submitted.

Consider adding form validation logic before submission:

onPressed: () {
  if (_formKey.currentState!.validate()) {
    final List<String> categoryIds =
        selectedCategories.map((category) => category.id!).toList();

    widget.model.createAgendaItem(
      title: titleController.text,
      duration: durationController.text,
      description: descController.text,
      urls: urls,
      categories: categoryIds,
      attachments: attachments,
    );
    Navigator.of(context).pop();
  }
},

Also, wrap your form fields within a Form widget and assign _formKey accordingly.


116-122: ⚠️ Potential issue

Validate URL Format in _addUrl Method

When adding URLs, there's no validation to check if the entered text is a valid URL. This might lead to invalid links being associated with the agenda item.

Implement URL validation before adding it to the list:

void _addUrl() {
  if (urlController.text.isNotEmpty && Uri.parse(urlController.text).isAbsolute) {
    setState(() {
      urls.add(urlController.text);
      urlController.clear();
    });
  } else {
    // Show an error message or indication
    print('Invalid URL entered.');
  }
}

326-356: ⚠️ Potential issue

Add Duration Format Validation

The duration input expects a mm:ss format, but there's no validation to enforce this pattern. Users might enter invalid durations, leading to errors when processing this data.

Add validation logic to ensure the duration follows the mm:ss format:

validator: (value) {
  if (value == null || value.isEmpty) {
    return AppLocalizations.of(context)!
        .strictTranslate('Please enter a duration');
  }
  final durationRegex = RegExp(r'^\d{2}:\d{2}$');
  if (!durationRegex.hasMatch(value)) {
    return AppLocalizations.of(context)!
        .strictTranslate('Duration must be in mm:ss format');
  }
  return null;
},

418-463: 🛠️ Refactor suggestion

Optimize Performance of Image Rendering in GridView

Rendering images using Image.memory can be resource-intensive, especially with large base64 strings. This might lead to performance issues in the UI.

Consider caching the images or using thumbnails to improve performance:

// Use Image.memory with a cache mechanism
Image.memory(
  imageData,
  fit: BoxFit.cover,
  width: double.infinity,
  height: double.infinity,
  cacheWidth: 100, // Adjust the width as needed for thumbnails
  cacheHeight: 100, // Adjust the height as needed for thumbnails
),

Alternatively, you might consider storing the file paths and using Image.file instead.

@palisadoes palisadoes merged commit c4fc7b6 into PalisadoesFoundation:main Nov 14, 2024
6 of 8 checks passed
@palisadoes palisadoes deleted the new-main branch November 14, 2024 14:40
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
ignore-sensitive-files-pr The contributor has a legitimate reason for editiing protected files
Projects
None yet
Development

Successfully merging this pull request may close these issues.

1 participant