From 00d7997f951597af142cf3a435f51c7e40d63baa Mon Sep 17 00:00:00 2001 From: Aaron LI Date: Thu, 14 Jul 2016 23:38:02 +0800 Subject: spline.py: workaround the spline fitting error when give constant input y data --- spline.py | 29 ++++++++++++++++++++++++----- 1 file changed, 24 insertions(+), 5 deletions(-) diff --git a/spline.py b/spline.py index 20ec1cf..352f905 100644 --- a/spline.py +++ b/spline.py @@ -2,9 +2,11 @@ # # Aaron LI # Created: 2016-07-10 -# Updated: 2016-07-13 +# Updated: 2016-07-14 # # Change logs: +# 2016-07-14: +# * Workaround the spline fitting error when give constant input y data # 2016-07-13: # * Improve the `np.array` usage a bit # @@ -16,9 +18,12 @@ and/or extrapolate a group of discrete data. * smoothing spline: R's mgcv::gam() """ +import sys + import numpy as np import rpy2.robjects as ro from rpy2.robjects.packages import importr +from rpy2.rinterface import RRuntimeError class Spline: @@ -81,10 +86,24 @@ class SmoothSpline(Spline): else: y = ro.FloatVector(self.y) weights = ro.FloatVector(self.weights) - self.spline = self.mgcv.gam( - ro.Formula("y ~ s(x, bs='ps')"), - data=ro.DataFrame({"x": x, "y": y}), - weights=weights, method="REML") + try: + self.spline = self.mgcv.gam( + ro.Formula("y ~ s(x, bs='ps')"), + data=ro.DataFrame({"x": x, "y": y}), + weights=weights, method="REML") + except RRuntimeError: + # NOTE: + # If the input y data is constant (e.g., the temperature profile + # has only ONE data point), then above smoothing spline fitting + # using method "REML" will failed with error: + # Error in gam.reparam(UrS, sp, grderiv) : + # NA/NaN/Inf in foreign function call (arg 3) + print("WARNING: 'mgcv.gam()' using method 'REML' failed!", + file=sys.stderr) + self.spline = self.mgcv.gam( + ro.Formula("y ~ s(x, bs='ps', sp=0.6)"), + data=ro.DataFrame({"x": x, "y": y}), + weights=weights) def eval(self, x): x = np.array(x, ndmin=1) -- cgit v1.2.2