1 /*
2  * Copyright (C) 2006 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package android.util;
18 
19 import android.content.res.Resources;
20 import android.content.res.XmlResourceParser;
21 import android.os.SystemClock;
22 import android.text.format.DateUtils;
23 
24 import com.android.internal.util.XmlUtils;
25 
26 import org.xmlpull.v1.XmlPullParser;
27 import org.xmlpull.v1.XmlPullParserException;
28 
29 import java.io.IOException;
30 import java.io.PrintWriter;
31 import java.text.SimpleDateFormat;
32 import java.util.ArrayList;
33 import java.util.Calendar;
34 import java.util.Collection;
35 import java.util.Date;
36 import java.util.TimeZone;
37 
38 import libcore.util.ZoneInfoDB;
39 
40 /**
41  * A class containing utility methods related to time zones.
42  */
43 public class TimeUtils {
TimeUtils()44     /** @hide */ public TimeUtils() {}
45     private static final boolean DBG = false;
46     private static final String TAG = "TimeUtils";
47 
48     /** Cached results of getTineZones */
49     private static final Object sLastLockObj = new Object();
50     private static ArrayList<TimeZone> sLastZones = null;
51     private static String sLastCountry = null;
52 
53     /** Cached results of getTimeZonesWithUniqueOffsets */
54     private static final Object sLastUniqueLockObj = new Object();
55     private static ArrayList<TimeZone> sLastUniqueZoneOffsets = null;
56     private static String sLastUniqueCountry = null;
57 
58     /** {@hide} */
59     private static SimpleDateFormat sLoggingFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
60 
61     /**
62      * Tries to return a time zone that would have had the specified offset
63      * and DST value at the specified moment in the specified country.
64      * Returns null if no suitable zone could be found.
65      */
getTimeZone(int offset, boolean dst, long when, String country)66     public static TimeZone getTimeZone(int offset, boolean dst, long when, String country) {
67         TimeZone best = null;
68         final Date d = new Date(when);
69 
70         TimeZone current = TimeZone.getDefault();
71         String currentName = current.getID();
72         int currentOffset = current.getOffset(when);
73         boolean currentDst = current.inDaylightTime(d);
74 
75         for (TimeZone tz : getTimeZones(country)) {
76             // If the current time zone is from the right country
77             // and meets the other known properties, keep it
78             // instead of changing to another one.
79 
80             if (tz.getID().equals(currentName)) {
81                 if (currentOffset == offset && currentDst == dst) {
82                     return current;
83                 }
84             }
85 
86             // Otherwise, take the first zone from the right
87             // country that has the correct current offset and DST.
88             // (Keep iterating instead of returning in case we
89             // haven't encountered the current time zone yet.)
90 
91             if (best == null) {
92                 if (tz.getOffset(when) == offset &&
93                     tz.inDaylightTime(d) == dst) {
94                     best = tz;
95                 }
96             }
97         }
98 
99         return best;
100     }
101 
102     /**
103      * Return list of unique time zones for the country. Do not modify
104      *
105      * @param country to find
106      * @return list of unique time zones, maybe empty but never null. Do not modify.
107      * @hide
108      */
getTimeZonesWithUniqueOffsets(String country)109     public static ArrayList<TimeZone> getTimeZonesWithUniqueOffsets(String country) {
110         synchronized(sLastUniqueLockObj) {
111             if ((country != null) && country.equals(sLastUniqueCountry)) {
112                 if (DBG) {
113                     Log.d(TAG, "getTimeZonesWithUniqueOffsets(" +
114                             country + "): return cached version");
115                 }
116                 return sLastUniqueZoneOffsets;
117             }
118         }
119 
120         Collection<TimeZone> zones = getTimeZones(country);
121         ArrayList<TimeZone> uniqueTimeZones = new ArrayList<TimeZone>();
122         for (TimeZone zone : zones) {
123             // See if we already have this offset,
124             // Using slow but space efficient and these are small.
125             boolean found = false;
126             for (int i = 0; i < uniqueTimeZones.size(); i++) {
127                 if (uniqueTimeZones.get(i).getRawOffset() == zone.getRawOffset()) {
128                     found = true;
129                     break;
130                 }
131             }
132             if (found == false) {
133                 if (DBG) {
134                     Log.d(TAG, "getTimeZonesWithUniqueOffsets: add unique offset=" +
135                             zone.getRawOffset() + " zone.getID=" + zone.getID());
136                 }
137                 uniqueTimeZones.add(zone);
138             }
139         }
140 
141         synchronized(sLastUniqueLockObj) {
142             // Cache the last result
143             sLastUniqueZoneOffsets = uniqueTimeZones;
144             sLastUniqueCountry = country;
145 
146             return sLastUniqueZoneOffsets;
147         }
148     }
149 
150     /**
151      * Returns the time zones for the country, which is the code
152      * attribute of the timezone element in time_zones_by_country.xml. Do not modify.
153      *
154      * @param country is a two character country code.
155      * @return TimeZone list, maybe empty but never null. Do not modify.
156      * @hide
157      */
getTimeZones(String country)158     public static ArrayList<TimeZone> getTimeZones(String country) {
159         synchronized (sLastLockObj) {
160             if ((country != null) && country.equals(sLastCountry)) {
161                 if (DBG) Log.d(TAG, "getTimeZones(" + country + "): return cached version");
162                 return sLastZones;
163             }
164         }
165 
166         ArrayList<TimeZone> tzs = new ArrayList<TimeZone>();
167 
168         if (country == null) {
169             if (DBG) Log.d(TAG, "getTimeZones(null): return empty list");
170             return tzs;
171         }
172 
173         Resources r = Resources.getSystem();
174         XmlResourceParser parser = r.getXml(com.android.internal.R.xml.time_zones_by_country);
175 
176         try {
177             XmlUtils.beginDocument(parser, "timezones");
178 
179             while (true) {
180                 XmlUtils.nextElement(parser);
181 
182                 String element = parser.getName();
183                 if (element == null || !(element.equals("timezone"))) {
184                     break;
185                 }
186 
187                 String code = parser.getAttributeValue(null, "code");
188 
189                 if (country.equals(code)) {
190                     if (parser.next() == XmlPullParser.TEXT) {
191                         String zoneIdString = parser.getText();
192                         TimeZone tz = TimeZone.getTimeZone(zoneIdString);
193                         if (tz.getID().startsWith("GMT") == false) {
194                             // tz.getID doesn't start not "GMT" so its valid
195                             tzs.add(tz);
196                             if (DBG) {
197                                 Log.d(TAG, "getTimeZone('" + country + "'): found tz.getID=="
198                                     + ((tz != null) ? tz.getID() : "<no tz>"));
199                             }
200                         }
201                     }
202                 }
203             }
204         } catch (XmlPullParserException e) {
205             Log.e(TAG, "Got xml parser exception getTimeZone('" + country + "'): e=", e);
206         } catch (IOException e) {
207             Log.e(TAG, "Got IO exception getTimeZone('" + country + "'): e=", e);
208         } finally {
209             parser.close();
210         }
211 
212         synchronized(sLastLockObj) {
213             // Cache the last result;
214             sLastZones = tzs;
215             sLastCountry = country;
216             return sLastZones;
217         }
218     }
219 
220     /**
221      * Returns a String indicating the version of the time zone database currently
222      * in use.  The format of the string is dependent on the underlying time zone
223      * database implementation, but will typically contain the year in which the database
224      * was updated plus a letter from a to z indicating changes made within that year.
225      *
226      * <p>Time zone database updates should be expected to occur periodically due to
227      * political and legal changes that cannot be anticipated in advance.  Therefore,
228      * when computing the UTC time for a future event, applications should be aware that
229      * the results may differ following a time zone database update.  This method allows
230      * applications to detect that a database change has occurred, and to recalculate any
231      * cached times accordingly.
232      *
233      * <p>The time zone database may be assumed to change only when the device runtime
234      * is restarted.  Therefore, it is not necessary to re-query the database version
235      * during the lifetime of an activity.
236      */
getTimeZoneDatabaseVersion()237     public static String getTimeZoneDatabaseVersion() {
238         return ZoneInfoDB.getInstance().getVersion();
239     }
240 
241     /** @hide Field length that can hold 999 days of time */
242     public static final int HUNDRED_DAY_FIELD_LEN = 19;
243 
244     private static final int SECONDS_PER_MINUTE = 60;
245     private static final int SECONDS_PER_HOUR = 60 * 60;
246     private static final int SECONDS_PER_DAY = 24 * 60 * 60;
247 
248     /** @hide */
249     public static final long NANOS_PER_MS = 1000000;
250 
251     private static final Object sFormatSync = new Object();
252     private static char[] sFormatStr = new char[HUNDRED_DAY_FIELD_LEN+10];
253     private static char[] sTmpFormatStr = new char[HUNDRED_DAY_FIELD_LEN+10];
254 
accumField(int amt, int suffix, boolean always, int zeropad)255     static private int accumField(int amt, int suffix, boolean always, int zeropad) {
256         if (amt > 999) {
257             int num = 0;
258             while (amt != 0) {
259                 num++;
260                 amt /= 10;
261             }
262             return num + suffix;
263         } else {
264             if (amt > 99 || (always && zeropad >= 3)) {
265                 return 3+suffix;
266             }
267             if (amt > 9 || (always && zeropad >= 2)) {
268                 return 2+suffix;
269             }
270             if (always || amt > 0) {
271                 return 1+suffix;
272             }
273         }
274         return 0;
275     }
276 
printFieldLocked(char[] formatStr, int amt, char suffix, int pos, boolean always, int zeropad)277     static private int printFieldLocked(char[] formatStr, int amt, char suffix, int pos,
278             boolean always, int zeropad) {
279         if (always || amt > 0) {
280             final int startPos = pos;
281             if (amt > 999) {
282                 int tmp = 0;
283                 while (amt != 0 && tmp < sTmpFormatStr.length) {
284                     int dig = amt % 10;
285                     sTmpFormatStr[tmp] = (char)(dig + '0');
286                     tmp++;
287                     amt /= 10;
288                 }
289                 tmp--;
290                 while (tmp >= 0) {
291                     formatStr[pos] = sTmpFormatStr[tmp];
292                     pos++;
293                     tmp--;
294                 }
295             } else {
296                 if ((always && zeropad >= 3) || amt > 99) {
297                     int dig = amt/100;
298                     formatStr[pos] = (char)(dig + '0');
299                     pos++;
300                     amt -= (dig*100);
301                 }
302                 if ((always && zeropad >= 2) || amt > 9 || startPos != pos) {
303                     int dig = amt/10;
304                     formatStr[pos] = (char)(dig + '0');
305                     pos++;
306                     amt -= (dig*10);
307                 }
308                 formatStr[pos] = (char)(amt + '0');
309                 pos++;
310             }
311             formatStr[pos] = suffix;
312             pos++;
313         }
314         return pos;
315     }
316 
formatDurationLocked(long duration, int fieldLen)317     private static int formatDurationLocked(long duration, int fieldLen) {
318         if (sFormatStr.length < fieldLen) {
319             sFormatStr = new char[fieldLen];
320         }
321 
322         char[] formatStr = sFormatStr;
323 
324         if (duration == 0) {
325             int pos = 0;
326             fieldLen -= 1;
327             while (pos < fieldLen) {
328                 formatStr[pos++] = ' ';
329             }
330             formatStr[pos] = '0';
331             return pos+1;
332         }
333 
334         char prefix;
335         if (duration > 0) {
336             prefix = '+';
337         } else {
338             prefix = '-';
339             duration = -duration;
340         }
341 
342         int millis = (int)(duration%1000);
343         int seconds = (int) Math.floor(duration / 1000);
344         int days = 0, hours = 0, minutes = 0;
345 
346         if (seconds > SECONDS_PER_DAY) {
347             days = seconds / SECONDS_PER_DAY;
348             seconds -= days * SECONDS_PER_DAY;
349         }
350         if (seconds > SECONDS_PER_HOUR) {
351             hours = seconds / SECONDS_PER_HOUR;
352             seconds -= hours * SECONDS_PER_HOUR;
353         }
354         if (seconds > SECONDS_PER_MINUTE) {
355             minutes = seconds / SECONDS_PER_MINUTE;
356             seconds -= minutes * SECONDS_PER_MINUTE;
357         }
358 
359         int pos = 0;
360 
361         if (fieldLen != 0) {
362             int myLen = accumField(days, 1, false, 0);
363             myLen += accumField(hours, 1, myLen > 0, 2);
364             myLen += accumField(minutes, 1, myLen > 0, 2);
365             myLen += accumField(seconds, 1, myLen > 0, 2);
366             myLen += accumField(millis, 2, true, myLen > 0 ? 3 : 0) + 1;
367             while (myLen < fieldLen) {
368                 formatStr[pos] = ' ';
369                 pos++;
370                 myLen++;
371             }
372         }
373 
374         formatStr[pos] = prefix;
375         pos++;
376 
377         int start = pos;
378         boolean zeropad = fieldLen != 0;
379         pos = printFieldLocked(formatStr, days, 'd', pos, false, 0);
380         pos = printFieldLocked(formatStr, hours, 'h', pos, pos != start, zeropad ? 2 : 0);
381         pos = printFieldLocked(formatStr, minutes, 'm', pos, pos != start, zeropad ? 2 : 0);
382         pos = printFieldLocked(formatStr, seconds, 's', pos, pos != start, zeropad ? 2 : 0);
383         pos = printFieldLocked(formatStr, millis, 'm', pos, true, (zeropad && pos != start) ? 3 : 0);
384         formatStr[pos] = 's';
385         return pos + 1;
386     }
387 
388     /** @hide Just for debugging; not internationalized. */
formatDuration(long duration, StringBuilder builder)389     public static void formatDuration(long duration, StringBuilder builder) {
390         synchronized (sFormatSync) {
391             int len = formatDurationLocked(duration, 0);
392             builder.append(sFormatStr, 0, len);
393         }
394     }
395 
396     /** @hide Just for debugging; not internationalized. */
formatDuration(long duration, PrintWriter pw, int fieldLen)397     public static void formatDuration(long duration, PrintWriter pw, int fieldLen) {
398         synchronized (sFormatSync) {
399             int len = formatDurationLocked(duration, fieldLen);
400             pw.print(new String(sFormatStr, 0, len));
401         }
402     }
403 
404     /** @hide Just for debugging; not internationalized. */
formatDuration(long duration, PrintWriter pw)405     public static void formatDuration(long duration, PrintWriter pw) {
406         formatDuration(duration, pw, 0);
407     }
408 
409     /** @hide Just for debugging; not internationalized. */
formatDuration(long time, long now, PrintWriter pw)410     public static void formatDuration(long time, long now, PrintWriter pw) {
411         if (time == 0) {
412             pw.print("--");
413             return;
414         }
415         formatDuration(time-now, pw, 0);
416     }
417 
418     /** @hide Just for debugging; not internationalized. */
formatUptime(long time)419     public static String formatUptime(long time) {
420         final long diff = time - SystemClock.uptimeMillis();
421         if (diff > 0) {
422             return time + " (in " + diff + " ms)";
423         }
424         if (diff < 0) {
425             return time + " (" + -diff + " ms ago)";
426         }
427         return time + " (now)";
428     }
429 
430     /**
431      * Convert a System.currentTimeMillis() value to a time of day value like
432      * that printed in logs. MM-DD HH:MM:SS.MMM
433      *
434      * @param millis since the epoch (1/1/1970)
435      * @return String representation of the time.
436      * @hide
437      */
logTimeOfDay(long millis)438     public static String logTimeOfDay(long millis) {
439         Calendar c = Calendar.getInstance();
440         if (millis >= 0) {
441             c.setTimeInMillis(millis);
442             return String.format("%tm-%td %tH:%tM:%tS.%tL", c, c, c, c, c, c);
443         } else {
444             return Long.toString(millis);
445         }
446     }
447 
448     /** {@hide} */
formatForLogging(long millis)449     public static String formatForLogging(long millis) {
450         if (millis <= 0) {
451             return "unknown";
452         } else {
453             return sLoggingFormat.format(new Date(millis));
454         }
455     }
456 }
457