231 lines
7.7 KiB
Java
231 lines
7.7 KiB
Java
![]() |
/*
|
||
|
* Copyright (C) 2020 The Android Open Source Project
|
||
|
*
|
||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||
|
* you may not use this file except in compliance with the License.
|
||
|
* You may obtain a copy of the License at
|
||
|
*
|
||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||
|
*
|
||
|
* Unless required by applicable law or agreed to in writing, software
|
||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||
|
* See the License for the specific language governing permissions and
|
||
|
* limitations under the License.
|
||
|
*/
|
||
|
package com.android.telephony;
|
||
|
|
||
|
import android.net.Uri;
|
||
|
import android.os.Build;
|
||
|
import android.telecom.PhoneAccount;
|
||
|
import android.telephony.PhoneNumberUtils;
|
||
|
import android.text.TextUtils;
|
||
|
import android.util.Base64;
|
||
|
import android.util.Log;
|
||
|
|
||
|
import com.android.internal.telephony.util.TelephonyUtils;
|
||
|
|
||
|
import java.security.MessageDigest;
|
||
|
import java.security.NoSuchAlgorithmException;
|
||
|
|
||
|
/**
|
||
|
* A copy of {@link android.telephony.Rlog} to be used within the telephony mainline module.
|
||
|
*
|
||
|
* @hide
|
||
|
*/
|
||
|
public final class Rlog {
|
||
|
|
||
|
private static final boolean USER_BUILD = TelephonyUtils.IS_USER;
|
||
|
|
||
|
private Rlog() {
|
||
|
}
|
||
|
|
||
|
private static int log(int priority, String tag, String msg) {
|
||
|
return Log.logToRadioBuffer(priority, tag, msg);
|
||
|
}
|
||
|
|
||
|
public static int v(String tag, String msg) {
|
||
|
return log(Log.VERBOSE, tag, msg);
|
||
|
}
|
||
|
|
||
|
public static int v(String tag, String msg, Throwable tr) {
|
||
|
return log(Log.VERBOSE, tag,
|
||
|
msg + '\n' + Log.getStackTraceString(tr));
|
||
|
}
|
||
|
|
||
|
public static int d(String tag, String msg) {
|
||
|
return log(Log.DEBUG, tag, msg);
|
||
|
}
|
||
|
|
||
|
public static int d(String tag, String msg, Throwable tr) {
|
||
|
return log(Log.DEBUG, tag,
|
||
|
msg + '\n' + Log.getStackTraceString(tr));
|
||
|
}
|
||
|
|
||
|
public static int i(String tag, String msg) {
|
||
|
return log(Log.INFO, tag, msg);
|
||
|
}
|
||
|
|
||
|
public static int i(String tag, String msg, Throwable tr) {
|
||
|
return log(Log.INFO, tag,
|
||
|
msg + '\n' + Log.getStackTraceString(tr));
|
||
|
}
|
||
|
|
||
|
public static int w(String tag, String msg) {
|
||
|
return log(Log.WARN, tag, msg);
|
||
|
}
|
||
|
|
||
|
public static int w(String tag, String msg, Throwable tr) {
|
||
|
return log(Log.WARN, tag,
|
||
|
msg + '\n' + Log.getStackTraceString(tr));
|
||
|
}
|
||
|
|
||
|
public static int w(String tag, Throwable tr) {
|
||
|
return log(Log.WARN, tag, Log.getStackTraceString(tr));
|
||
|
}
|
||
|
|
||
|
public static int e(String tag, String msg) {
|
||
|
return log(Log.ERROR, tag, msg);
|
||
|
}
|
||
|
|
||
|
public static int e(String tag, String msg, Throwable tr) {
|
||
|
return log(Log.ERROR, tag,
|
||
|
msg + '\n' + Log.getStackTraceString(tr));
|
||
|
}
|
||
|
|
||
|
public static int println(int priority, String tag, String msg) {
|
||
|
return log(priority, tag, msg);
|
||
|
}
|
||
|
|
||
|
public static boolean isLoggable(String tag, int level) {
|
||
|
return Log.isLoggable(tag, level);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Redact personally identifiable information for production users.
|
||
|
* @param tag used to identify the source of a log message
|
||
|
* @param pii the personally identifiable information we want to apply secure hash on.
|
||
|
* @return If tag is loggable in verbose mode or pii is null, return the original input.
|
||
|
* otherwise return a secure Hash of input pii
|
||
|
*/
|
||
|
public static String pii(String tag, Object pii) {
|
||
|
String val = String.valueOf(pii);
|
||
|
if (pii == null || TextUtils.isEmpty(val) || isLoggable(tag, Log.VERBOSE)) {
|
||
|
return val;
|
||
|
}
|
||
|
return "[" + secureHash(val.getBytes()) + "]";
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Redact personally identifiable information for production users.
|
||
|
* @param enablePiiLogging set when caller explicitly want to enable sensitive logging.
|
||
|
* @param pii the personally identifiable information we want to apply secure hash on.
|
||
|
* @return If enablePiiLogging is set to true or pii is null, return the original input.
|
||
|
* otherwise return a secure Hash of input pii
|
||
|
*/
|
||
|
public static String pii(boolean enablePiiLogging, Object pii) {
|
||
|
String val = String.valueOf(pii);
|
||
|
if (pii == null || TextUtils.isEmpty(val) || enablePiiLogging) {
|
||
|
return val;
|
||
|
}
|
||
|
return "[" + secureHash(val.getBytes()) + "]";
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Generates an obfuscated string for a calling handle in {@link Uri} format, or a raw phone
|
||
|
* phone number in {@link String} format.
|
||
|
* @param pii The information to obfuscate.
|
||
|
* @return The obfuscated string.
|
||
|
*/
|
||
|
public static String piiHandle(Object pii) {
|
||
|
StringBuilder sb = new StringBuilder();
|
||
|
if (pii instanceof Uri) {
|
||
|
Uri uri = (Uri) pii;
|
||
|
String scheme = uri.getScheme();
|
||
|
|
||
|
if (!TextUtils.isEmpty(scheme)) {
|
||
|
sb.append(scheme).append(":");
|
||
|
}
|
||
|
|
||
|
String textToObfuscate = uri.getSchemeSpecificPart();
|
||
|
if (PhoneAccount.SCHEME_TEL.equals(scheme)) {
|
||
|
obfuscatePhoneNumber(sb, textToObfuscate);
|
||
|
} else if (PhoneAccount.SCHEME_SIP.equals(scheme)) {
|
||
|
for (int i = 0; i < textToObfuscate.length(); i++) {
|
||
|
char c = textToObfuscate.charAt(i);
|
||
|
if (c != '@' && c != '.') {
|
||
|
c = '*';
|
||
|
}
|
||
|
sb.append(c);
|
||
|
}
|
||
|
} else {
|
||
|
sb.append("***");
|
||
|
}
|
||
|
} else if (pii instanceof String) {
|
||
|
String number = (String) pii;
|
||
|
obfuscatePhoneNumber(sb, number);
|
||
|
}
|
||
|
|
||
|
return sb.toString();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Obfuscates a phone number, allowing NUM_DIALABLE_DIGITS_TO_LOG digits to be exposed for the
|
||
|
* phone number.
|
||
|
* @param sb String buffer to write obfuscated number to.
|
||
|
* @param phoneNumber The number to obfuscate.
|
||
|
*/
|
||
|
private static void obfuscatePhoneNumber(StringBuilder sb, String phoneNumber) {
|
||
|
int numDigitsToLog = USER_BUILD ? 0 : 2;
|
||
|
int numDigitsToObfuscate = getDialableCount(phoneNumber) - numDigitsToLog;
|
||
|
for (int i = 0; i < phoneNumber.length(); i++) {
|
||
|
char c = phoneNumber.charAt(i);
|
||
|
boolean isDialable = PhoneNumberUtils.isDialable(c);
|
||
|
if (isDialable) {
|
||
|
numDigitsToObfuscate--;
|
||
|
}
|
||
|
sb.append(isDialable && numDigitsToObfuscate >= 0 ? "*" : c);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Determines the number of dialable characters in a string.
|
||
|
* @param toCount The string to count dialable characters in.
|
||
|
* @return The count of dialable characters.
|
||
|
*/
|
||
|
private static int getDialableCount(String toCount) {
|
||
|
int numDialable = 0;
|
||
|
for (char c : toCount.toCharArray()) {
|
||
|
if (PhoneNumberUtils.isDialable(c)) {
|
||
|
numDialable++;
|
||
|
}
|
||
|
}
|
||
|
return numDialable;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Returns a secure hash (using the SHA1 algorithm) of the provided input.
|
||
|
*
|
||
|
* @return "****" if the build type is user, otherwise the hash
|
||
|
* @param input the bytes for which the secure hash should be computed.
|
||
|
*/
|
||
|
private static String secureHash(byte[] input) {
|
||
|
// Refrain from logging user personal information in user build.
|
||
|
if (USER_BUILD) {
|
||
|
return "****";
|
||
|
}
|
||
|
|
||
|
MessageDigest messageDigest;
|
||
|
|
||
|
try {
|
||
|
messageDigest = MessageDigest.getInstance("SHA-1");
|
||
|
} catch (NoSuchAlgorithmException e) {
|
||
|
return "####";
|
||
|
}
|
||
|
|
||
|
byte[] result = messageDigest.digest(input);
|
||
|
return Base64.encodeToString(
|
||
|
result, Base64.URL_SAFE | Base64.NO_PADDING | Base64.NO_WRAP);
|
||
|
}
|
||
|
}
|