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

Grid LRU caching with Looker size estimates #5214

Draft
wants to merge 10 commits into
base: develop
Choose a base branch
from

Conversation

benjaminpkane
Copy link
Contributor

@benjaminpkane benjaminpkane commented Dec 4, 2024

What changes are proposed in this pull request?

Adds a maximum LRU cache size in bytes for Grid Lookers. Currently hardcoded to 1GB

How is this patch tested? If it is not, please explain why.

Hook test

What areas of FiftyOne does this PR affect?

  • App: FiftyOne application changes
  • Build: Build and test infrastructure changes
  • Core: Core fiftyone Python library changes
  • Documentation: FiftyOne documentation changes
  • Other

Summary by CodeRabbit

Release Notes

  • New Features

    • Enhanced numeral formatting error handling for better user feedback.
    • Improved user feedback in the RangeSlider component when no bounds are available.
    • Introduced a custom hook, useLookerCache, for optimized cache management.
  • Bug Fixes

    • Updated logic in the Grid component for improved performance during fast scrolling.
    • Enhanced error handling and state management in the VideoLooker class.
  • Documentation

    • Expanded guidance for developing JavaScript plugins, including new examples and sections for clarity.
  • Tests

    • Updated test cases for Lightning queries to reflect changes in expected results for float and group datasets.

Copy link
Contributor

coderabbitai bot commented Dec 4, 2024

Important

Review skipped

Draft detected.

Please check the settings in the CodeRabbit UI or the .coderabbit.yaml file in this repository. To trigger a single review, invoke the @coderabbitai review command.

You can disable this status message by setting the reviews.review_status to false in the CodeRabbit configuration file.

Walkthrough

The pull request introduces several modifications across multiple components and files. Key changes include enhancements to error handling in numeral formatting, updates to the RangeSlider component for improved user feedback, and the introduction of a custom hook for cache management in the Grid component. Additionally, modifications in the AbstractLooker and VideoLooker classes enhance size estimation capabilities. The documentation has been updated to provide clearer guidance on developing plugins. Overall, these changes focus on improving functionality, error handling, and user experience across the application.

Changes

File Change Summary
app/packages/core/src/components/Common/utils.tsx Modified getFormatter function for better error handling of numeral formatting, returning original value if formatted result is "NaN".
app/packages/core/src/components/Filters/NumericFieldFilter/RangeSlider.tsx Introduced nonfinitesText state variable to conditionally display messages when no bounds are available, enhancing user feedback.
app/packages/core/src/components/Filters/NumericFieldFilter/state.ts Replaced nonfiniteAtom import with nonfiniteData, renamed hasNonfinites selector to nonfinitesText, and updated logic to filter non-finite values.
app/packages/core/src/components/Grid/Grid.tsx Updated render method in spotlight to replace soft with zooming, simplifying looker instance retrieval and enhancing error handling.
app/packages/core/src/components/Grid/useLookerCache.ts Introduced useLookerCache hook for managing Looker cache with LRU strategy, including size estimation and loading management.
app/packages/core/src/components/Grid/useRefreshers.ts Removed LRUCache implementation, now utilizing useLookerCache for cache management.
app/packages/core/src/components/Grid/useSelect.ts Updated store parameter type to use lookerStore from useRefreshers, introduced retained set for tracking updated item IDs.
app/packages/core/src/plugins/SchemaIO/components/TableView.tsx Added fontSize property to TableCell components for styling; updated tooltip rendering logic and action handling in rows.
app/packages/looker/src/lookers/abstract.ts Added getSizeBytesEstimate method for size estimation; updated dispatchEvent and dispatchImpliedEvents methods for clarity and consistency.
app/packages/looker/src/lookers/video.ts Introduced getSizeBytesEstimate method for video data size estimation; modified loadOverlays and pluckOverlays methods for better overlay management.
app/packages/spotlight/src/row.ts Updated show method to replace hidden parameter with zooming, streamlining visibility management.
app/packages/spotlight/src/section.ts Removed #dirty property, added #itemIds set, and updated rendering logic to simplify item updates.
app/packages/spotlight/src/types.ts Modified Render type signature to replace two boolean parameters with one, simplifying the interface.
docs/source/plugins/developing_plugins.rst Expanded guidance on developing JavaScript plugins, added new sections, and included example code snippets for clarity.
fiftyone/server/lightning.py Enhanced lightning query resolver with new variable is_frames, updated filter construction logic, and improved handling of non-finite numbers.
fiftyone/server/view.py Added checks for "range" key existence in _make_scalar_expression and _make_query functions to prevent KeyError exceptions.
tests/unittests/lightning_tests.py Updated float query tests and expanded group dataset tests to reflect changes in expected results for Lightning queries.

