All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.gridkit.jvmtool.stacktrace.analytics.TimeRangeChecker Maven / Gradle / Ivy

package org.gridkit.jvmtool.stacktrace.analytics;

import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Arrays;
import java.util.Calendar;
import java.util.Date;
import java.util.TimeZone;
import java.util.regex.Matcher;
import java.util.regex.Pattern;

public class TimeRangeChecker {

    private static String DATE_FORMAT = "yyyy.MM.dd_HH:mm:ss";
    
    private static String DATE_PATTERN = "(((((((\\d\\d)?\\d\\d[.])?\\d\\d[.])?\\d\\d[_])?\\d\\d[:])?\\d\\d[:])\\d\\d)";
    
    private long[] whiteCache = new long[0];
    private long[] blackCache = new long[0];
    private int[] lowerBound;
    private int[] upperBound;
    private TimeZone tz;
    private boolean inverted = false;
    
    public TimeRangeChecker(String lower, String upper, TimeZone tz) {
        lowerBound = parse(lower);
        upperBound = parse(upper);
        this.tz = tz;
        if (tz == null) {
            throw new NullPointerException("tz is null");
        }
        
        for(int i = 0; i != lowerBound.length; ++i) {
            if ((lowerBound[i] == -1 && upperBound[i] != -1) || (lowerBound[i] != -1 && upperBound[i] == -1)) {
                throw new IllegalArgumentException("Bounds length mismatch '" + lower + "', '" + upper + "'");
            }
            else {
                if (lowerBound[i] == upperBound[i]) {
                    continue;
                }
                if (lowerBound[i] > upperBound[i]) {
                    inverted = true;
                }
                break;
            }
        }
    }
    
    private int[] parse(String bound) {
        try {
            Matcher m = Pattern.compile(DATE_PATTERN).matcher(bound);
            if (!m.matches()) {
                throw new IllegalArgumentException("Cannot parser time range bound '" + bound + "'");
            }
            int[] p = new int[7];
            Arrays.fill(p, -1);
            if (m.group(7) != null) {
                p[0] = Integer.parseInt(m.group(7));
            }
            if (m.group(6) != null) {
                p[1] = Integer.parseInt(substring(m.group(6), 2, 1));            
            }
            if (m.group(5) != null) {
                p[2] = Integer.parseInt(substring(m.group(5), 2, 1));            
            }
            if (m.group(4) != null) {
                p[3] = Integer.parseInt(substring(m.group(4), 2, 1));            
            }
            if (m.group(3) != null) {
                p[4] = Integer.parseInt(substring(m.group(3), 2, 1));            
            }
            if (m.group(2) != null) {
                p[5] = Integer.parseInt(substring(m.group(2), 2, 1));            
            }
            if (m.group(1) != null) {
                p[6] = Integer.parseInt(substring(m.group(1), 2, 0));            
            }
            return p;
        }
        catch(NumberFormatException e) {
            throw new IllegalArgumentException("Cannot parser time range bound '" + bound + "'");
        }
    }

    private String substring(String txt, int n, int m) {
        return txt.substring(txt.length() - n -m, txt.length() - m);
    }
    
    // Exposed for testing
    boolean isCached(long timestamp) {
        return match(whiteCache, timestamp) || match(blackCache, timestamp);
    }

