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

Dev -> main #89

Open
wants to merge 415 commits into
base: main
Choose a base branch
from
Open

Dev -> main #89

wants to merge 415 commits into from

Conversation

Prometheo
Copy link
Collaborator

@Prometheo Prometheo commented Oct 21, 2024

Summary by CodeRabbit

  • New Features

    • Automated deployment workflow introduced to streamline production releases.
    • Multiple new API endpoints added for enhanced account, donation, round, project, and voting functionalities.
    • Enhanced token metadata now displays live pricing and additional details from external services.
  • API Enhancements

    • API documentation updated with refined pagination and increased rate limits for improved responsiveness.
  • Infrastructure & Performance

    • Improved backend task scheduling and revised settings for CORS and service management to boost overall stability and performance.

Copy link

coderabbitai bot commented Mar 17, 2025

Walkthrough

This pull request introduces extensive modifications across the project. A new GitHub Actions workflow automates production deployments to EC2 via AWS CodeDeploy. Several .gitignore and README updates refine configuration and documentation. Multiple apps receive model, migration, management command, and serializer enhancements—adding fields (e.g. chain, near_social_profile_data), refactoring APIs with new endpoints and custom pagination, and updating admin interfaces. Additional Celery tasks, settings changes, and dependency upgrades further improve deployment, asynchronous processing, and external integrations.

Changes

File(s) Change Summary
.github/workflows/deploy-prod.yml New CI/CD workflow for production branch deployment to EC2 using AWS CodeDeploy (checkout, AWS credential configuration, zipping, S3 upload).
.gitignore, README.md New ignore entries (e.g. dump.rdb, pyrightconfig.json) and updated API documentation with additional endpoints, revised pagination, and increased rate limits.
accounts/... Admin, API, management commands, migrations, models, and serializers updated: added fields (chain, near_social_profile_data), introduced async methods, refactored endpoints with custom pagination, and included new management commands.
activities/... Admin and migrations updated: added autocomplete for related fields; modified tx_hash and action_result properties in models and migrations.
api/pagination.py, api/urls.py New custom pagination class and extensive URL routing updates for accounts, lists, pots, donations, grants, and related API endpoints.
base/... New StatsResponseSerializer, scheduled Celery tasks and updated task routing, TwoDecimalPlacesField, and enhanced settings with environment-conditioned variables and CORS configurations.
chains/... New Chain model with complete migrations, admin registration, and app configuration to support blockchain chain management.
donations/... Admin, API, migrations, models, and serializers refactored: field renaming from ft to token, extended endpoints/documentation, added management commands, and improved data migration for donation fields.
grantpicks/... New models, migrations, API views, serializers, and tests for rounds, projects, votes, and applications; endpoints restructured for granular operations.
indexer_app/... Numerous updates including admin display enhancements, new examples, asynchronous handler functions, management commands, tasks, logging, and utility functions for processing blockchain events.
lists/... API endpoints renamed/extended (detail, registrations, random registration), migrations and models updated (new on_chain_id, unique constraints), and serializers now include paginated responses and nested relations.
nadabot/... New app configuration, models, migrations, serializers, tests, utility functions, and views for managing nadabot registries and provider data.
pots/... Admin forms, API views, migrations, models (with update_configs methods), serializers, and utilities extensively refactored to support pot factories, applications, payouts, and additional constraint logic; regex patterns updated.
pyproject.toml Dependency updates and additions: upgraded near-lake-framework, and added drf-spectacular, django-extensions, psutil, and stellar-sdk.
scripts/after_install*.sh Deployment scripts refined to run collectstatic unconditionally with graceful Gunicorn reloads and service restarts, bypassing full stop/start sequences.
tokens/... Token admin, migrations, models, and serializers updated; new fields (name, symbol, icon, coingecko_id) added and primary key renamed from id to account, with methods to fetch USD prices integrated.

Sequence Diagram(s)

sequenceDiagram
    participant Dev as Developer
    participant GitHub as GitHub Actions
    participant AWS as AWS (S3/CodeDeploy)
    participant EC2 as EC2 Instance
    Dev->>GitHub: Push to prod branch
    GitHub->>GitHub: Trigger deploy-prod workflow
    GitHub->>GitHub: Clone repository (actions/checkout)
    GitHub->>AWS: Assume IAM role & configure AWS credentials
    GitHub->>GitHub: Set env vars, create deploy package (zip)
    GitHub->>AWS: Upload zip file to S3 bucket
    GitHub->>AWS: Create deployment in AWS CodeDeploy
    AWS->>EC2: Deploy package to EC2 instance
    EC2-->>GitHub: Acknowledge deployment
Loading
sequenceDiagram
    participant Client as API Client
    participant Router as DRF URL Router
    participant API as Accounts/Donation Endpoint
    participant Logic as Business Logic & Serializer
    Client->>Router: Send API request (e.g. /v1/accounts/<id>)
    Router->>API: Route request to AccountDetailAPI/AccountsListAPI
    API->>Logic: Process request, handle pagination, error-checking
    Logic->>API: Return serialized response data
    API-->>Client: Send API response (with JSON payload)
Loading

Poem

Oh, what a joyful day,
As changes hop along our way,
A rabbit scurries through the code so fleet,
With workflows, endpoints, migrations neat.
New fields and tasks bring a delightful tune,
Bounding forward under the techy moon.
Hop on, dear coder, let progress bloom!

Tip

⚡🧪 Multi-step agentic review comment chat (experimental)
  • We're introducing multi-step agentic chat in review comments. This experimental feature enhances review discussions with the CodeRabbit agentic chat by enabling advanced interactions, including the ability to create pull requests directly from comments.
    - To enable this feature, set early_access to true under in the settings.
✨ Finishing Touches
  • 📝 Generate Docstrings

🪧 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 generate docstrings to generate docstrings for this 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.

CodeRabbit Configuration File (.coderabbit.yaml)

  • You can programmatically configure CodeRabbit by adding a .coderabbit.yaml file to the root of your repository.
  • Please see the configuration documentation for more information.
  • If your editor has YAML language server enabled, you can add the path at the top of this file to enable auto-completion and validation: # yaml-language-server: $schema=https://coderabbit.ai/integrations/schema.v2.json

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

@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.

Actionable comments posted: 26

🧹 Nitpick comments (140)
nadabot/migrations/0003_alter_group_rule_type_alter_group_rule_val_and_more.py (1)

18-24: Inconsistent formatting in choice field values

