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

Add Dandiset star functionality with UI components #2123

Open
wants to merge 27 commits into
base: master
Choose a base branch
from
Open

Conversation

bendichter
Copy link
Member

@bendichter bendichter commented Dec 26, 2024

fix #1119

Feature developed with Cursor

  • Introduced DandisetStar model and admin interface for managing starred Dandisets.
  • Implemented star/unstar actions in the DandisetViewSet and corresponding API endpoints.
  • Added star count and starred status to DandisetSerializer.
  • Created StarButton component for toggling star status in the UI.
  • Developed StarredDandisetsView to display Dandisets starred by the user.
  • Updated DandisetList and DandisetLandingView to include star functionality.
  • Enhanced DandisetFilterBackend to support ordering by star count.
  • Update existing tests and add new tests
  • Add package-lock.json to ignore list for spell check

I haven't been able to log in on my dev server so I can't test that behavior, but the unlogged in web UI and the admin interface appear to work as expected

- Introduced DandisetStar model and admin interface for managing starred Dandisets.
- Implemented star/unstar actions in the DandisetViewSet and corresponding API endpoints.
- Added star count and starred status to DandisetSerializer.
- Created StarButton component for toggling star status in the UI.
- Developed StarredDandisetsView to display Dandisets starred by the user.
- Updated DandisetList and DandisetLandingView to include star functionality.
- Enhanced DandisetFilterBackend to support ordering by star count.
@bendichter bendichter marked this pull request as draft December 26, 2024 16:50
- Updated test cases in test_dandiset.py and test_version.py to include 'star_count' and 'is_starred' fields in the expected responses.
- Ensured consistency across various test scenarios for Dandiset and Version APIs regarding star functionality.
- Implemented new test cases in `test_dandiset.py` to verify star/unstar actions, star count, and starred status for Dandisets.
- Updated API responses in `DandisetViewSet` to return the current star count when starring or unstarring a Dandiset.
- Added tests for unauthenticated access to star functionality and listing starred Dandisets.
…ionality

- Added 'star_count' and 'is_starred' fields to the Dandiset interface in types.
- Updated StarredDandisetsView to process and display starred Dandisets with their most recent published version.
- Refactored data handling in StarredDandisetsView to ensure proper mapping of Dandiset data.
Updated the way Dandiset properties are managed by creating a shallow copy of the Dandiset object and removing 'most_recent_published_version' and 'draft_version' before returning the data.
Simplified the mapping of Dandiset objects by directly returning the original Dandiset instead of creating a shallow copy and removing specific properties. This change improves code clarity and maintains the integrity of the Dandiset data structure.
Copy link
Member

@waxlamp waxlamp left a comment

Choose a reason for hiding this comment

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

Very interesting!

I'm a dinosaur who's not ready to trust AI-generated code just yet, so I'll want to get careful reviews from @jjnesbitt and @mvandenburgh (both to identify problems with the code, and so we have team members with a mental model of how it works) before merging. Furthermore, I think this will benefit from some enhancements (like using a service layer to manage stars), as well as some minor stuff that I can push fixes for myself. Finally, I would like for @jtomeck to help out with the frontend design here as well.

Thanks @bendichter for kicking this feature off!

web/package-lock.json Outdated Show resolved Hide resolved
web/yarn.lock Outdated Show resolved Hide resolved
@bendichter
Copy link
Member Author

@waxlamp yes, I mentioned Cursor because I agree that this will need careful eyes on it before integrating. But looking at the changes the majority of them seem straightforward

.pre-commit-config.yaml Outdated Show resolved Hide resolved
@bendichter
Copy link
Member Author

ok yeah the npm-related issues are my fault. I am used to installing with npm and having large automatically generated files as a result of that. I'll make sure to use yarn in the future.

- Removed legacy star/unstar methods from the Dandiset model.
- Introduced new star_dandiset and unstar_dandiset service functions to handle starring and unstarring Dandisets, including transaction management and star count updates.
- Updated DandisetViewSet to integrate star/unstar actions, returning the updated star count in API responses.
- Enhanced the Dandiset model to support star functionality, ensuring proper user authentication checks.
@bendichter bendichter marked this pull request as ready for review December 27, 2024 01:23
@bendichter
Copy link
Member Author