Possibly related PRs

Suggested labels

bug, enhancement, app

Suggested reviewers

  • minhtuev
  • ritch

"In the code we hop and play,
Enhancing features every day.
From formatting to sliders bright,
We bring improvements to the light!
With tests and docs, we pave the way,
For better plugins, come what may!" 🐰✨


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

❤️ Share
🪧 Tips

Chat

There are 3 ways to chat with CodeRabbit:

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

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

CodeRabbit Commands (Invoked using PR comments)

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

Other keywords and placeholders

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

Documentation and Community

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

@benjaminpkane benjaminpkane marked this pull request as ready for review December 4, 2024 19:41
@benjaminpkane benjaminpkane added enhancement Code enhancement app Issues related to App features labels Dec 4, 2024
Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

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

Actionable comments posted: 4

🧹 Outside diff range and nitpick comments (11)
app/packages/looker/src/lookers/video.ts (2)

161-172: Consider optimizing the size estimation implementation

The size estimation logic could be improved in terms of performance and maintainability:

  1. Consider caching the size calculation result since first frame overlays are unlikely to change frequently
  2. The iteration could be replaced with a more efficient reduce operation
  getSizeBytesEstimate() {
    let size = super.getSizeBytesEstimate();
-    if (!this.firstFrame.overlays?.length) {
-      return size;
-    }
-
-    for (let index = 0; index < this.firstFrame.overlays.length; index++) {
-      size += this.firstFrame.overlays[index].getSizeBytes();
-    }
-
-    return size;
+    return size + (this.firstFrame.overlays?.reduce(
+      (acc, overlay) => acc + overlay.getSizeBytes(),
+      0
+    ) ?? 0);
  }

161-172: Consider adding memory management documentation

The class uses WeakRef for frame management which is good for memory, but the relationship between size estimation and frame references should be documented.

Add a class-level JSDoc comment explaining the memory management strategy:

/**
 * VideoLooker handles video frame management with memory-efficient practices:
 * - Uses WeakRef for frame caching to allow garbage collection
 * - Implements size estimation for LRU cache management
 * - First frame is kept in memory for quick access
 */
export class VideoLooker extends AbstractLooker<VideoState, VideoSample> {

Also applies to: 26-27

app/packages/core/src/components/Grid/useLookerCache.ts (2)

18-18: Remove unnecessary statement 'reset;'

The standalone reset; statement inside the useMemo callback does not have any effect and can be safely removed. The dependency array [reset] already ensures that the memoized value recalculates when reset changes.

Apply this diff to remove the unnecessary statement:

   /** CLEAR CACHE WHEN reset CHANGES */
-  reset;
   /** CLEAR CACHE WHEN reset CHANGES */

27-28: Avoid using console.log in production code

The console.log statement on line 27 may lead to unnecessary logging in the production environment. Consider removing it or using a logging library with appropriate log levels.

Apply this diff to remove the console log:

   sizeCalculation: (looker) => {
-    console.log(looker.getSizeBytesEstimate());
     return looker.getSizeBytesEstimate();
   },
app/packages/core/src/components/Grid/useSelect.ts (1)

19-37: Optimize the update logic and avoid potential null references