The internal values for "Diminishing_returns" and "Increasing_returns" use underscores, while their display labels use spaces. For consistency with the other choices (which don't use underscores), consider standardizing the format.

                choices=[
                    ("Highest", "Highest"),
                    ("Lowest", "Lowest"),
                    ("Sum", "Sum"),
-                   ("Diminishing_returns", "Diminishing Returns"),
-                   ("Increasing_returns", "Increasing Returns"),
+                   ("Diminishing returns", "Diminishing Returns"),
+                   ("Increasing returns", "Increasing Returns"),
                ],
grantpicks/admin.py (1)

1-1: Unused Import Warning: Consider Removing or Marking for Future Use
The import on line 1 (from django.contrib import admin) is currently unused. Since this file is a new scaffold for future model registrations, you might want to either keep it (with an appropriate # noqa comment if needed) or remove it until the models get registered.

🧰 Tools
🪛 Ruff (0.8.2)

1-1: django.contrib.admin imported but unused

Remove unused import: django.contrib.admin

(F401)

nadabot/tests.py (1)

1-1: Unused Import Notice: TestCase
The TestCase import on line 1 from django.test is not currently used in this file. If you plan to add tests shortly, you might retain it; otherwise, consider removing it to clear the warning.

🧰 Tools
🪛 Ruff (0.8.2)

1-1: django.test.TestCase imported but unused

Remove unused import: django.test.TestCase

(F401)

nadabot/views.py (1)

1-1: Unused Import Warning: render
The render function imported on line 1 from django.shortcuts is not used in this file yet. If this file is intended as a placeholder for future view implementations, you may either leave it or remove it until an actual view makes use of it.

🧰 Tools
🪛 Ruff (0.8.2)

1-1: django.shortcuts.render imported but unused

Remove unused import: django.shortcuts.render

(F401)

nadabot/admin.py (1)

1-1: Unused Import Warning: django.contrib.admin
The import on line 1 is currently not utilized in this admin file. If models will be registered shortly, it is acceptable to retain; otherwise, consider removing the unused import to resolve static analysis warnings.

🧰 Tools
🪛 Ruff (0.8.2)

1-1: django.contrib.admin imported but unused

Remove unused import: django.contrib.admin

(F401)

chains/views.py (1)

1-1: Unused Import Warning: render
The render import on line 1 from django.shortcuts is currently unused. Since the file is a starter template for future view definitions, consider removing it until an actual view is implemented—or annotate it with a # noqa comment if you prefer to keep it.

🧰 Tools
🪛 Ruff (0.8.2)

1-1: django.shortcuts.render imported but unused

Remove unused import: django.shortcuts.render

(F401)

nadabot/api.py (1)

1-1: Minimal API Stub: Consider Adding Context and Docstrings

This new file acts as a basic placeholder for the Nadabot API. In future iterations, consider adding module-level docstrings and placeholder functions or endpoints to clarify its purpose and guide subsequent development.

chains/tests.py (1)

1-3: Remove Unused Import in Test File

The TestCase import from django.test is currently unused, as indicated by static analysis (F401). Either remove this import until tests are implemented or add a dummy test to utilize it.

- from django.test import TestCase
+ # from django.test import TestCase  # Temporarily commented out until tests are added.
🧰 Tools
🪛 Ruff (0.8.2)

1-1: django.test.TestCase imported but unused

Remove unused import: django.test.TestCase

(F401)

chains/apps.py (1)

1-7: Django AppConfig Implementation Looks Good

The ChainsConfig class is implemented following Django conventions by setting the default auto field and the app name. Optionally, consider adding a docstring to explain the configuration’s role within the project.

grantpicks/views.py (1)

1-3: Clean Up Unused Import in Views Module

The render import from django.shortcuts is not used in the current placeholder. To reduce linter warnings and improve clarity, remove the unused import until view logic is implemented.

- from django.shortcuts import render
🧰 Tools
🪛 Ruff (0.8.2)

1-1: django.shortcuts.render imported but unused

Remove unused import: django.shortcuts.render

(F401)

grantpicks/tests.py (1)

1-3: Address Unused Import in Grantpicks Test File

Similar to the chains/tests.py file, TestCase is imported but not utilized here. Please remove the import or add at least one dummy test case to validate basic functionality.

- from django.test import TestCase
🧰 Tools
🪛 Ruff (0.8.2)

1-1: django.test.TestCase imported but unused

Remove unused import: django.test.TestCase

(F401)

donations/migrations/0009_donation_token.py (1)

18-24: Consider the implications of CASCADE deletion behavior.

The migration correctly adds a foreign key relationship between donations and tokens, but be aware that using CASCADE for the on_delete parameter means if a token is deleted, all associated donations will also be deleted. This may or may not be the intended behavior depending on your data model requirements.

If you want to preserve donation records even when tokens are deleted, consider using SET_NULL instead:

field=models.ForeignKey(
    help_text="Donation token.",
    null=True,
    -on_delete=django.db.models.deletion.CASCADE,
    +on_delete=django.db.models.deletion.SET_NULL,
    related_name="donations",
    to="tokens.token",
),
accounts/management/commands/fetchsocialprofiledata.py (2)

13-13: Potential memory issue with large datasets.

Loading all account IDs into memory at once could be problematic if there are many accounts in the system.

Use a queryset iterator instead to process accounts in batches:

- account_addresses = [account.id for account in Account.objects.all()]
+ # Process accounts in batches to reduce memory usage
+ batch_size = 100
+ total_accounts = Account.objects.count()
+ processed = 0
+ 
+ for account in Account.objects.iterator():
+     processed += 1
+     self.stdout.write(f"Processing account {processed} of {total_accounts}...")

23-23: Consider making the rate limit delay configurable.

The 1-second delay between requests is hardcoded, which limits flexibility.

Make the delay configurable through a command-line argument:

class Command(BaseCommand):
    help = "Fetch social profile data for all accounts."
+
+    def add_arguments(self, parser):
+        parser.add_argument(
+            '--delay',
+            type=float,
+            default=1.0,
+            help='Delay between requests in seconds (default: 1.0)'
+        )

    def handle(self, *args, **options):
        # Get all account addresses
        account_addresses = [account.id for account in Account.objects.all()]
+        delay = options['delay']
+        self.stdout.write(f"Using {delay} second(s) delay between requests")

        # Fetch social profile data for each account
        for address in account_addresses:
            account = Account.objects.get(id=address)
            account.fetch_near_social_profile_data()
            self.stdout.write(
                self.style.SUCCESS(f"Fetched social profile data for {address}")
            )
            # wait to avoid rate limiting
-            time.sleep(1)
+            time.sleep(delay)
api/pagination.py (3)

7-9: Fix typo in comment

There's a typo in the comment - "ovveeride" should be "override".

-# ovveeride PageNumberPagination to add page_size_query_param alias
+# override PageNumberPagination to add page_size_query_param alias

5-6: Remove unnecessary empty lines

There are multiple consecutive empty lines that don't add to readability.

 from rest_framework.pagination import PageNumberPagination

-

-

 # ovveeride PageNumberPagination to add page_size_query_param alias

8-9: Consider documenting default page size

The implementation looks good. You might want to explicitly set or document the default page size for clarity.

 class CustomSizePageNumberPagination(PageNumberPagination):
     page_size_query_param = 'page_size'
+    # Default page size is inherited from PageNumberPagination (typically 100)
nadabot/serializers.py (2)

6-9: Consider specifying fields explicitly instead of using __all__

Using fields = "__all__" includes all model fields in the serializer, which may expose sensitive data or include fields that aren't necessary for API responses. As the model grows, explicit field specification provides better control over what's exposed in your API.

class NadabotSerializer(ModelSerializer):
    class Meta:
        model = NadabotRegistry
-        fields = "__all__"  # TODO: potentially adjust this e.g. for formatting of datetimes, adding convenience fields etc
+        fields = [
+            "account", "owner", "default_human_threshold", 
+            "created_at", "updated_at", "admins", 
+            "blacklisted_accounts", "source_metadata"
+        ]  # Explicitly list required fields

12-15: Consider specifying fields explicitly instead of using __all__

Similarly to the NadabotSerializer, using fields = "__all__" in the ProviderSerializer includes all model fields in the serializer. As noted in your TODO comment, you might want to adjust this for formatting of datetimes or adding convenience fields.

class ProviderSerializer(ModelSerializer):
    class Meta:
        model = Provider
-        fields = "__all__"  # TODO: potentially adjust this e.g. for formatting of datetimes, adding convenience fields etc
+        fields = [
+            "id", "on_chain_id", "contract", "method_name", 
+            "name", "description", "status", "admin_notes",
+            "default_weight", "gas", "tags", "icon_url",
+            "external_url", "submitted_by", "submitted_at",
+            "stamp_validity_ms", "account_id_arg_name",
+            "custom_args", "registry"
+        ]  # Explicitly list required fields
nadabot/utils.py (2)

15-15: Remove debugging print statement

The print statement appears to be for debugging purposes and should be removed before deployment to production.

-    print(F"the TLA {pattern}, {receiver}")

Additionally, the acronym "TLA" is not explained and might be confusing to other developers.


12-14: Expand the function docstring

The current docstring is minimal. Consider expanding it to clarify the expected format of the receiver parameter and explain the meaning of the different patterns used.

def match_nadabot_registry_pattern(receiver):
-    """Matches nadabot subaccounts for registry."""
+    """
+    Matches nadabot subaccounts for registry.
+    
+    Args:
+        receiver (str): The account name to check against the pattern.
+            For testnet: Should match 'nadabot.testnet'
+            For mainnet: Should match pattern like 'v1.example.nadabot.near' or 'v1new.example.nadabot.near'
+    
+    Returns:
+        bool: True if the receiver matches the pattern, False otherwise.
+    """
indexer_app/management/commands/killspotindexer.py (1)

22-28: Fix unused variable warning

The worker variable is defined in the for loop but never used within the loop body. This triggers a static analysis warning.

-        for worker, tasks in active_tasks.items():
+        for _worker, tasks in active_tasks.items():
🧰 Tools
🪛 Ruff (0.8.2)

22-22: Loop control variable worker not used within loop body

Rename unused worker to _worker

(B007)

.github/workflows/deploy-prod.yml (3)

33-37: Improve shell script practices for environment variables.

The current shell script has several issues identified by static analysis:

  1. Individual redirects should be combined
  2. Variables should be double-quoted to prevent globbing and word splitting
-          echo "DATETIME=$(date +'%Y-%m-%d_%H-%M-%S')" >> $GITHUB_ENV
-          echo "FILENAME=django-indexer-prod-${DATETIME}.zip" >> $GITHUB_ENV
-          echo "S3_BUCKET=django-indexer-prod" >> $GITHUB_ENV
+          {
+            echo "DATETIME=$(date +'%Y-%m-%d_%H-%M-%S')"
+            echo "FILENAME=django-indexer-prod-${DATETIME}.zip"
+            echo "S3_BUCKET=django-indexer-prod"
+          } >> "$GITHUB_ENV"
🧰 Tools
🪛 actionlint (1.7.4)

33-33: shellcheck reported issue in this script: SC2129:style:1:1: Consider using { cmd1; cmd2; } >> file instead of individual redirects

(shellcheck)


33-33: shellcheck reported issue in this script: SC2086:info:1:49: Double quote to prevent globbing and word splitting

(shellcheck)


33-33: shellcheck reported issue in this script: SC2086:info:2:56: Double quote to prevent globbing and word splitting

(shellcheck)


33-33: shellcheck reported issue in this script: SC2086:info:3:41: Double quote to prevent globbing and word splitting

(shellcheck)


28-29: Clarify commented-out appspec.yml step.

The commented-out step for generating an appspec.yml file appears to copy the file to itself (cp appspec.yml appspec.yml), which would have no effect. If this is meant to be a placeholder, consider clarifying its purpose or removing it entirely.


44-51: Add deployment verification step.

The workflow creates a deployment but doesn't verify if it was successful. Consider adding a step to check the deployment status after creation to ensure it completes successfully.

      - name: Create CodeDeploy Deployment
        id: deploy
        run: |
          aws deploy create-deployment \
            --application-name django-indexer \
            --deployment-group-name django-indexer-prod \
            --deployment-config-name CodeDeployDefault.AllAtOnce \
            --s3-location bucket=${{ env.S3_BUCKET }},bundleType=zip,key=${{ env.FILENAME }}
+
+      - name: Wait for deployment to complete
+        run: |
+          DEPLOYMENT_ID=$(aws deploy get-deployment --deployment-id ${{ steps.deploy.outputs.stdout }} --query "deploymentInfo.deploymentId" --output text)
+          aws deploy wait deployment-successful --deployment-id $DEPLOYMENT_ID
+          echo "Deployment completed successfully"
chains/models.py (3)

5-21: Add unit tests for field constraints.

Your approach for unique and indexed fields on name and name_slug is clear. Consider adding unit tests to ensure these constraints (including slug population) behave as intended in all scenarios.


28-29: Consider using a positive field for EVM chain IDs.

evm_chain_id is an integer field, but chain IDs are typically non-negative. A PositiveIntegerField (with a suitable max value) might make the domain more explicit.


47-48: Optional: Provide more descriptive output in __str__.

Returning self.name is straightforward, but when debugging, a more detailed string (e.g., including evm_chain_id if EVM-compatible) can improve logging.

base/api.py (2)

7-7: Remove unused import of OpenApiParameter.

Static analysis indicates that OpenApiParameter is imported but not utilized. Removing it can improve code clarity.

Apply the following diff:

-from drf_spectacular.utils import (
-    OpenApiExample,
-    OpenApiParameter,
-    OpenApiResponse,
-    extend_schema,
-)
+from drf_spectacular.utils import (
+    OpenApiExample,
+    OpenApiResponse,
+    extend_schema,
+)
🧰 Tools
🪛 Ruff (0.8.2)

7-7: drf_spectacular.utils.OpenApiParameter imported but unused

Remove unused import: drf_spectacular.utils.OpenApiParameter

(F401)


36-59: Remember to maintain schema consistency for all response status codes.

You've defined a detailed schema for the 200 response and a 500 error description. If you return other status codes (e.g., 400), consider adding them to the OpenAPI specification for completeness.

scripts/after_install.sh (2)

62-66: Document rationale for skipping restarts when no migrations are found.

You commented out lines that previously performed a service restart if no migrations were found. Consider adding a brief comment clarifying why the step is no longer needed.


68-71: Confirm if collectstatic needs to run unconditionally.

collectstatic is now always run, even if no new static files exist. This is often safe but can be optimized by skipping it if no static files have changed.

nadabot/models.py (3)

13-19: Clarify usage of rule types.

The RuleType choices look valid. Consider describing usage scenarios in docstrings or comments to guide future contributors, especially for more complex rules like DIMINISHING_RETURNS.


68-97: Use UniqueConstraint meta instead of unique_together.

unique_together is valid but is considered legacy in recent Django versions. You can define UniqueConstraint(fields=["registry", "account"], name="unique_registry_account") in Meta for clarity.


241-280: Consider linking groups to constraints or rules.

Group references rule_type and rule_val; ensure there's a consistent domain or logic connecting providers and group rules, especially if new RuleType options are added later.

Would you like support drafting an additional data integrity constraint to validate group rules?

lists/migrations/0007_alter_list_cover_image_url_alter_list_description_and_more.py (1)

13-66: Consider using CharField for fields with fixed max_length.

Although TextField with a max_length is valid in Django, the database might not enforce the length constraint for text fields. If the text fields truly shouldn't exceed these particular lengths, switching to CharField could provide more direct validation at the database level for fields like description, admin_notes, registrant_notes, and tx_hash.

indexer_app/management/commands/populatedata.py (4)

2-2: Remove unused import.

The import date from datetime is not used. Consider removing it to clean up the code.

- from datetime import date, datetime
+ from datetime import datetime
🧰 Tools
🪛 Ruff (0.8.2)

2-2: datetime.date imported but unused

Remove unused import: datetime.date

(F401)


39-39: Use a more descriptive variable name.

The single-letter variable l is ambiguous. Choose a clearer name (e.g., current_list or lst) to improve readability and avoid confusion.

- for l in lists:
+ for current_list in lists:
🧰 Tools
🪛 Ruff (0.8.2)

39-39: Ambiguous variable name: l

(E741)


235-235: Remove assignment to unused variable or use it if needed.

Variable deployed_by is assigned but never used, which is flagged by static analysis. Remove it unless there's a future plan to use it.

- deployed_by = pot["deployed_by"]
🧰 Tools
🪛 Ruff (0.8.2)

235-235: Local variable deployed_by is assigned to but never used

Remove assignment to unused variable deployed_by

(F841)


26-548: Consider refactoring the monolithic handle method.

This command’s logic spans many responsibilities—pulling lists, registrations, donations, pots, and challenges. Splitting these into smaller helper methods can enhance readability, maintainability, and testability.

🧰 Tools
🪛 Ruff (0.8.2)

39-39: Ambiguous variable name: l

(E741)


235-235: Local variable deployed_by is assigned to but never used

Remove assignment to unused variable deployed_by

(F841)

README.md (4)

6-9: Address indentation per markdownlint recommendations.

The static analysis tool flagged that the list items under lines 6-9 use an indentation of 6 spaces instead of the expected 4. Adjusting the indentation will improve readability and consistency.

Apply a minimal diff to correct the indentation:

-      - [Base URL](#base-url)
-      - [Authorization](#authorization)
-      - [Error Responses](#error-responses)
-      - [Pagination](#pagination)
+    - [Base URL](#base-url)
+    - [Authorization](#authorization)
+    - [Error Responses](#error-responses)
+    - [Pagination](#pagination)
🧰 Tools
🪛 markdownlint-cli2 (0.17.2)

6-6: Unordered list indentation
Expected: 4; Actual: 6

(MD007, ul-indent)


7-7: Unordered list indentation
Expected: 4; Actual: 6

(MD007, ul-indent)


8-8: Unordered list indentation
Expected: 4; Actual: 6

(MD007, ul-indent)


9-9: Unordered list indentation
Expected: 4; Actual: 6

(MD007, ul-indent)


95-95: Rate limit update acknowledged.

Raising the rate limit to 500 requests/min is fine, but ensure your hosting environment can handle the increased traffic without performance degradation.


131-132: Redundant endpoint listing.

This section duplicates the earlier documentation of “donations_received.” To avoid confusion, consider either consolidating the references or clarifying their purpose if they serve different contexts.


133-134: Redundant endpoint listing.

Same duplication concern as above, now for the “donations_sent” endpoint.

tokens/models.py (4)

1-1: Remove unused import.

timedelta is needed, but reconsider the separate import from os import name at line 3, which appears unused.


3-3: Unused import “name.”

This import is not used in the file and may shadow the field named name. Removing it avoids confusion and satisfies static analysis.

-from os import name
🧰 Tools
🪛 Ruff (0.8.2)

3-3: os.name imported but unused

Remove unused import: os.name

(F401)


37-42: Icon field is appropriately typed.

Storing base64 data in a TextField is acceptable; just be mindful of potential DB bloat.

Consider storing icons in a dedicated storage and referencing by URL instead of embedding base64 strings.


106-128: Save method enhancement.

Auto-fetching the coin’s ID from CoinGecko is useful. Ensure symbol and 'Stellar' checks are correct for multi-token scenarios. If you intend to support multiple tokens, you might need a more flexible matching approach.

nadabot/migrations/0001_initial.py (2)

39-50: Consider using auto_now_add/auto_now for timestamps.

Currently, created_at and updated_at rely on the caller to populate them. You could simplify data consistency by enabling automatic timestamping, for example:

created_at = models.DateTimeField(auto_now_add=True)
updated_at = models.DateTimeField(auto_now=True)

This approach helps ensure they’re consistently populated without having to manage them manually.


264-328: Reassess usage of PositiveIntegerField as primary key.

While this works correctly, consider whether using Django’s auto-incrementing primary key generation (i.e., AutoField or BigAutoField) might avoid conflicts if the group ID is also managed externally or if data migrations occur. This is a minor design choice you may revisit before production.

indexer_app/tasks.py (9)

6-6: Remove unused import Path.

According to static analysis, pathlib.Path is unused. Removing it reduces clutter:

- from pathlib import Path
🧰 Tools
🪛 Ruff (0.8.2)

6-6: pathlib.Path imported but unused

Remove unused import: pathlib.Path

(F401)


11-11: Remove unused import worker_shutdown.

The static analysis indicates celery.signals.worker_shutdown is not referenced:

- from celery.signals import task_revoked, worker_shutdown
+ from celery.signals import task_revoked
🧰 Tools
🪛 Ruff (0.8.2)

11-11: celery.signals.worker_shutdown imported but unused

Remove unused import: celery.signals.worker_shutdown

(F401)


13-13: Remove unused DecimalField and Value.

Keep only the imports you actually use, such as Count, Q, and Sum:

- from django.db.models import Count, DecimalField, Q, Sum, Value
+ from django.db.models import Count, Q, Sum
🧰 Tools
🪛 Ruff (0.8.2)

13-13: django.db.models.DecimalField imported but unused

Remove unused import

(F401)


13-13: django.db.models.Value imported but unused

Remove unused import

(F401)


14-14: Remove unused Cast and NullIf.

Neither function is used in this file:

- from django.db.models.functions import Cast, NullIf
🧰 Tools
🪛 Ruff (0.8.2)

14-14: django.db.models.functions.Cast imported but unused

Remove unused import

(F401)


14-14: django.db.models.functions.NullIf imported but unused

Remove unused import

(F401)


18-18: Remove unused stellar_xdr and scval.

Only Address is used:

- from stellar_sdk import Address, stellar_xdr, scval
+ from stellar_sdk import Address
🧰 Tools
🪛 Ruff (0.8.2)

18-18: stellar_sdk.stellar_xdr imported but unused

Remove unused import

(F401)


18-18: stellar_sdk.scval imported but unused

Remove unused import

(F401)


23-23: Remove unused imports Round, RoundDeposit, ProjectContact.

Only StellarEvent is needed:

- from grantpicks.models import Round, RoundDeposit, StellarEvent, ProjectContact
+ from grantpicks.models import StellarEvent
🧰 Tools
🪛 Ruff (0.8.2)

23-23: grantpicks.models.Round imported but unused

Remove unused import

(F401)


23-23: grantpicks.models.RoundDeposit imported but unused

Remove unused import

(F401)


23-23: grantpicks.models.ProjectContact imported but unused

Remove unused import

(F401)


25-25: Remove unused imports PotApplication and PotApplicationStatus.

Only Pot and PotPayout are referenced:

- from pots.models import Pot, PotApplication, PotApplicationStatus, PotPayout
+ from pots.models import Pot, PotPayout
🧰 Tools
🪛 Ruff (0.8.2)

25-25: pots.models.PotApplication imported but unused

Remove unused import

(F401)


25-25: pots.models.PotApplicationStatus imported but unused

Remove unused import

(F401)


194-194: Use logger instead of raw print.

For better uniformity and debugging, log the message rather than printing:

- print(f"Processing pot: {pot.account}")
+ jobs_logger.info(f"Processing pot: {pot.account}")

440-440: Remove extraneous f in f-string with no placeholders.

- jobs_logger.info(f"Finished processing Stellar events.")
+ jobs_logger.info("Finished processing Stellar events.")
🧰 Tools
🪛 Ruff (0.8.2)

440-440: f-string without any placeholders

Remove extraneous f prefix

(F541)

pots/admin.py (6)

47-53: Use contextlib.suppress(KeyError) for cleaner code.
The try-except KeyError: pass can be replaced by contextlib.suppress(KeyError) for clarity:

+ from contextlib import suppress

 try:
     form.base_fields["admins"].queryset = obj.admins.all()
     form.base_fields["whitelisted_deployers"].queryset = obj.whitelisted_deployers.all()
 except KeyError:
     pass

+ with suppress(KeyError):
+     form.base_fields["admins"].queryset = obj.admins.all()
+     form.base_fields["whitelisted_deployers"].queryset = obj.whitelisted_deployers.all()

56-63: Clarify the purpose of commented-out permission methods.
If these methods are no longer needed, consider removing them. Otherwise, consider adding a brief comment explaining when and why they may be reintroduced.


89-93: Use contextlib.suppress(KeyError) instead of try-except.
Similar to lines 47-53, replacing the pass block with contextlib.suppress(KeyError) provides a cleaner pattern.

🧰 Tools
🪛 Ruff (0.8.2)

89-92: Use contextlib.suppress(KeyError) instead of try-except-pass

Replace with contextlib.suppress(KeyError)

(SIM105)


95-102: Reevaluate commented-out permission overrides.
As with PotFactoryAdmin, clarify whether these methods are intentionally deprecated or scheduled to be re-enabled.


125-133: Commented-out methods: reevaluate or remove them.
It appears you may no longer need specialized get_search_results, get_queryset, get_label, or overridden permission checks. If they are no longer necessary, removing them (or adding an explanatory comment) keeps the code clean.

Also applies to: 135-139, 140-144, 145-152


178-185: Same note for permission overrides.
If these are not needed anymore, remove. Otherwise, add a comment explaining when they might be reinstated.

api/urls.py (2)

53-93: Inconsistent path naming and delimiters in account endpoints.
Across these patterns, some paths use underscores (e.g., payouts_received), others use hyphens (list-registrations). Consider standardizing the path naming to improve consistency (e.g., use underscores consistently or convert all to hyphens).

- path("v1/accounts/<str:account_id>/list-registrations", AccountListRegistrationsAPI.as_view(), name="accounts_api_by_id_registrations"),
+ path("v1/accounts/<str:account_id>/list_registrations", AccountListRegistrationsAPI.as_view(), name="accounts_api_by_id_registrations"),

108-119: List endpoints: naming alignment.
Similar to the accounts endpoints, confirm consistent use of underscores or hyphens (e.g. "random_registration" vs. "list-registrations").

indexer_app/logging.py (1)

8-13: Useful memory monitoring utility for performance debugging

This function provides a standardized way to track memory usage during different stages of processing, which will be helpful for debugging performance issues and identifying memory leaks.

Consider adding a threshold warning when memory usage exceeds certain limits:

def log_memory_usage(stage):
    process = psutil.Process()
    memory_info = process.memory_info()
+   rss_mb = memory_info.rss / 1024 / 1024
+   vms_mb = memory_info.vms / 1024 / 1024
    logger.info(
-       f"{stage} - RSS: {memory_info.rss / 1024 / 1024:.2f} MB, VMS: {memory_info.vms / 1024 / 1024:.2f} MB"
+       f"{stage} - RSS: {rss_mb:.2f} MB, VMS: {vms_mb:.2f} MB"
    )
+   if rss_mb > 1000:  # Example threshold of 1GB
+       logger.warning(f"High memory usage detected at {stage}: {rss_mb:.2f} MB")
tokens/serializers.py (2)

6-10: Consider explicitly defining fields instead of using "all"

Using fields = "__all__" can lead to unintended exposure of sensitive fields or include fields that shouldn't be in the API. Consider explicitly listing the fields you want to serialize for better control and documentation.

class TokenSerializer(ModelSerializer):

    class Meta:
        model = Token
-        fields = "__all__"
+        fields = ['id', 'decimals', 'name', 'symbol', 'icon', 'reference']  # Add the actual fields you need

13-13: Example is helpful, consider moving to a dedicated constants file

The SIMPLE_TOKEN_EXAMPLE is useful for documentation or testing, but could be better placed in a dedicated constants file or test fixtures module if it's used across multiple files.

donations/migrations/0010_migrate_ft_to_token.py (1)

21-22: Consider adding a reverse migration

It's a good practice to provide a reverse migration function to enable rollbacks if needed.

operations = [
-    migrations.RunPython(migrate_ft_to_token),
+    migrations.RunPython(
+        migrate_ft_to_token,
+        # Reverse migration function
+        lambda apps, schema_editor: apps.get_model("donations", "Donation").objects.update(token=None)
+    ),
]
chains/migrations/0003_add_stellar_chain.py (1)

21-22: Add reverse migration function

Consider adding a reverse migration to allow rolling back these changes if needed.

operations = [
-    migrations.RunPython(create_stellar_chain),
+    migrations.RunPython(
+        create_stellar_chain,
+        # Reverse function to undo changes if needed
+        migrations.RunPython.noop
+    ),
]
chains/migrations/0001_initial.py (1)

37-39: Consider adding validators for URL fields

The rpc_url and explorer_url fields might benefit from validators to ensure they contain valid URLs with appropriate schemes (http/https) and formats.

-                ("rpc_url", models.URLField()),
-                ("explorer_url", models.URLField()),
+                ("rpc_url", models.URLField(validators=[django.core.validators.URLValidator(schemes=['http', 'https'])])),
+                ("explorer_url", models.URLField(validators=[django.core.validators.URLValidator(schemes=['http', 'https'])])),
pots/migrations/0008_migrate_ft_to_token.py (1)

21-23: Add reverse migration operation

The migration only provides a forward operation but lacks a reverse operation. This makes it difficult to roll back changes if needed.

     operations = [
-        migrations.RunPython(migrate_ft_to_token),
+        migrations.RunPython(
+            migrate_ft_to_token,
+            reverse_code=migrations.RunPython.noop
+        ),
     ]
grantpicks/migrations/0005_remove_votepair_project_votepair_projects_and_more.py (1)

22-24: Consider adding related_query_name to ManyToManyField

For consistency with other relation fields and to enable better query filtering, consider adding a related_query_name to the ManyToManyField.

             field=models.ManyToManyField(
-                related_name="vote_pairs_included_in", to="accounts.account"
+                related_name="vote_pairs_included_in",
+                related_query_name="vote_pair_included_in",
+                to="accounts.account"
             ),
grantpicks/migrations/0006_alter_round_compliance_period_ms_and_more.py (1)

16-21: Consider adding validators for reasonable time ranges

For time period fields, it might be helpful to add validators to ensure the values are within reasonable ranges. Even though the field is now a PositiveBigIntegerField, setting extremely large values could still cause issues in the application logic.

             field=models.PositiveBigIntegerField(
                 blank=True,
                 help_text="Compliance period in ms.",
                 null=True,
+                validators=[
+                    django.core.validators.MaxValueValidator(
+                        31536000000,  # Approximately 1 year in milliseconds
+                        message="Compliance period should not exceed 1 year"
+                    )
+                ],
                 verbose_name="compliance period in ms",
             ),
chains/migrations/0002_add_near_chain.py (2)

6-17: Missing reverse migration for data changes

The migration creates and assigns the NEAR chain, but lacks a reverse migration function to undo these changes if needed. Data migrations should typically include both forward and reverse operations.

Consider adding a reverse function that would undo this change:

 def create_near_chain(apps, schema_editor):
     Chain = apps.get_model("chains", "Chain")
     # Create the "near" chain
     near_chain, created = Chain.objects.get_or_create(
         name="NEAR", defaults={"evm_compat": False}
     )
 
     # Set "near" chain as the default for all existing accounts
     Account = apps.get_model("accounts", "Account")
     Account.objects.update(chain=near_chain)
     print("Updated all accounts to use NEAR chain")
+
+
+def reverse_near_chain(apps, schema_editor):
+    # This won't delete the chain but will set account.chain to NULL if possible
+    Account = apps.get_model("accounts", "Account")
+    Chain = apps.get_model("chains", "Chain")
+    try:
+        near_chain = Chain.objects.get(name="NEAR")
+        Account.objects.filter(chain=near_chain).update(chain=None)
+    except Chain.DoesNotExist:
+        pass

Then update the operations:

 operations = [
-    migrations.RunPython(create_near_chain),
+    migrations.RunPython(create_near_chain, reverse_near_chain),
 ]

9-11: Consider adding logging instead of print statements

Using print() statements in migrations is only visible during direct console execution. For better observability across all environments, consider using the logging module.

 def create_near_chain(apps, schema_editor):
+    import logging
+    logger = logging.getLogger(__name__)
     Chain = apps.get_model("chains", "Chain")
     # Create the "near" chain
     near_chain, created = Chain.objects.get_or_create(
         name="NEAR", defaults={"evm_compat": False}
     )
+    if created:
+        logger.info("Created new NEAR chain")
donations/migrations/0012_update_donations_net_amount.py (1)

23-27: Add reversal function for RunPython migration

Like the previous migration, this one lacks a reverse function. For data migrations, it's best practice to include a reverse operation, even if it's just a no-op function.

+def reverse_update_donations_net_amount(apps, schema_editor):
+    # This is a one-way migration as we're fixing data, not changing structure
+    pass

 class Migration(migrations.Migration):

     dependencies = [
         ("donations", "0011_remove_donation_ft_alter_donation_token"),
     ]

     operations = [
-        migrations.RunPython(update_donations_net_amount),
+        migrations.RunPython(update_donations_net_amount, reverse_update_donations_net_amount),
     ]
donations/admin.py (2)

24-24: Expanded filtering might increase maintenance overhead.

Adding "pot__account" to list_filter expands search usability, but ensure frequent filters remain relevant to average admin usage, as too many filters can complicate the UI.


29-29: Keep the comment concise or remove if self-descriptive.

The comment effectively describes the reason for the prefetch, but if the model names are self-explanatory, consider simplifying it to:

# Prefetch related fields to prevent N+1 queries
donations/api.py (1)

1-1: Consider import usage cleanup.

OpenApiParameter is not used in this file. Removing unused imports helps maintain clarity, reduces confusion, and adheres to best practices.

-from drf_spectacular.utils import (
-    OpenApiExample,
-    OpenApiParameter,
-    OpenApiResponse,
-    extend_schema,
-)
+from drf_spectacular.utils import (
+    OpenApiExample,
+    OpenApiResponse,
+    extend_schema,
+)
accounts/models.py (2)

3-3: Remove unused imports.

django.db is imported but never used. Removing unused imports improves readability and compliance with lint rules.

- from django import db
🧰 Tools
🪛 Ruff (0.8.2)

3-3: django.db imported but unused

Remove unused import: django.db

(F401)


71-132: Robust error handling and data enrichment.

  1. The approach for fetching NEAR social profile data is well-structured.
  2. Logging errors is good for transparency.
  3. Consider implementing a timeout with requests.get(..., timeout=...) or requests.post(..., timeout=...) to handle slow or unresponsive endpoints gracefully.
donations/models.py (5)

1-1: Remove unused import token.

This import overlaps with the Token model reference, and it is not actually used in the file.

- import token
🧰 Tools
🪛 Ruff (0.8.2)

1-1: token imported but unused

Remove unused import: token

(F401)


2-3: Remove unused imports.

Several imports (timedelta, Decimal, requests, settings) are not used in the file. Consider removing them to keep the code clean.

-from datetime import timedelta
-from decimal import Decimal
 import requests
-from django.conf import settings

Also applies to: 5-5, 7-7

🧰 Tools
🪛 Ruff (0.8.2)

2-2: datetime.timedelta imported but unused

Remove unused import: datetime.timedelta

(F401)


3-3: decimal.Decimal imported but unused

Remove unused import: decimal.Decimal

(F401)


68-74: Remove commented-out code if it's no longer needed.

If you've replaced the ft logic with the new token reference and won't need it in the future, consider removing this commented-out block.

-    # ft = models.ForeignKey(
-    #     Account,  # should probably be Token
-    #     on_delete=models.CASCADE,
-    #     related_name="ft_donations",
-    #     null=False,
-    #     help_text=_("Donation FT."),
-    # )

213-217: Remove unused local variables.

existing_*_usd variables are assigned but never used, creating dead code.

- existing_total_amount_usd = self.total_amount_usd
- existing_net_amount_usd = self.net_amount_usd
- existing_protocol_fee_usd = self.protocol_fee_usd
- existing_referrer_fee_usd = self.referrer_fee_usd
- existing_chef_fee_usd = self.chef_fee_usd
🧰 Tools
🪛 Ruff (0.8.2)

213-213: Local variable existing_total_amount_usd is assigned to but never used

Remove assignment to unused variable existing_total_amount_usd

(F841)


214-214: Local variable existing_net_amount_usd is assigned to but never used

Remove assignment to unused variable existing_net_amount_usd

(F841)


215-215: Local variable existing_protocol_fee_usd is assigned to but never used

Remove assignment to unused variable existing_protocol_fee_usd

(F841)


216-216: Local variable existing_referrer_fee_usd is assigned to but never used

Remove assignment to unused variable existing_referrer_fee_usd

(F841)


217-217: Local variable existing_chef_fee_usd is assigned to but never used

Remove assignment to unused variable existing_chef_fee_usd

(F841)


220-246: Consider refining broad exception handling.

Catching the general Exception might obscure specific errors. Use narrower exception classes or at least re-raise after logging to aid debugging.

except Exception as e:
    logger.error(f"Failed to calculate and save USD prices: {e}")
+    # Optionally re-raise for better visibility:
+    # raise
lists/api.py (1)

4-4: Remove unused import timezone if not used.

You can safely delete this import to keep the code concise.

-from django.utils import timezone
🧰 Tools
🪛 Ruff (0.8.2)

4-4: django.utils.timezone imported but unused

Remove unused import: django.utils.timezone

(F401)

donations/serializers.py (1)

5-5: Remove unused imports.

SerializerMethodField and EXAMPLE_POT_ID are not being used in this file.

- from rest_framework.serializers import (..., SerializerMethodField, ...)
- from pots.serializers import EXAMPLE_POT_ID, ...

Also applies to: 9-9

🧰 Tools
🪛 Ruff (0.8.2)

5-5: rest_framework.serializers.SerializerMethodField imported but unused

Remove unused import: rest_framework.serializers.SerializerMethodField

(F401)

grantpicks/api.py (3)

1-2: Remove unused imports.
Static analysis indicates that Q and timezone are unused. Removing them will help keep the code clean and avoid confusion.

-from django.db.models import Q, Count, Sum
+from django.db.models import Count, Sum
-from django.utils import timezone
🧰 Tools
🪛 Ruff (0.8.2)

1-1: django.db.models.Q imported but unused

Remove unused import: django.db.models.Q

(F401)


2-2: django.utils.timezone imported but unused

Remove unused import: django.utils.timezone

(F401)


15-29: Remove unused imports from accounts and donations.
The following imports are unused, leading to dead code and potential confusion:

  • PAGINATED_ACCOUNT_EXAMPLE (line 17)
  • AccountSerializer (line 18)
  • PaginatedAccountsResponseSerializer (line 19)
  • Donation (line 24)
  • PAGINATED_DONATION_EXAMPLE (line 26)
  • DonationSerializer (line 27)
  • PaginatedDonationsResponseSerializer (line 28)
  • StatsResponseSerializer (line 23)
-from accounts.serializers import (
-    PAGINATED_ACCOUNT_EXAMPLE,
-    AccountSerializer,
-    PaginatedAccountsResponseSerializer,
-)
-from base.api import StatsResponseSerializer
-from donations.models import Donation
-from donations.serializers import (
-    PAGINATED_DONATION_EXAMPLE,
-    DonationSerializer,
-    PaginatedDonationsResponseSerializer,
-)
🧰 Tools
🪛 Ruff (0.8.2)

17-17: accounts.serializers.PAGINATED_ACCOUNT_EXAMPLE imported but unused

Remove unused import

(F401)


18-18: accounts.serializers.AccountSerializer imported but unused

Remove unused import

(F401)


19-19: accounts.serializers.PaginatedAccountsResponseSerializer imported but unused

Remove unused import

(F401)


23-23: base.api.StatsResponseSerializer imported but unused

Remove unused import: base.api.StatsResponseSerializer

(F401)


24-24: donations.models.Donation imported but unused

Remove unused import: donations.models.Donation

(F401)


26-26: donations.serializers.PAGINATED_DONATION_EXAMPLE imported but unused

Remove unused import

(F401)


27-27: donations.serializers.DonationSerializer imported but unused

Remove unused import

(F401)


28-28: donations.serializers.PaginatedDonationsResponseSerializer imported but unused

Remove unused import

(F401)


217-240: Consider removing or clarifying commented-out code.
Lines 219, 225, and 226 contain commented-out references to project and votes. To keep the codebase free of clutter, remove or refactor this code if it is truly unnecessary, or add a comment explaining why it's intentionally left for future use.

-# project = Project.objects.get(id=project_id)
 
-# votes = round_obj.votes.filter(pairs__project_id=project_id)
-# vote_pairs = project.vote_pairs.all()
accounts/serializers.py (1)

2-2: Remove unused imports ModelSerializer and SerializerMethodField.
They are not used anywhere in the file. Unused imports may trigger linter errors and can clutter the code.

-from rest_framework.serializers import ModelSerializer, SerializerMethodField
🧰 Tools
🪛 Ruff (0.8.2)

2-2: rest_framework.serializers.ModelSerializer imported but unused

Remove unused import

(F401)


2-2: rest_framework.serializers.SerializerMethodField imported but unused

Remove unused import

(F401)

lists/serializers.py (4)

1-2: Remove unused import.
Static analysis indicates SerializerMethodField is unused. Consider removing it to keep imports clean and avoid confusion.

 from rest_framework import serializers
-from rest_framework.serializers import ModelSerializer, SerializerMethodField
+from rest_framework.serializers import ModelSerializer
🧰 Tools
🪛 Ruff (0.8.2)

2-2: rest_framework.serializers.SerializerMethodField imported but unused

Remove unused import: rest_framework.serializers.SerializerMethodField

(F401)


9-12: Prefer explicitly enumerating fields over __all__.
Using fields = '__all__' in ListUpvoteSerializer can inadvertently expose sensitive or future model fields. Enumerating the fields more explicitly may improve security and maintainability.


16-16: Consider marking the field read-only or providing a default.
registrations_count is set as required=False, but you might also consider read_only=True or a default value if it’s only for display or computed data.


35-37: Nested serializers look fine, but verify potential performance overhead.
Serializing large relationships can be expensive. If performance becomes an issue, consider a more selective approach or a separate endpoint for these nested resources.

pots/api.py (3)

1-7: Remove unused imports.
django.utils.timezone (line 4) appears to be unused, and static analysis also indicates some imports may be redundant. Consider pruning them:

 import os
 from django.conf import settings
-from django.utils import timezone
 from django.utils.decorators import method_decorator
 from django.views.decorators.cache import cache_page
 from drf_spectacular.utils import (
🧰 Tools
🪛 Ruff (0.8.2)

4-4: django.utils.timezone imported but unused

Remove unused import: django.utils.timezone

(F401)


35-35: PotApplication import is unused.
Remove it if the class is never referenced in this file.

-from .models import Pot, PotApplication, PotApplicationStatus, PotFactory
+from .models import Pot, PotApplicationStatus, PotFactory
🧰 Tools
🪛 Ruff (0.8.2)

35-35: .models.PotApplication imported but unused

Remove unused import: .models.PotApplication

(F401)


355-394: Consider file IO error handling in MpdaoVoterMixin.
Reading from last-snapshot-AllVoters.json could fail if the file is missing or corrupted. Adding graceful error handling might be beneficial.

accounts/api.py (3)

5-11: Remove unused OpenApiTypes.
It’s imported but not referenced in subsequent code. Keeping only required imports improves clarity.

 from drf_spectacular.utils import (
     OpenApiExample,
     OpenApiParameter,
     OpenApiResponse,
-    OpenApiTypes,
     extend_schema,
 )
🧰 Tools
🪛 Ruff (0.8.2)

9-9: drf_spectacular.utils.OpenApiTypes imported but unused

Remove unused import: drf_spectacular.utils.OpenApiTypes

(F401)


19-19: logger import appears unused.
Remove it if no logging calls exist in this file.

-from base.logging import logger
🧰 Tools
🪛 Ruff (0.8.2)

19-19: base.logging.logger imported but unused

Remove unused import: base.logging.logger

(F401)


50-93: DonorsAPI sorting logic is rudimentary but effective.
Only “most_donated_usd” is supported. If you plan to expand, consider a more scalable sorting approach or referencing a known set of fields.

grantpicks/models.py (2)

9-11: Remove unused import from pots.models
The import from pots.models import PotApplication is unused in this file.

Apply this diff to remove the unused import:

- from pots.models import PotApplication
🧰 Tools
🪛 Ruff (0.8.2)

11-11: pots.models.PotApplication imported but unused

Remove unused import: pots.models.PotApplication

(F401)


354-360: Flatten nested if statements
You can simplify the nested condition in save to a single if.

Proposed change:

-def save(self, *args, **kwargs):
-    if self._state.adding:  # If the account is being created (not updated)
-        if not self.chain_id:
-            self.chain = Chain.objects.get(name="stellar")
-    super().save(*args, **kwargs)
+def save(self, *args, **kwargs):
+    if self._state.adding and not self.chain_id:
+        self.chain = Chain.objects.get(name="stellar")
+    super().save(*args, **kwargs)
🧰 Tools
🪛 Ruff (0.8.2)

355-356: Use a single if statement instead of nested if statements

(SIM102)

pots/models.py (2)

1-12: Remove unused imports
timedelta, Decimal, and format_date are never used. Removing them reduces clutter and avoids confusion.

Proposed fix:

-from datetime import datetime, timedelta
+from datetime import datetime
-from decimal import Decimal
-from base.utils import format_date
🧰 Tools
🪛 Ruff (0.8.2)

1-1: datetime.timedelta imported but unused

Remove unused import: datetime.timedelta

(F401)


2-2: decimal.Decimal imported but unused

Remove unused import: decimal.Decimal

(F401)


12-12: base.utils.format_date imported but unused

Remove unused import: base.utils.format_date

(F401)


326-388: Long method in update_configs
Consider refactoring this method into smaller helpers for improved readability and maintainability.

grantpicks/serializers.py (1)

7-7: Remove the unused import of “Vote.”

Static analysis indicates that Vote is imported but never referenced. Cleaning up unused imports helps maintain clarity.

-from .models import Project, ProjectContact, ProjectContract, ProjectRepository, Round, Vote, VotePair
+from .models import Project, ProjectContact, ProjectContract, ProjectRepository, Round, VotePair
🧰 Tools
🪛 Ruff (0.8.2)

7-7: .models.Vote imported but unused

Remove unused import: .models.Vote

(F401)

pots/serializers.py (2)

5-5: Remove unused import of “settings.”

Static analysis flags django.conf.settings as unused. Removing it keeps the file tidy.

-from django.conf import settings
🧰 Tools
🪛 Ruff (0.8.2)

5-5: django.conf.settings imported but unused

Remove unused import: django.conf.settings

(F401)


8-8: Remove unused import of “NearSocialProfileDataSerializer.”

This import is never referenced. Clean up to reduce clutter.

-from accounts.serializers import SIMPLE_ACCOUNT_EXAMPLE, AccountSerializer, NearSocialProfileDataSerializer
+from accounts.serializers import SIMPLE_ACCOUNT_EXAMPLE, AccountSerializer
🧰 Tools
🪛 Ruff (0.8.2)

8-8: accounts.serializers.NearSocialProfileDataSerializer imported but unused

Remove unused import: accounts.serializers.NearSocialProfileDataSerializer

(F401)

indexer_app/handler.py (13)

1-1: Remove the unused asyncio import.

The asyncio module is never referenced in this file, producing a lint warning. Consider removing this import for cleanliness.

- import asyncio
🧰 Tools
🪛 Ruff (0.8.2)

1-1: asyncio imported but unused

Remove unused import: asyncio

(F401)


12-12: Remove the unused GetEventsResponse import.

This import from stellar_sdk.soroban_rpc is not utilized, so it should be removed.

- from stellar_sdk.soroban_rpc import GetEventsResponse
🧰 Tools
🪛 Ruff (0.8.2)

12-12: stellar_sdk.soroban_rpc.GetEventsResponse imported but unused

Remove unused import: stellar_sdk.soroban_rpc.GetEventsResponse

(F401)


16-16: Remove the unused StellarEvent import.

StellarEvent from grantpicks.models is not referenced in this file, so it should be removed.

- from grantpicks.models import StellarEvent
🧰 Tools
🪛 Ruff (0.8.2)

16-16: grantpicks.models.StellarEvent imported but unused

Remove unused import: grantpicks.models.StellarEvent

(F401)


65-65: Eliminate unused variable assignment.

start_time is set but never used in this function.

-    start_time = time.time()
🧰 Tools
🪛 Ruff (0.8.2)

65-65: Local variable start_time is assigned to but never used

Remove assignment to unused variable start_time

(F841)


71-75: Clean up commented-out code.

These commented lines can either be removed or reactivated if needed. Continuing to leave them in may add confusion.

Also applies to: 80-83


88-88: Rename or remove the unused loop variable.

shard_index is not used within the loop body. Renaming it to _shard_index or removing it prevents confusion.

- for shard_index, shard in enumerate(streamer_message.shards):
+ for _shard_index, shard in enumerate(streamer_message.shards):
🧰 Tools
🪛 Ruff (0.8.2)

88-88: Loop control variable shard_index not used within loop body

Rename unused shard_index to _shard_index

(B007)


89-89: Remove the redundant shard_start_time assignment.

This local variable is never used in the function body.

-        shard_start_time = time.time()
🧰 Tools
🪛 Ruff (0.8.2)

89-89: Local variable shard_start_time is assigned to but never used

Remove assignment to unused variable shard_start_time

(F841)


90-90: Rename or remove the unused loop variable.

outcome_index is never read. Renaming it to _outcome_index or removing it avoids lint warnings and confusion.

-        for outcome_index, receipt_execution_outcome in enumerate(
+        for _outcome_index, receipt_execution_outcome in enumerate(
🧰 Tools
🪛 Ruff (0.8.2)

90-90: Loop control variable outcome_index not used within loop body

Rename unused outcome_index to _outcome_index

(B007)


113-113: Remove the redundant log_processing_start assignment.

This variable is never used after initialization.

-            log_processing_start = time.time()
🧰 Tools
🪛 Ruff (0.8.2)

113-113: Local variable log_processing_start is assigned to but never used

Remove assignment to unused variable log_processing_start

(F841)


114-114: Rename or remove the unused log_index variable.

Currently log_index is not utilized. Renaming it to _log_index or removing it might help avoid confusion.

-            for log_index, log in enumerate(
+            for _log_index, log in enumerate(
🧰 Tools
🪛 Ruff (0.8.2)

114-114: Loop control variable log_index not used within loop body

Rename unused log_index to _log_index

(B007)


122-122: Replace the print statement with logger for consistency.

Using logger.info or logger.debug will maintain unified logging practices.

-                    print("parsa parsa...", parsed_log)
+                    logger.info(f"parsa parsa... {parsed_log}")

190-191: Combine nested if statements for clarity.

Per static analysis, merge these conditions into a single if check:

if event_name == "vote" and receiver_id == settings.NEAR_GRANTPICKS_CONTRACT_ID:
    ...
🧰 Tools
🪛 Ruff (0.8.2)

190-191: Use a single if statement instead of nested if statements

(SIM102)


237-237: Rename or remove the unused loop variable.

index is never used in the loop body. Renaming it to _index or removing it prevents confusion.

-            for index, action in enumerate(
+            for _index, action in enumerate(
🧰 Tools
🪛 Ruff (0.8.2)

237-237: Loop control variable index not used within loop body

Rename unused index to _index

(B007)

indexer_app/utils.py (15)

4-4: Remove the unused log import.

math.log is not referenced anywhere in this file.

- from math import log
🧰 Tools
🪛 Ruff (0.8.2)

4-4: math.log imported but unused

Remove unused import: math.log

(F401)


409-409: Eliminate the unused listObject variable.

The result of .aupdate() is never stored or referenced.

-        listObject = await List.objects.filter(on_chain_id=data["id"]).aupdate(
+        await List.objects.filter(on_chain_id=data["id"]).aupdate(
🧰 Tools
🪛 Ruff (0.8.2)

409-409: Local variable listObject is assigned to but never used

Remove assignment to unused variable listObject

(F841)


435-435: Remove the unused lst variable.

adelete() directly returns a tuple describing the deletion, and you don't reuse lst afterward.

-        lst = await List.objects.filter(on_chain_id=data["list_id"]).adelete()
+        await List.objects.filter(on_chain_id=data["list_id"]).adelete()
🧰 Tools
🪛 Ruff (0.8.2)

435-435: Local variable lst is assigned to but never used

Remove assignment to unused variable lst

(F841)


761-761: Remove the f prefix from this string.

There are no placeholders, so the f-string is unnecessary.

-        logger.info(f"Upvote removed successfully")
+        logger.info("Upvote removed successfully")
🧰 Tools
🪛 Ruff (0.8.2)

761-761: f-string without any placeholders

Remove extraneous f prefix

(F541)


897-897: Use not in for membership checks.

Change if not admin.id in data["admins"]: to a clearer if admin.id not in data["admins"]:.

-            if not admin.id in data["admins"]:
+            if admin.id not in data["admins"]:
🧰 Tools
🪛 Ruff (0.8.2)

897-897: Test for membership should be not in

Convert to not in

(E713)


1202-1202: Remove the unused reg variable.

This local variable is assigned but never used.

-        reg = await NadabotRegistry.objects.filter(account=receiverId).aupdate(
+        await NadabotRegistry.objects.filter(account=receiverId).aupdate(
🧰 Tools
🪛 Ruff (0.8.2)

1202-1202: Local variable reg is assigned to but never used

Remove assignment to unused variable reg

(F841)


1231-1231: Drop the unused provider variable.

This assignment is never referenced later in the code.

-        provider = await Provider.objects.aupdate_or_create(
+        await Provider.objects.aupdate_or_create(
🧰 Tools
🪛 Ruff (0.8.2)

1231-1231: Local variable provider is assigned to but never used

Remove assignment to unused variable provider

(F841)


1269-1269: Remove the unused stamp variable.

No code references this newly assigned stamp.

-        stamp = await Stamp.objects.aupdate_or_create(
+        await Stamp.objects.aupdate_or_create(
🧰 Tools
🪛 Ruff (0.8.2)

1269-1269: Local variable stamp is assigned to but never used

Remove assignment to unused variable stamp

(F841)


1288-1288: Use isinstance(rule, dict) for type checks.

Comparisons like type(rule) == dict should be replaced with isinstance(rule, dict).

-        if type(rule) == dict:
+        if isinstance(rule, dict):
🧰 Tools
🪛 Ruff (0.8.2)

1288-1288: Use is and is not for type comparisons, or isinstance() for isinstance checks

(E721)


1365-1365: Use isinstance(event_data, list) for type checks.

Comparisons like type(event_data) == list are less idiomatic.

-    if type(event_data) == list:
+    if isinstance(event_data, list):
🧰 Tools
🪛 Ruff (0.8.2)

1365-1365: Use is and is not for type comparisons, or isinstance() for isinstance checks

(E721)


1384-1384: Remove the unused submitted_at variable.

Once assigned, it is not referenced.

-            submitted_at = datetime.fromtimestamp(application_data['submited_ms'] / 1000)
🧰 Tools
🪛 Ruff (0.8.2)

1384-1384: Local variable submitted_at is assigned to but never used

Remove assignment to unused variable submitted_at

(F841)


1462-1462: Use isinstance(event_data, list) instead of type(event_data) == list.

This is more Pythonic and avoids subtle corner cases.

-    if type(event_data) == list:
+    if isinstance(event_data, list):
🧰 Tools
🪛 Ruff (0.8.2)

1462-1462: Use is and is not for type comparisons, or isinstance() for isinstance checks

(E721)


1631-1634: Use a ternary expression to streamline use_vault assignment.

Instead of an if-else block, do something like:

use_vault = True if chain_id == "NEAR" else event_data.get("use_vault", False)
🧰 Tools
🪛 Ruff (0.8.2)

1631-1634: Use ternary operator use_vault = True if chain_id == "NEAR" else event_data.get("use_vault", False) instead of if-else-block

Replace if-else-block with use_vault = True if chain_id == "NEAR" else event_data.get("use_vault", False)

(SIM108)


1726-1726: Use isinstance(event_data, list) again.

Consistently using isinstance is a standard Python practice.

-    if type(event_data) == list:
+    if isinstance(event_data, list):
🧰 Tools
🪛 Ruff (0.8.2)

1726-1726: Use is and is not for type comparisons, or isinstance() for isinstance checks

(E721)


1765-1765: Adopt isinstance(event_data, list) for type checks.

This is consistent with Python guidelines.

-    if type(event_data) == list:
+    if isinstance(event_data, list):
🧰 Tools
🪛 Ruff (0.8.2)

1765-1765: Use is and is not for type comparisons, or isinstance() for isinstance checks

(E721)

📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro

📥 Commits

Reviewing files that changed from the base of the PR and between 2297ed5 and 6018172.

📒 Files selected for processing (116)
  • .github/workflows/deploy-prod.yml (1 hunks)
  • .gitignore (3 hunks)
  • README.md (5 hunks)
  • accounts/admin.py (2 hunks)
  • accounts/api.py (2 hunks)
  • accounts/management/commands/fetchsocialprofiledata.py (1 hunks)
  • accounts/migrations/0002_account_near_social_profile_data.py (1 hunks)
  • accounts/migrations/0003_alter_account_options.py (1 hunks)
  • accounts/migrations/0004_account_chain.py (1 hunks)
  • accounts/migrations/0005_alter_account_chain.py (1 hunks)
  • accounts/migrations/0006_alter_account_near_social_profile_data.py (1 hunks)
  • accounts/models.py (3 hunks)
  • accounts/serializers.py (2 hunks)
  • activities/admin.py (2 hunks)
  • activities/migrations/0004_alter_activity_tx_hash.py (1 hunks)
  • activities/migrations/0005_alter_activity_action_result_alter_activity_tx_hash.py (1 hunks)
  • activities/models.py (1 hunks)
  • api/pagination.py (1 hunks)
  • api/urls.py (1 hunks)
  • base/api.py (2 hunks)
  • base/celery.py (1 hunks)
  • base/serializers.py (1 hunks)
  • base/settings.py (6 hunks)
  • chains/admin.py (1 hunks)
  • chains/apps.py (1 hunks)
  • chains/migrations/0001_initial.py (1 hunks)
  • chains/migrations/0002_add_near_chain.py (1 hunks)
  • chains/migrations/0003_add_stellar_chain.py (1 hunks)
  • chains/models.py (1 hunks)
  • chains/tests.py (1 hunks)
  • chains/views.py (1 hunks)
  • donations/admin.py (3 hunks)
  • donations/api.py (1 hunks)
  • donations/migrations/0008_alter_donation_tx_hash.py (1 hunks)
  • donations/migrations/0009_donation_token.py (1 hunks)
  • donations/migrations/0010_migrate_ft_to_token.py (1 hunks)
  • donations/migrations/0011_remove_donation_ft_alter_donation_token.py (1 hunks)
  • donations/migrations/0012_update_donations_net_amount.py (1 hunks)
  • donations/migrations/0013_alter_donation_chef_alter_donation_chef_fee_and_more.py (1 hunks)
  • donations/models.py (7 hunks)
  • donations/serializers.py (1 hunks)
  • grantpicks/admin.py (1 hunks)
  • grantpicks/api.py (1 hunks)
  • grantpicks/apps.py (1 hunks)
  • grantpicks/migrations/0001_initial.py (1 hunks)
  • grantpicks/migrations/0002_round_chain_rounddeposit_memo_and_more.py (1 hunks)
  • grantpicks/migrations/0003_alter_votepair_project.py (1 hunks)
  • grantpicks/migrations/0004_alter_round_on_chain_id_and_more.py (1 hunks)
  • grantpicks/migrations/0005_remove_votepair_project_votepair_projects_and_more.py (1 hunks)
  • grantpicks/migrations/0006_alter_round_compliance_period_ms_and_more.py (1 hunks)
  • grantpicks/models.py (1 hunks)
  • grantpicks/serializers.py (1 hunks)
  • grantpicks/tests.py (1 hunks)
  • grantpicks/views.py (1 hunks)
  • indexer_app/admin.py (1 hunks)
  • indexer_app/examples.txt (1 hunks)
  • indexer_app/handler.py (10 hunks)
  • indexer_app/logging.py (1 hunks)
  • indexer_app/management/commands/killspotindexer.py (1 hunks)
  • indexer_app/management/commands/populatedata.py (1 hunks)
  • indexer_app/management/commands/runspotindexer.py (1 hunks)
  • indexer_app/migrations/0003_alter_blockheight_block_timestamp.py (1 hunks)
  • indexer_app/models.py (1 hunks)
  • indexer_app/tasks.py (6 hunks)
  • indexer_app/utils.py (22 hunks)
  • lists/admin.py (1 hunks)
  • lists/api.py (1 hunks)
  • lists/migrations/0003_alter_listupvote_options.py (1 hunks)
  • lists/migrations/0004_list_on_chain_id.py (1 hunks)
  • lists/migrations/0005_alter_list_id.py (1 hunks)
  • lists/migrations/0006_alter_listregistration_unique_together.py (1 hunks)
  • lists/migrations/0007_alter_list_cover_image_url_alter_list_description_and_more.py (1 hunks)
  • lists/models.py (4 hunks)
  • lists/serializers.py (1 hunks)
  • nadabot/admin.py (1 hunks)
  • nadabot/api.py (1 hunks)
  • nadabot/apps.py (1 hunks)
  • nadabot/migrations/0001_initial.py (1 hunks)
  • nadabot/migrations/0002_rename_id_nadabotregistry_account.py (1 hunks)
  • nadabot/migrations/0003_alter_group_rule_type_alter_group_rule_val_and_more.py (1 hunks)
  • nadabot/models.py (1 hunks)
  • nadabot/serializers.py (1 hunks)
  • nadabot/tests.py (1 hunks)
  • nadabot/utils.py (1 hunks)
  • nadabot/views.py (1 hunks)
  • pots/admin.py (6 hunks)
  • pots/api.py (1 hunks)
  • pots/migrations/0003_alter_potapplication_options_and_more.py (1 hunks)
  • pots/migrations/0004_alter_potapplication_tx_hash_and_more.py (1 hunks)
  • pots/migrations/0005_alter_potapplication_updated_at.py (1 hunks)
  • pots/migrations/0006_alter_pot_custom_sybil_checks_and_more.py (1 hunks)
  • pots/migrations/0007_potpayout_token.py (1 hunks)
  • pots/migrations/0008_migrate_ft_to_token.py (1 hunks)
  • pots/migrations/0009_remove_potpayout_ft_alter_potpayout_token.py (1 hunks)
  • pots/migrations/0010_alter_potpayout_paid_at.py (1 hunks)
  • pots/migrations/0011_rename_id_pot_account_rename_id_potfactory_account.py (1 hunks)
  • pots/migrations/0012_alter_pot_base_currency_alter_pot_chef_and_more.py (1 hunks)
  • pots/migrations/0013_potpayoutchallenge_tx_hash.py (1 hunks)
  • pots/migrations/0014_alter_potapplication_unique_together_and_more.py (1 hunks)
  • pots/migrations/0015_alter_potpayout_options_and_more.py (1 hunks)
  • pots/migrations/0016_alter_potapplication_project.py (1 hunks)
  • pots/models.py (22 hunks)
  • pots/serializers.py (1 hunks)
  • pots/utils.py (1 hunks)
  • pyproject.toml (1 hunks)
  • scripts/after_install.sh (1 hunks)
  • scripts/after_install_dev.sh (2 hunks)
  • scripts/after_install_testnet.sh (2 hunks)
  • tokens/admin.py (1 hunks)
  • tokens/migrations/0002_token_icon_token_name_token_symbol.py (1 hunks)
  • tokens/migrations/0003_token_coingecko_id.py (1 hunks)
  • tokens/migrations/0004_alter_token_coingecko_id.py (1 hunks)
  • tokens/migrations/0005_alter_token_coingecko_id_alter_token_icon_and_more.py (1 hunks)
  • tokens/migrations/0006_rename_id_token_account.py (1 hunks)
  • tokens/models.py (2 hunks)
  • tokens/serializers.py (1 hunks)
🧰 Additional context used
🪛 Ruff (0.8.2)
nadabot/admin.py

1-1: django.contrib.admin imported but unused

Remove unused import: django.contrib.admin

(F401)

nadabot/tests.py

1-1: django.test.TestCase imported but unused

Remove unused import: django.test.TestCase

(F401)

nadabot/views.py

1-1: django.shortcuts.render imported but unused

Remove unused import: django.shortcuts.render

(F401)

grantpicks/views.py

1-1: django.shortcuts.render imported but unused

Remove unused import: django.shortcuts.render

(F401)

chains/views.py

1-1: django.shortcuts.render imported but unused

Remove unused import: django.shortcuts.render

(F401)

grantpicks/admin.py

1-1: django.contrib.admin imported but unused

Remove unused import: django.contrib.admin

(F401)

chains/tests.py

1-1: django.test.TestCase imported but unused

Remove unused import: django.test.TestCase

(F401)

indexer_app/management/commands/killspotindexer.py

22-22: Loop control variable worker not used within loop body

Rename unused worker to _worker

(B007)

nadabot/serializers.py

3-3: .models.Stamp imported but unused

Remove unused import: .models.Stamp

(F401)

donations/api.py

7-7: drf_spectacular.utils.OpenApiParameter imported but unused

Remove unused import: drf_spectacular.utils.OpenApiParameter

(F401)

grantpicks/tests.py

1-1: django.test.TestCase imported but unused

Remove unused import: django.test.TestCase

(F401)

accounts/models.py

3-3: django.db imported but unused

Remove unused import: django.db

(F401)

base/api.py

7-7: drf_spectacular.utils.OpenApiParameter imported but unused

Remove unused import: drf_spectacular.utils.OpenApiParameter

(F401)

donations/models.py

1-1: token imported but unused

Remove unused import: token

(F401)


2-2: datetime.timedelta imported but unused

Remove unused import: datetime.timedelta

(F401)


3-3: decimal.Decimal imported but unused

Remove unused import: decimal.Decimal

(F401)


5-5: requests imported but unused

Remove unused import: requests

(F401)


7-7: django.conf.settings imported but unused

Remove unused import: django.conf.settings

(F401)


75-75: Redefinition of unused token from line 1

(F811)


213-213: Local variable existing_total_amount_usd is assigned to but never used

Remove assignment to unused variable existing_total_amount_usd

(F841)


214-214: Local variable existing_net_amount_usd is assigned to but never used

Remove assignment to unused variable existing_net_amount_usd

(F841)


215-215: Local variable existing_protocol_fee_usd is assigned to but never used

Remove assignment to unused variable existing_protocol_fee_usd

(F841)


216-216: Local variable existing_referrer_fee_usd is assigned to but never used

Remove assignment to unused variable existing_referrer_fee_usd

(F841)


217-217: Local variable existing_chef_fee_usd is assigned to but never used

Remove assignment to unused variable existing_chef_fee_usd

(F841)

lists/api.py

4-4: django.utils.timezone imported but unused

Remove unused import: django.utils.timezone

(F401)

indexer_app/management/commands/populatedata.py

2-2: datetime.date imported but unused

Remove unused import: datetime.date

(F401)


39-39: Ambiguous variable name: l

(E741)


235-235: Local variable deployed_by is assigned to but never used

Remove assignment to unused variable deployed_by

(F841)

lists/serializers.py

2-2: rest_framework.serializers.SerializerMethodField imported but unused

Remove unused import: rest_framework.serializers.SerializerMethodField

(F401)

accounts/serializers.py

2-2: rest_framework.serializers.ModelSerializer imported but unused

Remove unused import

(F401)


2-2: rest_framework.serializers.SerializerMethodField imported but unused

Remove unused import

(F401)

donations/serializers.py

5-5: rest_framework.serializers.SerializerMethodField imported but unused

Remove unused import: rest_framework.serializers.SerializerMethodField

(F401)


9-9: pots.serializers.EXAMPLE_POT_ID imported but unused

Remove unused import: pots.serializers.EXAMPLE_POT_ID

(F401)

grantpicks/api.py

1-1: django.db.models.Q imported but unused

Remove unused import: django.db.models.Q

(F401)


2-2: django.utils.timezone imported but unused

Remove unused import: django.utils.timezone

(F401)


17-17: accounts.serializers.PAGINATED_ACCOUNT_EXAMPLE imported but unused

Remove unused import

(F401)


18-18: accounts.serializers.AccountSerializer imported but unused

Remove unused import

(F401)


19-19: accounts.serializers.PaginatedAccountsResponseSerializer imported but unused

Remove unused import

(F401)


23-23: base.api.StatsResponseSerializer imported but unused

Remove unused import: base.api.StatsResponseSerializer

(F401)


24-24: donations.models.Donation imported but unused

Remove unused import: donations.models.Donation

(F401)


26-26: donations.serializers.PAGINATED_DONATION_EXAMPLE imported but unused

Remove unused import

(F401)


27-27: donations.serializers.DonationSerializer imported but unused

Remove unused import

(F401)


28-28: donations.serializers.PaginatedDonationsResponseSerializer imported but unused

Remove unused import

(F401)

accounts/api.py

9-9: drf_spectacular.utils.OpenApiTypes imported but unused

Remove unused import: drf_spectacular.utils.OpenApiTypes

(F401)


19-19: base.logging.logger imported but unused

Remove unused import: base.logging.logger

(F401)

tokens/models.py

3-3: os.name imported but unused

Remove unused import: os.name

(F401)


23-23: Redefinition of unused name from line 3

(F811)

grantpicks/models.py

11-11: pots.models.PotApplication imported but unused

Remove unused import: pots.models.PotApplication

(F401)


355-356: Use a single if statement instead of nested if statements

(SIM102)

pots/admin.py

89-92: Use contextlib.suppress(KeyError) instead of try-except-pass

Replace with contextlib.suppress(KeyError)

(SIM105)

pots/api.py

4-4: django.utils.timezone imported but unused

Remove unused import: django.utils.timezone

(F401)


35-35: .models.PotApplication imported but unused

Remove unused import: .models.PotApplication

(F401)

pots/models.py

1-1: datetime.timedelta imported but unused

Remove unused import: datetime.timedelta

(F401)


2-2: decimal.Decimal imported but unused

Remove unused import: decimal.Decimal

(F401)


12-12: base.utils.format_date imported but unused

Remove unused import: base.utils.format_date

(F401)

indexer_app/handler.py

1-1: asyncio imported but unused

Remove unused import: asyncio

(F401)


10-10: django.core.cache.cache imported but unused

Remove unused import: django.core.cache.cache

(F401)


12-12: stellar_sdk.soroban_rpc.GetEventsResponse imported but unused

Remove unused import: stellar_sdk.soroban_rpc.GetEventsResponse

(F401)


16-16: grantpicks.models.StellarEvent imported but unused

Remove unused import: grantpicks.models.StellarEvent

(F401)


65-65: Local variable start_time is assigned to but never used

Remove assignment to unused variable start_time

(F841)


88-88: Loop control variable shard_index not used within loop body

Rename unused shard_index to _shard_index

(B007)


89-89: Local variable shard_start_time is assigned to but never used

Remove assignment to unused variable shard_start_time

(F841)


90-90: Loop control variable outcome_index not used within loop body

Rename unused outcome_index to _outcome_index

(B007)


113-113: Local variable log_processing_start is assigned to but never used

Remove assignment to unused variable log_processing_start

(F841)


114-114: Loop control variable log_index not used within loop body

Rename unused log_index to _log_index

(B007)


190-191: Use a single if statement instead of nested if statements

(SIM102)


237-237: Loop control variable index not used within loop body

Rename unused index to _index

(B007)

pots/serializers.py

5-5: django.conf.settings imported but unused

Remove unused import: django.conf.settings

(F401)


8-8: accounts.serializers.NearSocialProfileDataSerializer imported but unused

Remove unused import: accounts.serializers.NearSocialProfileDataSerializer

(F401)

indexer_app/tasks.py

6-6: pathlib.Path imported but unused

Remove unused import: pathlib.Path

(F401)


11-11: celery.signals.worker_shutdown imported but unused

Remove unused import: celery.signals.worker_shutdown

(F401)


13-13: django.db.models.DecimalField imported but unused

Remove unused import

(F401)


13-13: django.db.models.Value imported but unused

Remove unused import

(F401)


14-14: django.db.models.functions.Cast imported but unused

Remove unused import

(F401)


14-14: django.db.models.functions.NullIf imported but unused

Remove unused import

(F401)


18-18: stellar_sdk.stellar_xdr imported but unused

Remove unused import

(F401)


18-18: stellar_sdk.scval imported but unused

Remove unused import

(F401)


23-23: grantpicks.models.Round imported but unused

Remove unused import

(F401)


23-23: grantpicks.models.RoundDeposit imported but unused

Remove unused import

(F401)


23-23: grantpicks.models.ProjectContact imported but unused

Remove unused import

(F401)


25-25: pots.models.PotApplication imported but unused

Remove unused import

(F401)


25-25: pots.models.PotApplicationStatus imported but unused

Remove unused import

(F401)


440-440: f-string without any placeholders

Remove extraneous f prefix

(F541)

indexer_app/utils.py

4-4: math.log imported but unused

Remove unused import: math.log

(F401)


409-409: Local variable listObject is assigned to but never used

Remove assignment to unused variable listObject

(F841)


435-435: Local variable lst is assigned to but never used

Remove assignment to unused variable lst

(F841)


761-761: f-string without any placeholders

Remove extraneous f prefix

(F541)


897-897: Test for membership should be not in

Convert to not in

(E713)


1202-1202: Local variable reg is assigned to but never used

Remove assignment to unused variable reg

(F841)


1231-1231: Local variable provider is assigned to but never used

Remove assignment to unused variable provider

(F841)


1269-1269: Local variable stamp is assigned to but never used

Remove assignment to unused variable stamp

(F841)


1288-1288: Use is and is not for type comparisons, or isinstance() for isinstance checks

(E721)


1365-1365: Use is and is not for type comparisons, or isinstance() for isinstance checks

(E721)


1384-1384: Local variable submitted_at is assigned to but never used

Remove assignment to unused variable submitted_at

(F841)


1462-1462: Use is and is not for type comparisons, or isinstance() for isinstance checks

(E721)


1631-1634: Use ternary operator use_vault = True if chain_id == "NEAR" else event_data.get("use_vault", False) instead of if-else-block

Replace if-else-block with use_vault = True if chain_id == "NEAR" else event_data.get("use_vault", False)

(SIM108)


1726-1726: Use is and is not for type comparisons, or isinstance() for isinstance checks

(E721)


1765-1765: Use is and is not for type comparisons, or isinstance() for isinstance checks

(E721)

grantpicks/serializers.py

7-7: .models.Vote imported but unused

Remove unused import: .models.Vote

(F401)

🪛 actionlint (1.7.4)
.github/workflows/deploy-prod.yml

19-19: the runner of "actions/checkout@v3" action is too old to run on GitHub Actions. update the action's version to fix this issue

(action)


33-33: shellcheck reported issue in this script: SC2129:style:1:1: Consider using { cmd1; cmd2; } >> file instead of individual redirects

(shellcheck)


33-33: shellcheck reported issue in this script: SC2086:info:1:49: Double quote to prevent globbing and word splitting

(shellcheck)


33-33: shellcheck reported issue in this script: SC2086:info:2:56: Double quote to prevent globbing and word splitting

(shellcheck)


33-33: shellcheck reported issue in this script: SC2086:info:3:41: Double quote to prevent globbing and word splitting

(shellcheck)

🪛 LanguageTool
README.md

[duplication] ~10-~10: Possible typo: you repeated a word.
Context: ... - Pagination - API Endpoints - Account endpoints - [✅ Get all...

(ENGLISH_WORD_REPEAT_RULE)

🪛 markdownlint-cli2 (0.17.2)
README.md

6-6: Unordered list indentation
Expected: 4; Actual: 6

(MD007, ul-indent)


7-7: Unordered list indentation
Expected: 4; Actual: 6

(MD007, ul-indent)


8-8: Unordered list indentation
Expected: 4; Actual: 6

(MD007, ul-indent)


9-9: Unordered list indentation
Expected: 4; Actual: 6

(MD007, ul-indent)

🔇 Additional comments (230)
nadabot/migrations/0002_rename_id_nadabotregistry_account.py (3)

1-4: Standard Django Migration Header & Import Statement

The header and import statements are standard and correctly generated for a Django migration file.


6-11: Dependency Declaration Check

The migration correctly declares a dependency on the initial migration (0001_initial) of the nadabot app, ensuring proper order during schema changes.


12-18: Field Renaming Operation

The migration operation renames the field in the NadabotRegistry model from id to account. This change is consistent with similar renaming patterns in the tokens and pots migrations provided in the relevant snippets.

tokens/migrations/0003_token_coingecko_id.py (1)

12-23: Review of the migration adding coingecko_id field

The migration adds a new CharField coingecko_id to the Token model with some considerations:

  1. preserve_default=False means this field will be required for all new Token instances after migration. Ensure this is the intended behavior.

  2. The default value "near" seems specific for this migration. Consider if this is appropriate as a temporary default or if it should be more generic.

  3. The verbose_name "coingecko_id" is redundant since Django would use the field name automatically.

  4. Consider if unique=True should be added if each token should have a unique CoinGecko ID.

indexer_app/models.py (2)

19-19: Good update on the block_timestamp field.

Adding blank=True to the block_timestamp field is appropriate since the field already has null=True. This makes the field properly optional in both Django forms and database storage, which aligns with best practices for optional fields.


24-24: LGTM!

No substantial changes to the updated_at field; just a whitespace adjustment in the closing parenthesis.

indexer_app/admin.py (1)

8-8: Good admin interface improvement.

Adding block_timestamp to the list_display makes sense now that this field has been made optional in the model. This improves visibility of the timestamp data in the admin interface without requiring administrators to click into individual records.

indexer_app/migrations/0003_alter_blockheight_block_timestamp.py (1)

1-23: Migration looks correct and consistent.

This migration properly captures the changes made to the block_timestamp field in the BlockHeight model, adding the blank=True attribute while maintaining the existing attributes. The migration correctly depends on the previous 0002_blockheight_block_timestamp migration.

nadabot/migrations/0003_alter_group_rule_type_alter_group_rule_val_and_more.py (1)

1-52: Migration looks good apart from noted issues

This migration aligns with the project-wide standardization of field definitions by adding consistent help_text, verbose_name, and nullability options. This improves code documentation and database schema clarity, matching similar changes in other app migrations.

pots/migrations/0015_alter_potpayout_options_and_more.py (6)

105-115: Well-implemented model constraint for exclusivity.

The check constraint properly ensures that either pot or round must be present, but not both, enforcing a clean data model. This approach is preferable to application-level validation as it guarantees data integrity at the database level.


47-58: Good model evolution to support both pot and round-based payouts.

The migration effectively modifies the data model to support two different payout scenarios by making pot nullable and adding the round field. The related_name attributes are well-chosen, making it easy to navigate relationships from both sides.

Also applies to: 83-93


35-46: Well-structured on_chain_id implementation.

The on_chain_id field is properly defined with appropriate indexing and uniqueness constraints. This will facilitate efficient lookups when integrating with on-chain data. The db_index and unique constraints will ensure optimal database performance.


21-24: Good migration from unique_together to UniqueConstraint.

This change follows Django best practices by replacing the deprecated unique_together with explicit UniqueConstraint. This approach is more flexible and aligns with Django's recommendation for constraint definition.

Also applies to: 116-121


9-14: Verify dependency ordering.

The migration depends on several other migrations across different apps. While the dependencies look correct, ensure they're ordered appropriately if there are any cross-app foreign key relationships that might be affected.


122-132: Consistent constraint pattern across models.

The check constraint for potpayoutchallenge mirrors the one for potpayout, which maintains consistency in your data model design. This parallel structure will make the codebase more maintainable and easier to understand.

grantpicks/admin.py (1)

3-3: Placeholder for Admin Registrations
The comment on line 3 clearly indicates that models should be registered here. Once models are added, ensure that the necessary admin registration code is implemented.

nadabot/tests.py (1)

3-3: Test File Scaffold Confirmation
The placeholder comment on line 3 indicates where tests should be added. Once tests are implemented, ensure they provide adequate coverage for the corresponding functionality.

nadabot/views.py (1)

3-3: Views File Placeholder
The comment on line 3 serves as a reminder to create views. Please ensure that as view logic is added, the appropriate imports and error handling are implemented.

nadabot/admin.py (1)

3-3: Admin File Placeholder
The comment on line 3 effectively indicates that you'll register models here in the near future.

chains/views.py (1)

3-3: View File Scaffold Confirmation
The placeholder comment on line 3 is appropriate. As development progresses, ensure the addition of corresponding view logic and proper error handling.

nadabot/apps.py (1)

1-7: Django app configuration looks good.

The NadabotConfig class correctly sets up the Django application with appropriate default auto field and app name configuration. This follows Django's best practices for application configuration.

grantpicks/apps.py (1)

1-7: Django app configuration looks good.

The GrantpicksConfig class correctly sets up the Django application with appropriate default auto field and app name configuration. This follows Django's best practices for application configuration.

base/serializers.py (1)

4-8: Custom decimal field implementation looks good.

The TwoDecimalPlacesField correctly handles the formatting of decimal values to two decimal places, with proper handling of None values. This will ensure consistent decimal representation across serializers.

donations/migrations/0008_alter_donation_tx_hash.py (1)

1-23: Migration for transaction hash field looks good.

This migration correctly updates the tx_hash field definition to match the standardized approach used in other models throughout the application (as seen in the activities and pots migrations). The field specifications with max_length=64, null=True, help text and verbose name are consistent with the project's standards.

api/pagination.py (1)

11-24: OpenAPI parameters defined correctly

The pagination parameters are well-defined with clear descriptions, properly using the OpenAPI types and parameter locations.

chains/admin.py (1)

6-18: Admin registration looks good

The ChainAdmin class is properly configured with appropriate list_display, search_fields, list_filter, and ordering settings. This provides a good admin interface for managing Chain records.

pots/migrations/0007_potpayout_token.py (1)

15-25: Consider on_delete behavior implications

The migration correctly adds the token field as a ForeignKey with a helpful description. However, note that using CASCADE for on_delete means that if a Token is deleted, all related PotPayouts will also be deleted. Make sure this is the intended behavior, as it could lead to data loss.

If you want to preserve the PotPayout records when a Token is deleted, consider using SET_NULL instead (since the field already allows null=True):

             field=models.ForeignKey(
                 help_text="Payout token.",
                 null=True,
-                on_delete=django.db.models.deletion.CASCADE,
+                on_delete=django.db.models.deletion.SET_NULL,
                 related_name="pot_payouts",
                 to="tokens.token",
             ),
.gitignore (3)

67-67: Good addition for Redis database dumps

Adding dump.rdb to .gitignore is appropriate as these are Redis database dump files that shouldn't be committed to version control.


103-103: Good addition for Pyright configuration

Adding pyrightconfig.json to .gitignore is a good practice for Python projects using the Pyright type checker, as these configuration files are often user-specific.


137-137: Improved static directory exclusion

Using /static/ with a leading slash makes the gitignore rule more specific - it will only ignore the static directory at the root of the project, not any directory named "static" in subdirectories.

nadabot/serializers.py (1)

9-9: TODO comments should be addressed

There are TODO comments in both serializers about potentially adjusting fields for formatting datetimes or adding convenience fields. Consider creating issues to track these improvements for future implementation.

Also applies to: 15-15

tokens/migrations/0005_alter_token_coingecko_id_alter_token_icon_and_more.py (1)

12-56: Migration looks good

This migration properly modifies the fields to allow both null and blank values, which is consistent with optional fields in Django models. The help text and verbose_name attributes provide good context for each field.

nadabot/utils.py (1)

5-9: Good pattern differentiation based on environment

The conditional pattern selection based on environment is a good practice for handling different deployment scenarios.

tokens/migrations/0002_token_icon_token_name_token_symbol.py (2)

12-39: Migration looks good

This migration appropriately adds the new fields to the Token model with proper field types and attributes. The help text provides good context for each field's purpose.


16-21: Note on field constraints

This migration adds the fields with null=True but without blank=True. I noticed that migration 0005 later adds blank=True to these fields. While both approaches work, it might have been more efficient to add both constraints in a single migration.

Also applies to: 25-27, 32-37

grantpicks/migrations/0002_round_chain_rounddeposit_memo_and_more.py (4)

9-13: Dependencies are properly ordered with relevant migrations.

The migration correctly includes dependencies on previous migrations from both the accounts and chains applications, ensuring the database schema is applied in the correct order.


16-28: The chain field addition to Round model is well-structured.

The foreign key relationship to the chains.chain model is properly defined with:

  • Optional field (blank=True, null=True)
  • Descriptive help text
  • CASCADE deletion behavior
  • Appropriate related names for reverse lookups

Just be aware that using CASCADE for on_delete means if a Chain is deleted, all associated Rounds will be deleted as well. Verify this is the intended behavior.


29-35: The memo field addition to RoundDeposit is well-defined.

The text field is properly configured as optional with descriptive help text.


36-44: The approved_projects field alteration is properly structured.

The many-to-many relationship with accounts.account includes appropriate help text and a clear related_name for reverse lookups. This will allow easy bidirectional navigation between rounds and approved projects.

pots/migrations/0004_alter_potapplication_tx_hash_and_more.py (4)

12-22: Consistent tx_hash field definition for PotApplication.

The tx_hash field is properly standardized as a CharField with appropriate max_length, nullability, and descriptive text.


23-32: Consistent tx_hash field definition for PotApplicationReview.

The standardization maintains consistency with other transaction hash fields across models.


33-42: Consistent tx_hash field definition for PotPayout.

The standardization follows the same pattern, ensuring uniform handling of transaction hashes.


43-52: Consistent tx_hash field definition for PotPayoutChallengeAdminResponse.

The standardization completes the consistent field definition across all relevant models.

Consider verifying whether a maximum length of 64 characters is sufficient for all blockchain transaction hashes you'll be storing. Some blockchains may have transaction hashes with different lengths.

pyproject.toml (3)

12-12: Upgraded near-lake-framework dependency.

The version has been updated from ^0.0.7 to ^0.1.3, which appears to be a minor version upgrade. Ensure that this version is compatible with your codebase and that any breaking changes have been addressed.


20-20: Reformatted sentry-sdk dependency declaration.

Added a space after the extras key for improved readability.


23-26: Added new dependencies to enhance functionality.

Four new dependencies have been added:

  • drf-spectacular: For API documentation and schema generation
  • django-extensions: For additional Django development utilities
  • psutil: For system resource monitoring
  • stellar-sdk: For Stellar blockchain integration

These additions align with the broader project enhancements mentioned in the PR summary. Ensure that these dependencies are properly utilized in the codebase and that any configuration changes required for these packages are included in this PR.

indexer_app/management/commands/runspotindexer.py (3)

1-4: Appropriate imports for the management command.

The imports are focused and include only what's needed: the base command class and the specific Celery task to be invoked.


6-14: Well-structured command class with clear argument definition.

The command follows Django's management command conventions with:

  • Clear help text
  • Properly defined required argument with type validation
  • Descriptive help text for the argument

16-27:

✅ Verification successful

Good implementation with error handling for task invocation.

The handle method:

  • Correctly extracts the argument from options
  • Uses the Celery task's delay method for asynchronous execution
  • Provides clear success feedback
  • Properly catches and reports exceptions

Consider verifying that the related spot_index_near_events Celery task is properly defined and registered in the Celery app configuration.


🏁 Script executed:

#!/bin/bash
# Verify that the spot_index_near_events task is properly defined
grep -r "def spot_index_near_events" indexer_app/

Length of output: 113


Task Definition Verified: The asynchronous task invocation and error handling are implemented correctly.

  • The handle method correctly extracts the option, calls the Celery task asynchronously via delay(), and provides clear success/error feedback.
  • The grep check confirms that the spot_index_near_events task is defined in indexer_app/tasks.py (e.g., at the line def spot_index_near_events(start_block):).
  • Please ensure that the Celery task is properly registered in the Celery app configuration (via autodiscovery or explicit registration) so that it is correctly discovered at runtime.
pots/migrations/0016_alter_potapplication_project.py (1)

14-27:

❓ Verification inconclusive

Field name "project" points to accounts.account model

This migration alters the "project" field to reference the "accounts.account" model. While functionally correct, there's a potential semantic discrepancy between the field name "project" and the model it references. This might be confusing for future developers unless Account objects are conceptually treated as Projects in your domain model.

Otherwise, the migration is properly structured with appropriate nullability, cascade behavior, and related_name for reverse access.


🏁 Script executed:

#!/bin/bash
# Verify how Account and Project models are related in the codebase
echo "Checking Account model definition..."
rg "class Account" -A 10 --glob "*.py"

echo "Checking for Project model definitions..."
rg "class Project" -A 10 --glob "*.py"

echo "Checking PotApplication model usage of project field..."
rg "class PotApplication" -A 20 --glob "*.py"

Length of output: 20903


Clarification Needed: 'project' Field References Accounts Model

The migration alters the “project” field on the PotApplication model so that it references the accounts.Account model. Given that there is a distinct Project model (defined in grantpicks/models.py), this naming could be misleading unless it’s intended that Account objects conceptually represent projects in this context.

  • Please verify that using accounts.Account for the “project” field is intentional.
  • If Account objects are indeed meant to serve as projects, consider adding documentation or a clarifying comment to reduce future confusion.
  • Otherwise, you may need to revisit the field’s naming or target model to align with the domain model’s intent.
donations/migrations/0011_remove_donation_ft_alter_donation_token.py (1)

14-28: LGTM: Clean migration for token field refactoring

This migration properly removes the obsolete "ft" field and updates the "token" field to be a ForeignKey to "tokens.token" with appropriate cascade behavior and related_name. The migration depends on the previous migration (0010) which likely handled data migration from "ft" to "token".

This follows database schema best practices by:

  1. Creating a new field first (in previous migrations)
  2. Migrating data between fields (in migration 0010)
  3. Finally removing the old field and properly configuring the new one
indexer_app/management/commands/killspotindexer.py (4)

1-5: LGTM: Clean imports

The imports are appropriately chosen for the command's functionality, importing only what's needed from the Celery library and Django's command framework.


7-9: LGTM: Clear command help text

The command class provides a clear, descriptive help text that explains its purpose.


10-21: LGTM: Error handling for inactive workers

The command properly handles the case where no active tasks are found or the worker connection fails, providing a clear warning message.


29-33: LGTM: Clear success message

The command provides a clear success message indicating all tasks in the specified queue have been revoked.

pots/migrations/0003_alter_potapplication_options_and_more.py (1)

12-33: LGTM: Well-structured model options migration

This migration properly sets the verbose_name_plural for multiple models in the pots app, which will improve the readability and usability of the Django admin interface. The changes are consistent and follow Django's naming conventions.

Each model's plural form is appropriately formatted with proper capitalization and spacing (e.g., "Pot Applications" instead of "Potapplications").

accounts/admin.py (1)

14-14: Added near_social_profile_data to admin list display.

This change enhances the admin interface by displaying NEAR social profile data for each account. Make sure this field is properly defined in the Account model.

activities/admin.py (1)

22-22: Added autocomplete fields for better usability.

The addition of autocomplete_fields for signer and receiver is a good usability improvement for the admin interface, making it easier to select related objects.

grantpicks/migrations/0004_alter_round_on_chain_id_and_more.py (2)

13-35: Well-structured migration with appropriate constraints.

The changes to make on_chain_id an IntegerField with descriptive help text and adding unique constraints on appropriate field combinations improve data integrity. These constraints will prevent duplicate entries with the same chain/round and on_chain_id combinations.


1-1:

✅ Verification successful

Migration file has a future date.

The migration file indicates it was generated on 2024-11-05, which is in the future. While this might be intentional, typically Django migration files use the current date. Check if this future date could cause any issues with migration ordering or deployment.


🏁 Script executed:

#!/bin/bash
# Check all migration files in the grantpicks app to verify the ordering
echo "Checking migration files in grantpicks app:"
ls -la */migrations/*.py | grep grantpicks

Length of output: 5017


Migration header date verified

The migration file grantpicks/migrations/0004_alter_round_on_chain_id_and_more.py shows a header generated by Django on 2024-11-05 10:41. Given that the current timeline is March 2025, this date isn’t in the future relative to now. Since Django orders migrations by their file numbering rather than the header date, this discrepancy does not affect migration ordering or deployment. If the header’s date is confusing, you might consider regenerating the migration to reflect the current date—but technically, no changes are required.

chains/models.py (1)

38-45: Validate CheckConstraint logic with tests.

The evm_chain_id_check constraint is good. It might help to add test cases covering both True/False for evm_compat to confirm that constraints raise errors or allow records as expected.

base/api.py (1)

21-26: Serializer design looks good.

These fields capture the required statistics. The serializer is concise, typed, and well-suited for the StatsAPI.

scripts/after_install.sh (2)

53-55: Graceful approach to migrations.

Commenting out the service stop/start lines helps avoid downtime but be sure to handle migration failures or rollbacks if any issues occur after migrations are applied.

Also applies to: 59-60


73-78: Verify Gunicorn reload vs. full service restart.

Killing Gunicorn with --signal=HUP is typically safe for a graceful reload. Ensure no issues arise if the application state changes drastically due to migrations or environment updates. Continuing with a full restart might be safer for major changes.

nadabot/models.py (4)

7-10: TextChoices for ProviderStatus are well-structured.

Enums improve readability and reduce the chance of invalid status values. This is a good approach.


22-66: Add indexing or unique constraints if needed.

The NadabotRegistry references multiple fields important for identification (owner, account). If there's a chance a registry's owner could also be an admin, or if there's a global uniqueness requirement, consider covering these constraints or clarifying if multiple owners/admins can overlap.


98-220: Validate large integer usage and default values.

gas is a BigIntegerField, which is correct for large values. Consider making it strictly positive, or defining a default of 0 if negative usage is not expected. Similarly, confirm whether on_chain_id or default_weight should allow zero or negative values.


221-240: Check time unit consistency for verification dates.

Stamp.verified_at is a DateField, whereas Provider.submitted_at uses a DateTimeField in milliseconds. Ensure your logic for verifying stamps aligns with the date/time resolution your domain requires.

pots/utils.py (4)

6-9: Regex patterns look good.

The conditional assignment of BASE_PATTERN for each environment is straightforward and correct. No concerns here.


13-14: Good addition of the boundary marker in the regex pattern.

Using ^{BASE_PATTERN}$ prevents partial matches and enforces a stricter pattern match for pot factory accounts. This helps avoid accidental matches for longer strings.


19-20: Validate special characters in subaccounts.

Including - in the bracket ensures subaccounts with hyphens are accepted. This is valid, but confirm that your environment strictly allows hyphens in account names to avoid potential mismatch with real-world constraints.

Do you need hyphens for NEAR subaccounts or other chain constraints? If unsure, please verify it matches the actual naming rules of your environment.


24-26: Function is_relevant_account is straightforward.

This function’s logic looks concise and correct. No issues noted.

indexer_app/examples.txt (1)

17-18: Inconsistent timestamp field naming

There's an inconsistency between the timestamp field names:

  • POT DONATION uses donated_at: TimestampMs
  • DIRECT DONATION uses donated_at_ms: TimestampMs

Both have the same type but different names, which might cause confusion.

Consider using consistent naming for fields that represent the same concept:

// In POT DONATION
-pub donated_at: TimestampMs,
+pub donated_at_ms: TimestampMs,

// Or in DIRECT DONATION
-pub donated_at_ms: TimestampMs,
+pub donated_at: TimestampMs,

Also applies to: 48-49

scripts/after_install_testnet.sh (3)

37-37: Good addition of dependency locking

Adding poetry lock --no-update ensures dependency consistency without updating versions, which helps prevent unexpected changes during deployment.


54-56: Improved deployment strategy to minimize downtime

Commenting out the service stopping commands during migrations is a good approach to reduce unnecessary downtime.

Consider updating the comment to be more definitive since this appears to be a permanent improvement rather than a temporary change:

-    # COMMENTING OUT FOR NOW AS I BELIEVE STOPPING SERVICES IS UNNECESSARY
+    # We don't stop services during migrations to avoid unnecessary downtime

64-74: Improved service management with graceful reloading

These changes improve the deployment process by:

  1. Always running collectstatic regardless of migration status
  2. Using HUP signal for graceful Gunicorn reload instead of a full restart
  3. Only restarting the necessary Celery services

This approach minimizes downtime and follows best practices for zero-downtime deployments.

scripts/after_install_dev.sh (3)

37-37: Good addition of dependency locking

Adding poetry lock --no-update ensures dependency consistency without updating versions, which helps prevent unexpected changes during deployment.


54-56: Improved deployment strategy to minimize downtime

Commenting out the service stopping commands during migrations helps reduce unnecessary downtime, consistent with the changes in the testnet script.

Consider updating the comment to be more definitive since this appears to be a permanent improvement rather than a temporary change:

-    # COMMENTING OUT FOR NOW AS I BELIEVE STOPPING SERVICES CREATES UNNECESSARY DOWNTIME
+    # We don't stop services during migrations to avoid unnecessary downtime

64-74: Improved service management with graceful reloading

These changes improve the deployment process by:

  1. Always running collectstatic regardless of migration status
  2. Using HUP signal for graceful Gunicorn reload instead of a full restart
  3. Only restarting the necessary Celery services

This approach minimizes downtime and maintains consistency with the testnet environment.

tokens/admin.py (3)

8-17: Enhanced admin display with more token metadata

The updated list_display shows more relevant token information (account, name, symbol, coingecko_id, icon) which improves usability in the admin interface. The change from "id" to "account" in search_fields suggests a schema change where tokens are now referenced by account.


38-38: Updated search field to align with Token model changes

The change from token__id to token__account in search_fields is consistent with the changes in TokenAdmin, suggesting a schema change in how tokens are referenced.


25-32:

❓ Verification inconclusive

Verify the intentional removal of permission restrictions

Commenting out the permission methods (has_add_permission, has_change_permission, has_delete_permission) changes the behavior from explicitly denying these permissions to using Django's default permission system, which could unintentionally grant permissions that were previously restricted.

If this change is intentional, consider adding a comment explaining why the restrictions were removed. If not, these methods should be restored:

def has_add_permission(self, request):
    return False

def has_change_permission(self, request, obj=None):
    return False

def has_delete_permission(self, request, obj=None):
    return False

Let's check if any documentation or changes in the model suggest why these permissions were modified:

Also applies to: 41-48


🏁 Script executed:

#!/bin/bash
# Search for changes to the Token model that might explain the permission changes
rg -A 5 "class Token" --type py

Length of output: 1563


Attention: Confirm Permission Restriction Changes

It looks like the explicit permission restrictions in tokens/admin.py (lines 25–32 and 41–48) have been removed—switching from a definitive "deny" to relying on Django’s default behavior. Our search through the code (especially in tokens/models.py and related files) didn’t reveal any accompanying documentation or changes in the Token model that would clarify this decision.

  • Action Required:

    • If this change is intentional, please add an inline comment explaining the rationale for dropping the explicit permission denials.
    • If the removal was not intended, restore the methods as shown below:
    def has_add_permission(self, request):
        return False
    
    def has_change_permission(self, request, obj=None):
        return False
    
    def has_delete_permission(self, request, obj=None):
        return False
README.md (12)

14-15: Endpoints look good.

New paginated endpoints for donations received and sent improve the API’s utility by offering more granular insights into account-level donations.


17-18: Endpoints look good.

Additional paginated endpoints for pot applications and list-registrations extend functionality for account status lookups.


23-23: Endpoint documentation enhancement.

The new random registration endpoint is clearly documented and improves discoverability of the feature.


25-25: Endpoint looks good.

The donate contract config endpoint provides clarity for contract details.


112-112: Updated pagination looks good.

Switching to page and page_size aligns well with standard paginated API designs, and a default of 30 is reasonable.


139-140: Endpoint addition is valid.

The GET /accounts/{ACCOUNT_ID}/pot_applications (paginated) endpoint is well documented.


141-142: Status-based filtering is helpful.

Documenting available query params for pot applications fosters clarity in use cases.


165-166: Filter by category is clear.

Providing straightforward query param instructions for project categories is valuable for custom searches.


167-168: Endpoint addition is valid.

Documenting the random registration retrieval for a given list is a great addition.


169-170: Filter usage is clearly documented.

Clarifies how to retrieve specific statuses.


171-172: Clear section introduction.

This dedicated section for the donate contract config endpoint aids discoverability.


173-173: Endpoint addition is valid.

The documentation for GET /donate_contract_config is clear and consistent with the rest of the API reference.

base/celery.py (3)

38-52: New scheduled tasks appear consistent.

The 5-minute and 1-minute schedule intervals look appropriate for quickly updating pot statistics and processing Stellar events. Confirm concurrency control to avoid excessive load on your APIs if these tasks are CPU or network-intensive.


58-60: Expanded task routing is correct.

All newly introduced tasks are properly assigned to the beat_tasks queue, maintaining consistent routing architecture.


63-64: Queue naming is clear.

Defining SPOT_INDEXER_QUEUE_NAME aligns with naming conventions and clarifies usage.

tokens/models.py (8)

5-6: Dependency imports look fine.

Libraries (requests and django.conf.settings) are required for fetching data and reading configuration.


12-13: Logger and date utility usage is appropriate.

These imports facilitate structured logging and date formatting.


16-16: Model declaration is consistent.

Token model is well-structured. Ensure migrations are run after referencing Account as a primary key.


23-29: New name field integrates well.

No issues, though watch for the overshadowing import from os import name.

🧰 Tools
🪛 Ruff (0.8.2)

23-23: Redefinition of unused name from line 3

(F811)


30-36: Symbol field is valid.

Using a CharField for token symbols is standard.


48-54: CoinGecko field is logically named.

coingecko_id is clearly descriptive; no immediate concerns.


64-105: Price-fetching logic is comprehensive.

The fallback to existing records plus the external request to CoinGecko looks well-structured. Check for potential rate-limiting issues and appropriate error handling if repeated calls occur in a short timeframe.


17-22:

❓ Verification inconclusive

Primary key usage with account.

Replacing the default ID with a one-to-one relationship is a major change. Double-check related logic and references in your codebase to ensure it doesn’t break foreign keys or references to the old primary key.

You can run a quick search to confirm usage:


🏁 Script executed:

#!/bin/bash
# Searching code references to old Token id usage
rg -A 5 "Token.objects.get\(id="

Length of output: 881


Primary Key Change – Verify Affected Migration Queries

The change replaces the default auto-generated id with a one-to-one relationship to Account as the primary key. A search for Token.objects.get(id= revealed usage only in the migration scripts:

  • pots/migrations/0008_migrate_ft_to_token.py
  • donations/migrations/0010_migrate_ft_to_token.py

Please review these migration files to ensure they correctly handle the new primary key. In particular, verify whether the queries should reference the new key (e.g., using the related account field) or if they’re safe as-is because they operate on frozen historical models. Adjust them if needed so that foreign key relationships or other dependencies aren’t inadvertently broken.

nadabot/migrations/0001_initial.py (1)

371-420: Cleanly implemented unique constraint on (registry, account).

Declaring unique_together enforces a robust validation preventing duplicate blacklist entries. This is a well-structured approach. The rest of the BlackList model fields also seem to align with your intended logic.

pots/migrations/0014_alter_potapplication_unique_together_and_more.py (1)

99-107: Great use of check constraints for conditional foreign keys.

This pot_or_round constraint ensures either pot or round is populated, but not both. This enforces data integrity at the database level and simplifies validation logic in the application.

pots/admin.py (13)

25-29: Looks good for conditional queryset assignment.
The conditional logic correctly ensures the instance is present before assigning querysets to admins and whitelisted_deployers.


41-42: Nice enhancement to list and search fields.
Updating list_display and search_fields to use more descriptive fields improves admin usability.


74-74: Consistent approach for conditional queryset assignment.
Ensuring instance.pk exists before assigning the admins queryset mirrors the logic in PotFactoryForm.


82-83: Good expansion of list_display and search_fields.
Displaying name and searching by deployer__id provide quick filtering in the admin.


105-114: Form customization looks good.
Customizing the applicant widget’s placeholder is a helpful UX enhancement.


119-119: Explicit form usage is clear.
Declaring form = PotApplicationAdminForm makes the admin’s behavior easier to follow.


121-123: Appropriate search_fields and autocomplete_fields.
Searching on pot__account and applicant__id aligns well with typical usage patterns. Autocomplete fields further enhance admin usability.


159-159: New custom display method.
application_applicant_id in list_display adds clarity. Consider handling edge cases where obj.application or obj.application.applicant might be missing or invalid.


168-171: Improved usability with autocomplete_fields.
Allowing autocomplete for application and reviewer helps in data-heavy scenarios.


173-176: Potential None checks for application or applicant.
Ensure application and application.applicant are never None before accessing .id, or handle it gracefully.


190-191: Updated search fields for payouts.
Filtering by pot__account and recipient__id is straightforward and improves admin discoverability.


206-207: Challenger-based search is a good addition.
Ensures quick lookups for potential disputes or challenges.


222-223: Verify 'challenger__id' in search_fields.
PotPayoutChallengeAdminResponse may or may not have direct challenger relationship fields. Confirm this reference is valid to avoid runtime errors.

base/settings.py (8)

35-40: Expanded ALLOWED_HOSTS is suitable for multi-domain setups.
Double-check that all domains pointing to this app are included.


63-88: Environment-based addresses and TLA logic look coherent.
Defining environment-based contract IDs and addresses per environment variable is a good practice.


110-111: Addition of new apps to INSTALLED_APPS is consistent.
Enabling drf_spectacular, django_extensions, nadabot, chains, and grantpicks should integrate well with the rest of your codebase.

Also applies to: 123-125


131-131: Pagination and throttle updates.
Changing to PageNumberPagination and increasing the anon throttle to 500/minute can help moderate traffic. Also, setting DEFAULT_SCHEMA_CLASS to drf_spectacular is a good step for API docs.

Also applies to: 139-139, 141-141


144-149: SPECTACULAR_SETTINGS usage
Providing a unique TITLE such as "PotLock Indexer API dev" or "PotLock Indexer API testnet" helps distinguish environments in docs.


184-199: Environment-based CORS origins.
Specifying allowed origins for each environment helps keep your deployment secure.


235-238: CORS_ALLOWED_ORIGIN_REGEXES ensures flexible domain matching.
This allows more dynamic subdomains (e.g. potlock-next-[branch].vercel.app). Good approach for ephemeral environments.


390-390: Selective Sentry profiling based on environment.
Lowering the profiling sample rate in production is a prudent choice, balancing performance metrics vs. overhead.

api/urls.py (7)

2-6: Imports for API schema generation look good.
These additions from drf_spectacular are correct and align well with best practices for auto-generating API documentation.


23-28: Granular imports for lists, pots, and grantpicks.
It’s good to see modular slicing of these APIs. This improves clarity and maintainability.

Also applies to: 29-39


42-51: Schema endpoints are properly defined.
Exposing swagger-ui and redoc via drf_spectacular is standard. Great approach for documenting the API.


95-97: New endpoints for rounds, donation config, and donors.
These route definitions are clear. Ensure you add relevant unit tests for each endpoint to verify that queries, error handling, permissions, etc., function as intended.

Also applies to: 101-103, 106-106


121-143: Comprehensive pots endpoints introduced.
It’s great to see separate endpoints for applications, donations, sponsors, and payouts. Confirm that all these newly introduced endpoints handle permission checks (if any) and pagination for performance.


160-169: MPDAO endpoints.
These voter endpoints look straightforward. Make sure to verify access control if needed, since this might reveal confidential or privileged voter data.


8-19:

❓ Verification inconclusive

Confirm usage of imported classes.
All the account-related API classes are being imported together. Ensure that each view is referenced properly within the URL patterns to avoid unused imports.

Please run:

to verify alignment between these imported classes and their actual definitions.


🏁 Script executed:

#!/bin/bash
rg -A 5 'class Account' accounts/api.py

Length of output: 1926


Review Verification: Confirm API Imports Are Used

The grep results confirm that the account-related API classes—namely, AccountsListAPI, AccountDetailAPI, AccountActivePotsAPI, AccountPotApplicationsAPI, AccountDonationsReceivedAPI, AccountDonationsSentAPI, AccountPayoutsReceivedAPI, AccountListRegistrationsAPI, and AccountUpvotedListsAPI—all have corresponding definitions in accounts/api.py.

One note: the grep command did not reveal any definition for DonorsAPI because its class name doesn’t include the “Account” prefix. Please verify that DonorsAPI is:

  • Defined (possibly in accounts/api.py or another appropriate module), and
  • Properly referenced in the URL patterns within the API configuration.

If DonorsAPI is indeed defined and used, then all imports are justified. Otherwise, consider removing or correcting the import.

pots/migrations/0012_alter_pot_base_currency_alter_pot_chef_and_more.py (5)

133-162: PotApplication fields: text and timestamps.
Allowing blank=True, null=True for message and updated_at is appropriate for optional data. Confirm related forms or serializers handle these optional fields gracefully.


164-183: PotApplicationReview fields: notes and tx_hash.
The updated help_text clarifies usage. Good improvement for documentation.


185-193: PotFactory metadata changes.
Storing source metadata as a JSONField with blank=True, null=True is fine. Ensure upstream usage checks for empty/None.


195-226: PotPayout allowing null for amounts and payout date.
Indexing paid_at (line 211) is a good call for queries on payout date. Validate if any constraints require paid_at to always have a value after a successful payout.


228-260: PotPayoutChallengeAdminResponse fields updated.
The modifications look consistent with the rest of the changes, maintaining uniform “tx_hash” usage.

pots/migrations/0013_potpayoutchallenge_tx_hash.py (1)

13-22: New field “tx_hash” in PotPayoutChallenge.
This consolidates transaction-tracking capabilities across related pot models. Double-check that any references to the challenge’s transaction hash handle null=True and blanks correctly.

accounts/migrations/0003_alter_account_options.py (1)

1-18: Clear migration to improve database query performance on Account model

This migration adds default ordering by id to the Account model, which can help with query consistency and potentially optimize database queries.

pots/migrations/0010_alter_potpayout_paid_at.py (1)

13-22: Well-structured field modification with performance optimization

Adding a database index (db_index=True) to the paid_at field is a good optimization for a datetime field that will likely be used in filters and sorting operations. The nullable property makes sense for payouts that haven't been processed yet.

lists/migrations/0005_alter_list_id.py (1)

13-22: Clear separation between database and on-chain identifiers

Good job clearly documenting the distinction between database IDs and blockchain IDs. The help text provides important context that will help prevent confusion for developers working with this model in the future.

indexer_app/logging.py (1)

3-4:

✅ Verification successful

New dependency added: psutil

Make sure psutil is added to the project's requirements file to ensure it's installed in all environments.


🏁 Script executed:

#!/bin/bash
# Check if psutil is listed in requirements files
grep -r "psutil" --include="*requirements*.txt" .

Length of output: 49


Attention: psutil Dependency Missing

It appears that the new dependency, psutil, is not currently listed in any recognized dependency management files (e.g., requirements*.txt, setup.py, or pyproject.toml). Please add psutil to one of these files so that it gets installed in every environment where the project runs.

  • File: indexer_app/logging.py, Lines 3–4
  • Action: Update the project's dependency file with the appropriate psutil entry (e.g., add psutil==<version>).

🏁 Script executed:

#!/bin/bash
# Check if psutil is listed in recognized dependency management files
echo "Searching in requirements*.txt files:"
grep -r "psutil" --include="requirements*.txt" .
echo "Searching in setup.py:"
grep -r "psutil" --include="setup.py" .
echo "Searching in pyproject.toml:"
grep -r "psutil" --include="pyproject.toml" .

Length of output: 494


Dependency Verified: psutil is properly tracked

The new dependency psutil has been added to the pyproject.toml file (with the version constraint ^6.0.0), ensuring it's installed across all environments. No further changes are required regarding this dependency.

lists/migrations/0003_alter_listupvote_options.py (1)

1-17: Migration looks good

This migration properly sets the verbose_name_plural for the ListUpvote model to "List Upvotes", which follows Django's best practices for improving admin interface readability.

lists/migrations/0004_list_on_chain_id.py (1)

13-23:

❓ Verification inconclusive

Potential uniqueness constraint issue

The migration adds a new field on_chain_id with a uniqueness constraint and default value of 1, but sets preserve_default=False. If there are multiple existing List records in the database, they could all receive the default value 1, violating the uniqueness constraint during migration.


🏁 Script executed:

#!/bin/bash
# Check if there are multiple existing List records
echo "Checking if there are multiple List records that might cause uniqueness constraint issues..."
python -c "
import os, django
os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'your_project.settings')
django.setup()
from lists.models import List
count = List.objects.count()
print(f'Found {count} List records')
if count > 1:
    print('WARNING: Migration might fail due to uniqueness constraint violation')
else:
    print('No uniqueness constraint issue expected')
"

Length of output: 717


Manual Verification Required: Potential Uniqueness Constraint Issue in Migration

The migration adds the on_chain_id field with unique=True and a default value of 1. This setup risks a uniqueness violation if multiple existing List records are present. Note that the initial verification script failed due to Django not being available in the execution environment, so please verify manually in an environment where Django is installed.

  • Location: lists/migrations/0004_list_on_chain_id.py (lines 13-23)
  • Concern: Adding on_chain_id with a default of 1 may assign the same value to all existing records, causing a conflict under the uniqueness constraint.
  • Next Steps:
    • Ensure that, before applying the migration, unique values are generated for each record or modify the migration strategy (e.g., use a two-step process with a data migration and then add the constraint).
    • Re-run the check in an environment with Django properly configured to confirm the record count.
        migrations.AddField(
            model_name="list",
            name="on_chain_id",
            field=models.IntegerField(
                default=1,
                help_text="List ID in contract",
                unique=True,
                verbose_name="contract list ID",
            ),
            preserve_default=False,
        ),
activities/migrations/0004_alter_activity_tx_hash.py (1)

13-22:

❓ Verification inconclusive

Verify transaction hash length is sufficient

The migration sets max_length=64 for the transaction hash field. Ensure this is sufficient for all blockchain transaction hashes you might store. For reference, Ethereum transaction hashes are 66 characters (with '0x' prefix).


🏁 Script executed:

#!/bin/bash
# Check if there are any transaction hashes longer than the defined limit
echo "Checking if any existing transaction hashes exceed the new length limit..."
python -c "
import os, django
os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'your_project.settings')
django.setup()
from activities.models import Activity
long_hashes = Activity.objects.filter(tx_hash__isnull=False).extra(where=['LENGTH(tx_hash) > 64']).count()
print(f'Found {long_hashes} tx_hash values exceeding 64 characters')
if long_hashes > 0:
    print('WARNING: Some existing transaction hashes may be truncated')
else:
    print('No length issues expected')
"

Length of output: 788


Transaction Hash Length Verification – Manual Confirmation Required

The migration currently sets max_length=64 for the tx_hash field. However, note that Ethereum transaction hashes (when stored with the 0x prefix) are 66 characters long. The automated check script failed with a "ModuleNotFoundError: No module named 'django'", so we couldn’t verify existing data automatically.

  • Action Items:
    • Confirm whether your application stores Ethereum hashes with the 0x prefix.
    • If storing the full hash (66 characters) is necessary, adjust max_length accordingly.
    • Manually verify the length of transaction hash values in your database to ensure no truncation occurs.
lists/migrations/0006_alter_listregistration_unique_together.py (2)

13-17: Good data integrity implementation with unique_together constraint.

The migration correctly implements a uniqueness constraint to prevent duplicate registrations for the same list. This ensures that a registrant can only register once for a specific list, maintaining data integrity in your application.


8-11: Dependencies look properly established.

The migration properly depends on both the previous lists migration and the accounts migration that adds the NEAR social profile data field, ensuring correct execution order.

accounts/migrations/0002_account_near_social_profile_data.py (2)

13-21: JSONField is well-configured for storing NEAR social profile data.

The field implementation is solid with appropriate help text and null parameter. This will effectively store structured NEAR social data. The JSONField is a good choice for storing complex, potentially varying structured data like social profiles.


8-10: Dependencies correctly specified.

The migration properly depends on the initial migration of the accounts app, ensuring proper execution order.

tokens/migrations/0004_alter_token_coingecko_id.py (2)

13-22: Appropriate field configuration for CoinGecko ID.

The CharField with max_length=255 is appropriate for storing token identifiers. The null=True parameter allows for tokens without CoinGecko IDs, and the help text and verbose name provide good documentation.


8-10: Dependencies correctly specified.

The migration properly depends on the previous migration that added the coingecko_id field, ensuring proper execution order.

pots/migrations/0011_rename_id_pot_account_rename_id_potfactory_account.py (2)

8-10: Dependencies correctly specified.

The migration properly depends on the previous pot migration, ensuring proper execution order.


13-17:

✅ Verification successful

RenameField operations need verification for potential impact.

Renaming fields that might be primary keys (from 'id' to 'account') could have significant implications for your data models and related relationships. While the migration syntax is correct, ensure that:

  1. Related model foreign keys are updated accordingly
  2. Any code referencing these fields is updated
  3. Database constraints and indexes are properly maintained

This change suggests a shift from using simple IDs to using Account references as identifiers.

Also applies to: 18-22


🏁 Script executed:

#!/bin/bash
# Check for any foreign key references to the renamed fields
echo "Checking for foreign key references to Pot.id and PotFactory.id"
rg -A 2 -B 2 "ForeignKey\(.*Pot," --type py
rg -A 2 -B 2 "ForeignKey\(.*PotFactory," --type py

# Check models.py to see how these fields are defined after the rename
echo "Checking how account fields are defined in models.py"
cat pots/models.py | grep -A 10 "class Pot" | grep "account"
cat pots/models.py | grep -A 10 "class PotFactory" | grep "account"

Length of output: 723


Verified field renaming: No additional issues found.
The foreign key references weren't detected in the codebase, and the model definitions in pots/models.py show that the account field is now correctly implemented as a OneToOneField. All related references to the renaming from "id" to "account" appear to be updated, and there is no evidence of constraint or index issues arising from this change.

  • Confirmed that foreign key references for both Pot and PotFactory are not present.
  • Verified that the account field is correctly defined in both models.
pots/migrations/0009_remove_potpayout_ft_alter_potpayout_token.py (1)

1-29: Well-structured migration that properly finalizes the token relationship refactoring.

This migration appropriately completes the transition from the "ft" field to a proper foreign key relationship with the "token" model. The CASCADE deletion policy and related_name are correctly configured.

accounts/migrations/0004_account_chain.py (2)

9-12:

❓ Verification inconclusive

Potential circular migration dependency detected.

The chains.0002_add_near_chain migration (shown in relevant snippets) depends on this migration, while this migration depends on chains.0001_initial. This creates a circular dependency in the migration graph which could cause problems when applying migrations.


🏁 Script executed:

#!/bin/bash
# Check migration dependencies to verify the circular reference
echo "Examining migrations for circular dependencies..."
grep -r "dependencies.*chains.0002_add_near_chain" --include="*.py" ./accounts/migrations/
grep -r "dependencies.*accounts.0004_account_chain" --include="*.py" ./chains/migrations/

Length of output: 293


Action Required: Verify Migration Dependencies & Attribute Redundancy

The migration file accounts/migrations/0004_account_chain.py sets its dependencies to ("accounts", "0003_alter_account_options") and ("chains", "0001_initial"). Although it was originally flagged for a potential circular dependency (with the expectation that chains.0002_add_near_chain might depend on this migration), our automated grep checks did not reveal any direct references linking chains.0002_add_near_chain back to this migration.

Additionally, the migration defines related_name and related_query_name attributes, but later modifications in accounts/migrations/0005_alter_account_chain.py seem to duplicate this configuration. This raises a concern about redundancy which might lead to confusion about where these attributes are intended to be set.

Please manually verify the following:

  • Confirm that there is no circular dependency between accounts/migrations/0004_account_chain.py and chains/migrations/0002_add_near_chain.py.
  • Review whether redefining related_name and related_query_name in a subsequent migration is intentional or if it leads to unnecessary redundancy.

18-26:

❓ Verification inconclusive

Review redundancy in field attributes across migrations.

This migration defines the chain field with related_name="accounts" and related_query_name="account", but the related code snippets show that accounts.0005_alter_account_chain.py might redefine these same attributes. This could indicate redundancy in the migration sequence.


🏁 Script executed:

#!/bin/bash
# Compare field definitions between migrations
echo "Comparing chain field definitions between migrations..."
cat ./accounts/migrations/0005_alter_account_chain.py

Length of output: 886


Clarify and Confirm Field Attribute Changes Across Migrations

After comparing the field definitions in migrations 0004 and 0005, we observe that:

  • Migration 0004 sets the chain field with blank=True and null=True alongside attributes like help_text, on_delete, related_name="accounts", and related_query_name="account".
  • Migration 0005 redefines the field via an AlterField, reproducing the relationship attributes (related_name and related_query_name) but omitting blank and null.

This repetition of relationship attributes is typical in Django’s migration history, as each migration reflects a snapshot in time. However, the removal of blank=True and null=True in 0005 suggests an intentional change in the field’s nullability and form validation behavior.

Please verify that the removal of blank and null in the later migration is intended and that the consistent reproduction of related_name and related_query_name is deliberate. If so, the current state is correct; if not, consider adjusting the migration sequence.

lists/admin.py (1)

8-19: Improved admin interface with additional useful fields.

The addition of these fields to list_display enhances the admin interface by providing more context about each List instance, especially the on-chain identifier which helps correlate database records with blockchain data.

activities/models.py (1)

48-53: Field correctly made optional.

Making action_result optional by adding blank=True is appropriate if not all activities will have results, particularly for activities that are in progress or don't produce results.

accounts/migrations/0005_alter_account_chain.py (1)

18-24: Review CASCADE deletion behavior

Using CASCADE will delete all accounts if their associated chain is deleted. Consider if this is the intended behavior or if PROTECT might be more appropriate to prevent accidental data loss.

field=models.ForeignKey(
    help_text="Blockchain this account is located on.",
-    on_delete=django.db.models.deletion.CASCADE,
+    on_delete=django.db.models.deletion.PROTECT,
    related_name="accounts",
    related_query_name="account",
    to="chains.chain",
),
accounts/migrations/0006_alter_account_near_social_profile_data.py (1)

16-21: LGTM!

The JSONField is correctly configured with appropriate options for optional data storage and good documentation.

chains/migrations/0001_initial.py (1)

49-59: Well-designed constraint for EVM compatibility check

The check constraint ensures data integrity by enforcing that if evm_chain_id is not null, evm_compat must be true, and if evm_chain_id is null, evm_compat must be false. This is a good practice for maintaining data consistency.

grantpicks/migrations/0006_alter_round_compliance_period_ms_and_more.py (1)

12-32: Good improvement to field types and documentation

The migration properly updates the field types from a regular integer field to a PositiveBigIntegerField for the time periods, which is appropriate for large millisecond values. The added help text and verbose names improve the admin interface usability and documentation. This change will prevent potential integer overflow issues for large time periods.

activities/migrations/0005_alter_activity_action_result_alter_activity_tx_hash.py (2)

13-22: Fields updated with improved metadata and flexibility

This change converts the action_result field to a JSONField with null/blank support, improving flexibility for storing structured activity results while maintaining good documentation through help text and verbose name.


23-33: Transaction hash field standardization aligns with other models

The tx_hash modification follows the same pattern applied to similar fields in other models (like donations and pot applications), ensuring consistent handling of transaction hashes across the application with appropriate length and null handling.

grantpicks/migrations/0003_alter_votepair_project.py (2)

18-22:

❓ Verification inconclusive

CASCADE deletion might cause data loss

Using CASCADE for the on_delete parameter means that if an account is deleted, all related vote pairs will also be deleted. Verify this is the intended behavior rather than using a softer approach like SET_NULL (which would require making the field nullable).


🌐 Web query:

What are the best practices for Django's on_delete parameter in ForeignKey relationships?

💡 Result:

Django's on_delete parameter in ForeignKey relationships is crucial for maintaining data integrity and defining how related objects behave when their referenced object is deleted. Here are the best practices based on common use cases and Django documentation:

Key on_delete Options and Their Use Cases

  1. CASCADE
    Deletes all related objects when the referenced object is removed.
    Example: A user's blog posts should be deleted if the user account is removed.

    author = models.ForeignKey(User, on_delete=models.CASCADE)  

    Use when: Child objects have no purpose without the parent[1][6].

  2. PROTECT
    Prevents deletion of the referenced object if related objects exist. Raises ProtectedError.
    Example: Preventing a category from being deleted while associated products exist.

    category = models.ForeignKey(Category, on_delete=models.PROTECT)  

    Use when: Data integrity is critical, and deletion should only occur after manual cleanup[1][3][6].

  3. RESTRICT
    Similar to PROTECT but allows deletion if the referenced object is part of a cascading delete. Raises RestrictedError.

    order = models.ForeignKey(Order, on_delete=models.RESTRICT)  

    Use when: You need nuanced control over deletion in complex relational setups[1][6].

  4. SET_NULL
    Sets the foreign key to NULL (requires null=True).
    Example: Preserving comments after a user deletes their account.

    user = models.ForeignKey(User, on_delete=models.SET_NULL, null=True)  

    Use when: Retaining orphaned data is acceptable[1][3][6].

  5. SET_DEFAULT
    Assigns the foreign key to a predefined default value.

    team = models.ForeignKey(Team, on_delete=models.SET_DEFAULT, default=1)  

    Use when: A sensible default exists (e.g., a "default" team for reassignment)[1][3][6].

  6. SET()
    Dynamically sets the foreign key via a function or value.

    def get_default_owner():  
        return User.objects.get(username="admin")  
    
    owner = models.ForeignKey(User, on_delete=models.SET(get_default_owner))  

    Use when: Custom logic is needed for reassignment[1][3][6].

  7. DO_NOTHING
    Leaves the foreign key untouched, risking data corruption. Rarely recommended.

    log_entry = models.ForeignKey(Log, on_delete=models.DO_NOTHING)  

    Use only when: External systems handle integrity (e.g., database triggers)[1][6].

Best Practices Summary

  • Prioritize data integrity: Use CASCADE for dependent data and PROTECT/RESTRICT for essential references.
  • Avoid DO_NOTHING unless manually handling integrity through external mechanisms[1][3][6].
  • Index foreign keys: Improve query performance with db_index=True[4].
  • Use SET_NULL cautiously: Ensure null=True to avoid migration errors[3][6].
  • Test migrations: Always verify deletion behavior in staging environments[6].

Performance Considerations

For large-scale applications:

  • Use db_constraint=False to avoid database-level constraints and reduce locking[4].
  • Prefer CASCADE over application-level deletion for efficiency in batched operations[4].

By aligning on_delete behavior with your application’s data lifecycle, you balance integrity, usability, and performance effectively.

Citations:


Action Item: Verify CASCADE Behavior for Vote Pairs

The migration currently sets the foreign key with on_delete=django.db.models.deletion.CASCADE, which means deleting an account will also remove all associated vote pairs. Django best practices support this approach when the child objects (vote pairs) are completely dependent on the parent (account).

  • Confirm the Business Logic: Ensure that it’s acceptable for vote pair records to be deleted with the account. If vote pairs need to be retained for audit or historical purposes, consider alternatives such as:

    • SET_NULL: (requires null=True) to mark them as orphaned.
    • PROTECT or RESTRICT: to prevent deletion of the account when related vote pairs exist.
  • Location: grantpicks/migrations/0003_alter_votepair_project.py, lines 18-22.

Please review and confirm that the cascade deletion is intentional. If changes are needed, update the on_delete parameter accordingly.


15-23:

❓ Verification inconclusive

Verify data handling for reinstated project field

According to the summary, this field was previously removed in favor of projects (ManyToManyField) and voted_project. Now it's being reinstated, which might cause data inconsistency issues if existing records don't have a valid project to reference.


🏁 Script executed:

#!/bin/bash
# Check the model history to understand field changes
echo "Checking previous changes to VotePair model:"
rg "VotePair|votepair" --include "*.py" | grep -E "class|field|ForeignKey"

# Look for any existing data migrations that might have handled this field
echo "Looking for related data migrations:"
rg "VotePair|votepair" --include "00*.py" | grep -A 5 -B 5 "RunPython"

Length of output: 1213


Action Required: Verify Data Migration for Reinstated "project" Field

The migration reintroduces the project ForeignKey on the VotePair model, which replaces a previous removal in favor of projects (ManyToManyField) and voted_project. However, we haven’t found clear evidence—via automated searches—that any data migration exists to safely transition legacy records to include a valid reference in the reinstated field.

  • Confirm manually that data migration steps (or fallback logic) are in place to handle legacy VotePair records that might otherwise reference a null or inconsistent project value.
  • If such handling is missing, please ensure that appropriate migration or data fixing routines are added to avoid data inconsistency issues during deployment.

To further investigate, consider running the following revised shell script (without the problematic --include flag) to search for related changes:

Please verify the outputs manually to ensure that all legacy data is correctly transitioned.


🏁 Script executed:

#!/bin/bash
echo "Checking VotePair model references:"
rg --glob "*.py" "class VotePair"

echo "Looking for migrations using RunPython (potential data migration handling):"
rg --glob "*.py" "RunPython" migrations

Length of output: 622


Attention: Confirm Data Migration for Reinstated "project" Field

The migration in grantpicks/migrations/0003_alter_votepair_project.py reintroduces the ForeignKey field project on the VotePair model, which was previously removed in favor of a ManyToManyField (projects) and a separate voted_project field. Our automated searches haven’t uncovered any accompanying data migration (for instance, a RunPython data fix) that handles legacy records for this reintroduced field.

  • Action Required: Manually verify that legacy VotePair records are correctly handled by a data migration or a fallback process ensuring that every record has a valid reference for the project field.
  • Recommendation: If such data handling is missing, please introduce migration logic to prevent potential data inconsistencies during deployment.
donations/admin.py (2)

15-15: Ensure consistency in naming convention.

token_address is a concise and meaningful replacement for the previous ft_address. The updated attribute is more descriptive of the underlying object, which helps maintain clarity in the code.


33-33: Great optimization for preventing N+1 queries.

Prefetching these related objects significantly reduces database calls.

donations/api.py (2)

53-53: Efficient use of caching.

Using cache_page(60 * 5) is a good approach to alleviate load, especially for repeated requests of the same data.


67-70: Structured error response is consistent.

Logging the error using logger.error and returning the response text with the matching status code is a clean approach. Great job ensuring transparency for debugging.

accounts/models.py (5)

21-29: Chain relationship is clearly defined.

Requiring an explicit chain reference ensures the codebase accurately represents the blockchain environment for each account.


57-62: Validate data integrity on JSONField usage.

Storing social profile data in near_social_profile_data fosters flexibility. However, consider input sanitization to prevent any potential injection or malformed data issues.


64-66: Default ordering by 'id' provides consistent listing.

Explicitly specifying ordering is a good practice, especially in admin or user-facing lists.


67-70: Async approach for profile data is beneficial.

Wrapping the sync method with sync_to_async helps avoid blocking the event loop when dealing with large or slow external calls.


139-147: Avoid potential infinite loops during creation.

By not immediately saving after fetching social profile data in save(), you effectively circumvent circular calls. Good logic and clarifying comment.

donations/models.py (2)

50-51: Approved the blank=True additions.

Allowing these fields to be optional appears consistent with your requirement to make them user-optional or system-populated. This change looks appropriate.

Also applies to: 65-66, 80-81, 88-89, 102-103, 116-117, 130-133, 135-156, 158-179, 183-186


202-203: Providing a to_dict method is fine, but be aware of sensitive data.

model_to_dict serializes all fields by default. Confirm that no sensitive fields slip through inadvertently.

Would you like to exclude any fields from to_dict, or is this the intended behavior?

lists/api.py (1)

1-261: Overall API enhancements look good.

The new endpoints, filtering parameters, and pagination changes appear well-structured and maintainable. Nice usage of drf_spectacular for documentation.

🧰 Tools
🪛 Ruff (0.8.2)

4-4: django.utils.timezone imported but unused

Remove unused import: django.utils.timezone

(F401)

donations/serializers.py (1)

15-42: Explicit field definitions and nested serializers are well-structured.

Defining each field explicitly in DonationSerializer provides clarity and control over serialization. The nested serializers for donor, token, and other related entities are a solid approach.

Also applies to: 44-97

grantpicks/api.py (1)

348-405: Validate the rounds_approved_in relationship usage and ensure the account/project alignment.
The code in ProjectStatsAPI relies on account.rounds_approved_in.count() and filters for votes where pairs__voted_project=account. Confirm that rounds_approved_in exists on the Account model and that an Account is indeed treated as a "project" in vote relationships. This may require additional documentation or data-model checks.

lists/models.py (4)

16-20: Migrate from PositiveIntegerField to AutoField carefully.
Switching to AutoField can impact the existing database schema. Ensure that a safe data migration is performed to avoid conflicts during deployment.


34-38: Reintroducing the admins ManyToManyField.
Make sure that the reintroduction of admins aligns with existing admin functionality. This may require a data migration to populate or adjust relationships.


49-55: Optionality added to description and cover_image_url.
Allowing blank=True can improve usability. Verify that any existing client or API usage does not depend on these fields always being present.


187-187: unique_together for list and registrant.
Enforcing uniqueness here can prevent duplicate registrations. Confirm that no existing data violates this constraint.

accounts/serializers.py (4)

54-60: Validate ProjectInPairSerializer model choice.
This serializer uses Account as its model but refers to the field project_id. Ensure that the Account model or related logic truly matches the intent of a "project" relationship.


61-83: Nested NearSocialProfileDataSerializer within AccountSerializer.
This approach offers clear data organization. If there's a need to partially update nested fields, confirm if you require a custom update() implementation.


85-114: Verify example data coherence.
The SIMPLE_ACCOUNT_EXAMPLE includes a detailed nested structure (NFT, linktree, background image). Confirm that it accurately reflects real-world usage, especially if documentation references simpler or different fields.


124-128: PaginatedAccountsResponseSerializer usage.
The pagination structure looks consistent. Ensure that the corresponding view or API references this serializer to deliver the intended response format.

lists/serializers.py (9)

19-33: Narrowed fields approach is commendable.
Specifying each field instead of using __all__ ensures better clarity and helps avoid accidental data leakage.


49-60: Field enumeration for ListRegistrationSerializer looks good.
Explicitly listing fields can help avoid unexpected serialization of future model properties.


62-63: Nested Account serialization is appropriate.
No issues observed; references appear correct.


66-78: Example dictionary is helpful.
No concerns. Good practice to provide sample data for clarity.


80-85: Pagination example is well-defined.
Sample data fosters better understanding for developers integrating with the API.


88-92: Paginated response serializer is clearly defined.
Implementation follows typical REST conventions.


95-106: Example registration data is clear.
No issues—this aids in documentation.


108-113: Pagination example for registrations is well-structured.
Looks consistent with the rest of the serializers.


116-120: PaginatedListRegistrationsResponseSerializer is good.
Exposes only the necessary fields for pagination.

pots/api.py (9)

58-86: Confirm cache invalidation strategy.
cache_page(60 * 5) is used in PotsListAPI; ensure pot data changes are properly invalidated or accepted as stale within that period. Otherwise, this looks fine for read operations.


88-116: Pagination approach for PotFactoriesAPI is consistent.
Straightforward paginated fetch with minimal logic. No issues found.


118-149: PotDetailAPI implementation is clear.
404 response for missing pot is handled. Recommended to confirm whether caching is appropriate if pot details change frequently.


152-219: Filters and status checks for PotApplicationsAPI are well-implemented.
Both search and status queries are validated carefully. No major concerns.


221-257: PotDonationsAPI usage of prefetch-related is optimal.
Selecting related data in a single query likely improves performance.


259-299: PotSponsorsAPI sponsor extraction is clear.
Retrieves distinct donors where matching_pool=True. Straightforward approach.


302-353: PotPayoutsAPI logic for optional search is consistent.
Likewise, it correctly handles pot not found. Good use of param checks.


397-493: MpdaoVotersListAPI handles pagination in-memory.
For large voter sets, this may become a bottleneck, but it’s acceptable for smaller sets. Implementation is otherwise straightforward.


495-537: MpdaoVoterDetailAPI is logically consistent.
Provides 404 for missing data in the local JSON, then attempts to return partial fallback if no record is found.

accounts/api.py (9)

95-124: AccountsListAPI simply paginates all accounts.
Implementation looks good. The 5-minute cache ensures consistent performance for read requests.


126-160: AccountDetailAPI fetch and 404 is handled properly.
Straightforward detail retrieval. Validate the cache duration if account data changes rapidly.


163-217: AccountActivePotsAPI checks pot status.
Approach is correct, but ensure pot round times are up-to-date. Good use of query filtering.


220-273: AccountPotApplicationsAPI appropriately filters by status.
Acceptable logic. The 400 error for invalid status is well-defined.


275-314: AccountDonationsReceivedAPI is straightforward.
Uses prefetch for related fields. Implementation looks fine.


316-356: AccountDonationsSentAPI leverages select_related.
Fetching extended relationships in one go is beneficial. No concerns.


358-397: AccountPayoutsReceivedAPI looks good.
Appropriate filter for paid_at__isnull=False. No further issues.


399-445: AccountListRegistrationsAPI filters by optional status.
Graceful 404 and 400 error handling are in place.


448-485: AccountUpvotedListsAPI logic is clear.
Retrieves upvoted lists and paginates them. Implementation is sound.

grantpicks/migrations/0001_initial.py (7)

11-14: All dependencies look good
No issues found with referencing the accounts and tokens migrations.


17-81: Models creation for ProjectContact, ProjectContract, ProjectFundingHistory, ProjectRepository, and StellarEvent
Everything looks standard.


83-168: Project model
Fields and relationships are well-defined with a clear approach to references and constraints.


170-488: Round model
Defining fields like use_whitelist, use_vault, and the constraints on deployed dates provides excellent flexibility and clarity.


490-590: RoundDeposit
Good use of unique_together for (round, on_chain_id) to prevent duplicate deposit records.


592-633: Vote model
Inclusion of unique_together for (round, voter, voted_at) helps avoid duplicate voting records.


635-668: VotePair
Implementation looks correct and straightforward for grouping votes together.

pots/models.py (1)

655-673: Potential null date usage in fetch_usd_prices
paid_at can be null or blank. Adding a guard condition or fallback date can help avoid errors when fetching token prices.

donations/migrations/0013_alter_donation_chef_alter_donation_chef_fee_and_more.py (2)

15-145: Well-structured migration with clear field definitions.

All altered fields appear to follow Django best practices with appropriate help_text and logical on_delete behavior. Using blank=True and null=True in tandem allows optional relationships, which is beneficial if external references might be missing. Ensure that these fields correctly match your business requirements, especially if any of these relationships should be mandatory in practice.


146-157: Reassess the index on a DecimalField.

Indexing a high-cardinality decimal column (total_amount_usd) might not always offer performance benefits and could increase storage overhead. It is still valid if frequent range queries or lookups by exact amount are expected. Verify that indexing this decimal field aligns with your query patterns.

grantpicks/serializers.py (1)

1-304: Comprehensive serializer definitions look good.

The nested serializers (e.g., RoundSerializer, ProjectSerializer) correctly handle complex relationships. The usage of SerializerMethodField in RoundSerializer to retrieve chain is appropriate. Ensure unit tests cover invalid data scenarios for these nested structures. Overall, the architecture is consistent with REST framework best practices.

🧰 Tools
🪛 Ruff (0.8.2)

7-7: .models.Vote imported but unused

Remove unused import: .models.Vote

(F401)

pots/serializers.py (1)

15-402: Robust serializer definitions for pots and associated models.

The multiple serializers (e.g., PotSerializer, PotApplicationSerializer) suitably handle nested relationships, decimal formatting, and example data. Caching logic in MpdaoSnapshotSerializer is neatly encapsulated to reduce external calls. Ensure that HTTP request failures or unexpected JSON responses from external APIs (e.g., in get_is_human) are handled gracefully in production.

Comment on lines +44 to +49
field=models.CharField(
blank=True,
help_text="Custom args as Base64VecU8.",
null=True,
verbose_name="custom args",
),
Copy link

Choose a reason for hiding this comment

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

⚠️ Potential issue

Missing max_length parameter for CharField

The custom_args field is defined as a CharField but doesn't specify a maximum length, which Django requires. This could cause database migration errors.

            field=models.CharField(
                blank=True,
                help_text="Custom args as Base64VecU8.",
                null=True,
+               max_length=255,  # Adjust based on expected data size
                verbose_name="custom args",
            ),
📝 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.

Suggested change
field=models.CharField(
blank=True,
help_text="Custom args as Base64VecU8.",
null=True,
verbose_name="custom args",
),
field=models.CharField(
blank=True,
help_text="Custom args as Base64VecU8.",
null=True,
max_length=255, # Adjust based on expected data size
verbose_name="custom args",
),

Comment on lines +12 to +17
operations = [
migrations.RenameField(
model_name="token",
old_name="id",
new_name="account",
),
Copy link

Choose a reason for hiding this comment

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

💡 Verification agent

🧩 Analysis chain

Significant architectural change detected.

Renaming the id field to account is an unconventional approach that changes the primary key field. While this appears to be a deliberate architectural decision (similar renames exist in other models based on the provided context), be aware this could have widespread implications:

  1. Any code directly referencing the id field will need updating
  2. Foreign key relationships may need adjustments
  3. Performance considerations for database indexing

Ensure all references to this model's primary key have been updated throughout the codebase:


🏁 Script executed:

#!/bin/bash
# Find references to Token.id that might need updating
rg "Token\(?\)?\.id" --type py
# Find potential ORM queries that might be affected
rg "Token\.objects\.get\(id=" --type py
rg "Token\.objects\.filter\(id=" --type py

Length of output: 310


Action Required: Address lingering references to the old primary key

The migration in tokens/migrations/0006_rename_id_token_account.py renames the primary key from id to account. However, verification scripts revealed that some code still directly references the old id field. In particular:

  • tokens/migrations/0008_migrate_ft_to_token.py contains:
    • token = Token.objects.get(id=payout.ft)
  • donations/migrations/0010_migrate_ft_to_token.py contains:
    • token = Token.objects.get(id=donation.ft)

Please review and update these references (and any other relevant code) to reflect the new primary key field (account). Ensure that any foreign key relationships or ORM queries that depend on the primary key are adjusted accordingly to avoid potential data integrity or performance issues.

Comment on lines +16 to +19
field=models.DateTimeField(
help_text="Application last update date.",
null=True,
verbose_name="updated at",
Copy link

Choose a reason for hiding this comment

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

🛠️ Refactor suggestion

Missing auto-update behavior for timestamp field.

The updated_at field is being configured without the auto_now=True attribute that's typically used for "last updated" timestamp fields. Without this attribute, the timestamp won't automatically update when the model is saved.

Consider modifying this migration to include the auto-update behavior:

field=models.DateTimeField(
    help_text="Application last update date.",
    null=True,
    verbose_name="updated at",
+   auto_now=True,
),

If manual control over the update timestamp is required, ensure that all code paths that modify PotApplication instances properly update this field.

📝 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.

Suggested change
field=models.DateTimeField(
help_text="Application last update date.",
null=True,
verbose_name="updated at",
field=models.DateTimeField(
help_text="Application last update date.",
null=True,
verbose_name="updated at",
auto_now=True,
),

Comment on lines +16 to +23
for address in account_addresses:
account = Account.objects.get(id=address)
account.fetch_near_social_profile_data()
self.stdout.write(
self.style.SUCCESS(f"Fetched social profile data for {address}")
)
# wait for 1 second to avoid rate limiting
time.sleep(1)
Copy link

Choose a reason for hiding this comment

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

🛠️ Refactor suggestion

Missing error handling and progress reporting.

The command lacks error handling and detailed progress reporting, which could lead to incomplete execution if any account processing fails.

Add error handling and improved progress reporting:

- # Fetch social profile data for each account
- for address in account_addresses:
-     account = Account.objects.get(id=address)
-     account.fetch_near_social_profile_data()
-     self.stdout.write(
-         self.style.SUCCESS(f"Fetched social profile data for {address}")
-     )
-     # wait for 1 second to avoid rate limiting
-     time.sleep(1)
+ total = len(account_addresses)
+ for i, address in enumerate(account_addresses, 1):
+     try:
+         self.stdout.write(f"Processing {i}/{total}: {address}")
+         account = Account.objects.get(id=address)
+         success = account.fetch_near_social_profile_data()
+         
+         if success:
+             self.stdout.write(
+                 self.style.SUCCESS(f"✓ Fetched social profile data for {address}")
+             )
+         else:
+             self.stdout.write(
+                 self.style.WARNING(f"⚠ No data found for {address}")
+             )
+     except Exception as e:
+         self.stdout.write(
+             self.style.ERROR(f"✗ Error processing {address}: {str(e)}")
+         )
+         
+     # wait for 1 second to avoid rate limiting
+     time.sleep(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.

Suggested change
for address in account_addresses:
account = Account.objects.get(id=address)
account.fetch_near_social_profile_data()
self.stdout.write(
self.style.SUCCESS(f"Fetched social profile data for {address}")
)
# wait for 1 second to avoid rate limiting
time.sleep(1)
total = len(account_addresses)
for i, address in enumerate(account_addresses, 1):
try:
self.stdout.write(f"Processing {i}/{total}: {address}")
account = Account.objects.get(id=address)
success = account.fetch_near_social_profile_data()
if success:
self.stdout.write(
self.style.SUCCESS(f"✓ Fetched social profile data for {address}")
)
else:
self.stdout.write(
self.style.WARNING(f"⚠ No data found for {address}")
)
except Exception as e:
self.stdout.write(
self.style.ERROR(f"✗ Error processing {address}: {str(e)}")
)
# wait for 1 second to avoid rate limiting
time.sleep(1)

@@ -0,0 +1,15 @@
from rest_framework.serializers import ModelSerializer

from .models import NadabotRegistry, Provider, Stamp
Copy link

Choose a reason for hiding this comment

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

⚠️ Potential issue

Remove unused import

The Stamp model is imported but not used in this file. Consider removing it to keep imports clean.

-from .models import NadabotRegistry, Provider, Stamp
+from .models import NadabotRegistry, Provider
📝 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.

Suggested change
from .models import NadabotRegistry, Provider, Stamp
from .models import NadabotRegistry, Provider
🧰 Tools
🪛 Ruff (0.8.2)

3-3: .models.Stamp imported but unused

Remove unused import: .models.Stamp

(F401)

Comment on lines +6 to +17
def update_donations_net_amount(apps, schema_editor):
Donation = apps.get_model("donations", "Donation")
for donation in Donation.objects.filter(net_amount="0"):
total_amount = int(donation.total_amount)
protocol_fee = int(donation.protocol_fee)
referrer_fee = int(donation.referrer_fee or 0)
chef_fee = int(donation.chef_fee or 0)

net_amount = total_amount - protocol_fee - referrer_fee - chef_fee
donation.net_amount = net_amount
donation.save()

Copy link

Choose a reason for hiding this comment

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

🛠️ Refactor suggestion

Missing error handling for string conversion failures

The migration converts string values to integers without any error handling. If any field contains non-numeric values, the migration will fail. Consider adding try/except blocks with appropriate logging.

 def update_donations_net_amount(apps, schema_editor):
+    import logging
+    logger = logging.getLogger(__name__)
     Donation = apps.get_model("donations", "Donation")
-    for donation in Donation.objects.filter(net_amount="0"):
-        total_amount = int(donation.total_amount)
-        protocol_fee = int(donation.protocol_fee)
-        referrer_fee = int(donation.referrer_fee or 0)
-        chef_fee = int(donation.chef_fee or 0)
-
-        net_amount = total_amount - protocol_fee - referrer_fee - chef_fee
-        donation.net_amount = net_amount
-        donation.save()
+    donations_to_update = Donation.objects.filter(net_amount="0")
+    logger.info(f"Found {donations_to_update.count()} donations with net_amount=0 to update")
+    
+    for donation in donations_to_update:
+        try:
+            total_amount = int(donation.total_amount)
+            protocol_fee = int(donation.protocol_fee)
+            referrer_fee = int(donation.referrer_fee or 0)
+            chef_fee = int(donation.chef_fee or 0)
+
+            net_amount = total_amount - protocol_fee - referrer_fee - chef_fee
+            donation.net_amount = str(net_amount)  # Convert back to string to match model field type
+            donation.save()
+        except (ValueError, TypeError) as e:
+            logger.error(f"Error updating donation {donation.id}: {str(e)}")
+            logger.error(f"Values: total={donation.total_amount}, protocol={donation.protocol_fee}, referrer={donation.referrer_fee}, chef={donation.chef_fee}")
📝 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.

Suggested change
def update_donations_net_amount(apps, schema_editor):
Donation = apps.get_model("donations", "Donation")
for donation in Donation.objects.filter(net_amount="0"):
total_amount = int(donation.total_amount)
protocol_fee = int(donation.protocol_fee)
referrer_fee = int(donation.referrer_fee or 0)
chef_fee = int(donation.chef_fee or 0)
net_amount = total_amount - protocol_fee - referrer_fee - chef_fee
donation.net_amount = net_amount
donation.save()
def update_donations_net_amount(apps, schema_editor):
import logging
logger = logging.getLogger(__name__)
Donation = apps.get_model("donations", "Donation")
donations_to_update = Donation.objects.filter(net_amount="0")
logger.info(f"Found {donations_to_update.count()} donations with net_amount=0 to update")
for donation in donations_to_update:
try:
total_amount = int(donation.total_amount)
protocol_fee = int(donation.protocol_fee)
referrer_fee = int(donation.referrer_fee or 0)
chef_fee = int(donation.chef_fee or 0)
net_amount = total_amount - protocol_fee - referrer_fee - chef_fee
donation.net_amount = str(net_amount) # Convert back to string to match model field type
donation.save()
except (ValueError, TypeError) as e:
logger.error(f"Error updating donation {donation.id}: {str(e)}")
logger.error(f"Values: total={donation.total_amount}, protocol={donation.protocol_fee}, referrer={donation.referrer_fee}, chef={donation.chef_fee}")

Comment on lines +45 to +46
def token_address(self, obj):
return obj.token.account
Copy link

Choose a reason for hiding this comment

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

💡 Verification agent

🧩 Analysis chain

Validate potential null references within token.

When calling obj.token.account, be cautious if token might be None. Consider adding a safety check or a fallback if token is not set.


🏁 Script executed:

#!/bin/bash
# Checking for lines where 'token_address' might be used without verifying obj.token is set
rg "token_address" -A 3

Length of output: 382


Potential null-reference issue in token_address method
In donations/admin.py at lines 45–46, accessing obj.token.account directly may raise an error if obj.token is None. Please update this method to include a null-check. For example:

def token_address(self, obj):
    return obj.token.account if obj.token else None

This update will ensure the method safely returns a fallback value (e.g., None) when token is not set.

Comment on lines +59 to +63
data.pop("total_donations_amount")
data.pop("net_donations_amount")
data.pop("total_donations_count")
data.pop("total_protocol_fees")
data.pop("total_referrer_fees")
Copy link

Choose a reason for hiding this comment

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

🛠️ Refactor suggestion

Check for potential key errors.

The code calls .pop() for several keys ("total_donations_amount", "net_donations_amount", etc.) without verifying their presence. While it’s likely these keys exist, consider a safer approach, such as:

data.pop("total_donations_amount", None)

and so on, to avoid KeyError in edge cases.

Comment on lines +21 to +25
on_chain_id = models.IntegerField(
_("contract list ID"),
null=False,
unique=True,
help_text=_("List ID in contract"),
Copy link

Choose a reason for hiding this comment

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

🛠️ Refactor suggestion

Confirm the new on_chain_id uniqueness constraint.
Introducing a non-null, unique integer field might require a migration step and data backfill. Verify that upstream code handles any uniqueness errors gracefully.

Comment on lines +209 to +213
approved_projects = models.ManyToManyField(
Account,
related_name="rounds_approved_in",
help_text=_("Projects Approved for round."),
)
Copy link

Choose a reason for hiding this comment

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

⚠️ Potential issue

Mismatch in approved_projects field
The approved_projects field currently references Account, but the migration definitions reference grantpicks.project. This mismatch will break migrations.

Proposed fix:

-    approved_projects = models.ManyToManyField(
-        Account,
-        related_name="rounds_approved_in",
-        help_text=_("Projects Approved for round."),
-    )
+    from grantpicks.models import Project
+    approved_projects = models.ManyToManyField(
+        Project,
+        related_name="rounds_approved_in",
+        help_text=_("Projects Approved for round."),
+    )
📝 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.

Suggested change
approved_projects = models.ManyToManyField(
Account,
related_name="rounds_approved_in",
help_text=_("Projects Approved for round."),
)
from grantpicks.models import Project
approved_projects = models.ManyToManyField(
Project,
related_name="rounds_approved_in",
help_text=_("Projects Approved for round."),
)

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants