From 43e4a3836a42526e91b5055719685b1d9effacf4 Mon Sep 17 00:00:00 2001 From: Peter Powers <pmpowers@usgs.gov> Date: Sun, 13 Dec 2020 14:29:27 -0700 Subject: [PATCH] mfd test docs and cleaning --- .../java/gov/usgs/earthquake/nshmp/Maths.java | 4 +- .../gov/usgs/earthquake/nshmp/mfd/Mfd.java | 247 +++++++---- .../gov/usgs/earthquake/nshmp/mfd/Mfds.java | 28 ++ .../earthquake/nshmp/model/Deserialize.java | 4 +- .../nshmp/model/FaultRuptureSet.java | 16 +- .../usgs/earthquake/nshmp/model/MfdTrees.java | 4 +- .../nshmp/mfd/IncrementalMfdBuilderTest.java | 141 ------- .../usgs/earthquake/nshmp/mfd/MfdTests.java | 397 +++++++++++++++++- .../usgs/earthquake/nshmp/mfd/Mfds2Test.java | 44 -- .../usgs/earthquake/nshmp/mfd/MfdsTests.java | 93 ++-- 10 files changed, 641 insertions(+), 337 deletions(-) delete mode 100644 src/test/java/gov/usgs/earthquake/nshmp/mfd/IncrementalMfdBuilderTest.java delete mode 100644 src/test/java/gov/usgs/earthquake/nshmp/mfd/Mfds2Test.java diff --git a/src/main/java/gov/usgs/earthquake/nshmp/Maths.java b/src/main/java/gov/usgs/earthquake/nshmp/Maths.java index b794ce31..fb47cf43 100644 --- a/src/main/java/gov/usgs/earthquake/nshmp/Maths.java +++ b/src/main/java/gov/usgs/earthquake/nshmp/Maths.java @@ -23,7 +23,7 @@ public final class Maths { public static final double PI_BY_2 = Math.PI / 2; /** Constant for 2π. */ - public static final double TWO_PI = 2 * Math.PI; + public static final double TWO_PI = 2.0 * Math.PI; /** Conversion multiplier for degrees to radians. */ public static final double TO_RADIANS = Math.toRadians(1.0); @@ -35,7 +35,7 @@ public final class Maths { * The precomputed √<span style="border-top:1px solid; padding:0 0.1em;" * >2</span>. */ - public static final double SQRT_2 = Math.sqrt(2); + public static final double SQRT_2 = Math.sqrt(2.0); /** * The precomputed √<span style="border-top:1px solid; padding:0 0.1em;" diff --git a/src/main/java/gov/usgs/earthquake/nshmp/mfd/Mfd.java b/src/main/java/gov/usgs/earthquake/nshmp/mfd/Mfd.java index 07ba4f0a..71ce10ff 100644 --- a/src/main/java/gov/usgs/earthquake/nshmp/mfd/Mfd.java +++ b/src/main/java/gov/usgs/earthquake/nshmp/mfd/Mfd.java @@ -1,21 +1,23 @@ package gov.usgs.earthquake.nshmp.mfd; -import static com.google.common.base.Preconditions.checkArgument; +import static com.google.common.base.Preconditions.checkNotNull; import static gov.usgs.earthquake.nshmp.Earthquakes.checkMagnitude; import static gov.usgs.earthquake.nshmp.Earthquakes.magToMoment; import static gov.usgs.earthquake.nshmp.mfd.Mfd.Type.GR; import static gov.usgs.earthquake.nshmp.mfd.Mfd.Type.GR_TAPER; import static gov.usgs.earthquake.nshmp.mfd.Mfd.Type.INCR; import static gov.usgs.earthquake.nshmp.mfd.Mfd.Type.SINGLE; +import static gov.usgs.earthquake.nshmp.mfd.Mfds.checkRate; import static gov.usgs.earthquake.nshmp.mfd.Mfds.gutenbergRichterRate; +import java.util.Arrays; import java.util.EnumSet; -import java.util.Map; import java.util.Optional; import java.util.Set; import java.util.function.Consumer; import gov.usgs.earthquake.nshmp.Maths; +import gov.usgs.earthquake.nshmp.Text; import gov.usgs.earthquake.nshmp.data.MutableXySequence; import gov.usgs.earthquake.nshmp.data.Sequences; import gov.usgs.earthquake.nshmp.data.XyPoint; @@ -57,8 +59,13 @@ import gov.usgs.earthquake.nshmp.data.XySequence; * * <p>MFD {@link Properties} hold references to the parameters used to * initialize an MFD, including its {@link Type}. Properties objects may be - * constructed directly and support a {@link Properties#toBuilder()} method. - * NSHM + * constructed directly and support a {@link Properties#toBuilder()} method. The + * the {@code Mfd.Type} will be {@code INCREMENTAL} if the MFD was created + * directly from magnitude and rate data or is the result of combining more than + * one MFD via {@link Mfds#combine(java.util.Collection)}. Similarly, properties + * <i>may</i> not align with an associated MFD downstream if a builder + * initialized with a set of properties has been used to create multiple MFDs + * with variably scaled rates or transformed in various ways. * * @author U.S. Geological Survey * @see Mfds @@ -74,18 +81,24 @@ public final class Mfd { this.props = props; } - /** The immutable values representing this MFD. */ + /** + * The immutable values representing this MFD. + */ public XySequence data() { return data; } - /** The properties used to initialize the MFD. */ + /** + * The properties used to initialize the MFD. + */ public Properties properties() { return props; } - // TODO need magnitude and rate checks - // TODO Arrays.stream(magnitudes).forEach(Earthquakes::checkMagnitude); + @Override + public String toString() { + return "MFD:" + props.toString() + Text.NEWLINE + data.toString(); + } /** * Create an MFD from copies of the supplied arrays. The returned MFD will be @@ -110,6 +123,7 @@ public final class Mfd { * @param xy data to wrap in an MFD */ public static Mfd create(XySequence xy) { + Mfds.checkValues(xy.xValues(), xy.yValues()); return new Mfd(XySequence.copyOf(xy), new Properties(INCR)); } @@ -131,7 +145,10 @@ public final class Mfd { * rounds each magnitude to 5 decimal places. This implementation constructs a * distribution that is {@code μ ± 2σ} wide and represented by {@code nm} * evenly distributed magnitude values. The initial distribution integrates to - * one. + * one. The MFD, once built, will be of type: {@link Type#SINGLE} as it + * represents a single magnitude combined with a model of uncertainty. In the + * context of a hazard model, a normal distribution of magnitudes is typically + * used to represent aleatory variability. * * @param μ the mean magnitude * @param nm the number of magnitudes in the resultant distribution @@ -155,10 +172,10 @@ public final class Mfd { * magnitude distribtion, this implementation rounds each magnitude to 4 * decimal places. * + * @param b the Gutenberg-Richter b-value + * @param Δm the magnitude step of the distribtion * @param mMin the minimum truncation magnitude * @param mMax the maximum truncation magnitude - * @param Δm the magnitude step of the distribtion - * @param b the Gutenberg-Richter b-value * @throws IllegalArgumentException if {@code mMin} or {@code mMax} is outside * the range {@code [-2.0..9.7]} * @throws IllegalArgumentException if {@code mMin > mMax} @@ -167,11 +184,11 @@ public final class Mfd { * {@code [0..2]} */ public static Mfd.Builder newGutenbergRichterBuilder( - double mMin, - double mMax, + double b, double Δm, - double b) { - return new Properties.GutenbergRichter(b, Δm, mMin, mMax).toBuilder(); + double mMin, + double mMax) { + return new Properties.GutenbergRichter(1.0, b, Δm, mMin, mMax).toBuilder(); } /** @@ -205,26 +222,40 @@ public final class Mfd { double mMin, double mMax, double mc) { - return new Properties.GrTaper(b, Δm, mMin, mMax, mc).toBuilder(); + return new Properties.TaperedGr(1.0, b, Δm, mMin, mMax, mc).toBuilder(); } /** Magnitude-frequency distribution (MFD) type identifier. */ public enum Type { - /** An MFD with a single magnitude and rate. */ + /** + * An MFD with a single magnitude and rate. + * + * @see Mfd#newSingleBuilder(double) + */ SINGLE, - /** An MFD defining multiple magnitudes with varying rates. */ + /** + * An MFD defining multiple magnitudes with varying rates. + * + * @see Mfd#create(double[], double[]) + * @see Mfd#create(XySequence) + */ INCR, - /** An incremental Gutenberg-Richter MFD. */ + /** + * An incremental Gutenberg-Richter MFD. + * + * @see Mfd#newGutenbergRichterBuilder(double, double, double, double) + */ GR, /** * An incremental Gutenberg-Richter MFD with a rate scaling hint. This type * of MFD is used with NSHM grid sources to indicate that rates above * {@code M=6.5} (the minimum magnitude for finite faults) should be set to - * zero if certain conditions are met. + * zero if certain conditions are met. This type of MFD can not be created + * directly. */ GR_MMAX_GR, @@ -232,11 +263,16 @@ public final class Mfd { * An incremental Gutenberg-Richter MFD with a rate scaling hint. This type * of MFD is used with NSHM grid sources to indicate rates above some * 'characterisitic' fault magnitude should be set to zero if certain - * conditions are met. + * conditions are met. This type of MFD can not be created directly. */ GR_MMAX_SINGLE, - /** A Gutenberg-Richter MFD with a tapered upper tail. */ + /** + * A Gutenberg-Richter MFD with a tapered upper tail. + * + * @see Mfd#newTaperedGutenbergRichterBuilder(double, double, double, + * double, double) + */ GR_TAPER; /** A set containing all Gutenberg-Richter MFD types. */ @@ -257,7 +293,7 @@ public final class Mfd { private final MutableXySequence mfd; /* - * TODO clean Developer notes: + * Developer notes: * * add constraints preconditions for params other than magnitude * @@ -296,6 +332,8 @@ public final class Mfd { * * values are checked in toBuilder methods allowing us to verify values set * via JSON deserialization + * + * TODO note GR bin centering in docs */ /** @@ -306,16 +344,16 @@ public final class Mfd { * @param rates to initialize builder with */ public static Builder from(double[] magnitudes, double[] rates) { - return new Builder(new Properties(INCR), magnitudes, Optional.of(rates)); + return new Builder(new Properties(INCR), magnitudes, rates); } /** * Create an {@code Mfd.Builder} initialized with the supplied sequence. * - * @param mfd to initialize builder with + * @param xy sequence to initialize builder with */ - public static Builder from(XySequence mfd) { - return new Builder(mfd); + public static Builder from(XySequence xy) { + return new Builder(xy); } /** @@ -327,23 +365,25 @@ public final class Mfd { return new Builder(mfd); } - private Builder(XySequence src) { + private Builder(XySequence xy) { + Mfds.checkValues(xy.xValues(), xy.yValues()); this.props = new Properties(INCR); - this.mfd = MutableXySequence.copyOf(src); + this.mfd = MutableXySequence.copyOf(xy); } - private Builder(Mfd src) { - this.props = src.props; - this.mfd = MutableXySequence.copyOf(src.data); + private Builder(Mfd mfd) { + this.props = mfd.props; + this.mfd = MutableXySequence.copyOf(mfd.data); } - private Builder( - Properties props, - double[] magnitudes, - Optional<double[]> rates) { + private Builder(Properties props, double[] magnitudes) { + this(props, magnitudes, new double[magnitudes.length]); + } + private Builder(Properties props, double[] magnitudes, double[] rates) { + Mfds.checkValues(Arrays.stream(magnitudes), Arrays.stream(rates)); this.props = props; - this.mfd = MutableXySequence.create(magnitudes, rates); + this.mfd = MutableXySequence.create(magnitudes, Optional.of(rates)); } /** @@ -385,7 +425,7 @@ public final class Mfd { * @return this {@code Builder} object */ public Builder scaleToCumulativeRate(double cumulativeRate) { - return scale(cumulativeRate / Mfds.cumulativeRate(mfd)); + return scale(checkRate(cumulativeRate) / Mfds.cumulativeRate(mfd)); } /** @@ -397,7 +437,7 @@ public final class Mfd { * @return this {@code Builder} object */ public Builder scaleToIncrementalRate(double incrementalRate) { - return scale(incrementalRate / mfd.y(0)); + return scale(checkRate(incrementalRate) / mfd.y(0)); } /** @@ -407,7 +447,7 @@ public final class Mfd { * @return this {@code Builder} object */ public Builder scaleToMomentRate(double momentRate) { - return scale(momentRate / Mfds.momentRate(mfd)); + return scale(checkRate(momentRate) / Mfds.momentRate(mfd)); } /** @@ -418,7 +458,7 @@ public final class Mfd { * @return this {@code Builder} object */ public Builder transform(Consumer<XyPoint> action) { - mfd.transform(action); + mfd.transform(checkNotNull(action)); return this; } } @@ -426,9 +466,6 @@ public final class Mfd { /** * Properties object associated with a MFD. A properties object wraps the * original parameters required to initialize a {@link Mfd.Builder}. - * Properties <i>may</i> not align with an associated MFD downstream include - * specific not include rate or uncertainty model information that may have - * been used to scale and/or create multiple MFDs. */ public static class Properties { @@ -436,6 +473,7 @@ public final class Mfd { /* No-arg constructor for deserialization. */ private Properties() { + // TODO test this as null for no-arg this.type = SINGLE; } @@ -476,8 +514,8 @@ public final class Mfd { * * @throws ClassCastException if properties are for other MFD type */ - public GrTaper getAsGrTaper() { - return (GrTaper) this; + public TaperedGr getAsGrTaper() { + return (TaperedGr) this; } /** Return a MFD builder initialized with this properties object. */ @@ -485,31 +523,47 @@ public final class Mfd { throw new UnsupportedOperationException(); } + @Override + public String toString() { + return type().toString(); + } + /** Properties of a single magnitude MFD. */ public static final class Single extends Properties { - private final double m; + private final double magnitude; private final double rate; /* No-arg constructor for deserialization. */ private Single() { - m = Double.NaN; + magnitude = Double.NaN; rate = 1.0; }; - public Single(double m, double rate) { + /** + * Create a new properties object for a single MFD. + * + * @param magnitude of the distribution + * @param rate of the sole magnitude in the distribution + */ + public Single(double magnitude, double rate) { super(SINGLE); - this.m = m; + this.magnitude = magnitude; this.rate = rate; } - public Single(double m) { - this(m, 1.0); + /** + * Create a new properties object for a single MFD with a rate of one. + * + * @param magnitude of the distribution + */ + public Single(double magnitude) { + this(magnitude, 1.0); } /** The sole magnitude of the MFD */ - public double m() { - return m; + public double magnitude() { + return magnitude; } /** The rate of the sole magnitude. */ @@ -522,17 +576,18 @@ public final class Mfd { @Override public Builder toBuilder() { - checkMagnitude(m); - checkArgument(rate >= 0.0); + /* Values checked in builder */ return new Builder( this, - new double[] { this.m }, - Optional.of(new double[] { this.rate == 0.0 ? 1.0 : this.rate })); + new double[] { this.magnitude }, + // TODO this is bad + new double[] { this.rate == 0.0 ? 1.0 : this.rate }); } // TODO note scale in docs public Builder toGaussianBuilder(int nm, double σ, int nσ) { - double μ = checkMagnitude(this.m); + /* Pre-check magnitudes; final arrays are checked in builder */ + double μ = checkMagnitude(this.magnitude); double mMin = checkMagnitude(μ - nσ * σ); double mMax = checkMagnitude(μ + nσ * σ); double Δm = (mMax - mMin) / (nm - 1); @@ -540,18 +595,18 @@ public final class Mfd { .centered() .scale(5) .build(); - Builder builder = new Builder(this, magnitudes, Optional.empty()); + Builder builder = new Builder(this, magnitudes); builder.mfd.forEach(p -> p.set(Maths.normalPdf(μ, σ, p.x()))); return builder; } @Override public String toString() { - return new StringBuilder() - .append(type()) - .append(": ") - .append(Map.of("m", m, "rate", rate)) + String props = new StringBuilder() + .append("magnitude=").append(magnitude) + .append(", rate=").append(rate) .toString(); + return Mfds.propsToString(type(), props); } } @@ -573,8 +628,10 @@ public final class Mfd { this.mMax = Double.NaN; } - private GutenbergRichter(Type type, double a, double b, double Δm, double mMin, - double mMax) { + private GutenbergRichter( + Type type, double a, double b, + double Δm, double mMin, double mMax) { + super(type); this.a = a; this.b = b; @@ -583,8 +640,17 @@ public final class Mfd { this.mMax = mMax; } - public GutenbergRichter(double b, double Δm, double mMin, double mMax) { - this(GR, 1.0, b, Δm, mMin, mMax); + /** + * Create a new properties object for a Gutenberg-Richter MFD. + * + * @param a value of the distribution + * @param b value of the distribution + * @param Δm magnitude bin width of the distribution + * @param mMin minimum magnitude of the distribution + * @param mMin maximum magnitude of the distribution + */ + public GutenbergRichter(double a, double b, double Δm, double mMin, double mMax) { + this(GR, a, b, Δm, mMin, mMax); } /** The Gutenberg-Richter {@code a}-value. */ @@ -619,33 +685,53 @@ public final class Mfd { double[] magnitudes = Sequences.arrayBuilder(mMin, mMax, this.Δm) .scale(4) .build(); - Builder builder = new Builder(this, magnitudes, Optional.empty()); + Builder builder = new Builder(this, magnitudes); builder.mfd.forEach(p -> p.set(gutenbergRichterRate(this.a, this.b, p.x()))); return builder; } @Override public String toString() { + return Mfds.propsToString(type(), propsString()); + } + + private String propsString() { return new StringBuilder() - .append(type()) - .append(": ") - .append(Map.of("a", a, "b", b, "Δm", Δm, "mMin", mMin, "mMax", mMax)) + .append("a=").append(a) + .append(", b=").append(b) + .append(", Δm=").append(Δm) + .append(", mMin=").append(mMin) + .append(", mMax=").append(mMax) .toString(); } } /** Tapered Gutenberg–Richter MFD properties. */ - public static final class GrTaper extends GutenbergRichter { + public static final class TaperedGr extends GutenbergRichter { private final double mc; /* Dummy, no-arg constructor for deserialization. */ - private GrTaper() { + private TaperedGr() { this.mc = Double.NaN; } - private GrTaper(double b, double Δm, double mMin, double mMax, double mc) { - super(GR_TAPER, 1.0, b, Δm, mMin, mMax); + // TODO note that inital a value may not agree with distribution if + // it was transformed or scaled prior to building. + + /** + * Create a new properties object for a tapered Gutenberg-Richter MFD with + * an initial {@code a}-value of one. + * + * @param a value of the distribution + * @param b value of the distribution + * @param Δm magnitude bin width of the distribution + * @param mMin minimum magnitude of the distribution + * @param mMin maximum magnitude of the distribution + * @param mc corner magnitude of the distribution + */ + public TaperedGr(double a, double b, double Δm, double mMin, double mMax, double mc) { + super(GR_TAPER, a, b, Δm, mMin, mMax); this.mc = mc; } @@ -663,12 +749,19 @@ public final class Mfd { double[] magnitudes = Sequences.arrayBuilder(mMin, mMax, Δm()) .scale(4) .build(); - Builder builder = new Builder(this, magnitudes, Optional.empty()); + Builder builder = new Builder(this, magnitudes); TaperFunction grTaper = new TaperFunction(Δm(), b(), mc); builder.mfd.forEach(p -> p.set( gutenbergRichterRate(a(), b(), p.x()) * grTaper.scale(p.x()))); return builder; } + + @Override + public String toString() { + return Mfds.propsToString( + type(), + super.propsString() + ", mc=" + mc); + } } private static final double M_MIN_MOMENT = magToMoment(4.0); diff --git a/src/main/java/gov/usgs/earthquake/nshmp/mfd/Mfds.java b/src/main/java/gov/usgs/earthquake/nshmp/mfd/Mfds.java index e3a21019..33321faa 100644 --- a/src/main/java/gov/usgs/earthquake/nshmp/mfd/Mfds.java +++ b/src/main/java/gov/usgs/earthquake/nshmp/mfd/Mfds.java @@ -7,6 +7,7 @@ import static java.lang.Math.log; import static java.util.stream.Collectors.toList; import java.util.Collection; +import java.util.stream.DoubleStream; import com.google.common.base.Converter; import com.google.common.collect.Range; @@ -15,6 +16,7 @@ import gov.usgs.earthquake.nshmp.Earthquakes; import gov.usgs.earthquake.nshmp.Maths; import gov.usgs.earthquake.nshmp.data.MutableXySequence; import gov.usgs.earthquake.nshmp.data.XySequence; +import gov.usgs.earthquake.nshmp.mfd.Mfd.Type; /** * Utility methods for working with magnitude frequency distributions (MFDs). @@ -27,6 +29,30 @@ public final class Mfds { private Mfds() {} + /** + * Ensure {@code rate ≥ 0.0}. + * + * @param rate to validate + * @return the validated rate + * @throws IllegalArgumentException if {@code rate} value is less than 0.0 + */ + public static double checkRate(double rate) { + return checkInRange(Range.atLeast(0.0), "Rate", rate); + } + + static void checkValues(DoubleStream magnitudes, DoubleStream rates) { + magnitudes.forEach(Earthquakes::checkMagnitude); + rates.forEach(Mfds::checkRate); + } + + static String propsToString(Type type, String propsString) { + return new StringBuilder(type.toString()) + .append(" {") + .append(propsString) + .append("}") + .toString(); + } + /** * Combine the supplied MFDs into a single MFD, summing the rates of duplicate * magnitudes. @@ -128,6 +154,8 @@ public final class Mfds { return moRate; } + // TODO take Mfd not XySequence; replace with sum()?? + /** * Returns the total moment rate of an MFD defined by the magnitudes * (x-vlaues) and incremental rates (y-values) in the supplied XySquence. diff --git a/src/main/java/gov/usgs/earthquake/nshmp/model/Deserialize.java b/src/main/java/gov/usgs/earthquake/nshmp/model/Deserialize.java index 4fe96e93..b393c007 100644 --- a/src/main/java/gov/usgs/earthquake/nshmp/model/Deserialize.java +++ b/src/main/java/gov/usgs/earthquake/nshmp/model/Deserialize.java @@ -36,9 +36,9 @@ import gov.usgs.earthquake.nshmp.geo.json.Properties; import gov.usgs.earthquake.nshmp.gmm.Gmm; import gov.usgs.earthquake.nshmp.gmm.UncertaintyModel; import gov.usgs.earthquake.nshmp.mfd.Mfd; -import gov.usgs.earthquake.nshmp.mfd.Mfd.Properties.GrTaper; import gov.usgs.earthquake.nshmp.mfd.Mfd.Properties.GutenbergRichter; import gov.usgs.earthquake.nshmp.mfd.Mfd.Properties.Single; +import gov.usgs.earthquake.nshmp.mfd.Mfd.Properties.TaperedGr; import gov.usgs.earthquake.nshmp.model.MfdConfig.AleatoryProperties; import gov.usgs.earthquake.nshmp.tree.LogicGroup; import gov.usgs.earthquake.nshmp.tree.LogicTree; @@ -415,7 +415,7 @@ class Deserialize { properties = GSON.fromJson(validateGr(obj), GutenbergRichter.class); break; case GR_TAPER: - properties = GSON.fromJson(validateGrTaper(obj), GrTaper.class); + properties = GSON.fromJson(validateGrTaper(obj), TaperedGr.class); break; case INCR: // TODO custom props object that we'll replace once MFDs built diff --git a/src/main/java/gov/usgs/earthquake/nshmp/model/FaultRuptureSet.java b/src/main/java/gov/usgs/earthquake/nshmp/model/FaultRuptureSet.java index 6fd5e69d..36cfe4b8 100644 --- a/src/main/java/gov/usgs/earthquake/nshmp/model/FaultRuptureSet.java +++ b/src/main/java/gov/usgs/earthquake/nshmp/model/FaultRuptureSet.java @@ -347,7 +347,7 @@ public class FaultRuptureSet implements RuptureSet { Single props = mBranch.value().getAsSingle(); String id = String.join(":", props.type().name(), mBranch.id(), rBranch.id()); double weight = mBranch.weight() * rBranch.weight(); - propsTree.addBranch(id, new Single(props.m(), rate), weight); + propsTree.addBranch(id, new Single(props.magnitude(), rate), weight); } } return propsTree.build(); @@ -418,7 +418,7 @@ public class FaultRuptureSet implements RuptureSet { double Δm = Maths.round(Rm / nm, 6); // System.out.println(Rm + " " + grProps.mMin + " " + mMax + " " + Δm); - return new Mfd.Properties.GutenbergRichter(grProps.b(), Δm, grProps.mMin(), mMax); + return new Mfd.Properties.GutenbergRichter(1.0, grProps.b(), Δm, grProps.mMin(), mMax); } /* @@ -644,10 +644,10 @@ public class FaultRuptureSet implements RuptureSet { double moRate = moBranch.isPresent() ? moBranch.orElseThrow().value() - : single.rate() * Earthquakes.magToMoment(single.m()); + : single.rate() * Earthquakes.magToMoment(single.magnitude()); /* Optional epistemic uncertainty. */ - boolean epistemic = hasEpistemic(mfdConfig, single.m()); + boolean epistemic = hasEpistemic(mfdConfig, single.magnitude()); /* Optional aleatory variability. */ boolean aleatory = mfdConfig.aleatoryProperties.isPresent(); @@ -657,7 +657,7 @@ public class FaultRuptureSet implements RuptureSet { for (Branch<Double> epiBranch : mfdConfig.epistemicTree.orElseThrow()) { - double mEpi = single.m() + epiBranch.value(); + double mEpi = single.magnitude() + epiBranch.value(); double weightEpi = mfdWt * epiBranch.weight(); String id = mfdBranchId(mfdId, epiBranch.id()); @@ -685,14 +685,14 @@ public class FaultRuptureSet implements RuptureSet { if (aleatory) { MfdConfig.AleatoryProperties aleaProps = mfdConfig.aleatoryProperties.orElseThrow(); - Mfd mfd = newGaussianBuilder(single.m(), aleaProps) + Mfd mfd = newGaussianBuilder(single.magnitude(), aleaProps) .scaleToMomentRate(moRate) .build(); mfdTree.addBranch(mfdId, mfd, mfdWt); } else { - Mfd mfd = Mfd.newSingleBuilder(single.m()) + Mfd mfd = Mfd.newSingleBuilder(single.magnitude()) .scaleToMomentRate(moRate) .build(); mfdTree.addBranch(mfdId, mfd, mfdWt); @@ -751,7 +751,7 @@ public class FaultRuptureSet implements RuptureSet { /* GR MFD builder with mMax override. */ static Mfd.Builder newGrBuilder(Mfd.Properties.GutenbergRichter gr, double mMax) { - return Mfd.newGutenbergRichterBuilder(gr.mMin(), mMax, gr.Δm(), gr.b()); + return Mfd.newGutenbergRichterBuilder(gr.b(), gr.Δm(), gr.mMin(), mMax); } /* Expected moment rate of a GR MFD. */ diff --git a/src/main/java/gov/usgs/earthquake/nshmp/model/MfdTrees.java b/src/main/java/gov/usgs/earthquake/nshmp/model/MfdTrees.java index 4c1a2516..5f765e57 100644 --- a/src/main/java/gov/usgs/earthquake/nshmp/model/MfdTrees.java +++ b/src/main/java/gov/usgs/earthquake/nshmp/model/MfdTrees.java @@ -91,7 +91,7 @@ class MfdTrees { */ private static Mfd mMaxGrMfd(Mfd.Properties props, double mMax) { GutenbergRichter gr = props.getAsGr(); - return Mfd.newGutenbergRichterBuilder(gr.mMin(), mMax, gr.Δm(), gr.b()) + return Mfd.newGutenbergRichterBuilder(gr.b(), gr.Δm(), gr.mMin(), mMax) .transform(p -> p.set(p.x() > gr.mMax() ? 0.0 : p.y())) .build(); } @@ -113,7 +113,7 @@ class MfdTrees { double[] weights = new double[mfdTree.size()]; for (int i = 0; i < mfdTree.size(); i++) { Branch<Mfd.Properties> mfd = mfdTree.get(i); - magnitudes[i] = mfd.value().getAsSingle().m(); + magnitudes[i] = mfd.value().getAsSingle().magnitude(); weights[i] = mfd.weight(); } return Mfd.create(magnitudes, weights); diff --git a/src/test/java/gov/usgs/earthquake/nshmp/mfd/IncrementalMfdBuilderTest.java b/src/test/java/gov/usgs/earthquake/nshmp/mfd/IncrementalMfdBuilderTest.java deleted file mode 100644 index 894f499a..00000000 --- a/src/test/java/gov/usgs/earthquake/nshmp/mfd/IncrementalMfdBuilderTest.java +++ /dev/null @@ -1,141 +0,0 @@ -/** - * - */ -package gov.usgs.earthquake.nshmp.mfd; - -/** - * @author U.S. Geological Survey - * - */ -class IncrementalMfdBuilderTest { - - // /** - // * @throws java.lang.Exception - // */ - // @BeforeAll - // public static void setUpBeforeClass() throws Exception {} - // - // /** - // * @throws java.lang.Exception - // */ - // @AfterAll - // public static void tearDownAfterClass() throws Exception {} - // - // /** - // * @throws java.lang.Exception - // */ - // @Before - // public void setUp() throws Exception {} - // - // /** - // * @throws java.lang.Exception - // */ - // @After - // public void tearDown() throws Exception {} - // - // /** - // * Test method for {@link - // gov.usgs.earthquake.nshmp.mfd.IncrementalMfdBuilder#builder()}. - // */ - // @Test - // public final void testBuilder() { - // fail("Not yet implemented"); // TODO - // } - // - // /** - // * Test method for {@link java.lang.Object#Object()}. - // */ - // @Test - // public final void testObject() { - // fail("Not yet implemented"); // TODO - // } - // - // /** - // * Test method for {@link java.lang.Object#getClass()}. - // */ - // @Test - // public final void testGetClass() { - // fail("Not yet implemented"); // TODO - // } - // - // /** - // * Test method for {@link java.lang.Object#hashCode()}. - // */ - // @Test - // public final void testHashCode() { - // fail("Not yet implemented"); // TODO - // } - // - // /** - // * Test method for {@link java.lang.Object#equals(java.lang.Object)}. - // */ - // @Test - // public final void testEquals() { - // fail("Not yet implemented"); // TODO - // } - // - // /** - // * Test method for {@link java.lang.Object#clone()}. - // */ - // @Test - // public final void testClone() { - // fail("Not yet implemented"); // TODO - // } - // - // /** - // * Test method for {@link java.lang.Object#toString()}. - // */ - // @Test - // public final void testToString() { - // fail("Not yet implemented"); // TODO - // } - // - // /** - // * Test method for {@link java.lang.Object#notify()}. - // */ - // @Test - // public final void testNotify() { - // fail("Not yet implemented"); // TODO - // } - // - // /** - // * Test method for {@link java.lang.Object#notifyAll()}. - // */ - // @Test - // public final void testNotifyAll() { - // fail("Not yet implemented"); // TODO - // } - // - // /** - // * Test method for {@link java.lang.Object#wait(long)}. - // */ - // @Test - // public final void testWaitLong() { - // fail("Not yet implemented"); // TODO - // } - // - // /** - // * Test method for {@link java.lang.Object#wait(long, int)}. - // */ - // @Test - // public final void testWaitLongInt() { - // fail("Not yet implemented"); // TODO - // } - // - // /** - // * Test method for {@link java.lang.Object#wait()}. - // */ - // @Test - // public final void testWait() { - // fail("Not yet implemented"); // TODO - // } - // - // /** - // * Test method for {@link java.lang.Object#finalize()}. - // */ - // @Test - // public final void testFinalize() { - // fail("Not yet implemented"); // TODO - // } - -} diff --git a/src/test/java/gov/usgs/earthquake/nshmp/mfd/MfdTests.java b/src/test/java/gov/usgs/earthquake/nshmp/mfd/MfdTests.java index 24c1aeb6..4688fd36 100644 --- a/src/test/java/gov/usgs/earthquake/nshmp/mfd/MfdTests.java +++ b/src/test/java/gov/usgs/earthquake/nshmp/mfd/MfdTests.java @@ -1,12 +1,405 @@ package gov.usgs.earthquake.nshmp.mfd; +import static org.junit.jupiter.api.Assertions.assertArrayEquals; +import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.junit.jupiter.api.Assertions.assertTrue; + +import java.util.Arrays; + import org.junit.jupiter.api.Test; +import gov.usgs.earthquake.nshmp.data.XySequence; +import gov.usgs.earthquake.nshmp.mfd.Mfd.Properties; +import gov.usgs.earthquake.nshmp.mfd.Mfd.Properties.GutenbergRichter; +import gov.usgs.earthquake.nshmp.mfd.Mfd.Properties.Single; +import gov.usgs.earthquake.nshmp.mfd.Mfd.Properties.TaperedGr; +import gov.usgs.earthquake.nshmp.mfd.Mfd.Type; + class MfdTests { + static final double[] M = { 5.05, 5.15, 5.25, 5.35, 5.45 }; + static final double[] R = { 0.1, 0.08, 0.06, 0.04, 0.02 }; + + @Test + void testCreate() { + /* Covers array and xy constructors. */ + Mfd mfd = Mfd.create(M, R); + assertArrayEquals(M, mfd.data().xValues().toArray()); + assertArrayEquals(R, mfd.data().yValues().toArray()); + assertEquals(Type.INCR, mfd.properties().type()); + } + + static final double[] GAUSS_M = { + 6.76, + 6.808, + 6.856, + 6.904, + 6.952, + 7.0, + 7.048, + 7.096, + 7.144, + 7.192, + 7.24 }; + + static final double[] GAUSS_R = { + 0.4499247209432322, + 0.9243402889954612, + 1.618217124860106, + 2.414096273012355, + 3.068917835861028, + 3.324519003345273, + 3.068917835861028, + 2.414096273012355, + 1.618217124860106, + 0.9243402889954612, + 0.4499247209432322 }; + + @Test + void testBuilderFrom() { + // TODO + } + + @Test + void testSingle() { + + /* Factory builder; covers props.toBuilder() */ + Mfd mfd = Mfd.newSingleBuilder(6.0).build(); + XySequence xy = mfd.data(); + assertTrue(xy.size() == 1); + assertEquals(6.0, xy.min().x()); + assertEquals(1.0, xy.min().y()); + + /* Factory builder; covers props.toGaussianBuilder() */ + mfd = Mfd.newGaussianBuilder(7.0, 11, 0.12, 2).build(); + xy = mfd.data(); + assertTrue(xy.size() == 11); + System.out.println(xy); + assertArrayEquals(GAUSS_M, xy.xValues().toArray()); + assertArrayEquals(GAUSS_R, xy.yValues().toArray()); + + /* Properties */ + Properties props = new Single(5.0, 2.0); + Single singleProps = props.getAsSingle(); // cover getAs + assertEquals(5.0, singleProps.magnitude()); + assertEquals(2.0, singleProps.rate()); + singleProps = new Single(7.0); + assertEquals(7.0, singleProps.magnitude()); + assertEquals(1.0, singleProps.rate()); + + /* props.toString() */ + assertEquals( + singleProps.type() + + " {magnitude=" + singleProps.magnitude() + + ", rate=" + singleProps.rate() + "}", + singleProps.toString()); + + // TODO changing rate=0 to 1 is probably not correct + // and this check should/will go away + singleProps = new Single(7.0, 0.0); + mfd = singleProps.toBuilder().build(); + xy = mfd.data(); + assertEquals(7.0, xy.min().x()); + assertEquals(1.0, xy.min().y()); + } + + static final double[] GR_M = { + 5.05, + 5.15, + 5.25, + 5.35, + 5.45 }; + + static final double[] GR_R = { + 8.912509381337459E-5, + 7.079457843841373E-5, + 5.623413251903491E-5, + 4.466835921509635E-5, + 3.5481338923357534E-5 }; + + @Test + void testGutenbergRichter() { + + /* Factory builder; covers props.toBuilder() */ + Mfd mfd = Mfd.newGutenbergRichterBuilder(1.0, 0.1, 5.0, 5.5).build(); + XySequence xy = mfd.data(); + assertTrue(xy.size() == 5); + assertArrayEquals(GR_M, xy.xValues().toArray()); + assertArrayEquals(GR_R, xy.yValues().toArray()); + + /* Properties */ + Properties props = new GutenbergRichter(1.0, 1.0, 0.1, 5.0, 5.5); + GutenbergRichter grProps = props.getAsGr(); // cover getAs + assertEquals(1.0, grProps.a()); + assertEquals(1.0, grProps.b()); + assertEquals(0.1, grProps.Δm()); + assertEquals(5.0, grProps.mMin()); + assertEquals(5.5, grProps.mMax()); + + /* props.toString() */ + assertEquals( + grProps.type() + " {a=" + grProps.a() + + ", b=" + grProps.b() + + ", Δm=" + grProps.Δm() + + ", mMin=" + grProps.mMin() + + ", mMax=" + grProps.mMax() + "}", + grProps.toString()); + } + + /* + * TODO something in the TaperedGR moment based scaling leads to differences + * in the 4th to 5th significant figure that should be better understood, and + * then this main method cleaned out. The results are still pretty close + * overall. + */ + public static void main(String[] args) { + double bVal = 1.0; + double aVal = Math.log10(150.0); + double Δm = 0.1; + double mMin = 5.0; + double mMax = 8.0; + double mc = 7.5; + + /* GR direct vs scaled builder comparison */ + System.out.println("Gutenberg Richter"); + Mfd mfd = new GutenbergRichter(aVal, bVal, Δm, mMin, mMax) + .toBuilder() + .build(); + + System.out.println(Arrays.toString(mfd.data().xValues().toArray())); + System.out.println(Arrays.toString(mfd.data().yValues().toArray())); + + double incrRate = Mfds.gutenbergRichterRate(aVal, bVal, 5.05); + mfd = Mfd.newGutenbergRichterBuilder(bVal, Δm, mMin, mMax) + .scaleToIncrementalRate(incrRate) + .build(); + + System.out.println(Arrays.toString(mfd.data().yValues().toArray())); + + /* GR direct vs scaled builder comparison */ + System.out.println("Tapered Gutenberg Richter"); + mfd = new TaperedGr(aVal, bVal, Δm, mMin, mMax, mc) + .toBuilder() + .build(); + + System.out.println(Arrays.toString(mfd.data().xValues().toArray())); + System.out.println(Arrays.toString(mfd.data().yValues().toArray())); + + incrRate = Mfds.gutenbergRichterRate(aVal, bVal, 5.05); + mfd = Mfd.newTaperedGutenbergRichterBuilder(bVal, Δm, mMin, mMax, mc) + .scaleToIncrementalRate(incrRate) + .build(); + + System.out.println(Arrays.toString(mfd.data().yValues().toArray())); + + } + + private static final double[] TAPERED_GR_M = { + 5.05, 5.15, 5.25, 5.35, 5.45, + 5.55, 5.65, 5.75, 5.85, 5.95, + 6.05, 6.15, 6.25, 6.35, 6.45, + 6.55, 6.65, 6.75, 6.85, 6.95, + 7.05, 7.15, 7.25, 7.35, 7.45, + 7.55, 7.65, 7.75, 7.85, 7.95 }; + + /* using scaleToInrementalRate */ + private static final double[] TAPERED_GR_R_SCALED = { + 3.6480433574236396E-4, + 3.0345403881799134E-4, + 2.5242909958337484E-4, + 2.099931150361607E-4, + 1.7470192588897718E-4, + 1.4535446859752395E-4, + 1.2095189730212225E-4, + 1.0066358352203966E-4, + 8.37988347826301E-5, + 6.978336683158799E-5, + 5.813972403773392E-5, + 4.847097401109892E-5, + 4.044710933063091E-5, + 3.37936743657631E-5, + 2.8282200517277984E-5, + 2.372208037314646E-5, + 1.9953542732997602E-5, + 1.684141264491656E-5, + 1.4269370950942086E-5, + 1.213450863835374E-5, + 1.034219228924177E-5, + 8.801777631086678E-6, + 7.4247305061395814E-6, + 6.128255653142835E-6, + 4.84871327802538E-6, + 3.5668582219695296E-6, + 2.3358516405761174E-6, + 1.2823409762916182E-6, + 5.437829885686744E-7, + 1.5949675112240326E-7 }; + + /* using a value directly from properties */ + private static final double[] TAPERED_GR_R_DIRECT = { + 3.648734771264746E-4, + 3.0351155247723867E-4, + 2.5247694248332143E-4, + 2.100329150418206E-4, + 1.7473503715378214E-4, + 1.4538201763727166E-4, + 1.2097482132130438E-4, + 1.006826622960901E-4, + 8.381471718000331E-5, + 6.979659287661517E-5, + 5.815074326255646E-5, + 4.848016071724236E-5, + 4.0454775273297684E-5, + 3.3800079282566184E-5, + 2.8287560844165223E-5, + 2.3726576420233308E-5, + 1.9957324528955875E-5, + 1.684460459870263E-5, + 1.4272075425536471E-5, + 1.213680849238659E-5, + 1.034415244546678E-5, + 8.803445832444015E-6, + 7.426137715686032E-6, + 6.1294171417451564E-6, + 4.849632254896958E-6, + 3.5675342487878286E-6, + 2.3362943546550987E-6, + 1.2825840184413838E-6, + 5.438860517858616E-7, + 1.5952698054966957E-7 }; + + @Test + void testTaperedGr() { + + /* Factory builder; covers props.toBuilder() */ + double incrRate = Mfds.gutenbergRichterRate(Math.log10(4.0), 0.8, 5.05); + Mfd mfd = Mfd.newTaperedGutenbergRichterBuilder(0.8, 0.1, 5.0, 8.0, 7.5) + .scaleToIncrementalRate(incrRate) + .build(); + XySequence xy = mfd.data(); + assertTrue(xy.size() == 30); + + System.out.println(Arrays.toString(xy.xValues().toArray())); + System.out.println(Arrays.toString(xy.yValues().toArray())); + + assertArrayEquals(TAPERED_GR_M, xy.xValues().toArray()); + assertArrayEquals(TAPERED_GR_R_SCALED, xy.yValues().toArray()); + + /* Properties */ + double aValue = Math.log10(4.0); + Properties props = new TaperedGr(aValue, 0.8, 0.1, 5.0, 8.0, 7.5); + TaperedGr grProps = props.getAsGrTaper(); // cover getAs + assertEquals(aValue, grProps.a()); + assertEquals(0.8, grProps.b()); + assertEquals(0.1, grProps.Δm()); + assertEquals(5.0, grProps.mMin()); + assertEquals(8.0, grProps.mMax()); + assertEquals(7.5, grProps.mc()); + assertTrue(xy.size() == 30); + mfd = props.toBuilder().build(); + xy = mfd.data(); + assertArrayEquals(TAPERED_GR_M, xy.xValues().toArray()); + assertArrayEquals(TAPERED_GR_R_DIRECT, xy.yValues().toArray()); + + /* props.toString() */ + assertEquals( + grProps.type() + " {a=" + grProps.a() + + ", b=" + grProps.b() + + ", Δm=" + grProps.Δm() + + ", mMin=" + grProps.mMin() + + ", mMax=" + grProps.mMax() + + ", mc=" + grProps.mc() + "}", + grProps.toString()); + } + + private static final double[] GR_M_TRANDSORM = TAPERED_GR_M; + + private static final double[] GR_R_TRANSFORM = { + 0.0013368764072006192, + 0.0010619186765762063, + 8.435119877855239E-4, + 6.700253882264454E-4, + 5.322200838503631E-4, + 4.2275743968966836E-4, + 3.3580817078525074E-4, + 2.667419115058385E-4, + 2.1188063169341338E-4, + 1.683027681452945E-4, + 1.3368764072006192E-4, + 1.0619186765762062E-4, + 8.435119877855238E-5, + 6.700253882264454E-5, + 5.322200838503631E-5, + 4.227574396896683E-5, + 3.358081707852507E-5, + 2.667419115058385E-5, + 2.1188063169341338E-5, + 1.683027681452945E-5, + 1.3368764072006191E-5, + 1.0619186765762062E-5, + 8.435119877855238E-6, + 6.700253882264454E-6, + 5.322200838503631E-6, + 4.227574396896683E-6, + 3.3580817078525076E-6, + 2.6674191150583848E-6, + 2.1188063169341338E-6, + 1.683027681452945E-6 }; + @Test - void test() { - // fail("Not yet implemented"); + void testBuilderTransforms() { + + /* tolerances set based on experimentaion with each transform */ + + double bVal = 1.0; + double aVal = Math.log10(150.0); + double Δm = 0.1; + double mMin = 5.0; + double mMax = 8.0; + + Mfd mfd = new GutenbergRichter(aVal, bVal, Δm, mMin, mMax) + .toBuilder() + .build(); + + /* Scale to incremental */ + double incrRate = Mfds.gutenbergRichterRate(aVal, bVal, 5.05); + Mfd scaleIncrMfd = Mfd.newGutenbergRichterBuilder(bVal, Δm, mMin, mMax) + .scaleToIncrementalRate(incrRate) + .build(); + + assertArrayEquals( + mfd.data().yValues().toArray(), + scaleIncrMfd.data().yValues().toArray(), + 1e-18); + + /* Scale to cumulative */ + double cumulativeRateExpected = mfd.data().yValues().sum() * 2.0; + Mfd scaleCumulativeMfd = Mfd.newGutenbergRichterBuilder(bVal, Δm, mMin, mMax) + .scaleToCumulativeRate(cumulativeRateExpected) + .build(); + double cumulativeRateActual = scaleCumulativeMfd.data().yValues().sum(); + assertEquals(cumulativeRateExpected, cumulativeRateActual, 1e-17); + + /* Scale to moment */ + double momentRateExpected = Mfds.momentRate(mfd.data()) * 2.0; + Mfd scaleMomentMfd = Mfd.newGutenbergRichterBuilder(bVal, Δm, mMin, mMax) + .scaleToMomentRate(momentRateExpected) + .build(); + double momentRateActual = Mfds.momentRate(scaleMomentMfd.data()); + /* very small delta relative to 1e16 moment rate values */ + assertEquals(momentRateExpected, momentRateActual, 10); + + /* Transform */ + double transformRateExpected = cumulativeRateExpected; + Mfd transformMfd = new GutenbergRichter(aVal, bVal, Δm, mMin, mMax) + .toBuilder() + .transform(xy -> xy.set(xy.y() * 2.0)) + .build(); + double transformRateActual = transformMfd.data().yValues().sum(); + System.out.println(transformRateExpected); + System.out.println(transformRateActual); + /* these values happen to match exactly */ + assertEquals(transformRateExpected, transformRateActual); } } diff --git a/src/test/java/gov/usgs/earthquake/nshmp/mfd/Mfds2Test.java b/src/test/java/gov/usgs/earthquake/nshmp/mfd/Mfds2Test.java deleted file mode 100644 index 50e6a8a2..00000000 --- a/src/test/java/gov/usgs/earthquake/nshmp/mfd/Mfds2Test.java +++ /dev/null @@ -1,44 +0,0 @@ -/** - * - */ -package gov.usgs.earthquake.nshmp.mfd; - -/** - * @author U.S. Geological Survey - * - */ -class Mfds2Test { - - // /** - // * @throws java.lang.Exception - // */ - // @BeforeAll - // public static void setUpBeforeClass() throws Exception {} - // - // /** - // * @throws java.lang.Exception - // */ - // @AfterAll - // public static void tearDownAfterClass() throws Exception {} - // - // /** - // * @throws java.lang.Exception - // */ - // @Before - // public void setUp() throws Exception {} - // - // /** - // * @throws java.lang.Exception - // */ - // @After - // public void tearDown() throws Exception {} - // - // /** - // * Test method for {@link gov.usgs.earthquake.nshmp.mfd.Mfds2#builder()}. - // */ - // @Test - // public final void testBuilder() { - // fail("Not yet implemented"); // TODO - // } - -} diff --git a/src/test/java/gov/usgs/earthquake/nshmp/mfd/MfdsTests.java b/src/test/java/gov/usgs/earthquake/nshmp/mfd/MfdsTests.java index cbcb65f6..20571e21 100644 --- a/src/test/java/gov/usgs/earthquake/nshmp/mfd/MfdsTests.java +++ b/src/test/java/gov/usgs/earthquake/nshmp/mfd/MfdsTests.java @@ -1,68 +1,43 @@ package gov.usgs.earthquake.nshmp.mfd; -class MfdsTests { +import static org.junit.jupiter.api.Assertions.assertDoesNotThrow; +import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.junit.jupiter.api.Assertions.assertThrows; + +import java.util.stream.DoubleStream; - private static final double MFD_TOL = 1e-10; +import org.junit.jupiter.api.Test; - // @Test - // final void testTaperedGR() { - // // IncrementalMfd tGR = Mfds.newTaperedGutenbergRichterMFD(5.05, 0.1, 30, - // // 4.0, 0.8, 7.5, 1.0); - // - // double incrRate = Mfds.incrRate(4.0, 0.8, 5.05); - // XySequence tGR = Mfd.newTaperedGutenbergRichterMfd(5.05, 7.95, 0.1, 0.8, - // 7.5) - // .scaleToIncrementalRate(incrRate) - // .build(); - // assertArrayEquals(TAPERED_GR_MAGS, tGR.xValues().toArray(), MFD_TOL); - // assertArrayEquals(TAPERED_GR_RATES, tGR.yValues().toArray(), MFD_TOL); - // } +import gov.usgs.earthquake.nshmp.mfd.Mfd.Type; + +class MfdsTests { - public static void main(String[] args) { - // TODO clean - // IncrementalMfd tGR = Mfd.newTaperedGutenbergRichterMFD(5.05, 0.1, 30, - // 4.0, 0.8, 7.5, 1.0); - // System.out.println(tGR.xValues()); - // System.out.println(tGR.yValues()); - // for (Point2D p : tGR) { - // System.out.println(p.getX() + " " + p.getY()); - // } + @Test + void testCheckRate() { + assertEquals(1.0, Mfds.checkRate(1.0)); + assertThrows(IllegalArgumentException.class, () -> { + Mfds.checkRate(-1.0); + }); } - private static final double[] TAPERED_GR_MAGS = { - 5.05, 5.15, 5.25, 5.35, 5.45, 5.55, 5.65, - 5.75, 5.85, 5.95, 6.05, 6.15, 6.25, 6.35, 6.45, 6.55, 6.65, 6.75, 6.85, 6.95, 7.05, 7.15, - 7.25, 7.35, 7.45, 7.55, 7.65, 7.75, 7.85, 7.95 }; + @Test + void testValues() { + assertThrows(IllegalArgumentException.class, () -> { + Mfds.checkValues(DoubleStream.of(-5.0), DoubleStream.of(1.0)); + }); + assertThrows(IllegalArgumentException.class, () -> { + Mfds.checkValues(DoubleStream.of(5.0), DoubleStream.of(-1.0)); + }); + assertDoesNotThrow(() -> { + Mfds.checkValues(DoubleStream.of(5.0), DoubleStream.of(1.0)); + }); + } + + @Test + void testPropsToString() { + assertEquals( + Type.SINGLE.name() + " {dummy}", + Mfds.propsToString(Type.SINGLE, "dummy")); + } - private static final double[] TAPERED_GR_RATES = { - 3.6487347712647455E-4, - 3.0351155247723856E-4, - 2.524769424833213E-4, - 2.1003291504182055E-4, - 1.747350371537821E-4, - 1.4538201763727163E-4, - 1.2097482132130435E-4, - 1.0068266229609008E-4, - 8.38147171800033E-5, - 6.979659287661515E-5, - 5.815074326255645E-5, - 4.848016071724235E-5, - 4.0454775273297684E-5, - 3.380007928256618E-5, - 2.828756084416522E-5, - 2.37265764202333E-5, - 1.995732452895587E-5, - 1.6844604598702625E-5, - 1.4272075425536466E-5, - 1.2136808492386587E-5, - 1.0344152445466779E-5, - 8.803445832444012E-6, - 7.426137715686029E-6, - 6.129417141745154E-6, - 4.849632254896957E-6, - 3.5675342487878273E-6, - 2.3362943546550987E-6, - 1.2825840184413836E-6, - 5.438860517858615E-7, - 1.5952698054966954E-7 }; } -- GitLab