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 com.google.common.annotations.VisibleForTesting;
023import org.apache.commons.lang3.Validate;
024import org.apache.commons.lang3.time.DateUtils;
025
026import java.text.DecimalFormat;
027import java.text.NumberFormat;
028import java.util.Date;
029import java.util.LinkedList;
030import java.util.concurrent.TimeUnit;
031
032/**
033 * A multipurpose stopwatch which can be used to time tasks and produce
034 * human readable output about task duration, throughput, estimated task completion,
035 * etc.
036 * <p>
037 * <p>
038 * <b>Thread Safety Note: </b> StopWatch is not intended to be thread safe.
039 * </p>
040 *
041 * @since HAPI FHIR 3.3.0
042 */
043public class StopWatch {
044
045        // TODO KHS it is risky for this to be a static field.  Safer to make it non-static, but that will require
046        // TODO KHS significant rework of StopWatchTest
047        private static Long ourNowForUnitTest;
048        private long myStarted = now();
049        private TaskTiming myCurrentTask;
050        private LinkedList<TaskTiming> myTasks;
051
052        /**
053         * Constructor
054         */
055        public StopWatch() {
056                super();
057        }
058
059        /**
060         * Constructor
061         *
062         * @param theStart The time to record as the start for this timer
063         */
064        public StopWatch(Date theStart) {
065                myStarted = theStart.getTime();
066        }
067
068        /**
069         * Constructor
070         *
071         * @param theStart The time that the stopwatch was started
072         */
073        public StopWatch(long theStart) {
074                myStarted = theStart;
075        }
076
077        private void addNewlineIfContentExists(StringBuilder theB) {
078                if (theB.length() > 0) {
079                        theB.append("\n");
080                }
081        }
082
083        /**
084         * Finish the counter on the current task (which was started by calling
085         * {@link #startTask(String)}. This method has no effect if no task
086         * is currently started so it's ok to call it more than once.
087         */
088        public void endCurrentTask() {
089                ensureTasksListExists();
090                if (myCurrentTask != null) {
091                        myCurrentTask.setEnd(now());
092                }
093                myCurrentTask = null;
094        }
095
096        private void ensureTasksListExists() {
097                if (myTasks == null) {
098                        myTasks = new LinkedList<>();
099                }
100        }
101
102        /**
103         * Returns a nice human-readable display of the time taken per
104         * operation. Note that this may not actually output the number
105         * of milliseconds if the time taken per operation was very long (over
106         * 10 seconds)
107         *
108         * @see #formatMillis(long)
109         */
110        public String formatMillisPerOperation(long theNumOperations) {
111                double millisPerOperation = (((double) getMillis()) / Math.max(1.0, theNumOperations));
112                return formatMillis(millisPerOperation);
113        }
114
115        /**
116         * Returns a string providing the durations of all tasks collected by {@link #startTask(String)}
117         */
118        public String formatTaskDurations() {
119
120                ensureTasksListExists();
121                StringBuilder b = new StringBuilder();
122
123                if (myTasks.size() > 0) {
124                        long delta = myTasks.getFirst().getStart() - myStarted;
125                        if (delta > 10) {
126                                addNewlineIfContentExists(b);
127                                b.append("Before first task");
128                                b.append(": ");
129                                b.append(formatMillis(delta));
130                        }
131                } else {
132                        b.append("No tasks");
133                }
134
135                TaskTiming last = null;
136                for (TaskTiming nextTask : myTasks) {
137
138                        if (last != null) {
139                                long delta = nextTask.getStart() - last.getEnd();
140                                if (delta > 10) {
141                                        addNewlineIfContentExists(b);
142                                        b.append("Between");
143                                        b.append(": ");
144                                        b.append(formatMillis(delta));
145                                }
146                        }
147
148                        addNewlineIfContentExists(b);
149                        b.append(nextTask.getTaskName());
150                        b.append(": ");
151                        long delta = nextTask.getMillis();
152                        b.append(formatMillis(delta));
153
154                        last = nextTask;
155                }
156
157                if (myTasks.size() > 0) {
158                        long delta = now() - myTasks.getLast().getEnd();
159                        if (delta > 10) {
160                                addNewlineIfContentExists(b);
161                                b.append("After last task");
162                                b.append(": ");
163                                b.append(formatMillis(delta));
164                        }
165                }
166
167                return b.toString();
168        }
169
170        /**
171         * Determine the current throughput per unit of time (specified in theUnit)
172         * assuming that theNumOperations operations have happened.
173         * <p>
174         * For example, if this stopwatch has 2 seconds elapsed, and this method is
175         * called for theNumOperations=30 and TimeUnit=SECONDS,
176         * this method will return 15
177         * </p>
178         *
179         * @see #getThroughput(long, TimeUnit)
180         */
181        public String formatThroughput(long theNumOperations, TimeUnit theUnit) {
182                double throughput = getThroughput(theNumOperations, theUnit);
183                return formatThroughput(throughput);
184        }
185
186        /**
187         * Given an amount of something completed so far, and a total amount, calculates how long it will take for something to complete
188         *
189         * @param theCompleteToDate The amount so far
190         * @param theTotal          The total (must be higher than theCompleteToDate
191         * @return A formatted amount of time
192         */
193        public String getEstimatedTimeRemaining(double theCompleteToDate, double theTotal) {
194                double millis = getMillis();
195                return formatEstimatedTimeRemaining(theCompleteToDate, theTotal, millis);
196        }
197
198        /**
199         * Given an amount of something completed so far, and a total amount, calculates how long it will take for something to complete
200         *
201         * @param theCompleteToDate The amount so far
202         * @param theTotal          The total (must be higher than theCompleteToDate
203         * @return A formatted amount of time
204         */
205        public static String formatEstimatedTimeRemaining(double theCompleteToDate, double theTotal, double millis) {
206                long millisRemaining = (long) (((theTotal / theCompleteToDate) * millis) - millis);
207                return formatMillis(millisRemaining);
208        }
209
210        public long getMillis(Date theNow) {
211                return theNow.getTime() - myStarted;
212        }
213
214        public long getMillis() {
215                long now = now();
216                return now - myStarted;
217        }
218
219        public long getMillisAndRestart() {
220                long now = now();
221                long retVal = now - myStarted;
222                myStarted = now;
223                return retVal;
224        }
225
226        /**
227         * @param theNumOperations Ok for this to be 0, it will be treated as 1
228         */
229        public long getMillisPerOperation(long theNumOperations) {
230                return (long) (((double) getMillis()) / Math.max(1.0, theNumOperations));
231        }
232
233        public Date getStartedDate() {
234                return new Date(myStarted);
235        }
236
237        /**
238         * Determine the current throughput per unit of time (specified in theUnit)
239         * assuming that theNumOperations operations have happened.
240         * <p>
241         * For example, if this stopwatch has 2 seconds elapsed, and this method is
242         * called for theNumOperations=30 and TimeUnit=SECONDS,
243         * this method will return 15
244         * </p>
245         *
246         * @see #formatThroughput(long, TimeUnit)
247         */
248        public double getThroughput(long theNumOperations, TimeUnit theUnit) {
249                long millis = getMillis();
250                return getThroughput(theNumOperations, millis, theUnit);
251        }
252
253        public void restart() {
254                myStarted = now();
255        }
256
257        /**
258         * Starts a counter for a sub-task
259         * <p>
260         * <b>Thread Safety Note: </b> This method is not threadsafe! Do not use subtasks in a
261         * multithreaded environment.
262         * </p>
263         *
264         * @param theTaskName Note that if theTaskName is blank or empty, no task is started
265         */
266        public void startTask(String theTaskName) {
267                endCurrentTask();
268                Validate.notBlank(theTaskName, "Task name must not be blank");
269                myCurrentTask = new TaskTiming().setTaskName(theTaskName).setStart(now());
270                myTasks.add(myCurrentTask);
271        }
272
273        /**
274         * Formats value in an appropriate format. See {@link #formatMillis(long)}}
275         * for a description of the format
276         *
277         * @see #formatMillis(long)
278         */
279        @Override
280        public String toString() {
281                return formatMillis(getMillis());
282        }
283
284        /**
285         * Format a throughput number (output does not include units)
286         */
287        public static String formatThroughput(double throughput) {
288                return new DecimalFormat("0.0").format(throughput);
289        }
290
291        /**
292         * Calculate throughput
293         *
294         * @param theNumOperations The number of operations completed
295         * @param theMillisElapsed The time elapsed
296         * @param theUnit          The unit for the throughput
297         */
298        public static double getThroughput(long theNumOperations, long theMillisElapsed, TimeUnit theUnit) {
299                if (theNumOperations <= 0) {
300                        return 0.0f;
301                }
302                long millisElapsed = Math.max(1, theMillisElapsed);
303                long periodMillis = theUnit.toMillis(1);
304
305                double denominator = ((double) millisElapsed) / ((double) periodMillis);
306
307                double throughput = (double) theNumOperations / denominator;
308                if (throughput > theNumOperations) {
309                        throughput = theNumOperations;
310                }
311
312                return throughput;
313        }
314
315        private static NumberFormat getDayFormat() {
316                return new DecimalFormat("0.0");
317        }
318
319        private static NumberFormat getTenDayFormat() {
320                return new DecimalFormat("0");
321        }
322
323        private static NumberFormat getSubMillisecondMillisFormat() {
324                return new DecimalFormat("0.000");
325        }
326
327        /**
328         * Append a right-aligned and zero-padded numeric value to a `StringBuilder`.
329         */
330        static void appendRightAlignedNumber(
331                        StringBuilder theStringBuilder, String thePrefix, int theNumberOfDigits, long theValueToAppend) {
332                theStringBuilder.append(thePrefix);
333                if (theNumberOfDigits > 1) {
334                        int pad = (theNumberOfDigits - 1);
335                        for (long xa = theValueToAppend; xa > 9 && pad > 0; xa /= 10) {
336                                pad--;
337                        }
338                        for (int xa = 0; xa < pad; xa++) {
339                                theStringBuilder.append('0');
340                        }
341                }
342                theStringBuilder.append(theValueToAppend);
343        }
344
345        /**
346         * Formats a number of milliseconds for display (e.g.
347         * in a log file), tailoring the output to how big
348         * the value actually is.
349         * <p>
350         * Example outputs:
351         * </p>
352         * <ul>
353         * <li>133ms</li>
354         * <li>00:00:10.223</li>
355         * <li>1.7 days</li>
356         * <li>64 days</li>
357         * </ul>
358         */
359        public static String formatMillis(long theMillis) {
360                return formatMillis((double) theMillis);
361        }
362
363        /**
364         * Formats a number of milliseconds for display (e.g.
365         * in a log file), tailoring the output to how big
366         * the value actually is.
367         * <p>
368         * Example outputs:
369         * </p>
370         * <ul>
371         * <li>133ms</li>
372         * <li>00:00:10.223</li>
373         * <li>1.7 days</li>
374         * <li>64 days</li>
375         * </ul>
376         */
377        public static String formatMillis(double theMillis) {
378                StringBuilder buf = new StringBuilder(20);
379                if (theMillis > 0.0 && theMillis < 1.0) {
380                        buf.append(getSubMillisecondMillisFormat().format(theMillis));
381                        buf.append("ms");
382                } else if (theMillis < (10 * DateUtils.MILLIS_PER_SECOND)) {
383                        buf.append((int) theMillis);
384                        buf.append("ms");
385                } else if (theMillis >= DateUtils.MILLIS_PER_DAY) {
386                        double days = theMillis / DateUtils.MILLIS_PER_DAY;
387                        if (days >= 10) {
388                                buf.append(getTenDayFormat().format(days));
389                                buf.append(" days");
390                        } else if (days != 1.0f) {
391                                buf.append(getDayFormat().format(days));
392                                buf.append(" days");
393                        } else {
394                                buf.append(getDayFormat().format(days));
395                                buf.append(" day");
396                        }
397                } else {
398                        long millisAsLong = (long) theMillis;
399                        appendRightAlignedNumber(
400                                        buf, "", 2, ((millisAsLong % DateUtils.MILLIS_PER_DAY) / DateUtils.MILLIS_PER_HOUR));
401                        appendRightAlignedNumber(
402                                        buf, ":", 2, ((millisAsLong % DateUtils.MILLIS_PER_HOUR) / DateUtils.MILLIS_PER_MINUTE));
403                        appendRightAlignedNumber(
404                                        buf, ":", 2, ((millisAsLong % DateUtils.MILLIS_PER_MINUTE) / DateUtils.MILLIS_PER_SECOND));
405                        if (theMillis <= DateUtils.MILLIS_PER_MINUTE) {
406                                appendRightAlignedNumber(buf, ".", 3, (millisAsLong % DateUtils.MILLIS_PER_SECOND));
407                        }
408                }
409                return buf.toString();
410        }
411
412        private static long now() {
413                if (ourNowForUnitTest != null) {
414                        return ourNowForUnitTest;
415                }
416                return System.currentTimeMillis();
417        }
418
419        @VisibleForTesting
420        public static void setNowForUnitTest(Long theNowForUnitTest) {
421                ourNowForUnitTest = theNowForUnitTest;
422        }
423
424        private static class TaskTiming {
425                private long myStart;
426                private long myEnd;
427                private String myTaskName;
428
429                public long getEnd() {
430                        if (myEnd == 0) {
431                                return now();
432                        }
433                        return myEnd;
434                }
435
436                public TaskTiming setEnd(long theEnd) {
437                        myEnd = theEnd;
438                        return this;
439                }
440
441                public long getMillis() {
442                        return getEnd() - getStart();
443                }
444
445                public long getStart() {
446                        return myStart;
447                }
448
449                public TaskTiming setStart(long theStart) {
450                        myStart = theStart;
451                        return this;
452                }
453
454                public String getTaskName() {
455                        return myTaskName;
456                }
457
458                public TaskTiming setTaskName(String theTaskName) {
459                        myTaskName = theTaskName;
460                        return this;
461                }
462        }
463}