Skip to content

Conversation

david-vicente
Copy link
Contributor

This PR follows the suggestion in this comment
As it is with_period only works with scalars, which contrasts with with_lengthscale that uses ARDTransform to deal with vector inputs. Any ideas on how should we deal with this?

@codecov
Copy link

codecov bot commented Nov 6, 2021

Codecov Report

Attention: Patch coverage is 0% with 2 lines in your changes missing coverage. Please review.

Project coverage is 78.89%. Comparing base (33d64d1) to head (928b9ec).
Report is 91 commits behind head on master.

Files with missing lines Patch % Lines
src/transform/with_period.jl 0.00% 2 Missing ⚠️

❗ There is a different number of reports uploaded between BASE (33d64d1) and HEAD (928b9ec). Click for more details.

HEAD has 1 upload less than BASE
Flag BASE (33d64d1) HEAD (928b9ec)
5 4
Additional details and impacted files
@@             Coverage Diff             @@
##           master     #401       +/-   ##
===========================================
- Coverage   89.23%   78.89%   -10.34%     
===========================================
  Files          52       53        +1     
  Lines        1198     1199        +1     
===========================================
- Hits         1069      946      -123     
- Misses        129      253      +124     

☔ View full report in Codecov by Sentry.
📢 Have feedback on the report? Share it here.

🚀 New features to boost your workflow:
  • ❄️ Test Analytics: Detect flaky tests, report on failures, and find test suite problems.

Copy link
Member

@devmotion devmotion left a comment

Choose a reason for hiding this comment

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

One of the motivations of with_lengthscale is that it provides a unified API for ScaleTransform and ARDTransform (although one can save allocations by using ARDTransform directly). This is not the case here since it will always just add a PeriodTransform. Usually, users are expected to use the transforms, eg there's also no with_linear or with_function as one can just use LinearTransform and FunctionTransform. Therefore maybe we should not add an additional function to the API but rather

  • tell users (how) to use PeriodTransform directly
  • add a convenience constructor PeriodKernel(r::Real)

@david-vicente
Copy link
Contributor Author

One of the motivations of with_lengthscale is that it provides a unified API for ScaleTransform and ARDTransform (although one can save allocations by using ARDTransform directly).

I thought that with_lengthscale was added because the convention when dealing with kernels is dividing the lengscale and ScaleTransform is a multiplication. The same happens with the period/frequency here. But I agree that adding a lot of different ways to achieve the same things ends up polluting the API.

@st--
Copy link
Member

st-- commented Nov 9, 2021

My motivation for introducing with_lengthscale was also to not have to think in terms of inverse lengthscale.
And when constructing a periodic kernel, I'd rather just write with_period(kernel, period) than kernel \circ<tab> PeriodicTransform(1/period).

add a convenience constructor PeriodKernel(r::Real)

I'm not quite sure what you mean by that, and how it's different from with_period? how would you pass in the base kernel?

What should the meaning of periodicity be in more than one dimension? Coordinate-wise (i.e., equivalent to a product of kernels operating on one of the dimensions each)?

@devmotion
Copy link
Member

So it seems the main motivation is to introduce a different parametrization of PeriodicTransform? Then we could just define

function PeriodicTransform(; period=nothing, frequency=period === nothing ? 1.0 : inv(period))
    return PeriodicTransform{typeof(frequency)}(frequency)
end

I am mainly questioning if it is worth adding a new function with_period just to avoid having to write inv explicitly. Maybe in general we could add more keyword argument based constructors of transforms, similar to the kernels, to make it clearer what the individual arguments mean and to allow different parameterizations.

@st--
Copy link
Member

st-- commented Nov 9, 2021

Hmm, having looked at the code again, I'm wondering about what the right way for defining periodic kernels actually is. PeriodicTransform only works on 1D inputs, and maps each Real to two values. (So the base kernel needs to handle 2D inputs correctly.) For distance-based kernels, the combination of Euclidean metric & PeriodicTransform can be done in closed form - this is effectively what the Sinus "distance" does, so in principle we could achieve the same by constructing e.g. SqExponentialKernel(; metric=Sinus()) except the Sinus metric doesn't do the sqrt(). Which is why PeriodicKernel is basically same as SqExponential, but without the square. And then it's kinda confusing that the period is set with the lengthscale transform, and the lengthscale is set as an argument to the period...

@st--
Copy link
Member

st-- commented Nov 9, 2021

That was quite rambly. So in bullet points:

  • it'd be great if we could construct all sorts of periodic kernels (based on any kernel supporting an Euclidean metric?) - PeriodicTransform does that (I think?)
  • it'd be great if we could avoid the intermediate ℝ → ℝ ² conversion required by PeriodicTransform - PeriodicKernel does this (but only for SqExponential base kernel)
  • it'd be great if we could directly specify the period and lengthscale in ways that make sense when just reading code (PeriodicTransform(; period=...) is good, with_lengthscale(PeriodicKernel(), period) not so good)

@devmotion
Copy link
Member

PeriodicTransform only works on 1D inputs, and maps each Real to two values.

This is the main reason why I still think that PeriodicTransform should map to complex numbers with cis or cispi (only available in more recent Julia versions). Then the "outer" dimensionality and structure would not be changed.

@st--
Copy link
Member

st-- commented Nov 9, 2021

This is the main reason why I still think that PeriodicTransform should map to complex numbers with cis or cispi (only available in more recent Julia versions). Then the "outer" dimensionality and structure would not be changed.

Then the base kernel would have to handle complex numbers, how would that affect speed e.g. when trying to run it on the GPU? It seems like it'd still be more extra effort than just computing the "equivalent distance" for the base kernel, no?

@devmotion
Copy link
Member

devmotion commented Nov 9, 2021

Then the base kernel would have to handle complex numbers, how would that affect speed e.g. when trying to run it on the GPU?

CuArrays of Complex{Float32} should be fine, in the memory every element is just like two Float32. Also many kernels should be able to handle complex inputs, and some others could be fixed easily (e.g., instead of dot we would use realdot from RealDot.jl for the dot product in the exponentiated kernel).

It seems like it'd still be more extra effort than just computing the "equivalent distance" for the base kernel, no?

You don't necessarily have to or want to use the PeriodicTransform in all cases. It still would make it more usable, in my opinion, in cases where you want to use the transform, in particular when dealing with multi-dimensional inputs.

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.

3 participants