1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
docs / android_logging.md [blame]
# Logging
[TOC]
## Overview
Logging used to be done using Android's
[android.util.Log](https://developer.android.com/reference/android/util/Log.html).
A wrapper on that is now available:
[org.chromium.base.Log](/base/android/java/src/org/chromium/base/Log.java). It
is designed to write logs as belonging to logical groups going beyond single
classes, and to make it easy to switch logging on or off for individual groups.
Usage:
```java
private static final String TAG = "YourModuleTag";
...
Log.i(TAG, "Logged INFO message.");
Log.d(TAG, "Some DEBUG info: %s", data);
```
Output:
```
I/cr_YourModuleTag: ( 999): Logged INFO message
D/cr_YourModuleTag: ( 999): [MyClass.java:42] Some DEBUG info: data.toString
```
Here, **TAG** will be a feature or package name, "MediaRemote" or "NFC" for
example. In most cases, the class name is not needed. It will be prepended by
the "cr\_" prefix to make obvious which logs are coming from Chrome.
### Verbose and Debug logs have special handling
* `Log.v` and `Log.d` Calls made using `org.chromium.base.Log` are stripped
out of production binaries using Proguard. There is no way to get those logs
in release builds.
* The file name and line number will be prepended to the log message.
For higher priority logs, those are not added for performance concerns.
### An exception trace is printed when the exception is the last parameter
As with `android.util.Log`, putting a throwable as last parameter will dump the
corresponding stack trace:
```java
Log.i(TAG, "An error happened: %s", e)
```
```
I/cr_YourModuleTag: ( 999): An error happened: This is the exception's message
I/cr_YourModuleTag: ( 999): java.lang.Exception: This is the exception's message
I/cr_YourModuleTag: ( 999): at foo.bar.MyClass.test(MyClass.java:42)
I/cr_YourModuleTag: ( 999): ...
```
Having the exception as last parameter doesn't prevent it from being used for
string formatting.
## Logging Best Practices
### Rule #1: Never log user data or PII (Personal Identification Information)
This is a huge concern, because other applications can access the log and
extract a lot of data from your own by doing so. Even if JellyBean restricted
this, people are going to run your application on rooted devices and allow some
apps to access it. Also anyone with USB access to the device can use ADB to get
the full logcat and get the same data right now.
If you really need to print something, print a series of Xs instead
(e.g. "XXXXXX"), or print a truncated hash of the data instead. Truncation is
required to make it harder for an attacker to recover the full data through
rainbow tables and similar methods.
Similarly, avoid dumping API keys, cookies, IP addresses, URLs, page content,
etc...
### Rule #2: Do not build debug logs in production code
The log methods are removed in release builds using Proguard. Because log
messages might not be written, the cost of creating them should also be avoided.
This can be done using three complementary ways:
#### Use string formatting instead of concatenations
```java
// BAD
Log.d(TAG, "I " + preference + " writing logs.");
// BETTER
Log.d(TAG, "I %s writing logs.", preference);
```
Proguard removes the method call itself, but doesn't do anything about the
arguments. The method's arguments will still be computed and provided as
input. The first call above will always lead to the creation of a
`StringBuilder` and a few concatenations, while the second just passes the
arguments and won't need that.
#### Guard expensive calls
Sometimes the values to log aren't readily available and need to be computed
specially. This should be avoided when logging is disabled.
```java
...
if (Log.isLoggable(TAG, Log.INFO)) {
Log.i(TAG, createThatExpensiveLogMessage(activity))
}
```
Because the variable is a `static final` that can be evaluated at compile
time, the Java compiler will optimize out all guarded calls from the
generated `.class` file. Changing it however requires editing each of the
files for which debug should be enabled and recompiling.
### Rule #3: Favor small log messages
This is still related to the global fixed-sized kernel buffer used to keep all
logs. Try to make your log information as terse as possible. This reduces the
risk of pushing interesting log data out of the buffer when something really
nasty happens. It's really better to have a single-line log message, than
several ones. I.e. don't use:
```java
Log.GROUP.d(TAG, "field1 = %s", value1);
Log.GROUP.d(TAG, "field2 = %s", value2);
Log.GROUP.d(TAG, "field3 = %s", value3);
```
Instead, write this as:
```java
Log.d(TAG, "field1 = %s, field2 = %s, field3 = %s", value1, value2, value3);
```
That doesn't seem to be much different if you count overall character counts,
but each independent log entry also implies a small, but non-trivial header, in
the kernel log buffer. And since every byte count, you can also try something
even shorter, as in:
```java
Log.d(TAG, "fields [%s,%s,%s]", value1, value2, value3);
```
## Filtering logs
Logcat allows filtering by specifying tags and the associated level:
```shell
adb logcat [TAG_EXPR:LEVEL]...
adb logcat cr_YourModuleTag:D *:S
```
This shows only logs having a level higher or equal to DEBUG for
`cr_YourModuleTag`, and SILENT (nothing is logged at this level or higher, so it
silences the tags) for everything else. You can persist a filter by setting an
environment variable:
```shell
export ANDROID_LOG_TAGS="cr_YourModuleTag:D *:S"
```
The syntax does not support tag expansion or regular expressions other than `*`
for all tags. Please use `grep` or a similar tool to refine your filters
further.
For more, see the [related page on developer.android.com]
(https://developer.android.com/tools/debugging/debugging-log.html#filteringOutput)
## Logs in JUnit tests
We use [robolectric](http://robolectric.org/) to run our JUnit tests. It
replaces some of the Android framework classes with "Shadow" classes
to ensure that we can run our code in a regular JVM. `android.util.Log` is one
of those replaced classes, and by default calling `Log` methods doesn't print
anything.
That default is not changed in the normal configuration, but if you need to
enable logging locally or for a specific test, just add those few lines to your
test:
```java
@Before
public void setUp() {
ShadowLog.stream = System.out;
// Your other setup here
}
```