I don't know what to do about the remaining CI check failures

Copy link
Member

@jjnesbitt jjnesbitt left a comment

Choose a reason for hiding this comment

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

Thanks for the work, @bendichter. I've done a first pass of this PR and identified some necessary changes, but nonetheless it's great to have the outline of this feature out of the way.

I'd be happy to address some of these changes myself, if that makes more sense. I'll still leave the comments either way.

dandiapi/api/services/dandiset/__init__.py Outdated Show resolved Hide resolved
dandiapi/api/services/dandiset/__init__.py Outdated Show resolved Hide resolved
dandiapi/api/views/dandiset.py Outdated Show resolved Hide resolved
operation_description='Star a dandiset. User must be authenticated.',
)
@action(methods=['POST'], detail=True)
def star(self, request, dandiset__pk) -> Response:
Copy link
Member

Choose a reason for hiding this comment

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

I think this PR should be merged after #2115, and one of the reasons is that this PR could take advantage of the new permission service layer. For example, a decorator that requires authentication could be used, similar to the require_dandiset_owner_or_403 decorator introduced in that PR.

Copy link
Member

Choose a reason for hiding this comment

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

@jjnesbitt do you still intend to do this, now that #2115 is merged?

Copy link
Member

Choose a reason for hiding this comment

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

Yes, good point. I'll see what can be done there.

dandiapi/api/views/serializers.py Show resolved Hide resolved
web/yarn.lock Outdated Show resolved Hide resolved
web/src/components/StarButton.vue Outdated Show resolved Hide resolved
web/src/router.ts Outdated Show resolved Hide resolved
@bendichter
Copy link
Member Author

bendichter commented Dec 29, 2024

@jjnesbitt thanks for the prompt and thorough review. I can confirm that I can replicate those UI bugs. I might have time to address them myself but feel free to take this over if you have the bandwidth to do so.

@jjnesbitt
Copy link
Member

jjnesbitt commented Dec 30, 2024

@jjnesbitt thanks for the prompt and thorough review. I can confirm that I can replicate those UI bugs. I might have time to address them myself but feel free to take this over if you have the bandwidth to do so.

I think I have the bandwidth, so today I can make a pass at changing someone of the things I mentioned.

@jjnesbitt jjnesbitt force-pushed the stars branch 2 times, most recently from a62f371 to c9d8c21 Compare December 30, 2024 18:54
To fix this behavior for the dandiset list endpoint, create a new method
on the DandisetViewSet class, `_get_dandiset_star_context`, which
collects dandiset star information. This information is then passed to
the DandisetListSerializer as context.
Comment on lines +82 to +84
serializer = VersionDetailSerializer(
instance=version, context=self.get_serializer_context()
)
Copy link
Member

Choose a reason for hiding this comment

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

@mvandenburgh @waxlamp I don't love this solution, but it works. I was curious how the other serializers work without this (e.g. the Dandiset retrieve endpoint), and it turns out the DRFViewSet class we inherit from (specifically ReadOnlyModelViewSet -> GenericViewSet -> generics.GenericAPIView) automatically populate this context when the get_serializer method is called. This means that on all untouched methods we inherit and use, it's populated, while on any methods we've overwritten, it's missing (unless we explicitly specify it, like it done here).

To me this suggests that we should do an overarching refactor of our view and serialization code, since I think we have a lot of technical debt there. But for now, this is what seems to be the easiest change that's in scope.

Copy link
Member

@mvandenburgh mvandenburgh left a comment

Choose a reason for hiding this comment

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

We'll also need to add audit events for starring/unstarring a dandiset.

@jjnesbitt
Copy link
Member

We'll also need to add audit events for starring/unstarring a dandiset.

For posterity: We met offline and agreed to exclude dandiset stars from audit events.

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.

GitHub stars on dandisets
5 participants