  • Null Check Enhancement: The check if (!instance) on line 22 prevents null reference errors, which is good. However, it's more concise to return early if the instance is null.
  • Simplify the Loop: Consider restructuring the loop to improve readability.

You can refactor the code block as follows:

const retained = new Set<string>();
spotlight?.updateItems((id) => {
  const instance = store.get(id.description);
  if (!instance) {
    return;
  }

  retained.add(id.description);
  instance.updateOptions({
    ...options,
    fontSize,
    selected: selected.has(id.description),
  });
});

-for (const id of store.keys()) {
+for (const id of store.keys()) {
  if (!retained.has(id)) {
    store.delete(id);
  }
}

This improves readability and ensures that all non-retained instances are deleted from the store.

app/packages/core/src/components/Grid/useLookerCache.test.ts (1)

31-34: Handle the possibility of 'looker' being undefined

The check for looker on line 32 is good practice. However, throwing a generic error may not provide enough context during test failures. Consider using an assertion or providing a clearer error message.

You can provide a more descriptive error message:

if (!looker) {
-  throw new Error("looker is missing");
+  throw new Error("Failed to retrieve 'looker' from the cache");
}

Alternatively, you can use an assertion library for better test clarity.

app/packages/core/src/components/Common/utils.tsx (1)

71-78: Consider extracting format string selection logic

The format string selection logic could be extracted into a separate function for better maintainability and testability.

Consider refactoring to:

+ const getFormatString = (fieldType: string, bounds: [number, number]) => {
+   if ([INT_FIELD, FRAME_NUMBER_FIELD, FRAME_SUPPORT_FIELD].includes(fieldType)) {
+     return "0a";
+   }
+   return bounds[1] - bounds[0] < 0.1 ? "0.0000a" : "0.00a";
+ };

  const str = numeral(v).format(
-   [INT_FIELD, FRAME_NUMBER_FIELD, FRAME_SUPPORT_FIELD].includes(fieldType)
-     ? "0a"
-     : bounds[1] - bounds[0] < 0.1
-     ? "0.0000a"
-     : "0.00a"
+   getFormatString(fieldType, bounds)
  );
app/packages/core/src/components/Filters/NumericFieldFilter/RangeSlider.tsx (1)

41-43: Consider using a template for i18n support

The text concatenation should be moved to a template for better internationalization support.

Consider using a template:

- <Box text={nonfinitesText ? `${nonfinitesText} present` : "No results"} />
+ <Box text={nonfinitesText ? t('nonfinites.present', { text: nonfinitesText }) : t('common.no_results')} />
app/packages/core/src/components/Grid/Grid.tsx (1)

Line range hint 73-86: LGTM! Performance optimization with improved cache handling.

The changes effectively optimize performance by:

  1. Utilizing cached looker instances
  2. Skipping looker creation during fast scrolling
  3. Simplifying control flow with early returns

Consider extracting the looker creation logic into a separate function for better maintainability:

-        const looker = createLooker.current?.(
-          { ...result, symbol: id },
-          {
-            fontSize: getFontSize(),
-          }
-        );
+        const createNewLooker = () => {
+          const looker = createLooker.current?.(
+            { ...result, symbol: id },
+            {
+              fontSize: getFontSize(),
+            }
+          );
+          looker.addEventListener("selectthumbnail", ({ detail }) =>
+            selectSample.current?.(detail)
+          );
+          return looker;
+        };
+        const looker = createNewLooker();
app/packages/spotlight/src/section.ts (1)

303-305: Consider memory management for large datasets

While the filtering logic effectively prevents duplicate items, for very large datasets, consider implementing a cleanup mechanism to prevent unbounded growth of the #itemIds Set.

- [...end.remainder, ...data.items].filter(
-   (i) => !this.#itemIds.has(i.id.description)
- ),
+ const newItems = [...end.remainder, ...data.items].filter(i => {
+   // Consider clearing #itemIds when it grows too large
+   if (this.#itemIds.size > MAX_CACHED_IDS) {
+     this.#itemIds.clear();
+   }
+   return !this.#itemIds.has(i.id.description);
+ }),
fiftyone/server/lightning.py (1)

138-141: Avoid variable shadowing of is_frames in list comprehension

In the list comprehension at lines 138-141, the loop variable is_frames shadows the outer variable is_frames, which can lead to confusion and potential errors. Consider renaming the loop variable to prevent shadowing.

Apply this diff to rename the loop variable:

flattened = [
    (collection, item, is_frame_field)
    for collection, sublist, is_frame_field in zip(
        collections, queries, is_frames
    )
    for item in sublist
]
📜 Review details

Configuration used: .coderabbit.yaml
Review profile: CHILL

📥 Commits

Reviewing files that changed from the base of the PR and between a31a55a and 6e86c74.

📒 Files selected for processing (18)
  • app/packages/core/src/components/Common/utils.tsx (1 hunks)
  • app/packages/core/src/components/Filters/NumericFieldFilter/RangeSlider.tsx (1 hunks)
  • app/packages/core/src/components/Filters/NumericFieldFilter/state.ts (2 hunks)
  • app/packages/core/src/components/Grid/Grid.tsx (2 hunks)
  • app/packages/core/src/components/Grid/useLookerCache.test.ts (1 hunks)
  • app/packages/core/src/components/Grid/useLookerCache.ts (1 hunks)
  • app/packages/core/src/components/Grid/useRefreshers.ts (2 hunks)
  • app/packages/core/src/components/Grid/useSelect.ts (2 hunks)
  • app/packages/core/src/plugins/SchemaIO/components/TableView.tsx (2 hunks)
  • app/packages/looker/src/lookers/abstract.ts (2 hunks)
  • app/packages/looker/src/lookers/video.ts (1 hunks)
  • app/packages/spotlight/src/row.ts (2 hunks)
  • app/packages/spotlight/src/section.ts (4 hunks)
  • app/packages/spotlight/src/types.ts (1 hunks)
  • docs/source/plugins/developing_plugins.rst (2 hunks)
  • fiftyone/server/lightning.py (11 hunks)
  • fiftyone/server/view.py (2 hunks)
  • tests/unittests/lightning_tests.py (27 hunks)
🧰 Additional context used
📓 Path-based instructions (14)
app/packages/core/src/components/Common/utils.tsx (1)

Pattern **/*.{ts,tsx}: Review the Typescript and React code for conformity with best practices in React, Recoil, Graphql, and Typescript. Highlight any deviations.

app/packages/core/src/components/Filters/NumericFieldFilter/RangeSlider.tsx (1)

Pattern **/*.{ts,tsx}: Review the Typescript and React code for conformity with best practices in React, Recoil, Graphql, and Typescript. Highlight any deviations.

app/packages/spotlight/src/types.ts (1)

Pattern **/*.{ts,tsx}: Review the Typescript and React code for conformity with best practices in React, Recoil, Graphql, and Typescript. Highlight any deviations.

app/packages/looker/src/lookers/video.ts (1)

Pattern **/*.{ts,tsx}: Review the Typescript and React code for conformity with best practices in React, Recoil, Graphql, and Typescript. Highlight any deviations.

app/packages/core/src/components/Grid/useRefreshers.ts (1)

Pattern **/*.{ts,tsx}: Review the Typescript and React code for conformity with best practices in React, Recoil, Graphql, and Typescript. Highlight any deviations.

app/packages/core/src/components/Grid/Grid.tsx (1)

Pattern **/*.{ts,tsx}: Review the Typescript and React code for conformity with best practices in React, Recoil, Graphql, and Typescript. Highlight any deviations.

app/packages/core/src/plugins/SchemaIO/components/TableView.tsx (1)

Pattern **/*.{ts,tsx}: Review the Typescript and React code for conformity with best practices in React, Recoil, Graphql, and Typescript. Highlight any deviations.

app/packages/core/src/components/Grid/useLookerCache.test.ts (1)

Pattern **/*.{ts,tsx}: Review the Typescript and React code for conformity with best practices in React, Recoil, Graphql, and Typescript. Highlight any deviations.

app/packages/core/src/components/Grid/useLookerCache.ts (1)

Pattern **/*.{ts,tsx}: Review the Typescript and React code for conformity with best practices in React, Recoil, Graphql, and Typescript. Highlight any deviations.

app/packages/spotlight/src/section.ts (1)

Pattern **/*.{ts,tsx}: Review the Typescript and React code for conformity with best practices in React, Recoil, Graphql, and Typescript. Highlight any deviations.

app/packages/spotlight/src/row.ts (1)

Pattern **/*.{ts,tsx}: Review the Typescript and React code for conformity with best practices in React, Recoil, Graphql, and Typescript. Highlight any deviations.

app/packages/core/src/components/Filters/NumericFieldFilter/state.ts (1)

Pattern **/*.{ts,tsx}: Review the Typescript and React code for conformity with best practices in React, Recoil, Graphql, and Typescript. Highlight any deviations.

app/packages/core/src/components/Grid/useSelect.ts (1)

Pattern **/*.{ts,tsx}: Review the Typescript and React code for conformity with best practices in React, Recoil, Graphql, and Typescript. Highlight any deviations.

app/packages/looker/src/lookers/abstract.ts (1)

Pattern **/*.{ts,tsx}: Review the Typescript and React code for conformity with best practices in React, Recoil, Graphql, and Typescript. Highlight any deviations.

🔇 Additional comments (21)
app/packages/looker/src/lookers/abstract.ts (2)

194-209: LGTM! Well-implemented size estimation.

The implementation correctly:

  1. Accounts for RGBA image data (4 bytes per pixel)
  2. Handles null/undefined cases
  3. Includes overlay sizes in the total estimate

228-239: LGTM! Improved state transition handling.

The changes enhance readability and reliability:

  1. Clear parameter naming with previous and next
  2. Explicit load state detection using overlaysPrepared and dimensions
  3. Proper sequencing of event dispatching
app/packages/looker/src/lookers/video.ts (1)

161-172: Verify memory usage patterns

The size estimation is crucial for the LRU cache implementation. Let's verify how this estimation is used across the codebase.

✅ Verification successful

Let me gather more information about the cache implementation and memory usage patterns.


Size estimation implementation is correctly used in LRU cache

The size estimation is properly integrated into the LRU cache implementation:

  • Used as sizeCalculation callback in the LRU cache configuration
  • Respects the MAX_LRU_CACHE_SIZE limit
  • Properly handles cleanup through the dispose callback
  • The video looker correctly extends the base implementation by adding overlay sizes
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Search for references to getSizeBytesEstimate to understand usage patterns
rg -A 3 "getSizeBytesEstimate" 

# Look for LRU cache implementations that might use this estimation
ast-grep --pattern 'class $_ implements $_Cache {
  $$$
  getSizeBytesEstimate($$$) {
    $$$
  }
  $$$
}'

Length of output: 1861


Script:

#!/bin/bash
# Look for cache size limits or memory configurations
rg -A 5 "maxSize|cacheSize|memoryLimit" app/packages/core/src/components/Grid/

# Check the base class implementation
ast-grep --pattern 'class AbstractLooker {
  $$$
  getSizeBytesEstimate() {
    $$$
  }
  $$$
}'