    public boolean evaluate(long timestamp) {
        if (match(whiteCache, timestamp)) {
            return true;
        }
        if (match(blackCache, timestamp)) {
            return false;
        }
        
        SimpleDateFormat fmt = new SimpleDateFormat(DATE_FORMAT);
        fmt.setTimeZone(tz);
        String date = fmt.format(timestamp);
        
        int[] p = parse(date);
        int[] rl = Arrays.copyOf(p, p.length);
        int[] ru = Arrays.copyOf(p, p.length);
        
        boolean match = match(p, lowerBound, upperBound);

        for(int j = 0; j != rl.length; ++j) {
            if (lowerBound[j] != -1) {
                rl[j] = lowerBound[j];
                ru[j] = upperBound[j];
            }
        }
        if (match) {
            // match
            long ldate = toDate(rl);
            long udate = toDate(ru);
            if (ldate == -1 || udate == -1) {
                // do not cache
                return true;
            }
            if (udate > ldate) {
                whiteCache = add(whiteCache, ldate, udate);
            }
            else {
                if (timestamp >= ldate && timestamp >= udate) {
                    udate = adjust(udate, 1);
                    whiteCache = add(whiteCache, ldate, udate);
                }
                else {
                    ldate = adjust(ldate, -1);
                    whiteCache = add(whiteCache, ldate, udate);
                }
            }
            return true;
        }
        else {
            // do not match
            long ldate = toDate(rl);
            long udate = toDate(ru);
            if (ldate == -1 || udate == -1) {
                // do not cache
                return false;
            }
            if (timestamp >= ldate && timestamp >= udate) {
                ldate = adjust(ldate, 1);
                blackCache = add(blackCache, udate, ldate);
            }
            else {
                udate = adjust(udate, -1);
                blackCache = add(blackCache, udate, ldate);
            }
            return false;
        }        
    }

    private boolean match(int[] p, int[] lower, int[] upper) {
        boolean matchUpper = true;
        boolean matchLower = true;
        for(int i = 0; i != lowerBound.length; ++i) {
            if (lowerBound[i] != -1) {
                if (checkInRange(p, i, matchLower, matchUpper)) {                    
                    if (upperBound[i] != p[i]) {
                        matchUpper = false;
                    }
                    if (lowerBound[i] != p[i]) {
                        matchLower = false;
                    }
                    if (!matchLower && !matchUpper) {
                        return true;
                    }
                    continue;
                }
                else {
                    return false;
                }
            }
        }
        return !matchUpper;
    }

    private boolean checkInRange(int[] p, int i, boolean matchLower, boolean matchUpper) {
        if (inverted) {
            return (((matchLower) && lowerBound[i] <= p[i]) || ((matchUpper) && upperBound[i] >= p[i]));
        }
        else {
            return ((!matchLower) || lowerBound[i] <= p[i]) && ((!matchUpper) || upperBound[i] >= p[i]);
        }
    }

    private static int[] FIELDS = {Calendar.YEAR, Calendar.YEAR, Calendar.MONTH, Calendar.DAY_OF_MONTH, Calendar.HOUR_OF_DAY, Calendar.MINUTE, Calendar.SECOND};
    
    private long adjust(long ldate, int step) {
        Calendar cal = Calendar.getInstance(tz);
        cal.setTime(new Date(ldate));
        int n = -1;
        for(int i = 0; i != lowerBound.length; ++i) {
            if (lowerBound[i] == -1) {
                n = i;
            }
        }
        if (n == -1) {
            cal.add(Calendar.YEAR, 100 * step);
        }
        else if (n == 0) {
            cal.add(Calendar.YEAR, 10 * step);
        }
        else {
            cal.add(FIELDS[n], step);
        }
        return cal.getTimeInMillis();        
    }

    private long[] add(long[] cache, long ldate, long udate) {
//        SimpleDateFormat fmt = new SimpleDateFormat(DATE_FORMAT);
//        fmt.setTimeZone(tz);
//        String l = fmt.format(ldate);
//        String u = fmt.format(udate);
        long[] ncache = Arrays.copyOf(cache, cache.length + 2);
        ncache[cache.length] = ldate;
        ncache[cache.length + 1] = udate;
        return ncache;
    }

    private long toDate(int[] r) {
        try {
            Object[] args = new Object[7];
            for(int i = 0; i != 7; ++i) {
                args[i] = r[i];
            }
            SimpleDateFormat fmt = new SimpleDateFormat(DATE_FORMAT);
            fmt.setTimeZone(tz);
            String date = String.format("%02d%02d.%02d.%02d_%02d:%02d:%02d", args);
            return fmt.parse(date).getTime();
        } catch (ParseException e) {
            return -1;
        }
    }

    private boolean match(long[] cache, long timestamp) {
        int n = 0;
        while(n < cache.length) {
            if (cache[n] <= timestamp && cache[n + 1] > timestamp) {
                return true;
            }
            n += 2;
        }
        return false;
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy