1 /*
2  * Copyright (C) 2012 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.telephony;
18 
19 import android.os.Build;
20 import android.text.TextUtils;
21 import android.util.Log;
22 
23 import android.util.Base64;
24 
25 import java.security.MessageDigest;
26 import java.security.NoSuchAlgorithmException;
27 
28 
29 /**
30  * A class to log strings to the RADIO LOG.
31  *
32  * @hide
33  */
34 public final class Rlog {
35 
36     private static final boolean USER_BUILD = Build.TYPE.equals("user");
37 
Rlog()38     private Rlog() {
39     }
40 
v(String tag, String msg)41     public static int v(String tag, String msg) {
42         return Log.println_native(Log.LOG_ID_RADIO, Log.VERBOSE, tag, msg);
43     }
44 
v(String tag, String msg, Throwable tr)45     public static int v(String tag, String msg, Throwable tr) {
46         return Log.println_native(Log.LOG_ID_RADIO, Log.VERBOSE, tag,
47                 msg + '\n' + Log.getStackTraceString(tr));
48     }
49 
d(String tag, String msg)50     public static int d(String tag, String msg) {
51         return Log.println_native(Log.LOG_ID_RADIO, Log.DEBUG, tag, msg);
52     }
53 
d(String tag, String msg, Throwable tr)54     public static int d(String tag, String msg, Throwable tr) {
55         return Log.println_native(Log.LOG_ID_RADIO, Log.DEBUG, tag,
56                 msg + '\n' + Log.getStackTraceString(tr));
57     }
58 
i(String tag, String msg)59     public static int i(String tag, String msg) {
60         return Log.println_native(Log.LOG_ID_RADIO, Log.INFO, tag, msg);
61     }
62 
i(String tag, String msg, Throwable tr)63     public static int i(String tag, String msg, Throwable tr) {
64         return Log.println_native(Log.LOG_ID_RADIO, Log.INFO, tag,
65                 msg + '\n' + Log.getStackTraceString(tr));
66     }
67 
w(String tag, String msg)68     public static int w(String tag, String msg) {
69         return Log.println_native(Log.LOG_ID_RADIO, Log.WARN, tag, msg);
70     }
71 
w(String tag, String msg, Throwable tr)72     public static int w(String tag, String msg, Throwable tr) {
73         return Log.println_native(Log.LOG_ID_RADIO, Log.WARN, tag,
74                 msg + '\n' + Log.getStackTraceString(tr));
75     }
76 
w(String tag, Throwable tr)77     public static int w(String tag, Throwable tr) {
78         return Log.println_native(Log.LOG_ID_RADIO, Log.WARN, tag, Log.getStackTraceString(tr));
79     }
80 
e(String tag, String msg)81     public static int e(String tag, String msg) {
82         return Log.println_native(Log.LOG_ID_RADIO, Log.ERROR, tag, msg);
83     }
84 
e(String tag, String msg, Throwable tr)85     public static int e(String tag, String msg, Throwable tr) {
86         return Log.println_native(Log.LOG_ID_RADIO, Log.ERROR, tag,
87                 msg + '\n' + Log.getStackTraceString(tr));
88     }
89 
println(int priority, String tag, String msg)90     public static int println(int priority, String tag, String msg) {
91         return Log.println_native(Log.LOG_ID_RADIO, priority, tag, msg);
92     }
93 
isLoggable(String tag, int level)94     public static boolean isLoggable(String tag, int level) {
95         return Log.isLoggable(tag, level);
96     }
97 
98     /**
99      * Redact personally identifiable information for production users.
100      * @param tag used to identify the source of a log message
101      * @param pii the personally identifiable information we want to apply secure hash on.
102      * @return If tag is loggable in verbose mode or pii is null, return the original input.
103      * otherwise return a secure Hash of input pii
104      */
pii(String tag, Object pii)105     public static String pii(String tag, Object pii) {
106         String val = String.valueOf(pii);
107         if (pii == null || TextUtils.isEmpty(val) || isLoggable(tag, Log.VERBOSE)) {
108             return val;
109         }
110         return "[" + secureHash(val.getBytes()) + "]";
111     }
112 
113     /**
114      * Redact personally identifiable information for production users.
115      * @param enablePiiLogging set when caller explicitly want to enable sensitive logging.
116      * @param pii the personally identifiable information we want to apply secure hash on.
117      * @return If enablePiiLogging is set to true or pii is null, return the original input.
118      * otherwise return a secure Hash of input pii
119      */
pii(boolean enablePiiLogging, Object pii)120     public static String pii(boolean enablePiiLogging, Object pii) {
121         String val = String.valueOf(pii);
122         if (pii == null || TextUtils.isEmpty(val) || enablePiiLogging) {
123             return val;
124         }
125         return "[" + secureHash(val.getBytes()) + "]";
126     }
127 
128     /**
129      * Returns a secure hash (using the SHA1 algorithm) of the provided input.
130      *
131      * @return "****" if the build type is user, otherwise the hash
132      * @param input the bytes for which the secure hash should be computed.
133      */
secureHash(byte[] input)134     private static String secureHash(byte[] input) {
135         // Refrain from logging user personal information in user build.
136         if (USER_BUILD) {
137             return "****";
138         }
139 
140         MessageDigest messageDigest;
141 
142         try {
143             messageDigest = MessageDigest.getInstance("SHA-1");
144         } catch (NoSuchAlgorithmException e) {
145             return "####";
146         }
147 
148         byte[] result = messageDigest.digest(input);
149         return Base64.encodeToString(
150                 result, Base64.URL_SAFE | Base64.NO_PADDING | Base64.NO_WRAP);
151     }
152 }
153 
154