# Find cache eviction or cleanup logic
rg -A 5 "evict|cleanup|dispose" app/packages/core/src/components/Grid/

Length of output: 1360

app/packages/core/src/components/Grid/useSelect.ts (1)

34-37: ⚠️ Potential issue

Ensure proper iteration and deletion of stale entries

When deleting entries from the store during iteration, ensure that you're not mutating the collection while iterating over it, which can lead to inconsistent behavior.

Consider collecting the keys to delete and removing them after the iteration:

-for (const id of store.keys()) {
-  if (retained.has(id)) continue;
-  store.delete(id);
-}
+const keysToDelete = [];
+for (const id of store.keys()) {
+  if (!retained.has(id)) {
+    keysToDelete.push(id);
+  }
+}
+for (const id of keysToDelete) {
+  store.delete(id);
+}

Likely invalid or redundant comment.

app/packages/spotlight/src/types.ts (1)

41-41: Update documentation to reflect changed 'Render' signature

The Render type's signature has changed, replacing soft: boolean and disable: boolean with zooming: boolean. Ensure that all implementations of the Render function are updated accordingly, and update any associated documentation or comments to prevent confusion.

No code changes needed here, but verify that all usages of Render are consistent with the new signature.

Run the following script to find all implementations of Render and check their parameters:

app/packages/core/src/components/Grid/useLookerCache.test.ts (1)

42-45: Verify cache reset behavior upon 'reset' prop change

The test checks that the cache sizes reset after rerendering with a new reset prop. Ensure that this behavior aligns with the intended functionality of the useLookerCache hook.

Confirm that resetting the cache on reset prop change is the desired behavior. If so, the test correctly validates this functionality.

app/packages/core/src/components/Filters/NumericFieldFilter/state.ts (2)

28-38: Improve the logic for generating 'nonfinitesText'

The new selector nonfinitesText filters out entries where the key is "none" and the value is falsy. Ensure that this logic correctly captures all relevant non-finite values and that the resulting text is accurate.

The implementation appears correct. It filters out irrelevant entries and joins the keys of valid non-finite values.


28-38: ⚠️ Potential issue

Update key in selectors to avoid duplication

The key for the nonfinitesText selector is set to "nonfinitesText", but there is another selector with the same key "hasBounds". Ensure that all selector keys are unique to prevent conflicts in the Recoil state management.

Apply this diff to correct the key for the hasDefaultRange selector:

-export const hasDefaultRange = selectorFamily({
-  key: "hasBounds",
+export const hasDefaultRange = selectorFamily({
+  key: "hasDefaultRange",

This ensures that each selector has a unique key.

Likely invalid or redundant comment.

app/packages/core/src/components/Grid/useRefreshers.ts (2)

7-7: LGTM!

The import of the new useLookerCache hook is correctly placed.


79-79: LGTM!

The replacement of the manual cache implementation with the useLookerCache hook is a good improvement for maintainability.

app/packages/core/src/components/Filters/NumericFieldFilter/RangeSlider.tsx (1)

38-38: LGTM!

The addition of the nonfinitesText state variable using Recoil is correct.

app/packages/spotlight/src/row.ts (1)

Line range hint 151-166: LGTM! Improved method signature with better state management.

The changes effectively:

  1. Remove the redundant hidden parameter
  2. Simplify the show method signature
  3. Properly propagate the zooming state to render
app/packages/core/src/plugins/SchemaIO/components/TableView.tsx (1)

224-224: Same font-size issue as noted above.

app/packages/spotlight/src/section.ts (2)

49-49: LGTM: New Set for tracking processed items

The addition of #itemIds as a private Set to track processed items is a good approach for preventing duplicates.


450-453: LGTM: Proper tracking of processed item IDs

The implementation correctly adds item IDs to the tracking Set as they are processed.

docs/source/plugins/developing_plugins.rst (1)

3135-3146: LGTM: Clear and helpful getting started guide

The new section provides excellent guidance for new plugin developers by:

  • Referencing the hello-world-plugin-js template repository
  • Introducing the fiftyone-js-plugin-build package
  • Explaining the build process setup
fiftyone/server/lightning.py (1)

519-525: ⚠️ Potential issue

Fix variable assignment error in _parse_result function

The variable value is used before being assigned when checking "value" in value, which will result in a NameError. Assign value = data[0] before using it.

Apply this diff to fix the variable assignment:

def _parse_result(data):
    if data and data[0]:
+       value = data[0]
        if "value" in value:
            value = value["value"]
            return (
                value
                if not isinstance(value, float) or math.isfinite(value)
                else None
            )

Likely invalid or redundant comment.

fiftyone/server/view.py (2)

617-619: Ensure safe access to "range" in _make_scalar_expression

Adding the check if "range" in args before unpacking args["range"] prevents potential KeyError exceptions, enhancing the robustness of the function when "range" is not provided.


644-651: Enhance robustness by checking "range" in arguments

In the handling of datetime and numeric fields, verifying the presence of "range" in args before unpacking prevents exceptions and ensures correct behavior when "range" is absent.

tests/unittests/lightning_tests.py (2)

Line range hint 538-969: Update tests to reflect changes in float handling

The test cases have been updated to account for non-finite float values. Setting max and min to None when non-finite values are present ensures that the tests accurately reflect the new behavior.


Line range hint 1063-1080: Add test cases for float fields in group datasets

Including confidence values in the classifications fields enhances test coverage for float handling in group datasets, ensuring min and max calculations are properly validated.

Comment on lines +205 to +207
for (let index = 0; index < this.sampleOverlays.length; index++) {
size += this.sampleOverlays[index].getSizeBytes();
}
Copy link
Contributor

Choose a reason for hiding this comment

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

For overlays, we might want to add one more case in sizer for bitmaps since they have the largest memory footprint. Something along the lines of:

if (typeof object === "object" && object.constructor.name === "ImageBitmap") {
    size += object.height * object.width * 4;
}

@benjaminpkane benjaminpkane marked this pull request as draft December 11, 2024 17:33
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
app Issues related to App features enhancement Code enhancement
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants