-
Notifications
You must be signed in to change notification settings - Fork 7
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
feat: add 'transforms' JS crate and include in augurs JS bindings #195
Conversation
WalkthroughThe changes in this pull request involve modifications to several Changes
Possibly related PRs
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? 🪧 TipsChatThere are 3 ways to chat with CodeRabbit:
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)
Other keywords and placeholders
CodeRabbit Configuration File (
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Actionable comments posted: 0
🧹 Outside diff range and nitpick comments (7)
js/augurs-transforms-js/src/lib.rs (2)
1-1
: Typographical error: 'Javascript' should be 'JavaScript'In the module documentation, 'Javascript' should be correctly capitalized as 'JavaScript'.
33-38
: Consider handling empty data inPowerTransform::new
Adding a check for empty
opts.data
can improve error handling. If the input data is empty, returning a meaningful error prevents potential runtime issues.js/testpkg/transforms.test.ts (2)
1-13
: Consider adding JSDoc documentation for the test suite.The initialization code is well-structured, but adding a brief JSDoc comment describing the test suite's purpose would improve maintainability.
+/** + * Test suite for PowerTransform class from @bsull/augurs/transforms. + * Validates both transform and inverse transform operations. + */ import { webcrypto } from 'node:crypto'
50-62
: Add edge case tests for the PowerTransform.While the basic functionality is tested, consider adding tests for:
- Empty arrays
- Arrays with negative values
- Arrays with zero values
- Arrays with NaN/Infinity
Cargo.toml (2)
47-47
: Fix formatting of argmin dependency.There's a spacing issue in the argmin dependency declaration.
-argmin= "0.10.0" +argmin = "0.10.0"
Line range hint
15-17
: Fix author email format.The author email is missing a closing bracket.
- "Ben Sully <ben.sully@grafana.com", + "Ben Sully <ben.sully@grafana.com>",crates/augurs-forecaster/src/transforms.rs (1)
Line range hint
154-182
: Consider adding documentation for public API.Since these methods are now part of the public API, they should have comprehensive documentation including examples.
+ /// Applies the transformation to the given time series. + /// + /// # Arguments + /// + /// * `input` - An iterator over f64 values to transform + /// + /// # Returns + /// + /// A boxed iterator over the transformed values + /// + /// # Examples + /// + /// ``` + /// use augurs_forecaster::transforms::Transform; + /// + /// let data = vec![1.0, 2.0, 3.0]; + /// let transform = Transform::log(); + /// let transformed: Vec<_> = transform.transform(data.into_iter()).collect(); + /// ``` pub fn transform<'a, T>(&'a self, input: T) -> Box<dyn Iterator<Item = f64> + 'a>
📜 Review details
Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro
📒 Files selected for processing (8)
Cargo.toml
(1 hunks)crates/augurs-forecaster/Cargo.toml
(1 hunks)crates/augurs-forecaster/src/transforms.rs
(2 hunks)js/augurs-transforms-js/Cargo.toml
(1 hunks)js/augurs-transforms-js/src/lib.rs
(1 hunks)js/justfile
(1 hunks)js/package.json.tmpl
(1 hunks)js/testpkg/transforms.test.ts
(1 hunks)
✅ Files skipped from review due to trivial changes (1)
- js/augurs-transforms-js/Cargo.toml
🔇 Additional comments (5)
js/testpkg/transforms.test.ts (1)
27-47
: LGTM! Well-implemented custom matcher.
The custom matcher toAllBeCloseTo
is well-implemented with clear error messages and appropriate tolerance handling.
crates/augurs-forecaster/src/transforms.rs (1)
Line range hint 154-167
: Verify the impact of increased visibility.
The transform and inverse_transform methods have been made public to support JS bindings. The implementation looks solid with proper error handling.
Also applies to: 169-182
✅ Verification successful
Based on the search results, I can generate the final response as I have enough information to verify the impact of the visibility changes:
Visibility change is safe and aligned with intended usage
The increased visibility of transform
and inverse_transform
methods is intentionally supporting the JS bindings (js/augurs-transforms-js/src/lib.rs
) while maintaining proper encapsulation. All existing usages are within the crate or through the intended public JS interface. The implementation includes proper error handling and type safety.
- ✓ Used internally in
crates/augurs-forecaster/src/forecaster.rs
- ✓ Exposed correctly through JS bindings in
js/augurs-transforms-js/src/lib.rs
- ✓ No unexpected usages or implementations found in the codebase
🏁 Scripts executed
The following scripts were executed for the analysis:
Script:
#!/bin/bash
# Description: Check for any existing usages of these methods that might be affected
# by the visibility change
# Search for any direct usage of these methods
rg -A 2 "transform\(" --type rust
# Search for any implementations or traits using these methods
ast-grep --pattern 'impl $_ for $_ {
$$$
transform($$$) {
$$$
}
$$$
}'
Length of output: 4693
crates/augurs-forecaster/Cargo.toml (1)
16-19
: LGTM! Workspace dependency management looks good.
The transition to workspace-level dependency management is a good practice that will help maintain consistency across the project.
Let's verify workspace dependency consistency:
✅ Verification successful
Let me verify the workspace dependencies more specifically by checking the root Cargo.toml for workspace dependencies and comparing with the specific crate's usage.
Workspace dependency configuration is correctly implemented
The verification confirms that:
- The dependencies (argmin, augurs-core, itertools, thiserror) are properly defined in the root workspace's dependencies section
- The augurs-forecaster crate correctly references these dependencies using the
.workspace = true
syntax - The versions in the workspace match the intended dependency management structure
🏁 Scripts executed
The following scripts were executed for the analysis:
Script:
#!/bin/bash
# Description: Check for any inconsistent dependency versions across workspace
# Check if all workspace members use workspace dependencies consistently
rg "^\w+\s*=\s*\"[^\"]+\"" -g "Cargo.toml" -A 1
Length of output: 10941
Script:
#!/bin/bash
# Check workspace dependencies in root Cargo.toml
echo "=== Checking root Cargo.toml workspace dependencies ==="
rg "^\[workspace.dependencies\]" -A 50 Cargo.toml
# Check augurs-forecaster's specific dependency declarations
echo -e "\n=== Checking augurs-forecaster's dependencies ==="
cat crates/augurs-forecaster/Cargo.toml
Length of output: 2823
js/package.json.tmpl (1)
29-30
: LGTM! Export configuration follows established patterns.
The new transforms export is properly configured and maintains alphabetical ordering with other exports.
Let's verify the export path consistency:
✅ Verification successful
Let me verify the content of the justfile to ensure we have all the necessary build targets.
Export configuration is correctly aligned with build targets
All exported JS files in package.json.tmpl have corresponding build targets in the justfile. The transforms module is properly included in both the exports and the build targets list, maintaining consistency with the established pattern.
🏁 Scripts executed
The following scripts were executed for the analysis:
Script:
#!/bin/bash
# Description: Verify that all exported JS files have corresponding build targets
# Extract export paths and check against build targets
jq -r '.exports | to_entries[] | .value' js/package.json.tmpl | sed 's/^\.\///' | while read -r file; do
echo "Checking $file..."
if ! grep -q "$(basename "$file" .js)" js/justfile; then
echo "Warning: $file might be missing a build target"
fi
done
Length of output: 1350
Script:
#!/bin/bash
# Check the content of justfile for build targets
cat js/justfile
Length of output: 1077
js/justfile (1)
10-11
: LGTM! Build target properly integrated.
The transforms module is correctly added to the build process, maintaining consistency with other modules.
Let's verify the build target configuration:
✅ Verification successful
Build target properly integrated and verified
All build targets in the justfile have corresponding directories in the codebase:
- The
transforms
module is correctly added to the build process - Directory
js/augurs-transforms-js
exists - The structure is consistent with other modules (changepoint, clustering, core, etc.)
🏁 Scripts executed
The following scripts were executed for the analysis:
Script:
#!/bin/bash
# Description: Verify build target directory structure
# Check if all build targets have corresponding directories
while read -r target; do
target_dir="js/augurs-${target}-js"
if [ ! -d "$target_dir" ]; then
echo "Warning: Missing directory for target: $target"
fi
done < <(grep "build-inner" js/justfile | sed -n 's/.*"\\([^"]*\\)".*/\\1/p')
Length of output: 179
Script:
#!/bin/bash
# Let's try a simpler approach to verify the build targets and directory structure
# First, let's see the content of justfile
echo "=== Content of justfile ==="
cat js/justfile
# Then check for the corresponding directories
echo -e "\n=== Checking directories ==="
fd -t d "augurs-.*-js$" js/
Length of output: 1461
This provides a way to access the power transform functions from JS. I'd like to add the Forecaster functionality to the Prophet bindings somehow but that limits the Prophet APIs to only return yhat and the bounds. That's probably enough for many cases but it's not ideal. This is a quick workaround to get the transform functionality into the JS package really.
This is required otherwise the WASM build assumes some features are being provided in the environment, which isn't true.
7cd0929
to
80cfe32
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Actionable comments posted: 0
🧹 Outside diff range and nitpick comments (4)
crates/augurs-forecaster/src/transforms.rs (4)
154-167
: Enhance documentation with error handling examplesWhile the documentation is good, consider adding examples that demonstrate:
- Error handling scenarios (e.g., invalid inputs)
- Edge cases (e.g., NaN, infinity)
- Return type behavior in error conditions
/// # Example /// /// ``` /// use augurs_forecaster::transforms::Transform; /// /// let data = vec![1.0, 2.0, 3.0]; /// let transform = Transform::log(); /// let transformed: Vec<_> = transform.transform(data.into_iter()).collect(); +/// +/// // Error handling example +/// let invalid_data = vec![-1.0]; // Invalid for log transform +/// let result: Vec<_> = transform.transform(invalid_data.into_iter()).collect(); +/// assert!(result[0].is_nan()); // Returns NaN for invalid inputs /// ```Also applies to: 183-196
Line range hint
449-457
: Standardize error handling across transformationsThe error handling approach is inconsistent between the direct functions and their iterator implementations. While the functions return
Result
, the iterators silently convert errors toNaN
. This could lead to hard-to-debug issues.Consider either:
- Making all transformations return
Result
consistently, or- Adding error reporting capabilities to the iterator implementations
impl<T> Iterator for BoxCox<T> where T: Iterator<Item = f64>, { type Item = f64; fn next(&mut self) -> Option<Self::Item> { - self.inner - .next() - .map(|x| box_cox(x, self.lambda).unwrap_or(f64::NAN)) + self.inner.next().map(|x| { + match box_cox(x, self.lambda) { + Ok(val) => val, + Err(_) => { + // Log error or use a custom error handling strategy + f64::NAN + } + } + }) } }Also applies to: 516-524
Line range hint
293-308
: Optimize MinMaxScale performanceThe current implementation recalculates scaling factors for each item. Consider pre-calculating these values to improve performance.
struct MinMaxScale<T> { inner: T, params: MinMaxScaleParams, + scale_factor: f64, + offset: f64, } impl<T> Iterator for MinMaxScale<T> where T: Iterator<Item = f64>, { type Item = f64; fn next(&mut self) -> Option<Self::Item> { - let Self { - params: - MinMaxScaleParams { - data_min, - data_max, - scaled_min, - scaled_max, - }, - .. - } = self; - self.inner.next().map(|x| { - *scaled_min + ((x - *data_min) * (*scaled_max - *scaled_min)) / (*data_max - *data_min) - }) + self.inner.next().map(|x| self.offset + (x * self.scale_factor)) } } trait MinMaxScaleExt: Iterator<Item = f64> { fn min_max_scale(self, params: MinMaxScaleParams) -> MinMaxScale<Self> where Self: Sized, { + let scale_factor = (params.scaled_max - params.scaled_min) / (params.data_max - params.data_min); + let offset = params.scaled_min - (params.data_min * scale_factor); MinMaxScale { inner: self, params, + scale_factor, + offset, } } }
Line range hint
601-700
: Enhance test coverage for edge casesWhile the test coverage is good, consider adding tests for:
- NaN handling
- Infinity handling
- Error conditions (e.g., invalid lambda values)
#[cfg(test)] mod test { + #[test] + fn test_box_cox_edge_cases() { + let data = vec![f64::NAN, f64::INFINITY, f64::NEG_INFINITY]; + let lambda = 0.5; + let results: Vec<_> = data.into_iter().box_cox(lambda).collect(); + assert!(results.iter().all(|x| x.is_nan())); + } + + #[test] + fn test_yeo_johnson_invalid_lambda() { + let data = vec![1.0]; + let lambda = f64::NAN; + let results: Vec<_> = data.into_iter().yeo_johnson(lambda).collect(); + assert!(results.iter().all(|x| x.is_nan())); + } }
📜 Review details
Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro
📒 Files selected for processing (8)
Cargo.toml
(2 hunks)crates/augurs-forecaster/Cargo.toml
(1 hunks)crates/augurs-forecaster/src/transforms.rs
(2 hunks)js/augurs-transforms-js/Cargo.toml
(1 hunks)js/augurs-transforms-js/src/lib.rs
(1 hunks)js/justfile
(1 hunks)js/package.json.tmpl
(1 hunks)js/testpkg/transforms.test.ts
(1 hunks)
🚧 Files skipped from review as they are similar to previous changes (7)
- js/package.json.tmpl
- js/justfile
- crates/augurs-forecaster/Cargo.toml
- Cargo.toml
- js/augurs-transforms-js/Cargo.toml
- js/augurs-transforms-js/src/lib.rs
- js/testpkg/transforms.test.ts
🔇 Additional comments (1)
crates/augurs-forecaster/src/transforms.rs (1)
Line range hint 1-700
: Overall implementation looks good!
The transformations are well-implemented with proper mathematical formulas, good test coverage, and clear documentation. The suggested improvements are mostly optimizations and enhancements rather than critical issues.
This provides a way to access the power transform functions from JS.
I'd like to add the Forecaster functionality to the Prophet bindings
somehow but that limits the Prophet APIs to only return yhat and
the bounds. That's probably enough for many cases but it's not ideal.
This is a quick workaround to get the transform functionality into the
JS package really.
Summary by CodeRabbit
New Features
augurs-transforms-js
package for JavaScript bindings related to data transformations.PowerTransform
structure for applying power transformations, including methods for transformation and inverse transformation.Dependency Updates
argmin
as a workspace dependency.Bug Fixes
Transform
enum.Documentation
package.json.tmpl
to include a new export for transforms.Tests
PowerTransform
class to validate its functionality.