001/*
002 * #%L
003 * HAPI FHIR - Core Library
004 * %%
005 * Copyright (C) 2014 - 2025 Smile CDR, Inc.
006 * %%
007 * Licensed under the Apache License, Version 2.0 (the "License");
008 * you may not use this file except in compliance with the License.
009 * You may obtain a copy of the License at
010 *
011 * http://www.apache.org/licenses/LICENSE-2.0
012 *
013 * Unless required by applicable law or agreed to in writing, software
014 * distributed under the License is distributed on an "AS IS" BASIS,
015 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
016 * See the License for the specific language governing permissions and
017 * limitations under the License.
018 * #L%
019 */
020package ca.uhn.fhir.util;
021
022import ca.uhn.fhir.i18n.HapiLocalizer;
023import ca.uhn.fhir.i18n.Msg;
024
025import java.util.Locale;
026import java.util.TimeZone;
027import java.util.concurrent.Callable;
028import java.util.concurrent.atomic.AtomicInteger;
029
030import static org.apache.commons.lang3.StringUtils.defaultString;
031
032public class TestUtil {
033        private static final org.slf4j.Logger ourLog = org.slf4j.LoggerFactory.getLogger(TestUtil.class);
034
035        private static SleepUtil ourSleepUtil = new SleepUtil();
036
037        private static boolean ourShouldRandomizeTimezones = true;
038
039        public static void setShouldRandomizeTimezones(boolean theShouldRandomizeTimezones) {
040                ourShouldRandomizeTimezones = theShouldRandomizeTimezones;
041        }
042
043        /**
044         * <b>THIS IS FOR UNIT TESTS ONLY - DO NOT CALL THIS METHOD FROM USER CODE</b>
045         * <p>
046         * When we run the unit tests in cobertura, JUnit doesn't seem to clean up static fields which leads to
047         * tons of memory being used by the end and the JVM crashes in Travis. Manually clearing all of the
048         * static fields seems to solve this.
049         */
050        public static void randomizeLocaleAndTimezone() {
051                HapiLocalizer.setOurFailOnMissingMessage(true);
052
053                doRandomizeLocaleAndTimezone();
054        }
055
056        /**
057         * Set some system properties randomly after each test.. this is kind of hackish,
058         * but it helps us make sure we don't have any tests that depend on a particular
059         * environment
060         */
061        public static void doRandomizeLocaleAndTimezone() {
062                //              Locale[] availableLocales = {Locale.CANADA, Locale.GERMANY, Locale.TAIWAN};
063                Locale[] availableLocales = {Locale.US};
064                Locale.setDefault(availableLocales[(int) (Math.random() * availableLocales.length)]);
065                ourLog.info("Tests are running in locale: " + Locale.getDefault().getDisplayName());
066                if (Math.random() < 0.5) {
067                        ourLog.info("Tests are using WINDOWS line endings and ISO-8851-1");
068                        System.setProperty("file.encoding", "ISO-8859-1");
069                        System.setProperty("line.separator", "\r\n");
070                } else {
071                        ourLog.info("Tests are using UNIX line endings and UTF-8");
072                        System.setProperty("file.encoding", "UTF-8");
073                        System.setProperty("line.separator", "\n");
074                }
075
076                if (ourShouldRandomizeTimezones) {
077                        String availableTimeZones[] = {"GMT+08:00", "GMT-05:00", "GMT+00:00", "GMT+03:30"};
078                        String timeZone = availableTimeZones[(int) (Math.random() * availableTimeZones.length)];
079                        TimeZone.setDefault(TimeZone.getTimeZone(timeZone));
080                }
081
082                ourLog.info("Tests are using time zone: {}", TimeZone.getDefault().getID());
083        }
084
085        /**
086         * <b>THIS IS FOR UNIT TESTS ONLY - DO NOT CALL THIS METHOD FROM USER CODE</b>
087         * <p>
088         * Wait for an atomicinteger to hit a given site and fail if it never does
089         */
090        public static void waitForSize(int theTarget, AtomicInteger theInteger) {
091                long start = System.currentTimeMillis();
092                while (theInteger.get() != theTarget && (System.currentTimeMillis() - start) <= 15000) {
093                        try {
094                                Thread.sleep(50);
095                        } catch (InterruptedException theE) {
096                                throw new Error(Msg.code(1778) + theE);
097                        }
098                }
099                if ((System.currentTimeMillis() - start) >= 15000) {
100                        throw new IllegalStateException(Msg.code(1779) + "Size " + theInteger.get() + " is != target " + theTarget);
101                }
102        }
103
104        /**
105         * <b>THIS IS FOR UNIT TESTS ONLY - DO NOT CALL THIS METHOD FROM USER CODE</b>
106         * <p>
107         * Wait for an atomicinteger to hit a given site and fail if it never does
108         */
109        public static void waitForSize(int theTarget, Callable<Integer> theSource) throws Exception {
110                long start = System.currentTimeMillis();
111                while (theSource.call() != theTarget && (System.currentTimeMillis() - start) <= 15000) {
112                        try {
113                                Thread.sleep(50);
114                        } catch (InterruptedException theE) {
115                                throw new Error(Msg.code(1780) + theE);
116                        }
117                }
118                if ((System.currentTimeMillis() - start) >= 15000) {
119                        throw new IllegalStateException(Msg.code(1781) + "Size " + theSource.call() + " is != target " + theTarget);
120                }
121        }
122
123        /**
124         * <b>THIS IS FOR UNIT TESTS ONLY - DO NOT CALL THIS METHOD FROM USER CODE</b>
125         * <p>
126         * Strip \r chars from a string to account for line ending platform differences
127         */
128        public static String stripReturns(String theString) {
129                return defaultString(theString).replace("\r", "");
130        }
131
132        /**
133         * <b>THIS IS FOR UNIT TESTS ONLY - DO NOT CALL THIS METHOD FROM USER CODE</b>
134         * <p>
135         * Strip \r chars from a string to account for line ending platform differences
136         */
137        public static String stripWhitespace(String theString) {
138                return stripReturns(theString).replace(" ", "");
139        }
140
141        /**
142         *
143         * In production code, instead of this static method, it is better to use an instance of SleepUtil.
144         * Since SleepUtil isn't using static methods, it is easier to mock for unit test and avoid unnecessary waits in
145         * unit tests
146         */
147        public static void sleepAtLeast(long theMillis) {
148                ourSleepUtil.sleepAtLeast(theMillis);
149        }
150
151        /**
152         * In production code, instead of this static method, it is better to use an instance of SleepUtil.
153         * Since SleepUtil isn't using static methods, it is easier to mock for unit test and avoid unnecessary waits in
154         * unit tests
155         */
156        public static void sleepAtLeast(long theMillis, boolean theLogProgress) {
157                ourSleepUtil.sleepAtLeast(theMillis, theLogProgress);
158        }
159}